From 1a6dca8ec8837e6f0a25b2ecc500b4c256ba0fb4 Mon Sep 17 00:00:00 2001 From: Patryk Osmaczko Date: Thu, 18 May 2023 23:42:22 +0200 Subject: [PATCH] feat: enable multiple db connections with sqlcipher part of: status-im/status-desktop#10558 --- sqlite/sqlite.go | 79 ++++++++++++++++++++++++++++-------------------- 1 file changed, 47 insertions(+), 32 deletions(-) diff --git a/sqlite/sqlite.go b/sqlite/sqlite.go index 0a93d6a17f7..1a6d4c1c579 100644 --- a/sqlite/sqlite.go +++ b/sqlite/sqlite.go @@ -2,11 +2,13 @@ package sqlite import ( "database/sql" + "database/sql/driver" "errors" "fmt" "os" + "runtime" - _ "github.com/mutecomm/go-sqlcipher" // We require go sqlcipher that overrides default implementation + sqlcipher "github.com/mutecomm/go-sqlcipher" // We require go sqlcipher that overrides default implementation "github.com/status-im/status-go/protocol/sqlite" ) @@ -76,41 +78,54 @@ func EncryptDB(unencryptedPath string, encryptedPath string, key string, kdfIter } func openDB(path string, key string, kdfIterationsNumber int) (*sql.DB, error) { - db, err := sql.Open("sqlite3", path) + driverName := fmt.Sprintf("sqlcipher_with_extensions-%d", len(sql.Drivers())) + sql.Register(driverName, &sqlcipher.SQLiteDriver{ + ConnectHook: func(conn *sqlcipher.SQLiteConn) error { + if _, err := conn.Exec("PRAGMA foreign_keys=ON", []driver.Value{}); err != nil { + return errors.New("failed to set `foreign_keys` pragma") + } + + keyString := fmt.Sprintf("PRAGMA key = '%s'", key) + if _, err := conn.Exec(keyString, []driver.Value{}); err != nil { + return errors.New("failed to set `key` pragma") + } + + if kdfIterationsNumber <= 0 { + kdfIterationsNumber = sqlite.ReducedKDFIterationsNumber + } + + if _, err := conn.Exec(fmt.Sprintf("PRAGMA kdf_iter = '%d'", kdfIterationsNumber), []driver.Value{}); err != nil { + return errors.New("failed to set `kdf_iter` pragma") + } + + // readers do not block writers and faster i/o operations + if _, err := conn.Exec("PRAGMA journal_mode=WAL", []driver.Value{}); err != nil && path != InMemoryPath { + return errors.New("failed to set `journal_mode` pragma") + } + + // workaround to mitigate the issue of "database is locked" errors during concurrent write operations + if _, err := conn.Exec("PRAGMA busy_timeout=60000", []driver.Value{}); err != nil { + return errors.New("failed to set `busy_timeout` pragma") + } + + return nil + }, + }) + + db, err := sql.Open(driverName, path) if err != nil { return nil, err } - // Disable concurrent access as not supported by the driver - db.SetMaxOpenConns(1) - - if _, err = db.Exec("PRAGMA foreign_keys=ON"); err != nil { - return nil, err - } - keyString := fmt.Sprintf("PRAGMA key = '%s'", key) - if _, err = db.Exec(keyString); err != nil { - return nil, errors.New("failed to set key pragma") - } - - if kdfIterationsNumber <= 0 { - kdfIterationsNumber = sqlite.ReducedKDFIterationsNumber - } - - if _, err = db.Exec(fmt.Sprintf("PRAGMA kdf_iter = '%d'", kdfIterationsNumber)); err != nil { - return nil, err - } - - // readers do not block writers and faster i/o operations - // https://www.sqlite.org/draft/wal.html - // must be set after db is encrypted - var mode string - err = db.QueryRow("PRAGMA journal_mode=WAL").Scan(&mode) - if err != nil { - return nil, err - } - if mode != WALMode && path != InMemoryPath { - return nil, fmt.Errorf("unable to set journal_mode to WAL. actual mode %s", mode) - } + nproc := func() int { + maxProcs := runtime.GOMAXPROCS(0) + numCPU := runtime.NumCPU() + if maxProcs < numCPU { + return maxProcs + } + return numCPU + }() + db.SetMaxOpenConns(nproc) return db, nil }