-
Notifications
You must be signed in to change notification settings - Fork 562
/
scan.go
118 lines (113 loc) · 2.66 KB
/
scan.go
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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
package clickhouse
import (
"context"
"errors"
"fmt"
"reflect"
"github.com/ClickHouse/clickhouse-go/v2/lib/proto"
)
func (ch *clickhouse) Select(ctx context.Context, dest interface{}, query string, args ...interface{}) error {
value := reflect.ValueOf(dest)
if value.Kind() != reflect.Ptr {
return &OpError{
Op: "Select",
Err: errors.New("must pass a pointer, not a value, to Select destination"),
}
}
if value.IsNil() {
return &OpError{
Op: "Select",
Err: errors.New("nil pointer passed to Select destination"),
}
}
direct := reflect.Indirect(value)
if direct.Kind() != reflect.Slice {
return fmt.Errorf("must pass a slice to Select destination")
}
var (
base = direct.Type().Elem()
rows, err = ch.Query(ctx, query, args...)
)
if err != nil {
return err
}
defer rows.Close()
for rows.Next() {
elem := reflect.New(base)
if err := rows.ScanStruct(elem.Interface()); err != nil {
return err
}
direct.Set(reflect.Append(direct, elem.Elem()))
}
return rows.Err()
}
func scan(block *proto.Block, row int, dest ...interface{}) error {
columns := block.Columns
if len(columns) != len(dest) {
return &OpError{
Op: "Scan",
Err: fmt.Errorf("expected %d destination arguments in Scan, not %d", len(columns), len(dest)),
}
}
for i, d := range dest {
if err := columns[i].ScanRow(d, row-1); err != nil {
return &OpError{
Err: err,
ColumnName: block.ColumnsNames()[i],
}
}
}
return nil
}
func structToScannableValues(columns []string, dest interface{}) ([]interface{}, error) {
var (
v = reflect.ValueOf(dest)
t = reflect.TypeOf(dest)
)
if v.Kind() != reflect.Ptr {
return nil, &OpError{
Op: "ScanStruct",
Err: errors.New("must pass a pointer, not a value, to ScanStruct destination"),
}
}
if v.IsNil() {
return nil, &OpError{
Op: "ScanStruct",
Err: errors.New("nil pointer passed to ScanStruct destination"),
}
}
if v = reflect.Indirect(v); t.Kind() == reflect.Ptr {
t = t.Elem()
}
if v.Kind() != reflect.Struct {
return nil, &OpError{
Op: "ScanStruct",
Err: errors.New("ScanStruct expects a struct dest"),
}
}
var (
names = make(map[string]interface{}, len(columns))
values = make([]interface{}, 0, len(columns))
)
for i := 0; i < v.NumField(); i++ {
var (
f = t.Field(i)
name = f.Name
)
if tn := f.Tag.Get("ch"); len(tn) != 0 {
name = tn
}
names[name] = v.Field(i).Addr().Interface()
}
for _, name := range columns {
v, found := names[name]
if !found {
return nil, &OpError{
Op: "ScanStruct",
Err: fmt.Errorf("missing destination name %q in %T", name, dest),
}
}
values = append(values, v)
}
return values, nil
}