mirror of https://github.com/jackc/pgx.git
added MarshalJSON and UnmarshalJSON to timestamp and added their tests (based on timestamptz implementation)
parent
34eddf9983
commit
c542df4fb4
|
@ -3,6 +3,7 @@ package pgtype
|
||||||
import (
|
import (
|
||||||
"database/sql/driver"
|
"database/sql/driver"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
@ -66,6 +67,55 @@ func (ts Timestamp) Value() (driver.Value, error) {
|
||||||
return ts.Time, nil
|
return ts.Time, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ts Timestamp) MarshalJSON() ([]byte, error) {
|
||||||
|
if !ts.Valid {
|
||||||
|
return []byte("null"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var s string
|
||||||
|
|
||||||
|
switch ts.InfinityModifier {
|
||||||
|
case Finite:
|
||||||
|
s = ts.Time.Format(time.RFC3339Nano)
|
||||||
|
case Infinity:
|
||||||
|
s = "infinity"
|
||||||
|
case NegativeInfinity:
|
||||||
|
s = "-infinity"
|
||||||
|
}
|
||||||
|
|
||||||
|
return json.Marshal(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ts *Timestamp) UnmarshalJSON(b []byte) error {
|
||||||
|
var s *string
|
||||||
|
err := json.Unmarshal(b, &s)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if s == nil {
|
||||||
|
*ts = Timestamp{}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
switch *s {
|
||||||
|
case "infinity":
|
||||||
|
*ts = Timestamp{Valid: true, InfinityModifier: Infinity}
|
||||||
|
case "-infinity":
|
||||||
|
*ts = Timestamp{Valid: true, InfinityModifier: -Infinity}
|
||||||
|
default:
|
||||||
|
// PostgreSQL uses ISO 8601 for to_json function and casting from a string to timestamptz
|
||||||
|
tim, err := time.Parse(time.RFC3339Nano, *s)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
*ts = Timestamp{Time: tim, Valid: true}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
type TimestampCodec struct{}
|
type TimestampCodec struct{}
|
||||||
|
|
||||||
func (TimestampCodec) FormatSupported(format int16) bool {
|
func (TimestampCodec) FormatSupported(format int16) bool {
|
||||||
|
|
|
@ -62,3 +62,50 @@ func TestTimestampCodecDecodeTextInvalid(t *testing.T) {
|
||||||
err := plan.Scan([]byte(`eeeee`), &ts)
|
err := plan.Scan([]byte(`eeeee`), &ts)
|
||||||
require.Error(t, err)
|
require.Error(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestTimestampMarshalJSON(t *testing.T) {
|
||||||
|
successfulTests := []struct {
|
||||||
|
source pgtype.Timestamp
|
||||||
|
result string
|
||||||
|
}{
|
||||||
|
{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: time.Date(2012, 3, 29, 10, 5, 45, 555*1000*1000, time.UTC), Valid: true}, result: "\"2012-03-29T10:05:45.555Z\""},
|
||||||
|
{source: pgtype.Timestamp{InfinityModifier: pgtype.Infinity, Valid: true}, result: "\"infinity\""},
|
||||||
|
{source: pgtype.Timestamp{InfinityModifier: pgtype.NegativeInfinity, Valid: true}, result: "\"-infinity\""},
|
||||||
|
}
|
||||||
|
for i, tt := range successfulTests {
|
||||||
|
r, err := tt.source.MarshalJSON()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%d: %v", i, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if string(r) != tt.result {
|
||||||
|
t.Errorf("%d: expected %v to convert to %v, but it was %v", i, tt.source, tt.result, string(r))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTimestampUnmarshalJSON(t *testing.T) {
|
||||||
|
successfulTests := []struct {
|
||||||
|
source string
|
||||||
|
result pgtype.Timestamp
|
||||||
|
}{
|
||||||
|
{source: "null", result: pgtype.Timestamp{}},
|
||||||
|
{source: "\"2012-03-29T10:05:45Z\"", result: pgtype.Timestamp{Time: time.Date(2012, 3, 29, 10, 5, 45, 0, time.UTC), Valid: true}},
|
||||||
|
{source: "\"2012-03-29T10:05:45.555Z\"", result: pgtype.Timestamp{Time: time.Date(2012, 3, 29, 10, 5, 45, 555*1000*1000, time.UTC), Valid: true}},
|
||||||
|
{source: "\"infinity\"", result: pgtype.Timestamp{InfinityModifier: pgtype.Infinity, Valid: true}},
|
||||||
|
{source: "\"-infinity\"", result: pgtype.Timestamp{InfinityModifier: pgtype.NegativeInfinity, Valid: true}},
|
||||||
|
}
|
||||||
|
for i, tt := range successfulTests {
|
||||||
|
var r pgtype.Timestamp
|
||||||
|
err := r.UnmarshalJSON([]byte(tt.source))
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("%d: %v", i, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !r.Time.Equal(tt.result.Time) || r.Valid != tt.result.Valid || r.InfinityModifier != tt.result.InfinityModifier {
|
||||||
|
t.Errorf("%d: expected %v to convert to %v, but it was %v", i, tt.source, tt.result, r)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue