Merge pull request #2216 from pconstantinou/master

Timestamp incorrectly adds 'Z' when serializing from JSON to indicate GMT, fixes bug #2215
pull/2236/head
Jack Christensen 2025-01-18 10:17:43 -06:00 committed by GitHub
commit 9cce05944a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 53 additions and 11 deletions

View File

@ -12,6 +12,7 @@ import (
) )
const pgTimestampFormat = "2006-01-02 15:04:05.999999999" const pgTimestampFormat = "2006-01-02 15:04:05.999999999"
const jsonISO8601 = "2006-01-02T15:04:05.999999999"
type TimestampScanner interface { type TimestampScanner interface {
ScanTimestamp(v Timestamp) error ScanTimestamp(v Timestamp) error
@ -76,7 +77,7 @@ func (ts Timestamp) MarshalJSON() ([]byte, error) {
switch ts.InfinityModifier { switch ts.InfinityModifier {
case Finite: case Finite:
s = ts.Time.Format(time.RFC3339Nano) s = ts.Time.Format(jsonISO8601)
case Infinity: case Infinity:
s = "infinity" s = "infinity"
case NegativeInfinity: case NegativeInfinity:
@ -104,15 +105,23 @@ func (ts *Timestamp) UnmarshalJSON(b []byte) error {
case "-infinity": case "-infinity":
*ts = Timestamp{Valid: true, InfinityModifier: -Infinity} *ts = Timestamp{Valid: true, InfinityModifier: -Infinity}
default: default:
// PostgreSQL uses ISO 8601 wihout timezone for to_json function and casting from a string to timestampt // Parse time with or without timezonr
tim, err := time.Parse(time.RFC3339Nano, *s+"Z") tss := *s
if err != nil { // PostgreSQL uses ISO 8601 without timezone for to_json function and casting from a string to timestampt
return err tim, err := time.Parse(time.RFC3339Nano, tss)
if err == nil {
*ts = Timestamp{Time: tim, Valid: true}
return nil
} }
tim, err = time.ParseInLocation(jsonISO8601, tss, time.UTC)
*ts = Timestamp{Time: tim, Valid: true} if err == nil {
*ts = Timestamp{Time: tim, Valid: true}
return nil
}
ts.Valid = false
return fmt.Errorf("cannot unmarshal %s to timestamp with layout %s or %s (%w)",
*s, time.RFC3339Nano, jsonISO8601, err)
} }
return nil return nil
} }

View File

@ -2,12 +2,14 @@ package pgtype_test
import ( import (
"context" "context"
"encoding/json"
"testing" "testing"
"time" "time"
pgx "github.com/jackc/pgx/v5" pgx "github.com/jackc/pgx/v5"
"github.com/jackc/pgx/v5/pgtype" "github.com/jackc/pgx/v5/pgtype"
"github.com/jackc/pgx/v5/pgxtest" "github.com/jackc/pgx/v5/pgxtest"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
) )
@ -100,13 +102,24 @@ func TestTimestampCodecDecodeTextInvalid(t *testing.T) {
} }
func TestTimestampMarshalJSON(t *testing.T) { func TestTimestampMarshalJSON(t *testing.T) {
tsStruct := struct {
TS pgtype.Timestamp `json:"ts"`
}{}
tm := time.Date(2012, 3, 29, 10, 5, 45, 0, time.UTC)
tsString := "\"" + tm.Format("2006-01-02T15:04:05") + "\"" // `"2012-03-29T10:05:45"`
var pgt pgtype.Timestamp
_ = pgt.Scan(tm)
successfulTests := []struct { successfulTests := []struct {
source pgtype.Timestamp source pgtype.Timestamp
result string result string
}{ }{
{source: pgtype.Timestamp{}, result: "null"}, {source: pgtype.Timestamp{}, result: "null"},
{source: pgtype.Timestamp{Time: time.Date(2012, 3, 29, 10, 5, 45, 0, time.UTC), Valid: true}, result: "\"2012-03-29T10:05:45Z\""}, {source: pgtype.Timestamp{Time: tm, Valid: true}, result: tsString},
{source: pgtype.Timestamp{Time: time.Date(2012, 3, 29, 10, 5, 45, 555*1000*1000, time.UTC), Valid: true}, result: "\"2012-03-29T10:05:45.555Z\""}, {source: pgt, result: tsString},
{source: pgtype.Timestamp{Time: tm.Add(time.Second * 555 / 1000), Valid: true}, result: `"2012-03-29T10:05:45.555"`},
{source: pgtype.Timestamp{InfinityModifier: pgtype.Infinity, Valid: true}, result: "\"infinity\""}, {source: pgtype.Timestamp{InfinityModifier: pgtype.Infinity, Valid: true}, result: "\"infinity\""},
{source: pgtype.Timestamp{InfinityModifier: pgtype.NegativeInfinity, Valid: true}, result: "\"-infinity\""}, {source: pgtype.Timestamp{InfinityModifier: pgtype.NegativeInfinity, Valid: true}, result: "\"-infinity\""},
} }
@ -116,12 +129,32 @@ func TestTimestampMarshalJSON(t *testing.T) {
t.Errorf("%d: %v", i, err) t.Errorf("%d: %v", i, err)
} }
if string(r) != tt.result { if !assert.Equal(t, tt.result, string(r)) {
t.Errorf("%d: expected %v to convert to %v, but it was %v", i, tt.source, tt.result, string(r)) t.Errorf("%d: expected %v to convert to %v, but it was %v", i, tt.source, tt.result, string(r))
} }
tsStruct.TS = tt.source
b, err := json.Marshal(tsStruct)
assert.NoErrorf(t, err, "failed to marshal %v %s", tt.source, err)
t2 := tsStruct
t2.TS = pgtype.Timestamp{} // Clear out the value so that we can compare after unmarshalling
err = json.Unmarshal(b, &t2)
assert.NoErrorf(t, err, "failed to unmarshal %v with %s", tt.source, err)
assert.True(t, tsStruct.TS.Time.Unix() == t2.TS.Time.Unix())
} }
} }
func TestTimestampUnmarshalJSONErrors(t *testing.T) {
tsStruct := struct {
TS pgtype.Timestamp `json:"ts"`
}{}
goodJson1 := []byte(`{"ts":"2012-03-29T10:05:45"}`)
assert.NoError(t, json.Unmarshal(goodJson1, &tsStruct))
goodJson2 := []byte(`{"ts":"2012-03-29T10:05:45Z"}`)
assert.NoError(t, json.Unmarshal(goodJson2, &tsStruct))
badJson := []byte(`{"ts":"2012-03-29"}`)
assert.Error(t, json.Unmarshal(badJson, &tsStruct))
}
func TestTimestampUnmarshalJSON(t *testing.T) { func TestTimestampUnmarshalJSON(t *testing.T) {
successfulTests := []struct { successfulTests := []struct {
source string source string