Commit 3d2c58b7 authored by gwenn's avatar gwenn

Try to introduce a Value interface for scan.

Split tests.
parent c1750896
package sqlite_test
import (
. "github.com/gwenn/gosqlite"
"testing"
)
func BenchmarkScan(b *testing.B) {
b.StopTimer()
db, _ := Open("")
defer db.Close()
db.Exec("DROP TABLE IF EXISTS test")
db.Exec("CREATE TABLE test (id INTEGER PRIMARY KEY AUTOINCREMENT, float_num REAL, int_num INTEGER, a_string TEXT)")
db.Begin()
s, _ := db.Prepare("INSERT INTO test (float_num, int_num, a_string) VALUES (?, ?, ?)")
for i := 0; i < 1000; i++ {
s.Exec(float64(i)*float64(3.14), i, "hello")
}
s.Finalize()
db.Commit()
b.StartTimer()
for i := 0; i < b.N; i++ {
cs, _ := db.Prepare("SELECT float_num, int_num, a_string FROM test")
var fnum float64
var inum int64
var sstr string
for Must(cs.Next()) {
cs.Scan(&fnum, &inum, &sstr)
}
cs.Finalize()
}
}
func BenchmarkNamedScan(b *testing.B) {
b.StopTimer()
db, _ := Open("")
defer db.Close()
db.Exec("DROP TABLE IF EXISTS test")
db.Exec("CREATE TABLE test (id INTEGER PRIMARY KEY AUTOINCREMENT, float_num REAL, int_num INTEGER, a_string TEXT)")
db.Begin()
s, _ := db.Prepare("INSERT INTO test (float_num, int_num, a_string) VALUES (?, ?, ?)")
for i := 0; i < 1000; i++ {
s.Exec(float64(i)*float64(3.14), i, "hello")
}
s.Finalize()
db.Commit()
b.StartTimer()
for i := 0; i < b.N; i++ {
cs, _ := db.Prepare("SELECT float_num, int_num, a_string FROM test")
var fnum float64
var inum int64
var sstr string
for Must(cs.Next()) {
cs.NamedScan("float_num", &fnum, "int_num", &inum, "a_string", &sstr)
}
cs.Finalize()
}
}
func BenchmarkInsert(b *testing.B) {
db, _ := Open("")
defer db.Close()
db.Exec("DROP TABLE IF EXISTS test")
db.Exec("CREATE TABLE test (id INTEGER PRIMARY KEY AUTOINCREMENT," +
" float_num REAL, int_num INTEGER, a_string TEXT)")
s, _ := db.Prepare("INSERT INTO test (float_num, int_num, a_string)" +
" VALUES (?, ?, ?)")
defer s.Finalize()
db.Begin()
for i := 0; i < b.N; i++ {
s.Exec(float64(i)*float64(3.14), i, "hello")
}
db.Commit()
}
package sqlite_test
import (
. "github.com/gwenn/gosqlite"
"testing"
)
func createIndex(db *Conn, t *testing.T) {
err := db.Exec("DROP INDEX IF EXISTS test_index;" +
"CREATE INDEX test_index on test(a_string)")
if err != nil {
t.Fatalf("error creating index: %s", err)
}
}
func TestDatabases(t *testing.T) {
db := open(t)
defer db.Close()
databases, err := db.Databases()
if err != nil {
t.Fatalf("error looking for databases: %s", err)
}
if len(databases) != 1 {
t.Errorf("Expected one database but got %d\n", len(databases))
}
if _, ok := databases["main"]; !ok {
t.Errorf("Expected 'main' database\n")
}
}
func TestTables(t *testing.T) {
db := open(t)
defer db.Close()
tables, err := db.Tables()
if err != nil {
t.Fatalf("error looking for tables: %s", err)
}
if len(tables) != 0 {
t.Errorf("Expected no table but got %d\n", len(tables))
}
createTable(db, t)
tables, err = db.Tables()
if err != nil {
t.Fatalf("error looking for tables: %s", err)
}
if len(tables) != 1 {
t.Errorf("Expected one table but got %d\n", len(tables))
}
if tables[0] != "test" {
t.Errorf("Wrong table name: 'test' <> %s\n", tables[0])
}
}
func TestColumns(t *testing.T) {
db := open(t)
defer db.Close()
createTable(db, t)
columns, err := db.Columns("test")
if err != nil {
t.Fatalf("error listing columns: %s", err)
}
if len(columns) != 4 {
t.Fatalf("Expected 4 columns <> %d", len(columns))
}
column := columns[2]
if column.Name != "int_num" {
t.Errorf("Wrong column name: 'int_num' <> %s", column.Name)
}
}
func TestForeignKeys(t *testing.T) {
db := open(t)
defer db.Close()
err := db.Exec("CREATE TABLE parent (id INTEGER PRIMARY KEY);" +
"CREATE TABLE child (id INTEGER PRIMARY KEY, parentId INTEGER, " +
"FOREIGN KEY (parentId) REFERENCES parent(id));")
if err != nil {
t.Fatalf("error creating tables: %s", err)
}
fks, err := db.ForeignKeys("child")
if err != nil {
t.Fatalf("error listing FKs: %s", err)
}
if len(fks) != 1 {
t.Fatalf("Expected 1 FK <> %d", len(fks))
}
fk := fks[0]
if fk.From[0] != "parentId" || fk.Table != "parent" || fk.To[0] != "id" {
t.Errorf("Unexpected FK data: %#v", fk)
}
}
func TestIndexes(t *testing.T) {
db := open(t)
defer db.Close()
createTable(db, t)
createIndex(db, t)
indexes, err := db.Indexes("test")
if err != nil {
t.Fatalf("error listing indexes: %s", err)
}
if len(indexes) != 1 {
t.Fatalf("Expected one index <> %d", len(indexes))
}
index := indexes[0]
if index.Name != "test_index" {
t.Errorf("Wrong index name: 'test_index' <> %s", index.Name)
}
if index.Unique {
t.Errorf("Index 'test_index' is not unique")
}
columns, err := db.IndexColumns("test_index")
if err != nil {
t.Fatalf("error listing index columns: %s", err)
}
if len(columns) != 1 {
t.Fatalf("Expected one column <> %d", len(columns))
}
column := columns[0]
if column.Name != "a_string" {
t.Errorf("Wrong column name: 'a_string' <> %s", column.Name)
}
}
...@@ -581,6 +581,38 @@ func (s *Stmt) Scan(args ...interface{}) os.Error { ...@@ -581,6 +581,38 @@ func (s *Stmt) Scan(args ...interface{}) os.Error {
return nil return nil
} }
// Calls sqlite3_column_count and sqlite3_column_(blob|double|int|int64|text) depending on columns type.
// http://sqlite.org/c3ref/column_blob.html
func (s *Stmt) ScanNamedValues(values ...NamedValue) os.Error {
n := s.ColumnCount()
if n != len(values) { // What happens when the number of arguments is less than the number of columns?
return os.NewError(fmt.Sprintf("incorrect argument count for Stmt.ScanValues: have %d want %d", len(values), n))
}
for _, v := range values {
index, err := s.ColumnIndex(v.Name()) // How to look up only once for one statement ?
if err != nil {
return err
}
s.ScanValue(index, v)
}
return nil
}
// Calls sqlite3_column_count and sqlite3_column_(blob|double|int|int64|text) depending on columns type.
// http://sqlite.org/c3ref/column_blob.html
func (s *Stmt) ScanValues(values ...Value) os.Error {
n := s.ColumnCount()
if n != len(values) { // What happens when the number of arguments is less than the number of columns?
return os.NewError(fmt.Sprintf("incorrect argument count for Stmt.ScanValues: have %d want %d", len(values), n))
}
for i, v := range values {
s.ScanValue(i, v)
}
return nil
}
// Calls http://sqlite.org/c3ref/sql.html // Calls http://sqlite.org/c3ref/sql.html
func (s *Stmt) SQL() string { func (s *Stmt) SQL() string {
return C.GoString(C.sqlite3_sql(s.stmt)) return C.GoString(C.sqlite3_sql(s.stmt))
...@@ -616,8 +648,7 @@ func (s *Stmt) NamedScanColumn(name string, value interface{}, nullable bool) (b ...@@ -616,8 +648,7 @@ func (s *Stmt) NamedScanColumn(name string, value interface{}, nullable bool) (b
return s.ScanColumn(index, value, true) return s.ScanColumn(index, value, true)
} }
// The leftmost column is number 0. // The leftmost column/index is number 0.
// Index starts at 0.
// Set nullable to false to skip NULL type test. // Set nullable to false to skip NULL type test.
// Returns true when nullable is true and column is null. // Returns true when nullable is true and column is null.
// Calls sqlite3_column_(blob|double|int|int64|text) depending on args type. // Calls sqlite3_column_(blob|double|int|int64|text) depending on args type.
...@@ -685,6 +716,43 @@ func (s *Stmt) ScanColumn(index int, value interface{}, nullable bool) (bool, os ...@@ -685,6 +716,43 @@ func (s *Stmt) ScanColumn(index int, value interface{}, nullable bool) (bool, os
return isNull, nil return isNull, nil
} }
type NamedValue interface {
Value
Name() string
}
type Value interface {
setNull(bool)
setInt(int64) // Versus int?
setFloat(float64)
setText(string)
setBlob([]byte)
}
// The leftmost column/index is number 0.
// Calls sqlite3_column_(blob|double|int|int64|text) depending on columns type.
// http://sqlite.org/c3ref/column_blob.html
func (s *Stmt) ScanValue(index int, value Value) {
switch s.columnType(index) {
case C.SQLITE_NULL:
value.setNull(true)
case C.SQLITE_TEXT:
p := C.sqlite3_column_text(s.stmt, C.int(index))
n := C.sqlite3_column_bytes(s.stmt, C.int(index))
value.setText(C.GoStringN((*C.char)(unsafe.Pointer(p)), n))
case C.SQLITE_INTEGER:
value.setInt(int64(C.sqlite3_column_int64(s.stmt, C.int(index))))
case C.SQLITE_FLOAT:
value.setFloat(float64(C.sqlite3_column_double(s.stmt, C.int(index))))
case C.SQLITE_BLOB:
p := C.sqlite3_column_blob(s.stmt, C.int(index))
n := C.sqlite3_column_bytes(s.stmt, C.int(index))
value.setBlob((*[1 << 30]byte)(unsafe.Pointer(p))[0:n])
default:
panic("The column type is not one of SQLITE_INTEGER, SQLITE_FLOAT, SQLITE_TEXT, SQLITE_BLOB, or SQLITE_NULL")
}
}
// Calls http://sqlite.org/c3ref/finalize.html // Calls http://sqlite.org/c3ref/finalize.html
func (s *Stmt) Finalize() os.Error { func (s *Stmt) Finalize() os.Error {
rv := C.sqlite3_finalize(s.stmt) rv := C.sqlite3_finalize(s.stmt)
......
package sqlite_test package sqlite_test
import ( import (
"fmt"
. "github.com/gwenn/gosqlite" . "github.com/gwenn/gosqlite"
"strings" "strings"
"testing" "testing"
) )
func trace(d interface{}, t string) {
fmt.Printf("%s: %s\n", d, t)
}
func authorizer(d interface{}, action Action, arg1, arg2, arg3, arg4 string) Auth {
fmt.Printf("%s: %d, %s, %s, %s, %s\n", d, action, arg1, arg2, arg3, arg4)
return AUTH_OK
}
func profile(d interface{}, sql string, nanoseconds uint64) {
fmt.Printf("%s: %s = %d\n", d, sql, nanoseconds/1000)
}
func progressHandler(d interface{}) int {
fmt.Print("+")
return 0
}
func open(t *testing.T) *Conn { func open(t *testing.T) *Conn {
db, err := Open("", OPEN_READWRITE, OPEN_CREATE, OPEN_FULLMUTEX, OPEN_URI) db, err := Open("", OPEN_READWRITE, OPEN_CREATE, OPEN_FULLMUTEX, OPEN_URI)
if err != nil { if err != nil {
...@@ -33,15 +14,6 @@ func open(t *testing.T) *Conn { ...@@ -33,15 +14,6 @@ func open(t *testing.T) *Conn {
if db == nil { if db == nil {
t.Fatal("opened database is nil") t.Fatal("opened database is nil")
} }
//db.Trace(trace, "TRACE")
/*
err = db.SetAuthorizer(authorizer, "AUTH")
if err != nil {
t.Fatal("couldn't set an authorizer", err)
}
*/
//db.Profile(profile, "PROFILE")
//db.ProgressHandler(progressHandler, 20, nil)
return db return db
} }
...@@ -54,14 +26,6 @@ func createTable(db *Conn, t *testing.T) { ...@@ -54,14 +26,6 @@ func createTable(db *Conn, t *testing.T) {
} }
} }
func createIndex(db *Conn, t *testing.T) {
err := db.Exec("DROP INDEX IF EXISTS test_index;" +
"CREATE INDEX test_index on test(a_string)")
if err != nil {
t.Fatalf("error creating index: %s", err)
}
}
func TestVersion(t *testing.T) { func TestVersion(t *testing.T) {
v := Version() v := Version()
if !strings.HasPrefix(v, "3") { if !strings.HasPrefix(v, "3") {
...@@ -71,12 +35,9 @@ func TestVersion(t *testing.T) { ...@@ -71,12 +35,9 @@ func TestVersion(t *testing.T) {
func TestOpen(t *testing.T) { func TestOpen(t *testing.T) {
db := open(t) db := open(t)
db.Trace(nil, nil) if err := db.Close(); err != nil {
db.SetAuthorizer(nil, nil) t.Fatalf("Error closing database: %s", err)
db.Profile(nil, nil) }
db.ProgressHandler(nil, 0, nil)
db.BusyHandler(nil, nil)
db.Close()
} }
func TestEnableFKey(t *testing.T) { func TestEnableFKey(t *testing.T) {
...@@ -287,121 +248,6 @@ func TestInsertWithStatement(t *testing.T) { ...@@ -287,121 +248,6 @@ func TestInsertWithStatement(t *testing.T) {
} }
} }
func TestDatabases(t *testing.T) {
db := open(t)
defer db.Close()
databases, err := db.Databases()
if err != nil {
t.Fatalf("error looking for databases: %s", err)
}
if len(databases) != 1 {
t.Errorf("Expected one database but got %d\n", len(databases))
}
if _, ok := databases["main"]; !ok {
t.Errorf("Expected 'main' database\n")
}
}
func TestTables(t *testing.T) {
db := open(t)
defer db.Close()
tables, err := db.Tables()
if err != nil {
t.Fatalf("error looking for tables: %s", err)
}
if len(tables) != 0 {
t.Errorf("Expected no table but got %d\n", len(tables))
}
createTable(db, t)
tables, err = db.Tables()
if err != nil {
t.Fatalf("error looking for tables: %s", err)
}
if len(tables) != 1 {
t.Errorf("Expected one table but got %d\n", len(tables))
}
if tables[0] != "test" {
t.Errorf("Wrong table name: 'test' <> %s\n", tables[0])
}
}
func TestColumns(t *testing.T) {
db := open(t)
defer db.Close()
createTable(db, t)
columns, err := db.Columns("test")
if err != nil {
t.Fatalf("error listing columns: %s", err)
}
if len(columns) != 4 {
t.Fatalf("Expected 4 columns <> %d", len(columns))
}
column := columns[2]
if column.Name != "int_num" {
t.Errorf("Wrong column name: 'int_num' <> %s", column.Name)
}
}
func TestForeignKeys(t *testing.T) {
db := open(t)
defer db.Close()
err := db.Exec("CREATE TABLE parent (id INTEGER PRIMARY KEY);" +
"CREATE TABLE child (id INTEGER PRIMARY KEY, parentId INTEGER, " +
"FOREIGN KEY (parentId) REFERENCES parent(id));")
if err != nil {
t.Fatalf("error creating tables: %s", err)
}
fks, err := db.ForeignKeys("child")
if err != nil {
t.Fatalf("error listing FKs: %s", err)
}
if len(fks) != 1 {
t.Fatalf("Expected 1 FK <> %d", len(fks))
}
fk := fks[0]
if fk.From[0] != "parentId" || fk.Table != "parent" || fk.To[0] != "id" {
t.Errorf("Unexpected FK data: %#v", fk)
}
}
func TestIndexes(t *testing.T) {
db := open(t)
defer db.Close()
createTable(db, t)
createIndex(db, t)
indexes, err := db.Indexes("test")
if err != nil {
t.Fatalf("error listing indexes: %s", err)
}
if len(indexes) != 1 {
t.Fatalf("Expected one index <> %d", len(indexes))
}
index := indexes[0]
if index.Name != "test_index" {
t.Errorf("Wrong index name: 'test_index' <> %s", index.Name)
}
if index.Unique {
t.Errorf("Index 'test_index' is not unique")
}
columns, err := db.IndexColumns("test_index")
if err != nil {
t.Fatalf("error listing index columns: %s", err)
}
if len(columns) != 1 {
t.Fatalf("Expected one column <> %d", len(columns))
}
column := columns[0]
if column.Name != "a_string" {
t.Errorf("Wrong column name: 'a_string' <> %s", column.Name)
}
}
func TestBlob(t *testing.T) { func TestBlob(t *testing.T) {
db := open(t) db := open(t)
defer db.Close() defer db.Close()
...@@ -534,80 +380,3 @@ func TestLoadExtension(t *testing.T) { ...@@ -534,80 +380,3 @@ func TestLoadExtension(t *testing.T) {
} }
} }
*/ */
func BenchmarkScan(b *testing.B) {
b.StopTimer()
db, _ := Open("")
defer db.Close()
db.Exec("DROP TABLE IF EXISTS test")
db.Exec("CREATE TABLE test (id INTEGER PRIMARY KEY AUTOINCREMENT, float_num REAL, int_num INTEGER, a_string TEXT)")
db.Begin()
s, _ := db.Prepare("INSERT INTO test (float_num, int_num, a_string) VALUES (?, ?, ?)")
for i := 0; i < 1000; i++ {
s.Exec(float64(i)*float64(3.14), i, "hello")
}
s.Finalize()
db.Commit()
b.StartTimer()
for i := 0; i < b.N; i++ {
cs, _ := db.Prepare("SELECT float_num, int_num, a_string FROM test")
var fnum float64
var inum int64
var sstr string
for Must(cs.Next()) {
cs.Scan(&fnum, &inum, &sstr)
}
cs.Finalize()
}
}
func BenchmarkNamedScan(b *testing.B) {
b.StopTimer()
db, _ := Open("")
defer db.Close()
db.Exec("DROP TABLE IF EXISTS test")
db.Exec("CREATE TABLE test (id INTEGER PRIMARY KEY AUTOINCREMENT, float_num REAL, int_num INTEGER, a_string TEXT)")
db.Begin()
s, _ := db.Prepare("INSERT INTO test (float_num, int_num, a_string) VALUES (?, ?, ?)")
for i := 0; i < 1000; i++ {
s.Exec(float64(i)*float64(3.14), i, "hello")
}
s.Finalize()
db.Commit()
b.StartTimer()
for i := 0; i < b.N; i++ {
cs, _ := db.Prepare("SELECT float_num, int_num, a_string FROM test")
var fnum float64
var inum int64
var sstr string
for Must(cs.Next()) {
cs.NamedScan("float_num", &fnum, "int_num", &inum, "a_string", &sstr)
}
cs.Finalize()
}
}
func BenchmarkInsert(b *testing.B) {
db, _ := Open("")
defer db.Close()
db.Exec("DROP TABLE IF EXISTS test")
db.Exec("CREATE TABLE test (id INTEGER PRIMARY KEY AUTOINCREMENT," +
" float_num REAL, int_num INTEGER, a_string TEXT)")
s, _ := db.Prepare("INSERT INTO test (float_num, int_num, a_string)" +
" VALUES (?, ?, ?)")
defer s.Finalize()
db.Begin()
for i := 0; i < b.N; i++ {
s.Exec(float64(i)*float64(3.14), i, "hello")
}
db.Commit()
}
package sqlite_test
import (
"fmt"
. "github.com/gwenn/gosqlite"
"testing"
)
func trace(d interface{}, t string) {
fmt.Printf("%s: %s\n", d, t)
}
func authorizer(d interface{}, action Action, arg1, arg2, arg3, arg4 string) Auth {
fmt.Printf("%s: %d, %s, %s, %s, %s\n", d, action, arg1, arg2, arg3, arg4)
return AUTH_OK
}
func profile(d interface{}, sql string, nanoseconds uint64) {
fmt.Printf("%s: %s = %d\n", d, sql, nanoseconds/1000)
}
func progressHandler(d interface{}) int {
fmt.Print("+")
return 0
}
func TestNoTrace(t *testing.T) {
db, err := Open("")
if err != nil {
t.Fatalf("couldn't open database file: %s", err)
}
db.Trace(nil, nil)
db.SetAuthorizer(nil, nil)
db.Profile(nil, nil)
db.ProgressHandler(nil, 0, nil)
db.BusyHandler(nil, nil)
db.Close()
}
func TestTrace(t *testing.T) {
db, err := Open("")
db.Trace(trace, "TRACE")
err = db.SetAuthorizer(authorizer, "AUTH")
if err != nil {
t.Fatal("couldn't set an authorizer", err)
}
db.Profile(profile, "PROFILE")
db.ProgressHandler(progressHandler, 1, /*20*/ nil)
db.Exists("SELECT 1 WHERE 1 = ?", 1)
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment