Skip to content

Commit 1e51e93

Browse files
committed
intern error strings
This changes the error message cache to intern the error strings/messages. The reasoning for this change is that sqlite3 has a finite number of constant error strings, but potential misuse of the cache could lead to it growing unbounded with a large number of essentially duplicate error messages.
1 parent 23c3614 commit 1e51e93

File tree

2 files changed

+46
-4
lines changed

2 files changed

+46
-4
lines changed

error.go

+22-4
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ package sqlite3
1515
import "C"
1616
import (
1717
"sync"
18+
"sync/atomic"
1819
"syscall"
1920
)
2021

@@ -152,7 +153,11 @@ const (
152153
ErrWarningAutoIndex = ErrNoExtended(ErrWarning | 1<<8)
153154
)
154155

155-
var errStrCache sync.Map // int => string
156+
var errStrCache struct {
157+
sync.Map // int => string
158+
size atomic.Uint64 // size of cache
159+
intern sync.Map // interned error messages: string => string
160+
}
156161

157162
// errorString returns the result of sqlite3_errstr for result code rv,
158163
// which may be cached.
@@ -161,11 +166,24 @@ func errorString(rv int) string {
161166
return v.(string)
162167
}
163168
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" {
169+
if s == "unknown error" {
170+
// Prevent the cache from growing unbounded by ignoring invalid
171+
// error codes.
172+
return "unknown error"
173+
}
174+
if v, loaded := errStrCache.intern.LoadOrStore(s, s); loaded {
175+
s = v.(string)
176+
}
177+
// NB: This is limit is not precise and we may exceed
178+
// it if this function is called concurrently.
179+
//
180+
// TODO: make the limit less than a power of 2 to reduce
181+
// the chance that we resize the map if it is exceeded.
182+
if errStrCache.size.Load() < 1024 {
167183
if v, loaded := errStrCache.LoadOrStore(rv, s); loaded {
168184
s = v.(string)
185+
} else {
186+
errStrCache.size.Add(1)
169187
}
170188
}
171189
return s

error_test.go

+24
Original file line numberDiff line numberDiff line change
@@ -274,3 +274,27 @@ func TestError_SystemErrno(t *testing.T) {
274274
t.Errorf("expected SystemErrno to be a not exists error, but got %v", serr.SystemErrno)
275275
}
276276
}
277+
278+
func TestErrorStringCacheSize(t *testing.T) {
279+
for i := 0; i < 50_000; i++ {
280+
_ = errorString(i)
281+
}
282+
n := 0
283+
o := 0
284+
errStrCache.Range(func(_, v any) bool {
285+
n++
286+
return true
287+
})
288+
errStrCache.intern.Range(func(_, v any) bool {
289+
o++
290+
return true
291+
})
292+
if n > 1024 {
293+
t.Fatalf("len(errStrCache) should be capped at %d got: %d", 1024, n)
294+
}
295+
if o > 128 {
296+
// If sqlite3 adds a lot of new error messages this value
297+
// will need to be increased.
298+
t.Errorf("expected errStrMsgCache to be below %d got: %d", 128, o)
299+
}
300+
}

0 commit comments

Comments
 (0)