-
-
Notifications
You must be signed in to change notification settings - Fork 16
/
vtab_test.go
120 lines (105 loc) · 2.27 KB
/
vtab_test.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
package sqlite3_test
import (
"fmt"
"log"
"github.com/ncruces/go-sqlite3"
_ "github.com/ncruces/go-sqlite3/embed"
)
func ExampleCreateModule() {
db, err := sqlite3.Open(":memory:")
if err != nil {
log.Fatal(err)
}
defer db.Close()
err = sqlite3.CreateModule[seriesTable](db, "generate_series", nil,
func(db *sqlite3.Conn, module, schema, table string, arg ...string) (seriesTable, error) {
err := db.DeclareVTab(`CREATE TABLE x(value, start HIDDEN, stop HIDDEN, step HIDDEN)`)
return seriesTable{}, err
})
if err != nil {
log.Fatal(err)
}
stmt, _, err := db.Prepare(`SELECT rowid, value FROM generate_series(2, 10, 3)`)
if err != nil {
log.Fatal(err)
}
defer stmt.Close()
for stmt.Step() {
fmt.Println(stmt.ColumnInt(0), stmt.ColumnInt(1))
}
if err := stmt.Err(); err != nil {
log.Fatal(err)
}
// Output:
// 2 2
// 5 5
// 8 8
}
type seriesTable struct{}
func (seriesTable) BestIndex(idx *sqlite3.IndexInfo) error {
for i, cst := range idx.Constraint {
switch cst.Column {
case 1, 2, 3: // start, stop, step
if cst.Op == sqlite3.INDEX_CONSTRAINT_EQ && cst.Usable {
idx.ConstraintUsage[i] = sqlite3.IndexConstraintUsage{
ArgvIndex: cst.Column,
Omit: true,
}
}
}
}
idx.IdxNum = 1
idx.IdxStr = "idx"
return nil
}
func (seriesTable) Open() (sqlite3.VTabCursor, error) {
return &seriesCursor{}, nil
}
type seriesCursor struct {
start int64
stop int64
step int64
value int64
}
func (cur *seriesCursor) Filter(idxNum int, idxStr string, arg ...sqlite3.Value) error {
if idxNum != 1 || idxStr != "idx" {
return nil
}
cur.start = 0
cur.stop = 1000
cur.step = 1
if len(arg) > 0 {
cur.start = arg[0].Int64()
}
if len(arg) > 1 {
cur.stop = arg[1].Int64()
}
if len(arg) > 2 {
cur.step = arg[2].Int64()
}
cur.value = cur.start
return nil
}
func (cur *seriesCursor) Column(ctx sqlite3.Context, col int) error {
switch col {
case 0:
ctx.ResultInt64(cur.value)
case 1:
ctx.ResultInt64(cur.start)
case 2:
ctx.ResultInt64(cur.stop)
case 3:
ctx.ResultInt64(cur.step)
}
return nil
}
func (cur *seriesCursor) Next() error {
cur.value += cur.step
return nil
}
func (cur *seriesCursor) EOF() bool {
return cur.value > cur.stop
}
func (cur *seriesCursor) RowID() (int64, error) {
return cur.value, nil
}