mirror of
https://github.com/jackc/pgx.git
synced 2025-05-05 23:21:11 +00:00
Because reading a record type requires the decoder to be able to look up oid to type mapping and types such as hstore have types that are not fixed between different PostgreSQL servers it was necessary to restructure the pgtype system so all encoders and decodes take a *ConnInfo that includes oid/name/type information.
149 lines
3.3 KiB
Go
149 lines
3.3 KiB
Go
package pgtype
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"fmt"
|
|
"io"
|
|
"math"
|
|
"strconv"
|
|
|
|
"github.com/jackc/pgx/pgio"
|
|
)
|
|
|
|
type Float8 struct {
|
|
Float float64
|
|
Status Status
|
|
}
|
|
|
|
func (dst *Float8) Set(src interface{}) error {
|
|
switch value := src.(type) {
|
|
case float32:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case float64:
|
|
*dst = Float8{Float: value, Status: Present}
|
|
case int8:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case uint8:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case int16:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case uint16:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case int32:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case uint32:
|
|
*dst = Float8{Float: float64(value), Status: Present}
|
|
case int64:
|
|
f64 := float64(value)
|
|
if int64(f64) == value {
|
|
*dst = Float8{Float: f64, Status: Present}
|
|
} else {
|
|
return fmt.Errorf("%v cannot be exactly represented as float64", value)
|
|
}
|
|
case uint64:
|
|
f64 := float64(value)
|
|
if uint64(f64) == value {
|
|
*dst = Float8{Float: f64, Status: Present}
|
|
} else {
|
|
return fmt.Errorf("%v cannot be exactly represented as float64", value)
|
|
}
|
|
case int:
|
|
f64 := float64(value)
|
|
if int(f64) == value {
|
|
*dst = Float8{Float: f64, Status: Present}
|
|
} else {
|
|
return fmt.Errorf("%v cannot be exactly represented as float64", value)
|
|
}
|
|
case uint:
|
|
f64 := float64(value)
|
|
if uint(f64) == value {
|
|
*dst = Float8{Float: f64, Status: Present}
|
|
} else {
|
|
return fmt.Errorf("%v cannot be exactly represented as float64", value)
|
|
}
|
|
case string:
|
|
num, err := strconv.ParseFloat(value, 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
*dst = Float8{Float: float64(num), Status: Present}
|
|
default:
|
|
if originalSrc, ok := underlyingNumberType(src); ok {
|
|
return dst.Set(originalSrc)
|
|
}
|
|
return fmt.Errorf("cannot convert %v to Float8", value)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (dst *Float8) Get() interface{} {
|
|
switch dst.Status {
|
|
case Present:
|
|
return dst.Float
|
|
case Null:
|
|
return nil
|
|
default:
|
|
return dst.Status
|
|
}
|
|
}
|
|
|
|
func (src *Float8) AssignTo(dst interface{}) error {
|
|
return float64AssignTo(src.Float, src.Status, dst)
|
|
}
|
|
|
|
func (dst *Float8) DecodeText(ci *ConnInfo, src []byte) error {
|
|
if src == nil {
|
|
*dst = Float8{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
n, err := strconv.ParseFloat(string(src), 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
*dst = Float8{Float: n, Status: Present}
|
|
return nil
|
|
}
|
|
|
|
func (dst *Float8) DecodeBinary(ci *ConnInfo, src []byte) error {
|
|
if src == nil {
|
|
*dst = Float8{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
if len(src) != 8 {
|
|
return fmt.Errorf("invalid length for float4: %v", len(src))
|
|
}
|
|
|
|
n := int64(binary.BigEndian.Uint64(src))
|
|
|
|
*dst = Float8{Float: math.Float64frombits(uint64(n)), Status: Present}
|
|
return nil
|
|
}
|
|
|
|
func (src Float8) EncodeText(ci *ConnInfo, w io.Writer) (bool, error) {
|
|
switch src.Status {
|
|
case Null:
|
|
return true, nil
|
|
case Undefined:
|
|
return false, errUndefined
|
|
}
|
|
|
|
_, err := io.WriteString(w, strconv.FormatFloat(float64(src.Float), 'f', -1, 64))
|
|
return false, err
|
|
}
|
|
|
|
func (src Float8) EncodeBinary(ci *ConnInfo, w io.Writer) (bool, error) {
|
|
switch src.Status {
|
|
case Null:
|
|
return true, nil
|
|
case Undefined:
|
|
return false, errUndefined
|
|
}
|
|
|
|
_, err := pgio.WriteInt64(w, int64(math.Float64bits(src.Float)))
|
|
return false, err
|
|
}
|