19c6689752
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.
296 lines
6.4 KiB
Plaintext
296 lines
6.4 KiB
Plaintext
package pgtype
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
|
|
"github.com/jackc/pgx/pgio"
|
|
)
|
|
|
|
type <%= pgtype_array_type %> struct {
|
|
Elements []<%= pgtype_element_type %>
|
|
Dimensions []ArrayDimension
|
|
Status Status
|
|
}
|
|
|
|
func (dst *<%= pgtype_array_type %>) Set(src interface{}) error {
|
|
switch value := src.(type) {
|
|
<% go_array_types.split(",").each do |t| %>
|
|
case <%= t %>:
|
|
if value == nil {
|
|
*dst = <%= pgtype_array_type %>{Status: Null}
|
|
} else if len(value) == 0 {
|
|
*dst = <%= pgtype_array_type %>{Status: Present}
|
|
} else {
|
|
elements := make([]<%= pgtype_element_type %>, len(value))
|
|
for i := range value {
|
|
if err := elements[i].Set(value[i]); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
*dst = <%= pgtype_array_type %>{
|
|
Elements: elements,
|
|
Dimensions: []ArrayDimension{{Length: int32(len(elements)), LowerBound: 1}},
|
|
Status: Present,
|
|
}
|
|
}
|
|
<% end %>
|
|
default:
|
|
if originalSrc, ok := underlyingSliceType(src); ok {
|
|
return dst.Set(originalSrc)
|
|
}
|
|
return fmt.Errorf("cannot convert %v to <%= pgtype_element_type %>", value)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (dst *<%= pgtype_array_type %>) Get() interface{} {
|
|
switch dst.Status {
|
|
case Present:
|
|
return dst
|
|
case Null:
|
|
return nil
|
|
default:
|
|
return dst.Status
|
|
}
|
|
}
|
|
|
|
func (src *<%= pgtype_array_type %>) AssignTo(dst interface{}) error {
|
|
switch v := dst.(type) {
|
|
<% go_array_types.split(",").each do |t| %>
|
|
case *<%= t %>:
|
|
if src.Status == Present {
|
|
*v = make(<%= t %>, len(src.Elements))
|
|
for i := range src.Elements {
|
|
if err := src.Elements[i].AssignTo(&((*v)[i])); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
} else {
|
|
*v = nil
|
|
}
|
|
<% end %>
|
|
default:
|
|
if originalDst, ok := underlyingPtrSliceType(dst); ok {
|
|
return src.AssignTo(originalDst)
|
|
}
|
|
return fmt.Errorf("cannot decode %v into %T", src, dst)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (dst *<%= pgtype_array_type %>) DecodeText(ci *ConnInfo, src []byte) error {
|
|
if src == nil {
|
|
*dst = <%= pgtype_array_type %>{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
uta, err := ParseUntypedTextArray(string(src))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var elements []<%= pgtype_element_type %>
|
|
|
|
if len(uta.Elements) > 0 {
|
|
elements = make([]<%= pgtype_element_type %>, len(uta.Elements))
|
|
|
|
for i, s := range uta.Elements {
|
|
var elem <%= pgtype_element_type %>
|
|
var elemSrc []byte
|
|
if s != "NULL" {
|
|
elemSrc = []byte(s)
|
|
}
|
|
err = elem.DecodeText(ci, elemSrc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
elements[i] = elem
|
|
}
|
|
}
|
|
|
|
*dst = <%= pgtype_array_type %>{Elements: elements, Dimensions: uta.Dimensions, Status: Present}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (dst *<%= pgtype_array_type %>) DecodeBinary(ci *ConnInfo, src []byte) error {
|
|
if src == nil {
|
|
*dst = <%= pgtype_array_type %>{Status: Null}
|
|
return nil
|
|
}
|
|
|
|
var arrayHeader ArrayHeader
|
|
rp, err := arrayHeader.DecodeBinary(ci, src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(arrayHeader.Dimensions) == 0 {
|
|
*dst = <%= pgtype_array_type %>{Dimensions: arrayHeader.Dimensions, Status: Present}
|
|
return nil
|
|
}
|
|
|
|
elementCount := arrayHeader.Dimensions[0].Length
|
|
for _, d := range arrayHeader.Dimensions[1:] {
|
|
elementCount *= d.Length
|
|
}
|
|
|
|
elements := make([]<%= pgtype_element_type %>, elementCount)
|
|
|
|
for i := range elements {
|
|
elemLen := int(int32(binary.BigEndian.Uint32(src[rp:])))
|
|
rp += 4
|
|
var elemSrc []byte
|
|
if elemLen >= 0 {
|
|
elemSrc = src[rp:rp+elemLen]
|
|
rp += elemLen
|
|
}
|
|
err = elements[i].DecodeBinary(ci, elemSrc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
*dst = <%= pgtype_array_type %>{Elements: elements, Dimensions: arrayHeader.Dimensions, Status: Present}
|
|
return nil
|
|
}
|
|
|
|
func (src *<%= pgtype_array_type %>) EncodeText(ci *ConnInfo, w io.Writer) (bool, error) {
|
|
switch src.Status {
|
|
case Null:
|
|
return true, nil
|
|
case Undefined:
|
|
return false, errUndefined
|
|
}
|
|
|
|
if len(src.Dimensions) == 0 {
|
|
_, err := io.WriteString(w, "{}")
|
|
return false, err
|
|
}
|
|
|
|
err := EncodeTextArrayDimensions(w, src.Dimensions)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
// dimElemCounts is the multiples of elements that each array lies on. For
|
|
// example, a single dimension array of length 4 would have a dimElemCounts of
|
|
// [4]. A multi-dimensional array of lengths [3,5,2] would have a
|
|
// dimElemCounts of [30,10,2]. This is used to simplify when to render a '{'
|
|
// or '}'.
|
|
dimElemCounts := make([]int, len(src.Dimensions))
|
|
dimElemCounts[len(src.Dimensions)-1] = int(src.Dimensions[len(src.Dimensions)-1].Length)
|
|
for i := len(src.Dimensions) - 2; i > -1; i-- {
|
|
dimElemCounts[i] = int(src.Dimensions[i].Length) * dimElemCounts[i+1]
|
|
}
|
|
|
|
for i, elem := range src.Elements {
|
|
if i > 0 {
|
|
err = pgio.WriteByte(w, ',')
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
for _, dec := range dimElemCounts {
|
|
if i%dec == 0 {
|
|
err = pgio.WriteByte(w, '{')
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
}
|
|
|
|
elemBuf := &bytes.Buffer{}
|
|
null, err := elem.EncodeText(ci, elemBuf)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if null {
|
|
_, err = io.WriteString(w, `<%= text_null %>`)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
} else {
|
|
_, err = io.WriteString(w, QuoteArrayElementIfNeeded(elemBuf.String()))
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
for _, dec := range dimElemCounts {
|
|
if (i+1)%dec == 0 {
|
|
err = pgio.WriteByte(w, '}')
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
func (src *<%= pgtype_array_type %>) EncodeBinary(ci *ConnInfo, w io.Writer) (bool, error) {
|
|
return src.encodeBinary(ci, w, <%= element_oid %>)
|
|
}
|
|
|
|
func (src *<%= pgtype_array_type %>) encodeBinary(ci *ConnInfo, w io.Writer, elementOid int32) (bool, error) {
|
|
switch src.Status {
|
|
case Null:
|
|
return true, nil
|
|
case Undefined:
|
|
return false, errUndefined
|
|
}
|
|
|
|
arrayHeader := ArrayHeader{
|
|
ElementOid: elementOid,
|
|
Dimensions: src.Dimensions,
|
|
}
|
|
|
|
for i := range src.Elements {
|
|
if src.Elements[i].Status == Null {
|
|
arrayHeader.ContainsNull = true
|
|
break
|
|
}
|
|
}
|
|
|
|
err := arrayHeader.EncodeBinary(ci, w)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
elemBuf := &bytes.Buffer{}
|
|
|
|
for i := range src.Elements {
|
|
elemBuf.Reset()
|
|
|
|
null, err := src.Elements[i].EncodeBinary(ci, elemBuf)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if null {
|
|
_, err = pgio.WriteInt32(w, -1)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
} else {
|
|
_, err = pgio.WriteInt32(w, int32(elemBuf.Len()))
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
_, err = elemBuf.WriteTo(w)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
}
|
|
|
|
return false, err
|
|
}
|