Skip to content

Commit

Permalink
api: the Datetime type is immutable
Browse files Browse the repository at this point in the history
The patch forces the use of objects of type Datetime instead
of pointers.

Part of #238
  • Loading branch information
oleg-jukovec committed Jun 24, 2023
1 parent 5f76bc6 commit 22de648
Show file tree
Hide file tree
Showing 5 changed files with 87 additions and 75 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ Versioning](http://semver.org/spec/v2.0.0.html) except to the first release.
- Use msgpack/v5 instead of msgpack.v2 (#236)
- Call/NewCallRequest = Call17/NewCall17Request (#235)
- Use objects of the Decimal type instead of pointers (#238)
- Use objects of the Datetime type instead of pointers (#238)

### Deprecated

Expand Down
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ faster than other packages according to public benchmarks.
* [API reference](#api-reference)
* [Walking\-through example](#walking-through-example)
* [Migration to v2](#migration-to-v2)
* [datetime package](#datetime-package)
* [decimal package](#decimal-package)
* [multi package](#multi-package)
* [pool package](#pool-package)
Expand Down Expand Up @@ -149,6 +150,12 @@ by `Connect()`.

The article describes migration from go-tarantool to go-tarantool/v2.

#### datetime package

Now you need to use objects of the Datetime type instead of pointers to it. A
new constructor `MakeDatetime` returns an object. `NewDatetime` has been
removed.

#### decimal package

Now you need to use objects of the Decimal type instead of pointers to it. A
Expand Down
66 changes: 38 additions & 28 deletions datetime/datetime.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ package datetime
import (
"encoding/binary"
"fmt"
"reflect"
"time"

"github.com/vmihailenco/msgpack/v5"
Expand All @@ -35,7 +36,7 @@ import (

// Datetime external type. Supported since Tarantool 2.10. See more details in
// issue https://github.com/tarantool/tarantool/issues/5946.
const datetime_extId = 4
const datetimeExtID = 4

// datetime structure keeps a number of seconds and nanoseconds since Unix Epoch.
// Time is normalized by UTC, so time-zone offset is informative only.
Expand Down Expand Up @@ -93,41 +94,41 @@ const (
offsetMax = 14 * 60 * 60
)

// NewDatetime returns a pointer to a new datetime.Datetime that contains a
// MakeDatetime returns a datetime.Datetime object that contains a
// specified time.Time. It may return an error if the Time value is out of
// supported range: [-5879610-06-22T00:00Z .. 5879611-07-11T00:00Z] or
// an invalid timezone or offset value is out of supported range:
// [-12 * 60 * 60, 14 * 60 * 60].
//
// NOTE: Tarantool's datetime.tz value is picked from t.Location().String().
// "Local" location is unsupported, see ExampleNewDatetime_localUnsupported.
func NewDatetime(t time.Time) (*Datetime, error) {
func MakeDatetime(t time.Time) (Datetime, error) {
dt := Datetime{}
seconds := t.Unix()

if seconds < minSeconds || seconds > maxSeconds {
return nil, fmt.Errorf("time %s is out of supported range", t)
return dt, fmt.Errorf("time %s is out of supported range", t)
}

zone := t.Location().String()
_, offset := t.Zone()
if zone != NoTimezone {
if _, ok := timezoneToIndex[zone]; !ok {
return nil, fmt.Errorf("unknown timezone %s with offset %d",
return dt, fmt.Errorf("unknown timezone %s with offset %d",
zone, offset)
}
}

if offset < offsetMin || offset > offsetMax {
return nil, fmt.Errorf("offset must be between %d and %d hours",
return dt, fmt.Errorf("offset must be between %d and %d hours",
offsetMin, offsetMax)
}

dt := new(Datetime)
dt.time = t
return dt, nil
}

func intervalFromDatetime(dtime *Datetime) (ival Interval) {
func intervalFromDatetime(dtime Datetime) (ival Interval) {
ival.Year = int64(dtime.time.Year())
ival.Month = int64(dtime.time.Month())
ival.Day = int64(dtime.time.Day())
Expand Down Expand Up @@ -158,7 +159,7 @@ func daysInMonth(year int64, month int64) int64 {

// C implementation:
// https://github.com/tarantool/c-dt/blob/cec6acebb54d9e73ea0b99c63898732abd7683a6/dt_arithmetic.c#L74-L98
func addMonth(ival *Interval, delta int64, adjust Adjust) {
func addMonth(ival Interval, delta int64, adjust Adjust) Interval {
oldYear := ival.Year
oldMonth := ival.Month

Expand All @@ -172,16 +173,17 @@ func addMonth(ival *Interval, delta int64, adjust Adjust) {
}
}
if adjust == ExcessAdjust || ival.Day < 28 {
return
return ival
}

dim := daysInMonth(ival.Year, ival.Month)
if ival.Day > dim || (adjust == LastAdjust && ival.Day == daysInMonth(oldYear, oldMonth)) {
ival.Day = dim
}
return ival
}

func (dtime *Datetime) add(ival Interval, positive bool) (*Datetime, error) {
func (dtime Datetime) add(ival Interval, positive bool) (Datetime, error) {
newVal := intervalFromDatetime(dtime)

var direction int64
Expand All @@ -191,7 +193,7 @@ func (dtime *Datetime) add(ival Interval, positive bool) (*Datetime, error) {
direction = -1
}

addMonth(&newVal, direction*ival.Year*12+direction*ival.Month, ival.Adjust)
newVal = addMonth(newVal, direction*ival.Year*12+direction*ival.Month, ival.Adjust)
newVal.Day += direction * 7 * ival.Week
newVal.Day += direction * ival.Day
newVal.Hour += direction * ival.Hour
Expand All @@ -203,23 +205,23 @@ func (dtime *Datetime) add(ival Interval, positive bool) (*Datetime, error) {
int(newVal.Day), int(newVal.Hour), int(newVal.Min),
int(newVal.Sec), int(newVal.Nsec), dtime.time.Location())

return NewDatetime(tm)
return MakeDatetime(tm)
}

// Add creates a new Datetime as addition of the Datetime and Interval. It may
// return an error if a new Datetime is out of supported range.
func (dtime *Datetime) Add(ival Interval) (*Datetime, error) {
func (dtime Datetime) Add(ival Interval) (Datetime, error) {
return dtime.add(ival, true)
}

// Sub creates a new Datetime as subtraction of the Datetime and Interval. It
// may return an error if a new Datetime is out of supported range.
func (dtime *Datetime) Sub(ival Interval) (*Datetime, error) {
func (dtime Datetime) Sub(ival Interval) (Datetime, error) {
return dtime.add(ival, false)
}

// Interval returns an Interval value to a next Datetime value.
func (dtime *Datetime) Interval(next *Datetime) Interval {
func (dtime Datetime) Interval(next Datetime) Interval {
curIval := intervalFromDatetime(dtime)
nextIval := intervalFromDatetime(next)
_, curOffset := dtime.time.Zone()
Expand All @@ -236,11 +238,12 @@ func (dtime *Datetime) Interval(next *Datetime) Interval {
// If a Datetime created via unmarshaling Tarantool's datetime then we try to
// create a location with time.LoadLocation() first. In case of failure, we use
// a location created with time.FixedZone().
func (dtime *Datetime) ToTime() time.Time {
func (dtime Datetime) ToTime() time.Time {
return dtime.time
}

func (dtime *Datetime) MarshalMsgpack() ([]byte, error) {
func datetimeEncoder(e *msgpack.Encoder, v reflect.Value) ([]byte, error) {
dtime := v.Interface().(Datetime)
tm := dtime.ToTime()

var dt datetime
Expand Down Expand Up @@ -272,17 +275,25 @@ func (dtime *Datetime) MarshalMsgpack() ([]byte, error) {
return buf, nil
}

func (tm *Datetime) UnmarshalMsgpack(b []byte) error {
l := len(b)
if l != maxSize && l != secondsSize {
return fmt.Errorf("invalid data length: got %d, wanted %d or %d", len(b), secondsSize, maxSize)
func datetimeDecoder(d *msgpack.Decoder, v reflect.Value, extLen int) error {
if extLen != maxSize && extLen != secondsSize {
return fmt.Errorf("invalid data length: got %d, wanted %d or %d", extLen, secondsSize, maxSize)
}

b := make([]byte, extLen)
n, err := d.Buffered().Read(b)
if err != nil {
return err
}
if n < extLen {
return fmt.Errorf("msgpack: unexpected end of stream after %d datetime bytes", n)
}

var dt datetime
sec := binary.LittleEndian.Uint64(b)
dt.seconds = int64(sec)
dt.nsec = 0
if l == maxSize {
if extLen == maxSize {
dt.nsec = int32(binary.LittleEndian.Uint32(b[secondsSize:]))
dt.tzOffset = int16(binary.LittleEndian.Uint16(b[secondsSize+nsecSize:]))
dt.tzIndex = int16(binary.LittleEndian.Uint16(b[secondsSize+nsecSize+tzOffsetSize:]))
Expand Down Expand Up @@ -315,13 +326,12 @@ func (tm *Datetime) UnmarshalMsgpack(b []byte) error {
}
tt = tt.In(loc)

dtp, err := NewDatetime(tt)
if dtp != nil {
*tm = *dtp
}
ptr := v.Addr().Interface().(*Datetime)
*ptr, err = MakeDatetime(tt)
return err
}

func init() {
msgpack.RegisterExt(datetime_extId, (*Datetime)(nil))
msgpack.RegisterExtDecoder(datetimeExtID, Datetime{}, datetimeDecoder)
msgpack.RegisterExtEncoder(datetimeExtID, Datetime{}, datetimeEncoder)
}
Loading

0 comments on commit 22de648

Please sign in to comment.