2020-04-23 08:27:35 +02:00
|
|
|
## Implementation of KvStore based on sqlite3
|
|
|
|
|
|
|
|
{.push raises: [Defect].}
|
2020-02-20 11:49:34 +01:00
|
|
|
|
|
|
|
import
|
|
|
|
os,
|
|
|
|
sqlite3_abi,
|
|
|
|
./kvstore
|
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
export kvstore
|
|
|
|
|
2020-02-20 11:49:34 +01:00
|
|
|
type
|
2020-04-23 08:27:35 +02:00
|
|
|
SqStoreRef* = ref object of RootObj
|
2020-02-20 11:49:34 +01:00
|
|
|
env: ptr sqlite3
|
|
|
|
selectStmt, insertStmt, deleteStmt: ptr sqlite3_stmt
|
|
|
|
|
|
|
|
template checkErr(op, cleanup: untyped) =
|
|
|
|
if (let v = (op); v != SQLITE_OK):
|
|
|
|
cleanup
|
2020-04-23 08:27:35 +02:00
|
|
|
return err(sqlite3_errstr(v))
|
2020-02-20 11:49:34 +01:00
|
|
|
|
|
|
|
template checkErr(op) =
|
|
|
|
checkErr(op): discard
|
|
|
|
|
|
|
|
proc bindBlob(s: ptr sqlite3_stmt, n: int, blob: openarray[byte]): cint =
|
|
|
|
sqlite3_bind_blob(s, n.cint, unsafeAddr blob[0], blob.len.cint, nil)
|
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
proc get*(db: SqStoreRef, key: openarray[byte], onData: DataProc): KvResult[bool] =
|
2020-02-20 11:49:34 +01:00
|
|
|
checkErr sqlite3_reset(db.selectStmt)
|
|
|
|
checkErr sqlite3_clear_bindings(db.selectStmt)
|
|
|
|
checkErr bindBlob(db.selectStmt, 1, key)
|
|
|
|
|
|
|
|
let v = sqlite3_step(db.selectStmt)
|
|
|
|
case v
|
|
|
|
of SQLITE_ROW:
|
|
|
|
let
|
|
|
|
p = cast[ptr UncheckedArray[byte]](sqlite3_column_blob(db.selectStmt, 0))
|
|
|
|
l = sqlite3_column_bytes(db.selectStmt, 0)
|
|
|
|
onData(toOpenArray(p, 0, l-1))
|
2020-04-23 08:27:35 +02:00
|
|
|
ok(true)
|
2020-02-20 11:49:34 +01:00
|
|
|
of SQLITE_DONE:
|
2020-04-23 08:27:35 +02:00
|
|
|
ok(false)
|
2020-02-20 11:49:34 +01:00
|
|
|
else:
|
2020-04-23 08:27:35 +02:00
|
|
|
err(sqlite3_errstr(v))
|
2020-02-20 11:49:34 +01:00
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
proc put*(db: SqStoreRef, key, value: openarray[byte]): KvResult[void] =
|
2020-02-20 11:49:34 +01:00
|
|
|
checkErr sqlite3_reset(db.insertStmt)
|
|
|
|
checkErr sqlite3_clear_bindings(db.insertStmt)
|
|
|
|
|
|
|
|
checkErr bindBlob(db.insertStmt, 1, key)
|
|
|
|
checkErr bindBlob(db.insertStmt, 2, value)
|
|
|
|
|
|
|
|
if (let v = sqlite3_step(db.insertStmt); v != SQLITE_DONE):
|
2020-04-23 08:27:35 +02:00
|
|
|
err(sqlite3_errstr(v))
|
|
|
|
else:
|
|
|
|
ok()
|
2020-02-20 11:49:34 +01:00
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
proc contains*(db: SqStoreRef, key: openarray[byte]): KvResult[bool] =
|
2020-02-20 11:49:34 +01:00
|
|
|
checkErr sqlite3_reset(db.selectStmt)
|
|
|
|
checkErr sqlite3_clear_bindings(db.selectStmt)
|
|
|
|
|
|
|
|
checkErr bindBlob(db.selectStmt, 1, key)
|
|
|
|
|
|
|
|
let v = sqlite3_step(db.selectStmt)
|
|
|
|
case v
|
2020-04-23 08:27:35 +02:00
|
|
|
of SQLITE_ROW: ok(true)
|
|
|
|
of SQLITE_DONE: ok(false)
|
|
|
|
else: err(sqlite3_errstr(v))
|
2020-02-20 11:49:34 +01:00
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
proc del*(db: SqStoreRef, key: openarray[byte]): KvResult[void] =
|
2020-02-20 11:49:34 +01:00
|
|
|
checkErr sqlite3_reset(db.deleteStmt)
|
|
|
|
checkErr sqlite3_clear_bindings(db.deleteStmt)
|
|
|
|
|
|
|
|
checkErr bindBlob(db.deleteStmt, 1, key)
|
|
|
|
|
|
|
|
if (let v = sqlite3_step(db.deleteStmt); v != SQLITE_DONE):
|
2020-04-23 08:27:35 +02:00
|
|
|
err(sqlite3_errstr(v))
|
|
|
|
else:
|
|
|
|
ok()
|
2020-02-20 11:49:34 +01:00
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
proc close*(db: SqStoreRef) =
|
2020-02-20 11:49:34 +01:00
|
|
|
discard sqlite3_finalize(db.insertStmt)
|
|
|
|
discard sqlite3_finalize(db.selectStmt)
|
|
|
|
discard sqlite3_finalize(db.deleteStmt)
|
2020-04-23 08:27:35 +02:00
|
|
|
|
2020-02-20 11:49:34 +01:00
|
|
|
discard sqlite3_close(db.env)
|
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
db[] = SqStoreRef()[]
|
2020-02-20 11:49:34 +01:00
|
|
|
|
|
|
|
proc init*(
|
2020-04-23 08:27:35 +02:00
|
|
|
T: type SqStoreRef,
|
2020-02-20 11:49:34 +01:00
|
|
|
basePath: string,
|
2020-04-23 08:27:35 +02:00
|
|
|
name: string,
|
2020-02-20 11:49:34 +01:00
|
|
|
readOnly = false,
|
2020-04-23 08:27:35 +02:00
|
|
|
inMemory = false): KvResult[T] =
|
2020-02-20 11:49:34 +01:00
|
|
|
var
|
|
|
|
env: ptr sqlite3
|
|
|
|
|
|
|
|
let
|
|
|
|
name =
|
|
|
|
if inMemory: ":memory:"
|
2020-04-23 08:27:35 +02:00
|
|
|
else: basepath / name & ".sqlite3"
|
2020-02-20 11:49:34 +01:00
|
|
|
flags =
|
|
|
|
if readOnly: SQLITE_OPEN_READONLY
|
|
|
|
else: SQLITE_OPEN_READWRITE or SQLITE_OPEN_CREATE
|
|
|
|
|
|
|
|
if not inMemory:
|
2020-04-23 08:27:35 +02:00
|
|
|
try:
|
|
|
|
createDir(basePath)
|
|
|
|
except OSError, IOError:
|
|
|
|
return err("`sqlite: cannot create database directory")
|
2020-02-20 11:49:34 +01:00
|
|
|
|
|
|
|
checkErr sqlite3_open_v2(name, addr env, flags.cint, nil)
|
|
|
|
|
|
|
|
template prepare(q: string, cleanup: untyped): ptr sqlite3_stmt =
|
|
|
|
var s: ptr sqlite3_stmt
|
|
|
|
checkErr sqlite3_prepare_v2(env, q, q.len.cint, addr s, nil):
|
|
|
|
cleanup
|
|
|
|
discard sqlite3_close(env)
|
|
|
|
s
|
|
|
|
|
|
|
|
template checkExec(q: string) =
|
|
|
|
let s = prepare(q): discard
|
|
|
|
|
|
|
|
if (let x = sqlite3_step(s); x != SQLITE_DONE):
|
|
|
|
discard sqlite3_finalize(s)
|
|
|
|
discard sqlite3_close(env)
|
2020-04-23 08:27:35 +02:00
|
|
|
return err(sqlite3_errstr(x))
|
2020-02-20 11:49:34 +01:00
|
|
|
|
|
|
|
if (let x = sqlite3_finalize(s); x != SQLITE_OK):
|
|
|
|
discard sqlite3_close(env)
|
2020-04-23 08:27:35 +02:00
|
|
|
return err(sqlite3_errstr(x))
|
2020-02-20 11:49:34 +01:00
|
|
|
|
|
|
|
# TODO: check current version and implement schema versioning
|
|
|
|
checkExec "PRAGMA user_version = 1;"
|
|
|
|
|
|
|
|
checkExec """
|
|
|
|
CREATE TABLE IF NOT EXISTS kvstore(
|
|
|
|
key BLOB PRIMARY KEY,
|
|
|
|
value BLOB
|
|
|
|
) WITHOUT ROWID;
|
|
|
|
"""
|
|
|
|
|
|
|
|
let
|
|
|
|
selectStmt = prepare "SELECT value FROM kvstore WHERE key = ?;":
|
|
|
|
discard
|
|
|
|
insertStmt = prepare "INSERT OR REPLACE INTO kvstore(key, value) VALUES (?, ?);":
|
|
|
|
discard sqlite3_finalize(selectStmt)
|
|
|
|
deleteStmt = prepare "DELETE FROM kvstore WHERE key = ?;":
|
|
|
|
discard sqlite3_finalize(selectStmt)
|
|
|
|
discard sqlite3_finalize(insertStmt)
|
|
|
|
|
2020-04-23 08:27:35 +02:00
|
|
|
ok(SqStoreRef(
|
2020-02-20 11:49:34 +01:00
|
|
|
env: env,
|
|
|
|
selectStmt: selectStmt,
|
|
|
|
insertStmt: insertStmt,
|
|
|
|
deleteStmt: deleteStmt
|
2020-04-23 08:27:35 +02:00
|
|
|
))
|