-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathvalues_map.go
More file actions
97 lines (84 loc) · 2.15 KB
/
values_map.go
File metadata and controls
97 lines (84 loc) · 2.15 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
package llsr
import (
"database/sql"
"errors"
_ "github.com/lib/pq"
"github.com/lib/pq/oid"
"github.com/liquidm/llsr/decoderbufs"
)
var (
ErrUnknownOID = errors.New("Unknown OID")
)
// ValuesMap is used in Converter interface.
// It has true set to every oid which is value type in database.
type ValuesMap map[int]bool
func loadValuesMap(dbConfig *DatabaseConfig) (ValuesMap, error) {
valuesMap := make(ValuesMap)
if err := valuesMap.load(dbConfig); err != nil {
return nil, err
}
return valuesMap, nil
}
// Extract value from DatumMessage. Returned value is always a pointer.
// Returns ErrUnknownOID if value is of unonkown OID. If returned with error, value is []byte or nil.
func (v ValuesMap) Extract(m *decoderbufs.DatumMessage) (interface{}, error) {
var err error
var value interface{}
switch oid.Oid(*m.ColumnType) {
case oid.T_bool:
value = m.DatumBool
case oid.T_int2, oid.T_int4:
value = m.DatumInt32
case oid.T_int8, oid.T_oid:
value = m.DatumInt64
case oid.T_float4:
value = m.DatumFloat
case oid.T_float8, oid.T_numeric:
value = m.DatumDouble
case oid.T_char, oid.T_varchar, oid.T_bpchar, oid.T_text, oid.T_json, oid.T_xml, oid.T_uuid, oid.T_timestamp, oid.T_timestamptz, oid.T_date, oid.T_tstzrange:
value = m.DatumString
case oid.T_point:
value = m.DatumPoint
case oid.T_bytea:
value = m.DatumBytes
default:
if v[int(*m.ColumnType)] {
valueStr := string(m.DatumBytes)
value = &valueStr
} else {
err = ErrUnknownOID
value = m.DatumBytes
}
}
if value == nil {
return nil, err
}
return value, err
}
func (v ValuesMap) load(dbConfig *DatabaseConfig) error {
db, err := sql.Open("postgres", dbConfig.ToConnectionString())
if err != nil {
return err
}
defer db.Close()
oidQueries := []string{
"SELECT enumtypid FROM pg_enum;",
"SELECT DISTINCT typarray FROM pg_type WHERE typarray > 0;",
"SELECT oid FROM pg_type WHERE typname = 'citext'",
}
for _, query := range oidQueries {
rows, err := db.Query(query)
if err != nil {
return err
}
for rows.Next() {
var oid int
if err := rows.Scan(&oid); err != nil {
return err
}
v[oid] = true
}
rows.Close()
}
return nil
}