mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-25 19:42:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			93 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			93 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| 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()
 | |
| }
 |