Skip to content

Commit 07f9c9c

Browse files
committed
Implement number-named parameters. Close #187
1 parent ff38c8e commit 07f9c9c

File tree

2 files changed

+55
-3
lines changed

2 files changed

+55
-3
lines changed

sqlite3.go

+19-3
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,7 @@ type SQLiteStmt struct {
122122
c *SQLiteConn
123123
s *C.sqlite3_stmt
124124
nv int
125+
nn []string
125126
t string
126127
closed bool
127128
cls bool
@@ -382,7 +383,14 @@ func (c *SQLiteConn) Prepare(query string) (driver.Stmt, error) {
382383
t = strings.TrimSpace(C.GoString(tail))
383384
}
384385
nv := int(C.sqlite3_bind_parameter_count(s))
385-
ss := &SQLiteStmt{c: c, s: s, nv: nv, t: t}
386+
var nn []string
387+
for i := 0; i < nv; i++ {
388+
pn := C.GoString(C.sqlite3_bind_parameter_name(s, C.int(i+1)))
389+
if len(pn) > 1 && pn[0] == '$' && 48 <= pn[1] && pn[1] <= 57 {
390+
nn = append(nn, C.GoString(C.sqlite3_bind_parameter_name(s, C.int(i+1))))
391+
}
392+
}
393+
ss := &SQLiteStmt{c: c, s: s, nv: nv, nn: nn, t: t}
386394
runtime.SetFinalizer(ss, (*SQLiteStmt).Close)
387395
return ss, nil
388396
}
@@ -423,8 +431,16 @@ func (s *SQLiteStmt) bind(args []driver.Value) error {
423431
var vargs []bindArg
424432
narg := len(args)
425433
vargs = make([]bindArg, narg)
426-
for i, v := range args {
427-
vargs[i] = bindArg{i + 1, v}
434+
if len(s.nn) > 0 {
435+
for i, v := range s.nn {
436+
if pi, err := strconv.Atoi(v[1:]); err == nil {
437+
vargs[i] = bindArg{pi, args[i]}
438+
}
439+
}
440+
} else {
441+
for i, v := range args {
442+
vargs[i] = bindArg{i + 1, v}
443+
}
428444
}
429445

430446
for _, varg := range vargs {

sqlite3_test.go

+36
Original file line numberDiff line numberDiff line change
@@ -909,3 +909,39 @@ func TestVersion(t *testing.T) {
909909
t.Errorf("Version failed %q, %d, %q\n", s, n, id)
910910
}
911911
}
912+
913+
func TestNumberNamedParams(t *testing.T) {
914+
tempFilename := TempFilename()
915+
db, err := sql.Open("sqlite3", tempFilename)
916+
if err != nil {
917+
t.Fatal("Failed to open database:", err)
918+
}
919+
defer os.Remove(tempFilename)
920+
defer db.Close()
921+
922+
_, err = db.Exec(`
923+
create table foo (id integer, name text, extra text);
924+
`)
925+
if err != nil {
926+
t.Error("Failed to call db.Query:", err)
927+
}
928+
929+
_, err = db.Exec(`insert into foo(id, name, extra) values($1, $2, $2)`, 1, "foo")
930+
if err != nil {
931+
t.Error("Failed to call db.Exec:", err)
932+
}
933+
934+
row := db.QueryRow(`select id, extra from foo where id = $1 and extra = $2`, 1, "foo")
935+
if row == nil {
936+
t.Error("Failed to call db.QueryRow")
937+
}
938+
var id int
939+
var extra string
940+
err = row.Scan(&id, &extra)
941+
if err != nil {
942+
t.Error("Failed to db.Scan:", err)
943+
}
944+
if id != 1 || extra != "foo" {
945+
t.Error("Failed to db.QueryRow: not matched results")
946+
}
947+
}

0 commit comments

Comments
 (0)