Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
93 lines
2.4 KiB
93 lines
2.4 KiB
package pq |
|
|
|
import ( |
|
"math" |
|
"reflect" |
|
"time" |
|
|
|
"github.com/lib/pq/oid" |
|
) |
|
|
|
const headerSize = 4 |
|
|
|
type fieldDesc struct { |
|
// The object ID of the data type. |
|
OID oid.Oid |
|
// The data type size (see pg_type.typlen). |
|
// Note that negative values denote variable-width types. |
|
Len int |
|
// The type modifier (see pg_attribute.atttypmod). |
|
// The meaning of the modifier is type-specific. |
|
Mod int |
|
} |
|
|
|
func (fd fieldDesc) Type() reflect.Type { |
|
switch fd.OID { |
|
case oid.T_int8: |
|
return reflect.TypeOf(int64(0)) |
|
case oid.T_int4: |
|
return reflect.TypeOf(int32(0)) |
|
case oid.T_int2: |
|
return reflect.TypeOf(int16(0)) |
|
case oid.T_varchar, oid.T_text: |
|
return reflect.TypeOf("") |
|
case oid.T_bool: |
|
return reflect.TypeOf(false) |
|
case oid.T_date, oid.T_time, oid.T_timetz, oid.T_timestamp, oid.T_timestamptz: |
|
return reflect.TypeOf(time.Time{}) |
|
case oid.T_bytea: |
|
return reflect.TypeOf([]byte(nil)) |
|
default: |
|
return reflect.TypeOf(new(interface{})).Elem() |
|
} |
|
} |
|
|
|
func (fd fieldDesc) Name() string { |
|
return oid.TypeName[fd.OID] |
|
} |
|
|
|
func (fd fieldDesc) Length() (length int64, ok bool) { |
|
switch fd.OID { |
|
case oid.T_text, oid.T_bytea: |
|
return math.MaxInt64, true |
|
case oid.T_varchar, oid.T_bpchar: |
|
return int64(fd.Mod - headerSize), true |
|
default: |
|
return 0, false |
|
} |
|
} |
|
|
|
func (fd fieldDesc) PrecisionScale() (precision, scale int64, ok bool) { |
|
switch fd.OID { |
|
case oid.T_numeric, oid.T__numeric: |
|
mod := fd.Mod - headerSize |
|
precision = int64((mod >> 16) & 0xffff) |
|
scale = int64(mod & 0xffff) |
|
return precision, scale, true |
|
default: |
|
return 0, 0, false |
|
} |
|
} |
|
|
|
// ColumnTypeScanType returns the value type that can be used to scan types into. |
|
func (rs *rows) ColumnTypeScanType(index int) reflect.Type { |
|
return rs.colTyps[index].Type() |
|
} |
|
|
|
// ColumnTypeDatabaseTypeName return the database system type name. |
|
func (rs *rows) ColumnTypeDatabaseTypeName(index int) string { |
|
return rs.colTyps[index].Name() |
|
} |
|
|
|
// ColumnTypeLength returns the length of the column type if the column is a |
|
// variable length type. If the column is not a variable length type ok |
|
// should return false. |
|
func (rs *rows) ColumnTypeLength(index int) (length int64, ok bool) { |
|
return rs.colTyps[index].Length() |
|
} |
|
|
|
// ColumnTypePrecisionScale should return the precision and scale for decimal |
|
// types. If not applicable, ok should be false. |
|
func (rs *rows) ColumnTypePrecisionScale(index int) (precision, scale int64, ok bool) { |
|
return rs.colTyps[index].PrecisionScale() |
|
}
|
|
|