Skip to content

Commit b1cd85e

Browse files
committed
use extended error codes and handle closed db connections in lastError
This commit changes the error handling logic so that it respects the offending result code (instead of only relying on sqlite3_errcode) and changes the db connection to always report the extended result code (which eliminates the need to call sqlite3_extended_errcode). These changes make it possible to correctly and safely handle errors when the underlying db connection has been closed.
1 parent 376bff0 commit b1cd85e

9 files changed

+260
-86
lines changed

backup.go

+4-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,10 @@ func (destConn *SQLiteConn) Backup(dest string, srcConn *SQLiteConn, src string)
3636
runtime.SetFinalizer(bb, (*SQLiteBackup).Finish)
3737
return bb, nil
3838
}
39-
return nil, destConn.lastError()
39+
if destConn.db != nil {
40+
return nil, destConn.lastError(int(C.sqlite3_extended_errcode(destConn.db)))
41+
}
42+
return nil, Error{Code: 1, ExtendedCode: 1, err: "backup: destination connection is nil"}
4043
}
4144

4245
// Step to backs up for one step. Calls the underlying `sqlite3_backup_step`

error.go

+25-3
Original file line numberDiff line numberDiff line change
@@ -13,13 +13,16 @@ package sqlite3
1313
#endif
1414
*/
1515
import "C"
16-
import "syscall"
16+
import (
17+
"sync"
18+
"syscall"
19+
)
1720

1821
// ErrNo inherit errno.
1922
type ErrNo int
2023

2124
// ErrNoMask is mask code.
22-
const ErrNoMask C.int = 0xff
25+
const ErrNoMask = 0xff
2326

