mirror of
https://github.com/status-im/status-go.git
synced 2025-01-09 14:16:21 +00:00
e8c4b7647f
* chore(upgradeSQLCipher): Upgrading SQLCipher to version 5.4.5 Changes: ### github.com/mutecomm/go-sqlcipher 1. The improved crypto argorighms from go-sqlcipher v3 are merged in v4 Tags: v4.4.2-status.1 - merge `burn_stack` improvement v4.4.2-status.2 - merge `SHA1` improvement v4.4.2-status.4- merge 'AES' improvement 2. Fixed `go-sqlcipher` to support v3 database in compatibility mode (`sqlcipher` already supports this) (Tag: v4.4.2-status.3) 3. Upgrade `sqlcipher` to v5.4.5 (Tag: v4.5.4-status.1) ### github.com/status-im/migrate/v4 1. Upgrade `go-sqlcipher` version in `github.com/status-im/migrate/v4` ### status-go 1. Upgrade `go-sqlcipher` and `migrate` modules in status-go 2. Configure the DB connections to open the DB in v3 compatibility mode * chore(upgradeSQLCipher): Use sqlcipher v3 configuration to encrypt a plain text database * chore(upgradeSQLCipher): Scanning NULL BLOB value should return nil Fixing failing tests: TestSyncDeviceSuite/TestPairingSyncDeviceClientAsReceiver; TestSyncDeviceSuite/TestPairingSyncDeviceClientAsSender Considering the following configuration: 1. Table with BLOB column has 1 NULL value 2. Query the value 3. Rows.Scan(&dest sql.NullString) Expected: dest.Valid == false; dest.String == nil Actual: dest.Valid == true; dest.String == "" * chore: Bump go-sqlcipher version to include NULL BLOB fix
249 lines
6.6 KiB
Go
249 lines
6.6 KiB
Go
package sqlite
|
|
|
|
import (
|
|
"database/sql"
|
|
"database/sql/driver"
|
|
"errors"
|
|
"fmt"
|
|
"net/url"
|
|
"os"
|
|
"runtime"
|
|
"strings"
|
|
|
|
sqlcipher "github.com/mutecomm/go-sqlcipher/v4" // We require go sqlcipher that overrides default implementation
|
|
|
|
"github.com/status-im/status-go/protocol/sqlite"
|
|
)
|
|
|
|
const (
|
|
// The reduced number of kdf iterations (for performance reasons) which is
|
|
// used as the default value
|
|
// https://github.com/status-im/status-go/pull/1343
|
|
// https://notes.status.im/i8Y_l7ccTiOYq09HVgoFwA
|
|
ReducedKDFIterationsNumber = 3200
|
|
|
|
// WALMode for sqlite.
|
|
WALMode = "wal"
|
|
InMemoryPath = ":memory:"
|
|
)
|
|
|
|
// DecryptDB completely removes the encryption from the db
|
|
func DecryptDB(oldPath string, newPath string, key string, kdfIterationsNumber int) error {
|
|
|
|
db, err := openDB(oldPath, key, kdfIterationsNumber)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = db.Exec(`ATTACH DATABASE '` + newPath + `' AS plaintext KEY ''`)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = db.Exec(`SELECT sqlcipher_export('plaintext')`)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = db.Exec(`DETACH DATABASE plaintext`)
|
|
return err
|
|
}
|
|
|
|
// EncryptDB takes a plaintext database and adds encryption
|
|
func EncryptDB(unencryptedPath string, encryptedPath string, key string, kdfIterationsNumber int) error {
|
|
_ = os.Remove(encryptedPath)
|
|
|
|
db, err := OpenUnecryptedDB(unencryptedPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = db.Exec(`ATTACH DATABASE '` + encryptedPath + `' AS encrypted KEY '` + key + `'`)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if kdfIterationsNumber <= 0 {
|
|
kdfIterationsNumber = sqlite.ReducedKDFIterationsNumber
|
|
}
|
|
|
|
_, err = db.Exec(fmt.Sprintf("PRAGMA encrypted.kdf_iter = '%d'", kdfIterationsNumber))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err := db.Exec("PRAGMA encrypted.cipher_page_size = 1024"); err != nil {
|
|
fmt.Println("failed to set cipher_page_size pragma")
|
|
return err
|
|
}
|
|
if _, err := db.Exec("PRAGMA encrypted.cipher_hmac_algorithm = HMAC_SHA1"); err != nil {
|
|
fmt.Println("failed to set cipher_hmac_algorithm pragma")
|
|
return err
|
|
}
|
|
|
|
if _, err := db.Exec("PRAGMA encrypted.cipher_kdf_algorithm = PBKDF2_HMAC_SHA1"); err != nil {
|
|
fmt.Println("failed to set cipher_kdf_algorithm pragma")
|
|
return err
|
|
}
|
|
|
|
_, err = db.Exec(`SELECT sqlcipher_export('encrypted')`)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = db.Exec(`DETACH DATABASE encrypted`)
|
|
return err
|
|
}
|
|
|
|
func buildSqlcipherDSN(path string) (string, error) {
|
|
if path == InMemoryPath {
|
|
return InMemoryPath, nil
|
|
}
|
|
|
|
// Adding sqlcipher query parameter to the DSN
|
|
queryOperator := "?"
|
|
|
|
if queryStart := strings.IndexRune(path, '?'); queryStart != -1 {
|
|
params, err := url.ParseQuery(path[queryStart+1:])
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if len(params) > 0 {
|
|
queryOperator = "&"
|
|
}
|
|
}
|
|
|
|
// We need to set txlock=immediate to avoid "database is locked" errors during concurrent write operations
|
|
// This could happen when a read transaction is promoted to write transaction
|
|
// https://www.sqlite.org/lang_transaction.html
|
|
return path + queryOperator + "_txlock=immediate", nil
|
|
}
|
|
|
|
func openDB(path string, key string, kdfIterationsNumber int) (*sql.DB, error) {
|
|
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")
|
|
}
|
|
|
|
if _, err := conn.Exec(fmt.Sprintf("PRAGMA key = '%s'", key), []driver.Value{}); err != nil {
|
|
return errors.New("failed to set `key` pragma")
|
|
}
|
|
|
|
if kdfIterationsNumber <= 0 {
|
|
kdfIterationsNumber = sqlite.ReducedKDFIterationsNumber
|
|
}
|
|
|
|
if _, err := conn.Exec("PRAGMA cipher_page_size = 1024", nil); err != nil {
|
|
fmt.Println("failed to set cipher_page_size pragma")
|
|
return err
|
|
}
|
|
if _, err := conn.Exec("PRAGMA cipher_hmac_algorithm = HMAC_SHA1", nil); err != nil {
|
|
fmt.Println("failed to set cipher_hmac_algorithm pragma")
|
|
return err
|
|
}
|
|
|
|
if _, err := conn.Exec("PRAGMA cipher_kdf_algorithm = PBKDF2_HMAC_SHA1", nil); err != nil {
|
|
fmt.Println("failed to set cipher_kdf_algorithm pragma")
|
|
return err
|
|
}
|
|
|
|
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
|
|
},
|
|
})
|
|
|
|
dsn, err := buildSqlcipherDSN(path)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
db, err := sql.Open(driverName, dsn)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if path == InMemoryPath {
|
|
db.SetMaxOpenConns(1)
|
|
} else {
|
|
nproc := func() int {
|
|
maxProcs := runtime.GOMAXPROCS(0)
|
|
numCPU := runtime.NumCPU()
|
|
if maxProcs < numCPU {
|
|
return maxProcs
|
|
}
|
|
return numCPU
|
|
}()
|
|
db.SetMaxOpenConns(nproc)
|
|
db.SetMaxIdleConns(nproc)
|
|
}
|
|
|
|
return db, nil
|
|
}
|
|
|
|
// OpenDB opens not-encrypted database.
|
|
func OpenDB(path string, key string, kdfIterationsNumber int) (*sql.DB, error) {
|
|
return openDB(path, key, kdfIterationsNumber)
|
|
}
|
|
|
|
// OpenUnecryptedDB opens database with setting PRAGMA key.
|
|
func OpenUnecryptedDB(path string) (*sql.DB, error) {
|
|
db, err := sql.Open("sqlite3", 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
|
|
}
|
|
// 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 {
|
|
return nil, fmt.Errorf("unable to set journal_mode to WAL. actual mode %s", mode)
|
|
}
|
|
|
|
return db, nil
|
|
}
|
|
|
|
func ChangeEncryptionKey(path string, key string, kdfIterationsNumber int, newKey string) error {
|
|
if kdfIterationsNumber <= 0 {
|
|
kdfIterationsNumber = sqlite.ReducedKDFIterationsNumber
|
|
}
|
|
|
|
db, err := openDB(path, key, kdfIterationsNumber)
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
resetKeyString := fmt.Sprintf("PRAGMA rekey = '%s'", newKey)
|
|
if _, err = db.Exec(resetKeyString); err != nil {
|
|
return errors.New("failed to set rekey pragma")
|
|
}
|
|
|
|
return nil
|
|
}
|