forked from marcboeker/go-duckdb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
arrow.go
279 lines (228 loc) · 6.58 KB
/
arrow.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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
//go:build !no_duckdb_arrow
package duckdb
/*
#include <stdlib.h>
#include <duckdb.h>
#include <stdint.h>
#ifndef ARROW_C_DATA_INTERFACE
#define ARROW_C_DATA_INTERFACE
#define ARROW_FLAG_DICTIONARY_ORDERED 1
#define ARROW_FLAG_NULLABLE 2
#define ARROW_FLAG_MAP_KEYS_SORTED 4
struct ArrowSchema {
// Array type description
const char* format;
const char* name;
const char* metadata;
int64_t flags;
int64_t n_children;
struct ArrowSchema** children;
struct ArrowSchema* dictionary;
// Release callback
void (*release)(struct ArrowSchema*);
// Opaque producer-specific data
void* private_data;
};
struct ArrowArray {
// Array data description
int64_t length;
int64_t null_count;
int64_t offset;
int64_t n_buffers;
int64_t n_children;
const void** buffers;
struct ArrowArray** children;
struct ArrowArray* dictionary;
// Release callback
void (*release)(struct ArrowArray*);
// Opaque producer-specific data
void* private_data;
};
struct ArrowArrayStream {
void (*get_schema)(struct ArrowArrayStream*);
void (*get_next)(struct ArrowArrayStream*);
void (*get_last_error)(struct ArrowArrayStream*);
void (*release)(struct ArrowArrayStream*);
void* private_data;
};
#endif // ARROW_C_DATA_INTERFACE
*/
import "C"
import (
"context"
"database/sql/driver"
"errors"
"fmt"
"unsafe"
"github.com/apache/arrow-go/v18/arrow"
"github.com/apache/arrow-go/v18/arrow/array"
"github.com/apache/arrow-go/v18/arrow/cdata"
)
// Arrow exposes DuckDB Apache Arrow interface.
// https://duckdb.org/docs/api/c/api#arrow-interface
type Arrow struct {
c *Conn
}
// NewArrowFromConn returns a new Arrow from a DuckDB driver connection.
func NewArrowFromConn(driverConn driver.Conn) (*Arrow, error) {
dbConn, ok := driverConn.(*Conn)
if !ok {
return nil, fmt.Errorf("not a duckdb driver connection")
}
if dbConn.closed {
return nil, errClosedCon
}
return &Arrow{c: dbConn}, nil
}
// QueryContext prepares statements, executes them, returns Apache Arrow array.RecordReader as a result of the last
// executed statement. Arguments are bound to the last statement.
func (a *Arrow) QueryContext(ctx context.Context, query string, args ...any) (array.RecordReader, error) {
if a.c.closed {
return nil, errClosedCon
}
stmts, size, err := a.c.extractStmts(query)
if err != nil {
return nil, err
}
defer C.duckdb_destroy_extracted(&stmts)
// execute all statements without args, except the last one
for i := C.idx_t(0); i < size-1; i++ {
stmt, err := a.c.prepareExtractedStmt(stmts, i)
if err != nil {
return nil, err
}
// send nil args to execute statement and ignore result (using ExecContext since we're ignoring the result anyway)
_, err = stmt.ExecContext(ctx, nil)
stmt.Close()
if err != nil {
return nil, err
}
}
// prepare and execute last statement with args and return result
stmt, err := a.c.prepareExtractedStmt(stmts, size-1)
if err != nil {
return nil, err
}
defer stmt.Close()
res, err := a.execute(stmt, a.anyArgsToNamedArgs(args))
if err != nil {
return nil, err
}
defer C.duckdb_destroy_arrow(res)
sc, err := a.queryArrowSchema(res)
if err != nil {
return nil, err
}
var recs []arrow.Record
defer func() {
for _, r := range recs {
r.Release()
}
}()
rowCount := uint64(C.duckdb_arrow_row_count(*res))
var retrievedRows uint64
for retrievedRows < rowCount {
select {
case <-ctx.Done():
return nil, ctx.Err()
default:
}
rec, err := a.queryArrowArray(res, sc)
if err != nil {
return nil, err
}
recs = append(recs, rec)
retrievedRows += uint64(rec.NumRows())
}
return array.NewRecordReader(sc, recs)
}
// queryArrowSchema fetches the internal arrow schema from the arrow result.
func (a *Arrow) queryArrowSchema(res *C.duckdb_arrow) (*arrow.Schema, error) {
schema := C.calloc(1, C.sizeof_struct_ArrowSchema)
defer func() {
cdata.ReleaseCArrowSchema((*cdata.CArrowSchema)(schema))
C.free(schema)
}()
if state := C.duckdb_query_arrow_schema(
*res,
(*C.duckdb_arrow_schema)(unsafe.Pointer(&schema)),
); state == C.DuckDBError {
return nil, errors.New("duckdb_query_arrow_schema")
}
sc, err := cdata.ImportCArrowSchema((*cdata.CArrowSchema)(schema))
if err != nil {
return nil, fmt.Errorf("%w: ImportCArrowSchema", err)
}
return sc, nil
}
// queryArrowArray fetches an internal arrow array from the arrow result.
//
// This function can be called multiple time to get next chunks,
// which will free the previous out_array.
func (a *Arrow) queryArrowArray(res *C.duckdb_arrow, sc *arrow.Schema) (arrow.Record, error) {
arr := C.calloc(1, C.sizeof_struct_ArrowArray)
defer func() {
cdata.ReleaseCArrowArray((*cdata.CArrowArray)(arr))
C.free(arr)
}()
if state := C.duckdb_query_arrow_array(
*res,
(*C.duckdb_arrow_array)(unsafe.Pointer(&arr)),
); state == C.DuckDBError {
return nil, errors.New("duckdb_query_arrow_array")
}
rec, err := cdata.ImportCRecordBatchWithSchema((*cdata.CArrowArray)(arr), sc)
if err != nil {
return nil, fmt.Errorf("%w: ImportCRecordBatchWithSchema", err)
}
return rec, nil
}
func (a *Arrow) execute(s *Stmt, args []driver.NamedValue) (*C.duckdb_arrow, error) {
if s.closed {
return nil, errClosedCon
}
if err := s.bind(args); err != nil {
return nil, err
}
var res C.duckdb_arrow
if state := C.duckdb_execute_prepared_arrow(*s.stmt, &res); state == C.DuckDBError {
dbErr := C.GoString(C.duckdb_query_arrow_error(res))
C.duckdb_destroy_arrow(&res)
return nil, fmt.Errorf("duckdb_execute_prepared_arrow: %v", dbErr)
}
return &res, nil
}
func (a *Arrow) anyArgsToNamedArgs(args []any) []driver.NamedValue {
if len(args) == 0 {
return nil
}
values := make([]driver.Value, len(args))
for i, arg := range args {
values[i] = arg
}
return argsToNamedArgs(values)
}
// RegisterView registers an Arrow record reader as a view with the given name in DuckDB.
// The returned release function must be called to release the memory once the view is no longer needed.
func (a *Arrow) RegisterView(reader array.RecordReader, name string) (release func(), err error) {
if a.c.closed {
return nil, errClosedCon
}
// duckdb_state duckdb_arrow_scan(duckdb_connection connection, const char *table_name, duckdb_arrow_stream arrow);
stream := C.calloc(1, C.sizeof_struct_ArrowArrayStream)
release = func() {
C.free(stream)
}
cdata.ExportRecordReader(reader, (*cdata.CArrowArrayStream)(stream))
cName := C.CString(name)
defer C.free(unsafe.Pointer(cName))
if state := C.duckdb_arrow_scan(
a.c.duckdbCon,
cName,
(C.duckdb_arrow_stream)(stream),
); state == C.DuckDBError {
release()
return nil, errors.New("duckdb_arrow_scan")
}
return release, nil
}