Skip to content

Commit

Permalink
Always return a valid timezone in cursor
Browse files Browse the repository at this point in the history
Conversion functions into a Go `time.Time` expect to have a valid
`*time.Location` and will panic if passed in `nil`.

Before we used also here `nil` to signal no timezone is set (and thus
falling back to `time.Local` implicitly, but it's better to always
return a timezone explicitly and use `time.Local` where appropriate
then.

Signed-off-by: Dirkjan Bussink <[email protected]>
  • Loading branch information
dbussink committed Jan 15, 2025
1 parent 859632f commit f33200f
Show file tree
Hide file tree
Showing 4 changed files with 19 additions and 12 deletions.
2 changes: 1 addition & 1 deletion go/vt/vtgate/engine/fake_vcursor_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -164,7 +164,7 @@ func (t *noopVCursor) Environment() *vtenv.Environment {
}

func (t *noopVCursor) TimeZone() *time.Location {
return nil
return time.Local
}

func (t *noopVCursor) SQLMode() string {
Expand Down
4 changes: 2 additions & 2 deletions go/vt/vtgate/evalengine/fn_time.go
Original file line number Diff line number Diff line change
Expand Up @@ -256,7 +256,7 @@ func (call *builtinNow) constant() bool {

func (call *builtinSysdate) eval(env *ExpressionEnv) (eval, error) {
now := SystemTime()
if tz := env.currentTimezone(); tz != nil {
if tz := env.currentTimezone(); tz != time.Local {
now = now.In(tz)
}
return newEvalDateTime(datetime.NewDateTimeFromStd(now), int(call.prec), false), nil
Expand Down Expand Up @@ -701,7 +701,7 @@ func (b *builtinFromUnixtime) eval(env *ExpressionEnv) (eval, error) {
}

t := time.Unix(sec, frac)
if tz := env.currentTimezone(); tz != nil {
if tz := env.currentTimezone(); tz != time.Local {
t = t.In(tz)
}

Expand Down
11 changes: 6 additions & 5 deletions go/vt/vtgate/executorcontext/safe_session.go
Original file line number Diff line number Diff line change
Expand Up @@ -656,17 +656,18 @@ func (session *SafeSession) TimeZone() *time.Location {
session.mu.Unlock()

if !ok {
return nil
return time.Local
}

tz, err := sqltypes.DecodeStringSQL(zoneSQL)
if err != nil {
return nil
return time.Local
}

loc, _ := datetime.ParseTimeZone(tz)
// it's safe to ignore the error - if we get an error, loc will be nil,
// and this is exactly the behaviour we want anyway
loc, err := datetime.ParseTimeZone(tz)
if err != nil {
return time.Local
}
return loc
}

Expand Down
14 changes: 10 additions & 4 deletions go/vt/vtgate/executorcontext/safe_session_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -173,6 +173,10 @@ func TestTimeZone(t *testing.T) {
tz string
want string
}{
{
tz: "",
want: time.Local.String(),
},
{
tz: "'Europe/Amsterdam'",
want: "Europe/Amsterdam",
Expand All @@ -183,16 +187,18 @@ func TestTimeZone(t *testing.T) {
},
{
tz: "foo",
want: (*time.Location)(nil).String(),
want: time.Local.String(),
},
}

for _, tc := range testCases {
t.Run(tc.tz, func(t *testing.T) {
sysvars := map[string]string{}
if tc.tz != "" {
sysvars["time_zone"] = tc.tz
}
session := NewSafeSession(&vtgatepb.Session{
SystemVariables: map[string]string{
"time_zone": tc.tz,
},
SystemVariables: sysvars,
})

assert.Equal(t, tc.want, session.TimeZone().String())
Expand Down

0 comments on commit f33200f

Please sign in to comment.