2427
// ErrNoExtended is extended errno.
2528
type ErrNoExtended int
@@ -85,7 +88,7 @@ func (err Error) Error() string {
8588
if err.err != "" {
8689
str = err.err
8790
} else {
88-
str = C.GoString(C.sqlite3_errstr(C.int(err.Code)))
91+
str = errorString(int(err.Code))
8992
}
9093
if err.SystemErrno != 0 {
9194
str += ": " + err.SystemErrno.Error()
@@ -148,3 +151,22 @@ const (
148151
ErrNoticeRecoverRollback = ErrNoExtended(ErrNotice | 2<<8)
149152
ErrWarningAutoIndex = ErrNoExtended(ErrWarning | 1<<8)
150153
)
154+
155+
var errStrCache sync.Map // int => string
156+
157+
// errorString returns the result of sqlite3_errstr for result code rv,
158+
// which may be cached.
159+
func errorString(rv int) string {
160+
if v, ok := errStrCache.Load(rv); ok {
161+
return v.(string)
162+
}
163+
s := C.GoString(C.sqlite3_errstr(C.int(rv)))
164+
// Prevent the cache from growing unbounded by ignoring invalid
165+
// error codes.
166+
if s != "unknown error" {
167+
if v, loaded := errStrCache.LoadOrStore(rv, s); loaded {
168+
s = v.(string)
169+
}
170+
}
171+
return s
172+
}

sqlite3.go

+73-36
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,6 @@ _sqlite3_bind_blob(sqlite3_stmt *stmt, int n, void *p, int np) {
7878
return sqlite3_bind_blob(stmt, n, p, np, SQLITE_TRANSIENT);
7979
}
8080
81-
#include <stdio.h>
8281
#include <stdint.h>
8382
8483
static int
@@ -141,7 +140,7 @@ _sqlite3_prepare_v2_internal(sqlite3 *db, const char *zSql, int nBytes, sqlite3_
141140
142141
// Our own implementation of ctype.h's isspace (for simplicity and to avoid
143142
// whatever locale shenanigans are involved with the Libc's isspace).
144-
static int _sqlite3_isspace(unsigned char c) {
143+
static inline int _sqlite3_isspace(unsigned char c) {
145144
return c == ' ' || c - '\t' < 5;
146145
}
147146
@@ -209,6 +208,7 @@ static int _sqlite3_prepare_v2(sqlite3 *db, const char *zSql, int nBytes, sqlite
209208
// _sqlite3_exec_no_args executes all of the statements in zSql. None of the
210209
// statements are allowed to have positional arguments.
211210
int _sqlite3_exec_no_args(sqlite3 *db, const char *zSql, int nBytes, int64_t *rowid, int64_t *changes) {
211+
const char *end = zSql + nBytes;
212212
while (*zSql && nBytes > 0) {
213213
sqlite3_stmt *stmt;
214214
const char *tail;
@@ -226,12 +226,25 @@ int _sqlite3_exec_no_args(sqlite3 *db, const char *zSql, int nBytes, int64_t *ro
226226
*changes = sqlite3_changes64(db);
227227
*rowid = sqlite3_last_insert_rowid(db);
228228
229-
sqlite3_finalize(stmt);
229+
int rv2 = sqlite3_finalize(stmt);
230+
if (rv == SQLITE_OK) {
231+
rv = rv2;
232+
}
233+
230234
if (rv != SQLITE_OK && rv != SQLITE_DONE) {
235+
// Handle statements that are empty or consist only of a comment.
236+
if (rv == SQLITE_MISUSE && sqlite3_errcode(db) == 0) {
237+
rv = SQLITE_OK;
238+
}
231239
return rv;
232240
}
233241
234-
nBytes -= tail - zSql;
242+
// Trim leading space to handle statements with trailing whitespace.
243+
// This can save us an additional call to sqlite3_prepare_v2.
244+
while (tail < end && _sqlite3_isspace(*tail)) {
245+
tail++;
246+
}
247+
nBytes -= (tail - zSql);
235248
zSql = tail;
236249
}
237250
return SQLITE_OK;
@@ -386,6 +399,21 @@ static go_sqlite3_text_column _sqlite3_column_blob(sqlite3_stmt *stmt, int idx)
386399
}
387400
return r;
388401
}
402+
403+
typedef struct {
404+
const char *msg;
405+
int code;
406+
} go_sqlite3_db_error;
407+
408+
static go_sqlite3_db_error _sqlite3_db_error(sqlite3 *db) {
409+
if (db) {
410+
return (go_sqlite3_db_error){
411+
.msg = sqlite3_errmsg(db),
412+
.code = sqlite3_extended_errcode(db)
413+
};
414+
}
415+
return (go_sqlite3_db_error){ 0 };
416+
}
389417
*/
390418
import "C"
391419
import (
@@ -744,7 +772,7 @@ func (c *SQLiteConn) RegisterCollation(name string, cmp func(string, string) int
744772
defer C.free(unsafe.Pointer(cname))
745773
rv := C.sqlite3_create_collation(c.db, cname, C.SQLITE_UTF8, handle, (*[0]byte)(unsafe.Pointer(C.compareTrampoline)))
746774
if rv != C.SQLITE_OK {
747-
return c.lastError()
775+
return c.lastError(int(rv))
748776
}
749777
return nil
750778
}
@@ -886,7 +914,7 @@ func (c *SQLiteConn) RegisterFunc(name string, impl any, pure bool) error {
886914
}
887915
rv := sqlite3CreateFunction(c.db, cname, C.int(numArgs), C.int(opts), newHandle(c, &fi), C.callbackTrampoline, nil, nil)
888916
if rv != C.SQLITE_OK {
889-
return c.lastError()
917+
return c.lastError(int(rv))
890918
}
891919
return nil
892920
}
@@ -1015,7 +1043,7 @@ func (c *SQLiteConn) RegisterAggregator(name string, impl any, pure bool) error
10151043
}
10161044
rv := sqlite3CreateFunction(c.db, cname, C.int(stepNArgs), C.int(opts), newHandle(c, &ai), nil, C.stepTrampoline, C.doneTrampoline)
10171045
if rv != C.SQLITE_OK {
1018-
return c.lastError()
1046+
return c.lastError(int(rv))
10191047
}
10201048
return nil
10211049
}
@@ -1027,32 +1055,38 @@ func (c *SQLiteConn) AutoCommit() bool {
10271055
return int(C.sqlite3_get_autocommit(c.db)) != 0
10281056
}
10291057

1030-
func (c *SQLiteConn) lastError() error {
1031-
return lastError(c.db)
1058+
func (c *SQLiteConn) lastError(rv int) error {
1059+
return lastError(c.db, rv)
10321060
}
10331061

1034-
// Note: may be called with db == nil
1035-
func lastError(db *C.sqlite3) error {
1036-
rv := C.sqlite3_errcode(db) // returns SQLITE_NOMEM if db == nil
1037-
if rv == C.SQLITE_OK {
1062+
func lastError(db *C.sqlite3, rv int) error {
1063+
if rv == SQLITE_OK {
10381064
return nil
10391065
}
1040-
extrv := C.sqlite3_extended_errcode(db) // returns SQLITE_NOMEM if db == nil
1041-
errStr := C.GoString(C.sqlite3_errmsg(db)) // returns "out of memory" if db == nil
1066+
extrv := rv
1067+
// Convert the extended result code to a basic result code.
1068+
rv &= ErrNoMask
10421069

10431070
// https://www.sqlite.org/c3ref/system_errno.html
10441071
// sqlite3_system_errno is only meaningful if the error code was SQLITE_CANTOPEN,
10451072
// or it was SQLITE_IOERR and the extended code was not SQLITE_IOERR_NOMEM
10461073
var systemErrno syscall.Errno
1047-
if rv == C.SQLITE_CANTOPEN || (rv == C.SQLITE_IOERR && extrv != C.SQLITE_IOERR_NOMEM) {
1074+
if db != nil && (rv == C.SQLITE_CANTOPEN ||
1075+
(rv == C.SQLITE_IOERR && extrv != C.SQLITE_IOERR_NOMEM)) {
10481076
systemErrno = syscall.Errno(C.sqlite3_system_errno(db))
10491077
}
10501078

1079+
var msg string
1080+
if db != nil {
1081+
msg = C.GoString(C.sqlite3_errmsg(db))
1082+
} else {
1083+
msg = errorString(extrv)
1084+
}
10511085
return Error{
10521086
Code: ErrNo(rv),
10531087
ExtendedCode: ErrNoExtended(extrv),
10541088
SystemErrno: systemErrno,
1055-
err: errStr,
1089+
err: msg,
10561090
}
10571091
}
10581092

@@ -1091,7 +1125,7 @@ func (c *SQLiteConn) execArgs(ctx context.Context, query string, args []driver.N
10911125
rv := C._sqlite3_prepare_v2(c.db, (*C.char)(unsafe.Pointer(stringData(query))),
10921126
C.int(len(query)), &s.s, &sz)
10931127
if rv != C.SQLITE_OK {
1094-
return nil, c.lastError()
1128+
return nil, c.lastError(int(rv))
10951129
}
10961130
query = strings.TrimSpace(query[sz:])
10971131

@@ -1143,7 +1177,7 @@ func (c *SQLiteConn) execNoArgsSync(query string) (_ driver.Result, err error) {
11431177
rv := C._sqlite3_exec_no_args(c.db, (*C.char)(unsafe.Pointer(stringData(query))),
11441178
C.int(len(query)), &rowid, &changes)
11451179
if rv != C.SQLITE_OK {
1146-
err = c.lastError()
1180+
err = c.lastError(int(rv))
11471181
}
11481182
return &SQLiteResult{id: int64(rowid), changes: int64(changes)}, err
11491183
}
@@ -1197,8 +1231,6 @@ func (c *SQLiteConn) Query(query string, args []driver.Value) (driver.Rows, erro
11971231
return c.query(context.Background(), query, list)
11981232
}
11991233

1200-
var closedRows = &SQLiteRows{s: &SQLiteStmt{closed: true}}
1201-
12021234
func (c *SQLiteConn) query(ctx context.Context, query string, args []driver.NamedValue) (driver.Rows, error) {
12031235
s := SQLiteStmt{c: c, cls: true}
12041236
p := stringData(query)
@@ -1208,14 +1240,14 @@ func (c *SQLiteConn) query(ctx context.Context, query string, args []driver.Name
12081240
if rv == C.GO_SQLITE_MULTIPLE_QUERIES {
12091241
return nil, errors.New("query contains multiple SQL statements")
12101242
}
1211-
return nil, c.lastError()
1243+
return nil, c.lastError(int(rv))
12121244
}
12131245

12141246
// The sqlite3_stmt will be nil if the SQL was valid but did not
12151247
// contain a query. For now we're supporting this for the sake of
12161248
// backwards compatibility, but that may change in the future.
12171249
if s.s == nil {
1218-
return closedRows, nil
1250+
return &SQLiteRows{s: &SQLiteStmt{closed: true}}, nil
12191251
}
12201252

12211253
na := int(paramCount)
@@ -1744,7 +1776,7 @@ func (d *SQLiteDriver) Open(dsn string) (driver.Conn, error) {
17441776
if rv != 0 {
17451777
// Save off the error _before_ closing the database.
17461778
// This is safe even if db is nil.
1747-
err := lastError(db)
1779+
err := lastError(db, int(rv))
17481780
if db != nil {
17491781
C.sqlite3_close_v2(db)
17501782
}
@@ -1753,13 +1785,18 @@ func (d *SQLiteDriver) Open(dsn string) (driver.Conn, error) {
17531785
if db == nil {
17541786
return nil, errors.New("sqlite succeeded without returning a database")
17551787
}
1788+
rv = C.sqlite3_extended_result_codes(db, 1)
1789+
if rv != SQLITE_OK {
1790+
C.sqlite3_close_v2(db)
1791+
return nil, lastError(db, int(rv))
1792+
}
17561793

17571794
exec := func(s string) error {
17581795
cs := C.CString(s)
17591796
rv := C.sqlite3_exec(db, cs, nil, nil, nil)
17601797
C.free(unsafe.Pointer(cs))
17611798
if rv != C.SQLITE_OK {
1762-
return lastError(db)
1799+
return lastError(db, int(rv))
17631800
}
17641801
return nil
17651802
}
@@ -2066,7 +2103,7 @@ func (c *SQLiteConn) Close() (err error) {
20662103
runtime.SetFinalizer(c, nil)
20672104
rv := C.sqlite3_close_v2(c.db)
20682105
if rv != C.SQLITE_OK {
2069-
err = c.lastError()
2106+
err = lastError(nil, int(rv))
20702107
}
20712108
deleteHandles(c)
20722109
c.db = nil
@@ -2092,7 +2129,7 @@ func (c *SQLiteConn) prepare(ctx context.Context, query string) (driver.Stmt, er
20922129
var s *C.sqlite3_stmt
20932130
rv := C._sqlite3_prepare_v2_internal(c.db, (*C.char)(unsafe.Pointer(p)), C.int(len(query)), &s, nil)
20942131
if rv != C.SQLITE_OK {
2095-
return nil, c.lastError()
2132+
return nil, c.lastError(int(rv))
20962133
}
20972134
ss := &SQLiteStmt{c: c, s: s}
20982135
runtime.SetFinalizer(ss, (*SQLiteStmt).Close)
@@ -2160,7 +2197,7 @@ func (c *SQLiteConn) SetFileControlInt(dbName string, op int, arg int) error {
21602197
cArg := C.int(arg)
21612198
rv := C.sqlite3_file_control(c.db, cDBName, C.int(op), unsafe.Pointer(&cArg))
21622199
if rv != C.SQLITE_OK {
2163-
return c.lastError()
2200+
return c.lastError(int(rv))
21642201
}
21652202
return nil
21662203
}
@@ -2183,7 +2220,7 @@ func (s *SQLiteStmt) Close() error {
21832220
}
21842221
rv := C.sqlite3_finalize(stmt)
21852222
if rv != C.SQLITE_OK {
2186-
return conn.lastError()
2223+
return conn.lastError(int(rv))
21872224
}
21882225
return nil
21892226
}
@@ -2215,7 +2252,7 @@ func (s *SQLiteStmt) bind(args []driver.NamedValue) error {
22152252
if s.reset {
22162253
rv := C.sqlite3_reset(s.s)
22172254
if rv != C.SQLITE_ROW && rv != C.SQLITE_OK && rv != C.SQLITE_DONE {
2218-
return s.c.lastError()
2255+
return s.c.lastError(int(rv))
22192256
}
22202257
} else {
22212258
s.reset = true
@@ -2259,7 +2296,7 @@ func (s *SQLiteStmt) bind(args []driver.NamedValue) error {
22592296
rv = C._sqlite3_bind_text(s.s, n, (*C.char)(unsafe.Pointer(&b[0])), C.int(len(b)))
22602297
}
22612298
if rv != C.SQLITE_OK {
2262-
return s.c.lastError()
2299+
return s.c.lastError(int(rv))
22632300
}
22642301
}
22652302
return nil
@@ -2327,7 +2364,7 @@ func (s *SQLiteStmt) bindIndices(args []driver.NamedValue) error {
23272364
rv = C._sqlite3_bind_text(s.s, n, (*C.char)(unsafe.Pointer(&b[0])), C.int(len(b)))
23282365
}
23292366
if rv != C.SQLITE_OK {
2330-
return s.c.lastError()
2367+
return s.c.lastError(int(rv))
23312368
}
23322369
}
23332370
}
@@ -2437,7 +2474,7 @@ func (s *SQLiteStmt) execSync(args []driver.NamedValue) (driver.Result, error) {
24372474
var rowid, changes C.longlong
24382475
rv := C._sqlite3_step_row_internal(s.s, &rowid, &changes)
24392476
if rv != C.SQLITE_ROW && rv != C.SQLITE_OK && rv != C.SQLITE_DONE {
2440-
err := s.c.lastError()
2477+
err := s.c.lastError(int(rv))
24412478
C.sqlite3_reset(s.s)
24422479
C.sqlite3_clear_bindings(s.s)
24432480
return nil, err
@@ -2473,8 +2510,8 @@ func (rc *SQLiteRows) Close() error {
24732510
}
24742511
rv := C.sqlite3_reset(s.s)
24752512
if rv != C.SQLITE_OK {
2476-
s.mu.Unlock()
2477-
return s.c.lastError()
2513+
rc.s.mu.Unlock()
2514+
return rc.s.c.lastError(int(rv))
24782515
}
24792516
s.mu.Unlock()
24802517
return nil
@@ -2573,7 +2610,7 @@ func (rc *SQLiteRows) nextSyncLocked(dest []driver.Value) error {
25732610
if rv != C.SQLITE_ROW {
25742611
rv = C.sqlite3_reset(rc.s.s)
25752612
if rv != C.SQLITE_OK {
2576-
return rc.s.c.lastError()
2613+
return rc.s.c.lastError(int(rv))
25772614
}
25782615
return nil
25792616
}

0 commit comments

Comments
 (0)