Yet another SQLite binding based on:
There are two layers:
- one matching the SQLite API (with Backup, Blob, user-defined Function/Module, ...).
- and another implementing the "database/sql/driver" interface.
Caveat
With Go 1.6, some features do not work anymore without GODEBUG=cgocheck=0
(see Cgo major change).
It seems that the solution is a global variable/lock (see here).
"I would prefer not to" do this.
Custom build
If your OS does not bundle SQLite3 development files (or old ones):
- download and copy SQLite3 files
$ cp ~/Downloads/sqlite-amalgamation-xxx/sqlite3.{c,h} $GOPATH/src/github.com/gwenn/gosqlite
-#cgo linux freebsd pkg-config: sqlite3
-#cgo !linux,!freebsd LDFLAGS: -lsqlite3
+#cgo CFLAGS: -I.
+#cgo CFLAGS: -DSQLITE_ENABLE_COLUMN_METADATA=1
Features (not supported by database/sql/driver):
Named bind parameters.- Partial scan: scan values may be partially scanned (by index or name) or skipped/ignored by passing nil pointer(s).
- Null value: by default, empty string and zero time are bound to NULL for prepared statement's parameters (no need for NullString, NullTime but still supported).
- Null value: Stmt.Scan methods return default Go zero value (0, "", ...) for SQL NULL (no need for NullInt64, NullString, NullTime but still supported).
- Correctly retrieve the time returns by
select current_timestamp
statement or others expressions: in SQLite, expression affinity is NONE. - Full control over connection pool
- No restrictive converter
- Support for metadata
- Nested transaction support
Changes:
Open supports flags.
Conn.Exec handles multiple statements (separated by semicolons) properly.
Conn.Prepare can optionally bind as well.
Conn.Prepare can reuse already prepared Stmt.
Conn.Close ensures that all dangling statements are finalized.
Stmt.Exec is renamed in Stmt.Bind and a new Stmt.Exec method is introduced to bind and step.
Stmt.Bind uses native sqlite3_bind_x methods and failed if unsupported type.
Stmt.NamedBind can be used to bind by name.
Stmt.Next returns a (bool, os.Error) couple like Reader.Read.
Stmt.Scan uses native sqlite3_column_x methods.
Stmt.NamedScan is added. It's compliant with go-dbi.
Stmt.ScanByIndex/ScanByName are added to test NULL value.
Currently, the weak point of the binding is the Scan methods:
The original implementation is using this strategy:
- convert the stored value to a []byte by calling sqlite3_column_blob,
- convert the bytes to the desired Go type with correct feedback in case of illegal conversion,
- but apparently no support for NULL value.
Using the native sqlite3_column_x implies:
- optimal conversion from the storage type to Go type (when they match),
- lossy conversion when types mismatch (select cast('M' as int); --> 0),
- NULL value can be returned only for **type, otherwise a default value (0, false, "") is returned.
SQLite logs (SQLITE_CONFIG_LOG) can be activated by:
- ConfigLog function
- or
export SQLITE_LOG=1
Similar projects created after Jul 17, 2011:
https://github.com/mattn/go-sqlite3 (Nov 11, 2011)
https://github.com/mxk/go-sqlite (Feb 12, 2013)
https://github.com/crawshaw/sqlite (Mar 28, 2018)
https://github.com/bvinc/go-sqlite-lite (Aug 7, 2018)
Additions:
Conn.Exists
Conn.OneValue
Conn.OpenVfs
Conn.EnableFkey/IsFKeyEnabled
Conn.Changes/TotalChanges
Conn.LastInsertRowid
Conn.Interrupt
Conn.Begin/BeginTransaction(type)/Commit/Rollback
Conn.GetAutocommit
Conn.EnableLoadExtension/LoadExtension
Conn.IntegrityCheck
Stmt.Insert/ExecDml/Select/SelectOneRow
Stmt.BindParameterCount/BindParameterIndex(name)/BindParameterName(index)
Stmt.ClearBindings
Stmt.ColumnCount/ColumnNames/ColumnIndex(name)/ColumnName(index)/ColumnType(index)
Stmt.ReadOnly
Stmt.Busy
Blob:
ZeroBlobLength
Conn.NewBlobReader
Conn.NewBlobReadWriter
Meta:
Conn.Databases
Conn.Tables/Views/Indexes
Conn.Columns
Conn.ForeignKeys
Conn.TableIndexes/IndexColumns
Time:
JulianDay
JulianDayToUTC
JulianDayToLocalTime
UnixTime, JulianTime and TimeStamp used to persist go time in formats supported by SQLite3 date functions.
Trace:
Conn.BusyHandler
Conn.Profile
Conn.ProgressHandler
Conn.SetAuthorizer
Conn.Trace
Stmt.Status
Hook:
Conn.CommitHook
Conn.RollbackHook
Conn.UpdateHook
Function:
Conn.CreateScalarFunction
Conn.CreateAggregateFunction
Virtual Table (partial support):
Conn.CreateModule
Conn.DeclareVTab
GC:
Although Go is gced, there is no destructor (see http://www.airs.com/blog/archives/362).
In the gosqlite wrapper, no finalizer is used.
So users must ensure that C resources (database connections, prepared statements, BLOBs, Backups) are destroyed/deallocated by calling Conn.Close, Stmt.Finalize, BlobReader.Close, Backup.Close.
Therefore, sqlite3_close/sqlite3_next_stmt are used by Conn.Close to free the database connection and all dangling statements (not sqlite3_close_v2) (see http://sqlite.org/c3ref/close.html).
Benchmarks:
$ go test -bench . -benchmem
BenchmarkValuesScan 500000 6265 ns/op 74 B/op 3 allocs/op
BenchmarkScan 500000 4994 ns/op 41 B/op 4 allocs/op
BenchmarkNamedScan 500000 4960 ns/op 93 B/op 7 allocs/op
BenchmarkInsert 500000 4085 ns/op 16 B/op 1 allocs/op
BenchmarkNamedInsert 500000 4798 ns/op 64 B/op 4 allocs/op
BenchmarkDisabledCache 100000 19841 ns/op 117 B/op 3 allocs/op
BenchmarkEnabledCache 2000000 790 ns/op 50 B/op 1 allocs/op
BenchmarkLike 1000000 2605 ns/op 0 B/op 0 allocs/op
BenchmarkHalf 500000 4988 ns/op 33 B/op 1 allocs/op
BenchmarkRegexp 500000 5557 ns/op 8 B/op 1 allocs/op