mirror of https://github.com/jackc/pgx.git
wip
parent
0ca0ee7206
commit
83a5447cd2
|
@ -11,6 +11,7 @@ import (
|
||||||
type Date struct {
|
type Date struct {
|
||||||
// time.Time is embedded to hide internal implementation. Possibly do date
|
// time.Time is embedded to hide internal implementation. Possibly do date
|
||||||
// implementation at some point rather than simply delegating to time.Time.
|
// implementation at some point rather than simply delegating to time.Time.
|
||||||
|
// Also TODO handling Infinity and -Infinity
|
||||||
t time.Time
|
t time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,7 +22,7 @@ func (d *Date) DecodeText(r io.Reader) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
if size == -1 {
|
if size == -1 {
|
||||||
return fmt.Errorf("invalid length for int8: %v", size)
|
return fmt.Errorf("invalid length for date: %v", size)
|
||||||
}
|
}
|
||||||
|
|
||||||
buf := make([]byte, int(size))
|
buf := make([]byte, int(size))
|
||||||
|
|
|
@ -0,0 +1,96 @@
|
||||||
|
package pgtype
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/jackc/pgx/pgio"
|
||||||
|
)
|
||||||
|
|
||||||
|
const pgTimestamptzFormat = "2006-01-02 15:04:05.999999999Z07:00"
|
||||||
|
const microsecFromUnixEpochToY2K = 946684800 * 1000000
|
||||||
|
|
||||||
|
type Timestamptz struct {
|
||||||
|
// time.Time is embedded to handle Infinity and -Infinity
|
||||||
|
// TODO - infinity
|
||||||
|
t time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Timestamptz) DecodeText(r io.Reader) error {
|
||||||
|
size, err := pgio.ReadInt32(r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if size == -1 {
|
||||||
|
return fmt.Errorf("invalid length for timestamptz: %v", size)
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := make([]byte, int(size))
|
||||||
|
_, err = r.Read(buf)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
t.t, err = time.Parse(pgTimestamptzFormat, string(buf))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Timestamptz) DecodeBinary(r io.Reader) error {
|
||||||
|
size, err := pgio.ReadInt32(r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if size != 8 {
|
||||||
|
return fmt.Errorf("invalid length for timestamptz: %v", size)
|
||||||
|
}
|
||||||
|
|
||||||
|
microsecSinceY2K, err := pgio.ReadInt64(r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
microsecSinceUnixEpoch := microsecFromUnixEpochToY2K + microsecSinceY2K
|
||||||
|
t.t = time.Unix(microsecSinceUnixEpoch/1000000, (microsecSinceUnixEpoch%1000000)*1000)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t Timestamptz) EncodeText(w io.Writer) error {
|
||||||
|
buf := []byte(t.t.Format(pgTimestamptzFormat))
|
||||||
|
|
||||||
|
_, err := pgio.WriteInt32(w, int32(len(buf)))
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = w.Write(buf)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t Timestamptz) EncodeBinary(w io.Writer) error {
|
||||||
|
_, err := pgio.WriteInt32(w, 8)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
microsecSinceUnixEpoch := t.t.Unix()*1000000 + int64(t.t.Nanosecond())/1000
|
||||||
|
microsecSinceY2K := microsecSinceUnixEpoch - microsecFromUnixEpochToY2K
|
||||||
|
|
||||||
|
_, err = pgio.WriteInt64(w, microsecSinceY2K)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t Timestamptz) Time() time.Time {
|
||||||
|
return t.t
|
||||||
|
}
|
||||||
|
|
||||||
|
func TimestamptzFromTime(t time.Time) Timestamptz {
|
||||||
|
return Timestamptz{t: t}
|
||||||
|
}
|
31
values.go
31
values.go
|
@ -2033,13 +2033,7 @@ func encodeTime(w *WriteBuf, oid OID, value time.Time) error {
|
||||||
case DateOID:
|
case DateOID:
|
||||||
return pgtype.DateFromTime(value).EncodeBinary(w)
|
return pgtype.DateFromTime(value).EncodeBinary(w)
|
||||||
case TimestampTzOID, TimestampOID:
|
case TimestampTzOID, TimestampOID:
|
||||||
microsecSinceUnixEpoch := value.Unix()*1000000 + int64(value.Nanosecond())/1000
|
return pgtype.TimestamptzFromTime(value).EncodeBinary(w)
|
||||||
microsecSinceY2K := microsecSinceUnixEpoch - microsecFromUnixEpochToY2K
|
|
||||||
|
|
||||||
w.WriteInt32(8)
|
|
||||||
w.WriteInt64(microsecSinceY2K)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("cannot encode %s into oid %v", "time.Time", oid)
|
return fmt.Errorf("cannot encode %s into oid %v", "time.Time", oid)
|
||||||
}
|
}
|
||||||
|
@ -2060,19 +2054,26 @@ func decodeTimestampTz(vr *ValueReader) time.Time {
|
||||||
return zeroTime
|
return zeroTime
|
||||||
}
|
}
|
||||||
|
|
||||||
if vr.Type().FormatCode != BinaryFormatCode {
|
vr.err = errRewoundLen
|
||||||
|
|
||||||
|
var t pgtype.Timestamptz
|
||||||
|
var err error
|
||||||
|
switch vr.Type().FormatCode {
|
||||||
|
case TextFormatCode:
|
||||||
|
err = t.DecodeText(&valueReader2{vr})
|
||||||
|
case BinaryFormatCode:
|
||||||
|
err = t.DecodeBinary(&valueReader2{vr})
|
||||||
|
default:
|
||||||
vr.Fatal(ProtocolError(fmt.Sprintf("Unknown field description format code: %v", vr.Type().FormatCode)))
|
vr.Fatal(ProtocolError(fmt.Sprintf("Unknown field description format code: %v", vr.Type().FormatCode)))
|
||||||
return zeroTime
|
return time.Time{}
|
||||||
}
|
}
|
||||||
|
|
||||||
if vr.Len() != 8 {
|
if err != nil {
|
||||||
vr.Fatal(ProtocolError(fmt.Sprintf("Received an invalid size for an timestamptz: %d", vr.Len())))
|
vr.Fatal(err)
|
||||||
return zeroTime
|
return time.Time{}
|
||||||
}
|
}
|
||||||
|
|
||||||
microsecSinceY2K := vr.ReadInt64()
|
return t.Time()
|
||||||
microsecSinceUnixEpoch := microsecFromUnixEpochToY2K + microsecSinceY2K
|
|
||||||
return time.Unix(microsecSinceUnixEpoch/1000000, (microsecSinceUnixEpoch%1000000)*1000)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func decodeTimestamp(vr *ValueReader) time.Time {
|
func decodeTimestamp(vr *ValueReader) time.Time {
|
||||||
|
|
Loading…
Reference in New Issue