mirror of
https://github.com/jackc/pgx.git
synced 2025-05-10 17:41:05 +00:00
This adds PostgreSQL numeric mapping to and from github.com/shopspring/decimal. Makes pgtype.NullAssignTo public as external types need this functionality. Begin extraction of pgtype testing functionality so it can easily be used by external types.
211 lines
4.2 KiB
Go
211 lines
4.2 KiB
Go
package pgtype
|
|
|
|
import (
|
|
"database/sql/driver"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"io"
|
|
"time"
|
|
|
|
"github.com/jackc/pgx/pgio"
|
|
)
|
|
|
|
type Date struct {
|
|
Time time.Time
|
|
Status Status
|
|
InfinityModifier InfinityModifier
|
|
}
|
|
|
|
const (
|
|
negativeInfinityDayOffset = -2147483648
|
|
infinityDayOffset = 2147483647
|
|
)
|
|
|
|
func (dst *Date) Set(src interface{}) error {
|
|
if src == nil {
|
|
*dst = Date{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
switch value := src.(type) {
|
|
case time.Time:
|
|
*dst = Date{Time: value, Status: Present}
|
|
default:
|
|
if originalSrc, ok := underlyingTimeType(src); ok {
|
|
return dst.Set(originalSrc)
|
|
}
|
|
return fmt.Errorf("cannot convert %v to Date", value)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (dst *Date) Get() interface{} {
|
|
switch dst.Status {
|
|
case Present:
|
|
if dst.InfinityModifier != None {
|
|
return dst.InfinityModifier
|
|
}
|
|
return dst.Time
|
|
case Null:
|
|
return nil
|
|
default:
|
|
return dst.Status
|
|
}
|
|
}
|
|
|
|
func (src *Date) AssignTo(dst interface{}) error {
|
|
switch src.Status {
|
|
case Present:
|
|
switch v := dst.(type) {
|
|
case *time.Time:
|
|
if src.InfinityModifier != None {
|
|
return fmt.Errorf("cannot assign %v to %T", src, dst)
|
|
}
|
|
*v = src.Time
|
|
return nil
|
|
default:
|
|
if nextDst, retry := GetAssignToDstType(dst); retry {
|
|
return src.AssignTo(nextDst)
|
|
}
|
|
}
|
|
case Null:
|
|
return NullAssignTo(dst)
|
|
}
|
|
|
|
return fmt.Errorf("cannot decode %v into %T", src, dst)
|
|
}
|
|
|
|
func (dst *Date) DecodeText(ci *ConnInfo, src []byte) error {
|
|
if src == nil {
|
|
*dst = Date{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
sbuf := string(src)
|
|
switch sbuf {
|
|
case "infinity":
|
|
*dst = Date{Status: Present, InfinityModifier: Infinity}
|
|
case "-infinity":
|
|
*dst = Date{Status: Present, InfinityModifier: -Infinity}
|
|
default:
|
|
t, err := time.ParseInLocation("2006-01-02", sbuf, time.UTC)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
*dst = Date{Time: t, Status: Present}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (dst *Date) DecodeBinary(ci *ConnInfo, src []byte) error {
|
|
if src == nil {
|
|
*dst = Date{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
if len(src) != 4 {
|
|
return fmt.Errorf("invalid length for date: %v", len(src))
|
|
}
|
|
|
|
dayOffset := int32(binary.BigEndian.Uint32(src))
|
|
|
|
switch dayOffset {
|
|
case infinityDayOffset:
|
|
*dst = Date{Status: Present, InfinityModifier: Infinity}
|
|
case negativeInfinityDayOffset:
|
|
*dst = Date{Status: Present, InfinityModifier: -Infinity}
|
|
default:
|
|
t := time.Date(2000, 1, int(1+dayOffset), 0, 0, 0, 0, time.UTC)
|
|
*dst = Date{Time: t, Status: Present}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (src Date) EncodeText(ci *ConnInfo, w io.Writer) (bool, error) {
|
|
switch src.Status {
|
|
case Null:
|
|
return true, nil
|
|
case Undefined:
|
|
return false, errUndefined
|
|
}
|
|
|
|
var s string
|
|
|
|
switch src.InfinityModifier {
|
|
case None:
|
|
s = src.Time.Format("2006-01-02")
|
|
case Infinity:
|
|
s = "infinity"
|
|
case NegativeInfinity:
|
|
s = "-infinity"
|
|
}
|
|
|
|
_, err := io.WriteString(w, s)
|
|
return false, err
|
|
}
|
|
|
|
func (src Date) EncodeBinary(ci *ConnInfo, w io.Writer) (bool, error) {
|
|
switch src.Status {
|
|
case Null:
|
|
return true, nil
|
|
case Undefined:
|
|
return false, errUndefined
|
|
}
|
|
|
|
var daysSinceDateEpoch int32
|
|
switch src.InfinityModifier {
|
|
case None:
|
|
tUnix := time.Date(src.Time.Year(), src.Time.Month(), src.Time.Day(), 0, 0, 0, 0, time.UTC).Unix()
|
|
dateEpoch := time.Date(2000, 1, 1, 0, 0, 0, 0, time.UTC).Unix()
|
|
|
|
secSinceDateEpoch := tUnix - dateEpoch
|
|
daysSinceDateEpoch = int32(secSinceDateEpoch / 86400)
|
|
case Infinity:
|
|
daysSinceDateEpoch = infinityDayOffset
|
|
case NegativeInfinity:
|
|
daysSinceDateEpoch = negativeInfinityDayOffset
|
|
}
|
|
|
|
_, err := pgio.WriteInt32(w, daysSinceDateEpoch)
|
|
return false, err
|
|
}
|
|
|
|
// Scan implements the database/sql Scanner interface.
|
|
func (dst *Date) Scan(src interface{}) error {
|
|
if src == nil {
|
|
*dst = Date{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
switch src := src.(type) {
|
|
case string:
|
|
return dst.DecodeText(nil, []byte(src))
|
|
case []byte:
|
|
return dst.DecodeText(nil, src)
|
|
case time.Time:
|
|
*dst = Date{Time: src, Status: Present}
|
|
return nil
|
|
}
|
|
|
|
return fmt.Errorf("cannot scan %T", src)
|
|
}
|
|
|
|
// Value implements the database/sql/driver Valuer interface.
|
|
func (src Date) Value() (driver.Value, error) {
|
|
switch src.Status {
|
|
case Present:
|
|
if src.InfinityModifier != None {
|
|
return src.InfinityModifier.String(), nil
|
|
}
|
|
return src.Time, nil
|
|
case Null:
|
|
return nil, nil
|
|
default:
|
|
return nil, errUndefined
|
|
}
|
|
}
|