2020-04-27 15:16:11 +02:00
|
|
|
## Implementation of KvStore based on sqlite3
|
|
|
|
|
2023-02-21 18:54:30 +01:00
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
{.pragma: callback, gcsafe, raises: [Defect].}
|
|
|
|
else:
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
{.pragma: callback, gcsafe, raises: [].}
|
2020-04-27 15:16:11 +02:00
|
|
|
|
|
|
|
import
|
2022-11-28 21:15:53 +01:00
|
|
|
std/[os, options, strformat, typetraits],
|
2020-05-09 14:34:06 +02:00
|
|
|
sqlite3_abi,
|
2020-04-27 15:16:11 +02:00
|
|
|
./kvstore
|
|
|
|
|
2022-11-28 21:15:53 +01:00
|
|
|
export kvstore, typetraits
|
2020-04-27 15:16:11 +02:00
|
|
|
|
|
|
|
type
|
2020-10-13 21:44:42 +03:00
|
|
|
RawStmtPtr = ptr sqlite3_stmt
|
|
|
|
|
|
|
|
AutoDisposed[T: ptr|ref] = object
|
|
|
|
val: T
|
|
|
|
|
|
|
|
# TODO: These should become AutoDisposed
|
|
|
|
# This is currently considered risky due to the destructor
|
|
|
|
# problem found in FastStreams (triggered when objects in
|
|
|
|
# the GC heap have destructors)
|
|
|
|
Sqlite* = ptr sqlite3
|
|
|
|
SqliteStmt*[Params; Result] = distinct RawStmtPtr
|
|
|
|
NoParams* = tuple # this is the empty tuple
|
2023-02-21 18:54:30 +01:00
|
|
|
ResultHandler*[T] = proc(val: T) {.callback.}
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
SqStoreRef* = ref object
|
|
|
|
# Handle for a single database - from here, keyspaces and statements
|
|
|
|
# can be created
|
2020-10-13 21:44:42 +03:00
|
|
|
env: Sqlite
|
2020-11-27 20:49:08 +02:00
|
|
|
managedStmts: seq[RawStmtPtr]
|
2021-11-16 13:45:28 +02:00
|
|
|
readOnly*: bool
|
2020-10-13 21:44:42 +03:00
|
|
|
|
2020-12-18 14:25:46 +01:00
|
|
|
SqStoreCheckpointKind* {.pure.} = enum
|
|
|
|
passive, full, restart, truncate
|
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
SqKeyspace* = object of RootObj
|
|
|
|
# A Keyspace is a single key-value table - it is generally efficient to
|
|
|
|
# create separate keyspaces for each type of data stored
|
2022-11-28 21:15:53 +01:00
|
|
|
open: bool
|
2023-01-03 16:38:27 +01:00
|
|
|
env: Sqlite
|
|
|
|
getStmt, putStmt, delStmt, clearStmt, containsStmt,
|
2021-05-17 15:55:57 +02:00
|
|
|
findStmt0, findStmt1, findStmt2: RawStmtPtr
|
|
|
|
|
|
|
|
SqKeyspaceRef* = ref SqKeyspace
|
|
|
|
|
2022-08-19 01:24:19 +03:00
|
|
|
CustomFunction* =
|
2022-06-02 14:14:15 +02:00
|
|
|
proc (
|
|
|
|
a: openArray[byte],
|
|
|
|
b: openArray[byte]
|
2023-02-21 18:54:30 +01:00
|
|
|
): Result[seq[byte], cstring] {.noSideEffect, cdecl, callback.}
|
2022-06-02 14:14:15 +02:00
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
template dispose(db: Sqlite) =
|
|
|
|
discard sqlite3_close(db)
|
|
|
|
|
|
|
|
template dispose(db: RawStmtPtr) =
|
|
|
|
discard sqlite3_finalize(db)
|
|
|
|
|
2020-11-27 20:49:08 +02:00
|
|
|
template dispose*(db: SqliteStmt) =
|
|
|
|
discard sqlite3_finalize(RawStmtPtr db)
|
|
|
|
|
2022-08-19 01:24:19 +03:00
|
|
|
func isInsideTransaction*(db: SqStoreRef): bool =
|
|
|
|
sqlite3_get_autocommit(db.env) == 0
|
2022-11-16 10:44:00 -06:00
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
proc release[T](x: var AutoDisposed[T]): T =
|
|
|
|
result = x.val
|
|
|
|
x.val = nil
|
|
|
|
|
2020-10-16 19:50:01 +03:00
|
|
|
proc disposeIfUnreleased[T](x: var AutoDisposed[T]) =
|
2020-10-13 21:44:42 +03:00
|
|
|
mixin dispose
|
|
|
|
if x.val != nil:
|
|
|
|
dispose(x.release)
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2020-04-27 15:16:11 +02:00
|
|
|
template checkErr(op, cleanup: untyped) =
|
|
|
|
if (let v = (op); v != SQLITE_OK):
|
|
|
|
cleanup
|
|
|
|
return err($sqlite3_errstr(v))
|
|
|
|
|
|
|
|
template checkErr(op) =
|
|
|
|
checkErr(op): discard
|
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
proc prepareStmt*(db: SqStoreRef,
|
|
|
|
stmt: string,
|
|
|
|
Params: type,
|
2020-11-27 20:49:08 +02:00
|
|
|
Res: type,
|
|
|
|
managed = true): KvResult[SqliteStmt[Params, Res]] =
|
2020-10-13 21:44:42 +03:00
|
|
|
var s: RawStmtPtr
|
|
|
|
checkErr sqlite3_prepare_v2(db.env, stmt, stmt.len.cint, addr s, nil)
|
2020-11-27 20:49:08 +02:00
|
|
|
if managed: db.managedStmts.add s
|
2020-10-13 21:44:42 +03:00
|
|
|
ok SqliteStmt[Params, Res](s)
|
|
|
|
|
|
|
|
proc bindParam(s: RawStmtPtr, n: int, val: auto): cint =
|
2021-05-17 15:55:57 +02:00
|
|
|
when val is openArray[byte]|seq[byte]:
|
2020-10-13 21:44:42 +03:00
|
|
|
if val.len > 0:
|
|
|
|
sqlite3_bind_blob(s, n.cint, unsafeAddr val[0], val.len.cint, nil)
|
|
|
|
else:
|
|
|
|
sqlite3_bind_blob(s, n.cint, nil, 0.cint, nil)
|
2020-11-27 22:27:14 +01:00
|
|
|
elif val is array:
|
|
|
|
when val.items.typeof is byte:
|
|
|
|
# Prior to Nim 1.4 and view types array[N, byte] in tuples
|
2021-05-17 15:55:57 +02:00
|
|
|
# don't match with openArray[byte]
|
2020-11-27 22:27:14 +01:00
|
|
|
if val.len > 0:
|
|
|
|
sqlite3_bind_blob(s, n.cint, unsafeAddr val[0], val.len.cint, nil)
|
|
|
|
else:
|
|
|
|
sqlite3_bind_blob(s, n.cint, nil, 0.cint, nil)
|
|
|
|
else:
|
|
|
|
{.fatal: "Please add support for the '" & $typeof(val) & "' type".}
|
2021-05-26 11:34:46 +02:00
|
|
|
elif val is SomeInteger:
|
|
|
|
sqlite3_bind_int64(s, n.cint, val.clong)
|
|
|
|
elif val is Option:
|
|
|
|
if val.isNone():
|
|
|
|
sqlite3_bind_null(s, n.cint)
|
|
|
|
else:
|
|
|
|
bindParam(s, n, val.get())
|
2020-10-13 21:44:42 +03:00
|
|
|
else:
|
2020-11-27 22:27:14 +01:00
|
|
|
{.fatal: "Please add support for the '" & $typeof(val) & "' type".}
|
2020-10-13 21:44:42 +03:00
|
|
|
|
|
|
|
template bindParams(s: RawStmtPtr, params: auto) =
|
|
|
|
when params is tuple:
|
2022-11-28 21:15:53 +01:00
|
|
|
when params.type.arity > 0:
|
|
|
|
var i = 1
|
|
|
|
for param in fields(params):
|
|
|
|
checkErr bindParam(s, i, param)
|
|
|
|
inc i
|
2020-10-13 21:44:42 +03:00
|
|
|
else:
|
|
|
|
checkErr bindParam(s, 1, params)
|
|
|
|
|
|
|
|
proc exec*[P](s: SqliteStmt[P, void], params: P): KvResult[void] =
|
|
|
|
let s = RawStmtPtr s
|
|
|
|
bindParams(s, params)
|
|
|
|
|
|
|
|
let res =
|
|
|
|
if (let v = sqlite3_step(s); v != SQLITE_DONE):
|
|
|
|
err($sqlite3_errstr(v))
|
|
|
|
else:
|
|
|
|
ok()
|
|
|
|
|
2022-11-16 10:44:00 -06:00
|
|
|
# release implicit transaction
|
2020-10-13 21:44:42 +03:00
|
|
|
discard sqlite3_reset(s) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(s) # no errors possible
|
|
|
|
|
|
|
|
res
|
|
|
|
|
2021-05-26 11:34:46 +02:00
|
|
|
template readSimpleResult(s: RawStmtPtr, column: cint, T: type): auto =
|
|
|
|
when T is int64:
|
2020-10-13 21:44:42 +03:00
|
|
|
sqlite3_column_int64(s, column)
|
2021-05-26 11:34:46 +02:00
|
|
|
elif T is SomeInteger:
|
|
|
|
# sqlite integers are "up to" 8 bytes in size, so rather than silently
|
|
|
|
# truncate them, we support only 64-bit integers when reading and let the
|
|
|
|
# calling code deal with it - careful though, anything that is not an
|
|
|
|
# integer (ie TEXT) is returned as 0
|
|
|
|
{.fatal: "Use int64 for reading integers".}
|
2021-05-17 15:55:57 +02:00
|
|
|
elif T is openArray[byte]:
|
2020-10-13 21:44:42 +03:00
|
|
|
let
|
|
|
|
p = cast[ptr UncheckedArray[byte]](sqlite3_column_blob(s, column))
|
|
|
|
l = sqlite3_column_bytes(s, column)
|
|
|
|
toOpenArray(p, 0, l-1)
|
|
|
|
elif T is seq[byte]:
|
|
|
|
var res: seq[byte]
|
|
|
|
let len = sqlite3_column_bytes(s, column)
|
|
|
|
if len > 0:
|
|
|
|
res.setLen(len)
|
|
|
|
copyMem(addr res[0], sqlite3_column_blob(s, column), len)
|
|
|
|
res
|
2020-11-27 22:27:14 +01:00
|
|
|
elif T is array:
|
|
|
|
# array[N, byte]. "genericParams(T)[1]" requires 1.4 to handle nnkTypeOfExpr
|
2020-11-28 17:03:59 +01:00
|
|
|
when typeof(default(T)[0]) is byte:
|
2020-11-27 22:27:14 +01:00
|
|
|
var res: T
|
|
|
|
let colLen = sqlite3_column_bytes(s, column)
|
|
|
|
|
|
|
|
# truncate if the type is too small
|
|
|
|
# TODO: warning/error? We assume that users always properly dimension buffers
|
|
|
|
let copyLen = min(colLen, res.len)
|
|
|
|
if copyLen > 0:
|
|
|
|
copyMem(addr res[0], sqlite3_column_blob(s, column), copyLen)
|
|
|
|
res
|
|
|
|
else:
|
|
|
|
{.fatal: "Please add support for the '" & $(T) & "' type".}
|
2020-10-13 21:44:42 +03:00
|
|
|
else:
|
|
|
|
{.fatal: "Please add support for the '" & $(T) & "' type".}
|
|
|
|
|
2021-05-26 11:34:46 +02:00
|
|
|
template readResult(s: RawStmtPtr, column: cint, T: type): auto =
|
|
|
|
when T is Option:
|
|
|
|
if sqlite3_column_type(s, column) == SQLITE_NULL:
|
|
|
|
none(typeof(default(T).get()))
|
|
|
|
else:
|
|
|
|
some(readSimpleResult(s, column, typeof(default(T).get())))
|
|
|
|
else:
|
|
|
|
readSimpleResult(s, column, T)
|
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
template readResult(s: RawStmtPtr, T: type): auto =
|
|
|
|
when T is tuple:
|
|
|
|
var res: T
|
|
|
|
var i = cint 0
|
|
|
|
for field in fields(res):
|
|
|
|
field = readResult(s, i, typeof(field))
|
|
|
|
inc i
|
|
|
|
res
|
|
|
|
else:
|
|
|
|
readResult(s, 0.cint, T)
|
|
|
|
|
|
|
|
proc exec*[Params, Res](s: SqliteStmt[Params, Res],
|
|
|
|
params: Params,
|
|
|
|
onData: ResultHandler[Res]): KvResult[bool] =
|
|
|
|
let s = RawStmtPtr s
|
|
|
|
bindParams(s, params)
|
|
|
|
|
|
|
|
try:
|
|
|
|
var gotResults = false
|
|
|
|
while true:
|
|
|
|
let v = sqlite3_step(s)
|
|
|
|
case v
|
|
|
|
of SQLITE_ROW:
|
2022-11-28 21:15:53 +01:00
|
|
|
if onData != nil:
|
|
|
|
onData(readResult(s, Res))
|
2020-10-13 21:44:42 +03:00
|
|
|
gotResults = true
|
|
|
|
of SQLITE_DONE:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
return err($sqlite3_errstr(v))
|
|
|
|
return ok gotResults
|
|
|
|
finally:
|
|
|
|
# release implicit transaction
|
|
|
|
discard sqlite3_reset(s) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(s) # no errors possible
|
|
|
|
|
2022-01-28 14:23:41 +01:00
|
|
|
iterator exec*[Params, Res](s: SqliteStmt[Params, Res],
|
|
|
|
params: Params, item: var Res): KvResult[void] =
|
|
|
|
let s = RawStmtPtr s
|
|
|
|
|
|
|
|
# we use a mutable `res` variable here to avoid the code bloat that multiple
|
|
|
|
# `yield` statements cause when inlining the loop body
|
|
|
|
var res = KvResult[void].ok()
|
|
|
|
when params is tuple:
|
|
|
|
var i = 1
|
|
|
|
for param in fields(params):
|
|
|
|
if (let v = bindParam(s, i, param); v != SQLITE_OK):
|
|
|
|
res = KvResult[void].err($sqlite3_errstr(v))
|
|
|
|
break
|
|
|
|
|
|
|
|
inc i
|
|
|
|
else:
|
|
|
|
if (let v = bindParam(s, 1, params); v != SQLITE_OK):
|
|
|
|
res = KvResult[void].err($sqlite3_errstr(v))
|
|
|
|
|
|
|
|
defer:
|
|
|
|
# release implicit transaction
|
|
|
|
discard sqlite3_reset(s) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(s) # no errors possible
|
|
|
|
|
|
|
|
while res.isOk():
|
|
|
|
let v = sqlite3_step(s)
|
|
|
|
case v
|
|
|
|
of SQLITE_ROW:
|
|
|
|
item = readResult(s, Res)
|
|
|
|
yield KvResult[void].ok()
|
|
|
|
of SQLITE_DONE:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
res = KvResult[void].err($sqlite3_errstr(v))
|
|
|
|
|
|
|
|
if not res.isOk():
|
|
|
|
yield res
|
|
|
|
|
|
|
|
iterator exec*[Res](s: SqliteStmt[NoParams, Res], item: var Res): KvResult[void] =
|
|
|
|
for r in exec(s, (), item):
|
|
|
|
yield r
|
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
template exec*(s: SqliteStmt[NoParams, void]): KvResult[void] =
|
|
|
|
exec(s, ())
|
|
|
|
|
|
|
|
template exec*[Res](s: SqliteStmt[NoParams, Res],
|
|
|
|
onData: ResultHandler[Res]): KvResult[bool] =
|
|
|
|
exec(s, (), onData)
|
|
|
|
|
2020-11-27 20:49:08 +02:00
|
|
|
proc exec*[Params: tuple](db: SqStoreRef,
|
|
|
|
stmt: string,
|
|
|
|
params: Params): KvResult[void] =
|
|
|
|
let stmt = ? db.prepareStmt(stmt, Params, void, managed = false)
|
|
|
|
result = exec(stmt, params)
|
2020-10-13 21:44:42 +03:00
|
|
|
let finalizeStatus = sqlite3_finalize(RawStmtPtr stmt)
|
|
|
|
if finalizeStatus != SQLITE_OK and result.isOk:
|
|
|
|
return err($sqlite3_errstr(finalizeStatus))
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2021-05-27 11:31:34 +02:00
|
|
|
proc exec*[Params: tuple, Res](db: SqStoreRef,
|
|
|
|
stmt: string,
|
|
|
|
params: Params,
|
|
|
|
onData: ResultHandler[Res]): KvResult[bool] =
|
|
|
|
let stmt = ? db.prepareStmt(stmt, Params, Res, managed = false)
|
|
|
|
result = exec(stmt, params, onData)
|
|
|
|
let finalizeStatus = sqlite3_finalize(RawStmtPtr stmt)
|
|
|
|
if finalizeStatus != SQLITE_OK and result.isOk:
|
|
|
|
return err($sqlite3_errstr(finalizeStatus))
|
|
|
|
|
2020-11-27 20:49:08 +02:00
|
|
|
template exec*(db: SqStoreRef, stmt: string): KvResult[void] =
|
|
|
|
exec(db, stmt, ())
|
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
proc get*(db: SqKeyspaceRef,
|
|
|
|
key: openArray[byte],
|
|
|
|
onData: DataProc): KvResult[bool] =
|
2022-11-28 21:15:53 +01:00
|
|
|
if not db.open: return err("sqlite: database closed")
|
2021-05-17 15:55:57 +02:00
|
|
|
let getStmt = db.getStmt
|
2022-11-28 21:15:53 +01:00
|
|
|
if getStmt == nil: return ok(false) # no such table
|
2020-10-13 21:44:42 +03:00
|
|
|
checkErr bindParam(getStmt, 1, key)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-09-21 08:21:47 +02:00
|
|
|
let
|
2020-10-11 17:28:31 +03:00
|
|
|
v = sqlite3_step(getStmt)
|
2020-09-21 08:21:47 +02:00
|
|
|
res = case v
|
|
|
|
of SQLITE_ROW:
|
|
|
|
let
|
2020-10-11 17:28:31 +03:00
|
|
|
p = cast[ptr UncheckedArray[byte]](sqlite3_column_blob(getStmt, 0))
|
|
|
|
l = sqlite3_column_bytes(getStmt, 0)
|
2020-09-21 08:21:47 +02:00
|
|
|
onData(toOpenArray(p, 0, l-1))
|
|
|
|
ok(true)
|
|
|
|
of SQLITE_DONE:
|
|
|
|
ok(false)
|
|
|
|
else:
|
|
|
|
err($sqlite3_errstr(v))
|
|
|
|
|
|
|
|
# release implicit transaction
|
2020-10-11 17:28:31 +03:00
|
|
|
discard sqlite3_reset(getStmt) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(getStmt) # no errors possible
|
2020-09-21 08:21:47 +02:00
|
|
|
|
|
|
|
res
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
func nextPrefix(prefix: openArray[byte], next: var seq[byte]): bool =
|
|
|
|
# Return a seq that is greater than all strings starting with `prefix` when
|
|
|
|
# doing a lexicographical compare - we're looking for the string that
|
|
|
|
# increments the last byte by 1, removing any bytes from the back that
|
|
|
|
# cannot be incremented (0xff)
|
|
|
|
|
|
|
|
for i in 0..<prefix.len():
|
|
|
|
if prefix[^(i+1)] == high(byte):
|
|
|
|
if i == 0:
|
|
|
|
return false
|
|
|
|
else:
|
|
|
|
next = prefix[0..<i]
|
|
|
|
next[^1] += 1'u8
|
|
|
|
return true
|
|
|
|
|
|
|
|
false # Empty
|
|
|
|
|
|
|
|
proc find*(
|
|
|
|
db: SqKeyspaceRef,
|
|
|
|
prefix: openArray[byte],
|
|
|
|
onFind: KeyValueProc): KvResult[int] =
|
2022-11-28 21:15:53 +01:00
|
|
|
if not db.open: return err("sqlite: database closed")
|
2021-05-17 15:55:57 +02:00
|
|
|
var next: seq[byte] # extended lifetime of bound param
|
|
|
|
let findStmt =
|
|
|
|
if prefix.len == 0:
|
2022-11-28 21:15:53 +01:00
|
|
|
if db.findStmt0 == nil: return ok(0) # no such table
|
2021-05-17 15:55:57 +02:00
|
|
|
db.findStmt0 # all rows
|
|
|
|
else:
|
|
|
|
if not nextPrefix(prefix, next):
|
|
|
|
# For example when looking for the prefix [byte 255], there are no
|
|
|
|
# prefixes that lexicographically are greater, thus we use the
|
|
|
|
# query that only does the >= comparison
|
2022-11-28 21:15:53 +01:00
|
|
|
if db.findStmt1 == nil: return ok(0) # no such table
|
2021-05-17 15:55:57 +02:00
|
|
|
checkErr bindParam(db.findStmt1, 1, prefix)
|
|
|
|
db.findStmt1
|
|
|
|
else:
|
2022-11-28 21:15:53 +01:00
|
|
|
if db.findStmt2 == nil: return ok(0) # no such table
|
2021-05-17 15:55:57 +02:00
|
|
|
checkErr bindParam(db.findStmt2, 1, prefix)
|
|
|
|
checkErr bindParam(db.findStmt2, 2, next)
|
|
|
|
db.findStmt2
|
|
|
|
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
var
|
|
|
|
total = 0
|
|
|
|
while true:
|
|
|
|
let
|
|
|
|
v = sqlite3_step(findStmt)
|
|
|
|
case v
|
|
|
|
of SQLITE_ROW:
|
|
|
|
let
|
|
|
|
kp = cast[ptr UncheckedArray[byte]](sqlite3_column_blob(findStmt, 0))
|
|
|
|
kl = sqlite3_column_bytes(findStmt, 0)
|
|
|
|
vp = cast[ptr UncheckedArray[byte]](sqlite3_column_blob(findStmt, 1))
|
|
|
|
vl = sqlite3_column_bytes(findStmt, 1)
|
2022-11-28 21:15:53 +01:00
|
|
|
if onFind != nil:
|
|
|
|
onFind(kp.toOpenArray(0, kl - 1), vp.toOpenArray(0, vl - 1))
|
2021-05-17 15:55:57 +02:00
|
|
|
total += 1
|
|
|
|
of SQLITE_DONE:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
# release implicit transaction (could use a defer, but it's slow)
|
|
|
|
discard sqlite3_reset(findStmt) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(findStmt) # no errors possible
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
return err($sqlite3_errstr(v))
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
# release implicit transaction
|
|
|
|
discard sqlite3_reset(findStmt) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(findStmt) # no errors possible
|
|
|
|
|
|
|
|
ok(total)
|
|
|
|
|
|
|
|
proc put*(db: SqKeyspaceRef, key, value: openArray[byte]): KvResult[void] =
|
2022-11-28 21:15:53 +01:00
|
|
|
if not db.open: return err("sqlite: database closed")
|
2021-05-17 15:55:57 +02:00
|
|
|
let putStmt = db.putStmt
|
2022-11-28 21:15:53 +01:00
|
|
|
if putStmt == nil: return err("sqlite: cannot write to read-only database")
|
2020-10-13 21:44:42 +03:00
|
|
|
checkErr bindParam(putStmt, 1, key)
|
|
|
|
checkErr bindParam(putStmt, 2, value)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-09-21 08:21:47 +02:00
|
|
|
let res =
|
2020-10-11 17:28:31 +03:00
|
|
|
if (let v = sqlite3_step(putStmt); v != SQLITE_DONE):
|
2020-09-21 08:21:47 +02:00
|
|
|
err($sqlite3_errstr(v))
|
|
|
|
else:
|
|
|
|
ok()
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-09-21 08:21:47 +02:00
|
|
|
# release implict transaction
|
2020-10-11 17:28:31 +03:00
|
|
|
discard sqlite3_reset(putStmt) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(putStmt) # no errors possible
|
2020-09-21 08:21:47 +02:00
|
|
|
|
|
|
|
res
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
proc contains*(db: SqKeyspaceRef, key: openArray[byte]): KvResult[bool] =
|
2022-11-28 21:15:53 +01:00
|
|
|
if not db.open: return err("sqlite: database closed")
|
2021-05-17 15:55:57 +02:00
|
|
|
let containsStmt = db.containsStmt
|
2022-11-28 21:15:53 +01:00
|
|
|
if containsStmt == nil: return ok(false) # no such table
|
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
checkErr bindParam(containsStmt, 1, key)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-09-21 08:21:47 +02:00
|
|
|
let
|
2020-10-11 17:28:31 +03:00
|
|
|
v = sqlite3_step(containsStmt)
|
2020-09-21 08:21:47 +02:00
|
|
|
res = case v
|
|
|
|
of SQLITE_ROW: ok(true)
|
|
|
|
of SQLITE_DONE: ok(false)
|
|
|
|
else: err($sqlite3_errstr(v))
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-09-21 08:21:47 +02:00
|
|
|
# release implicit transaction
|
2020-10-11 17:28:31 +03:00
|
|
|
discard sqlite3_reset(containsStmt) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(containsStmt) # no errors possible
|
2020-09-21 08:21:47 +02:00
|
|
|
|
|
|
|
res
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2023-01-03 16:38:27 +01:00
|
|
|
proc del*(db: SqKeyspaceRef, key: openArray[byte]): KvResult[bool] =
|
2022-11-28 21:15:53 +01:00
|
|
|
if not db.open: return err("sqlite: database closed")
|
2021-05-17 15:55:57 +02:00
|
|
|
let delStmt = db.delStmt
|
2023-01-03 16:38:27 +01:00
|
|
|
if delStmt == nil: return ok(false) # no such table
|
2020-10-13 21:44:42 +03:00
|
|
|
checkErr bindParam(delStmt, 1, key)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-09-21 08:21:47 +02:00
|
|
|
let res =
|
2020-10-11 17:28:31 +03:00
|
|
|
if (let v = sqlite3_step(delStmt); v != SQLITE_DONE):
|
2020-09-21 08:21:47 +02:00
|
|
|
err($sqlite3_errstr(v))
|
|
|
|
else:
|
2023-01-03 16:38:27 +01:00
|
|
|
ok(sqlite3_changes(db.env) > 0)
|
2020-09-21 08:21:47 +02:00
|
|
|
|
|
|
|
# release implict transaction
|
2020-10-11 17:28:31 +03:00
|
|
|
discard sqlite3_reset(delStmt) # same return information as step
|
|
|
|
discard sqlite3_clear_bindings(delStmt) # no errors possible
|
2020-09-21 08:21:47 +02:00
|
|
|
|
|
|
|
res
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2023-01-03 16:38:27 +01:00
|
|
|
proc clear*(db: SqKeyspaceRef): KvResult[bool] =
|
|
|
|
if not db.open: return err("sqlite: database closed")
|
|
|
|
let clearStmt = db.clearStmt
|
|
|
|
if clearStmt == nil: return ok(false) # no such table
|
|
|
|
|
|
|
|
let res =
|
|
|
|
if (let v = sqlite3_step(clearStmt); v != SQLITE_DONE):
|
|
|
|
err($sqlite3_errstr(v))
|
|
|
|
else:
|
|
|
|
ok(sqlite3_changes(db.env) > 0)
|
|
|
|
|
|
|
|
# release implicit transaction
|
|
|
|
discard sqlite3_reset(clearStmt) # same return information as step
|
|
|
|
|
|
|
|
res
|
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
proc close*(db: var SqKeyspace) =
|
|
|
|
# Calling with null stmt is harmless
|
|
|
|
discard sqlite3_finalize(db.putStmt)
|
|
|
|
discard sqlite3_finalize(db.getStmt)
|
|
|
|
discard sqlite3_finalize(db.delStmt)
|
2023-01-03 16:38:27 +01:00
|
|
|
discard sqlite3_finalize(db.clearStmt)
|
2021-05-17 15:55:57 +02:00
|
|
|
discard sqlite3_finalize(db.containsStmt)
|
|
|
|
discard sqlite3_finalize(db.findStmt0)
|
|
|
|
discard sqlite3_finalize(db.findStmt1)
|
|
|
|
discard sqlite3_finalize(db.findStmt2)
|
|
|
|
db = SqKeyspace()
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
proc close*(db: SqKeyspaceRef) =
|
|
|
|
close(db[])
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2020-04-27 15:16:11 +02:00
|
|
|
proc close*(db: SqStoreRef) =
|
2020-11-27 20:49:08 +02:00
|
|
|
for stmt in db.managedStmts:
|
2020-10-13 21:44:42 +03:00
|
|
|
discard sqlite3_finalize(stmt)
|
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
# Lazy-v2-close allows closing the keyspaces in any order
|
|
|
|
discard sqlite3_close_v2(db.env)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
|
|
|
db[] = SqStoreRef()[]
|
|
|
|
|
2020-12-18 14:25:46 +01:00
|
|
|
proc checkpoint*(db: SqStoreRef, kind = SqStoreCheckpointKind.passive) =
|
|
|
|
let mode: cint = case kind
|
|
|
|
of SqStoreCheckpointKind.passive: SQLITE_CHECKPOINT_PASSIVE
|
|
|
|
of SqStoreCheckpointKind.full: SQLITE_CHECKPOINT_FULL
|
|
|
|
of SqStoreCheckpointKind.restart: SQLITE_CHECKPOINT_RESTART
|
|
|
|
of SqStoreCheckpointKind.truncate: SQLITE_CHECKPOINT_TRUNCATE
|
|
|
|
discard sqlite3_wal_checkpoint_v2(db.env, nil, mode, nil, nil)
|
|
|
|
|
2021-05-17 15:55:57 +02:00
|
|
|
template prepare(env: ptr sqlite3, q: string): ptr sqlite3_stmt =
|
|
|
|
block:
|
|
|
|
var s: ptr sqlite3_stmt
|
2022-11-28 21:15:53 +01:00
|
|
|
checkErr sqlite3_prepare_v2(env, cstring(q), q.len.cint, addr s, nil):
|
2021-05-17 15:55:57 +02:00
|
|
|
discard
|
|
|
|
s
|
|
|
|
|
|
|
|
template prepare(env: ptr sqlite3, q: string, cleanup: untyped): ptr sqlite3_stmt =
|
|
|
|
block:
|
|
|
|
var s: ptr sqlite3_stmt
|
2022-11-28 21:15:53 +01:00
|
|
|
checkErr sqlite3_prepare_v2(env, cstring(q), q.len.cint, addr s, nil)
|
2021-05-17 15:55:57 +02:00
|
|
|
s
|
|
|
|
|
|
|
|
template checkExec(s: ptr sqlite3_stmt) =
|
|
|
|
if (let x = sqlite3_step(s); x != SQLITE_DONE):
|
|
|
|
discard sqlite3_finalize(s)
|
|
|
|
return err($sqlite3_errstr(x))
|
|
|
|
|
|
|
|
if (let x = sqlite3_finalize(s); x != SQLITE_OK):
|
|
|
|
return err($sqlite3_errstr(x))
|
|
|
|
|
|
|
|
template checkExec(env: ptr sqlite3, q: string) =
|
|
|
|
block:
|
|
|
|
let s = prepare(env, q): discard
|
|
|
|
checkExec(s)
|
|
|
|
|
2020-10-13 21:44:42 +03:00
|
|
|
proc isClosed*(db: SqStoreRef): bool =
|
|
|
|
db.env != nil
|
|
|
|
|
2020-04-27 15:16:11 +02:00
|
|
|
proc init*(
|
|
|
|
T: type SqStoreRef,
|
|
|
|
basePath: string,
|
|
|
|
name: string,
|
|
|
|
readOnly = false,
|
2020-10-11 17:28:31 +03:00
|
|
|
inMemory = false,
|
2021-05-17 15:55:57 +02:00
|
|
|
manualCheckpoint = false): KvResult[T] =
|
2020-10-13 21:44:42 +03:00
|
|
|
var env: AutoDisposed[ptr sqlite3]
|
2020-10-16 19:50:01 +03:00
|
|
|
defer: disposeIfUnreleased(env)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
|
|
|
let
|
|
|
|
name =
|
|
|
|
if inMemory: ":memory:"
|
2021-12-20 13:14:50 +01:00
|
|
|
else: basePath / name & ".sqlite3"
|
2020-04-27 15:16:11 +02:00
|
|
|
flags =
|
2021-05-25 20:57:28 +02:00
|
|
|
# For some reason, opening multiple in-memory databases doesn't work if
|
|
|
|
# one of them is read-only - for now, disable read-only mode for them
|
|
|
|
if readOnly and not inMemory: SQLITE_OPEN_READONLY
|
2020-04-27 15:16:11 +02:00
|
|
|
else: SQLITE_OPEN_READWRITE or SQLITE_OPEN_CREATE
|
|
|
|
|
|
|
|
if not inMemory:
|
|
|
|
try:
|
|
|
|
createDir(basePath)
|
|
|
|
except OSError, IOError:
|
2021-05-17 15:55:57 +02:00
|
|
|
return err("sqlite: cannot create database directory")
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2022-06-28 11:03:02 +00:00
|
|
|
checkErr sqlite3_open_v2(cstring name, addr env.val, flags.cint, nil)
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2020-08-24 12:37:40 +02:00
|
|
|
template checkWalPragmaResult(journalModePragma: ptr sqlite3_stmt) =
|
|
|
|
if (let x = sqlite3_step(journalModePragma); x != SQLITE_ROW):
|
|
|
|
discard sqlite3_finalize(journalModePragma)
|
|
|
|
return err($sqlite3_errstr(x))
|
|
|
|
|
|
|
|
if (let x = sqlite3_column_type(journalModePragma, 0); x != SQLITE3_TEXT):
|
|
|
|
discard sqlite3_finalize(journalModePragma)
|
|
|
|
return err($sqlite3_errstr(x))
|
|
|
|
|
2022-06-28 11:03:02 +00:00
|
|
|
if (let x = cstring sqlite3_column_text(journalModePragma, 0);
|
2020-08-24 13:08:35 +02:00
|
|
|
x != "memory" and x != "wal"):
|
2020-08-24 12:37:40 +02:00
|
|
|
discard sqlite3_finalize(journalModePragma)
|
2020-08-24 13:34:45 +02:00
|
|
|
return err("Invalid pragma result: " & $x)
|
2020-08-24 12:37:40 +02:00
|
|
|
|
2021-05-25 20:57:28 +02:00
|
|
|
if not readOnly:
|
|
|
|
# user_version = 1: single kvstore table without rowid
|
|
|
|
# user_version = 2: single kvstore table with rowid
|
|
|
|
# user_version = 3: multiple named kvstore tables via openKvStore
|
|
|
|
checkExec env.val, "PRAGMA user_version = 3;"
|
2020-04-27 15:16:11 +02:00
|
|
|
|
2021-05-25 20:57:28 +02:00
|
|
|
let journalModePragma = prepare(env.val, "PRAGMA journal_mode = WAL;")
|
|
|
|
checkWalPragmaResult(journalModePragma)
|
|
|
|
checkExec journalModePragma
|
2020-12-18 14:25:46 +01:00
|
|
|
|
|
|
|
if manualCheckpoint:
|
|
|
|
checkErr sqlite3_wal_autocheckpoint(env.val, 0)
|
|
|
|
# In manual checkpointing mode, we relax synchronization to NORMAL -
|
|
|
|
# this is safe in WAL mode leaving us with a consistent database at all
|
|
|
|
# times, though potentially losing any data written between checkpoints.
|
|
|
|
# http://www3.sqlite.org/wal.html#performance_considerations
|
2021-05-17 15:55:57 +02:00
|
|
|
checkExec env.val, "PRAGMA synchronous = NORMAL;"
|
2020-04-27 15:16:11 +02:00
|
|
|
|
|
|
|
ok(SqStoreRef(
|
2020-10-13 21:44:42 +03:00
|
|
|
env: env.release,
|
2021-11-16 13:45:28 +02:00
|
|
|
readOnly: readOnly
|
2020-04-27 15:16:11 +02:00
|
|
|
))
|
2020-05-09 01:30:50 +02:00
|
|
|
|
2022-11-28 21:15:53 +01:00
|
|
|
proc hasTable*(db: SqStoreRef, name: string): KvResult[bool] =
|
|
|
|
let
|
|
|
|
sql = "SELECT name FROM sqlite_master WHERE type='table' AND name='" &
|
|
|
|
name & "';"
|
2023-02-21 18:54:30 +01:00
|
|
|
db.exec(sql, (), proc(_: openArray[byte]) {.callback.} = discard)
|
2022-11-28 21:15:53 +01:00
|
|
|
|
|
|
|
proc openKvStore*(
|
|
|
|
db: SqStoreRef, name = "kvstore", withoutRowid = false,
|
|
|
|
readOnly = false): KvResult[SqKeyspaceRef] =
|
2021-05-17 15:55:57 +02:00
|
|
|
## Open a new Key-Value store in the SQLite database
|
|
|
|
##
|
|
|
|
## withoutRowid: Create the table without rowid - this is more efficient when
|
|
|
|
## rows are small (<200 bytes) but very inefficient with larger
|
|
|
|
## rows (the row being the sum of key and value) - see
|
|
|
|
## https://www.sqlite.org/withoutrowid.html
|
|
|
|
##
|
2022-11-28 21:15:53 +01:00
|
|
|
let hasTable = if db.readOnly or readOnly:
|
|
|
|
? db.hasTable(name)
|
|
|
|
else:
|
2021-11-16 13:45:28 +02:00
|
|
|
let createSql = """
|
2022-11-28 21:15:53 +01:00
|
|
|
CREATE TABLE IF NOT EXISTS '""" & name & """' (
|
2021-05-17 15:55:57 +02:00
|
|
|
key BLOB PRIMARY KEY,
|
|
|
|
value BLOB
|
|
|
|
)"""
|
2021-11-16 13:45:28 +02:00
|
|
|
checkExec db.env,
|
|
|
|
if withoutRowid: createSql & " WITHOUT ROWID;" else: createSql & ";"
|
2022-11-28 21:15:53 +01:00
|
|
|
true
|
2021-05-17 15:55:57 +02:00
|
|
|
var
|
2023-01-03 16:38:27 +01:00
|
|
|
tmp = SqKeyspace(env: db.env)
|
2021-05-17 15:55:57 +02:00
|
|
|
defer:
|
|
|
|
# We'll "move" ownership to the return value, effectively disabling "close"
|
|
|
|
close(tmp)
|
2022-11-28 21:15:53 +01:00
|
|
|
tmp.open = true
|
|
|
|
if hasTable:
|
|
|
|
tmp.getStmt =
|
|
|
|
prepare(db.env, "SELECT value FROM '" & name & "' WHERE key = ?;")
|
|
|
|
tmp.putStmt =
|
|
|
|
prepare(db.env, "INSERT OR REPLACE INTO '" & name & "'(key, value) VALUES (?, ?);")
|
|
|
|
tmp.delStmt = prepare(db.env, "DELETE FROM '" & name & "' WHERE key = ?;")
|
2023-01-03 16:38:27 +01:00
|
|
|
tmp.clearStmt = prepare(db.env, "DELETE FROM '" & name & "';")
|
2022-11-28 21:15:53 +01:00
|
|
|
tmp.containsStmt = prepare(db.env, "SELECT 1 FROM '" & name & "' WHERE key = ?;")
|
|
|
|
tmp.findStmt0 = prepare(db.env, "SELECT key, value FROM '" & name & "';")
|
|
|
|
tmp.findStmt1 = prepare(db.env, "SELECT key, value FROM '" & name & "' WHERE key >= ?;")
|
|
|
|
tmp.findStmt2 = prepare(db.env, "SELECT key, value FROM '" & name & "' WHERE key >= ? and key < ?;")
|
2021-05-17 15:55:57 +02:00
|
|
|
|
|
|
|
var res = SqKeyspaceRef()
|
|
|
|
res[] = tmp
|
|
|
|
tmp = SqKeyspace() # make close harmless
|
|
|
|
ok res
|
2020-10-11 17:28:31 +03:00
|
|
|
|
2023-02-22 09:11:32 +01:00
|
|
|
proc customScalarBlobFunction(ctx: ptr sqlite3_context, n: cint, v: ptr ptr sqlite3_value) {.cdecl, callback.} =
|
2022-06-02 14:14:15 +02:00
|
|
|
let ptrs = cast[ptr UncheckedArray[ptr sqlite3_value]](v)
|
|
|
|
let blob1 = cast[ptr UncheckedArray[byte]](sqlite3_value_blob(ptrs[][0]))
|
|
|
|
let blob2 = cast[ptr UncheckedArray[byte]](sqlite3_value_blob(ptrs[][1]))
|
|
|
|
let blob1Len = sqlite3_value_bytes(ptrs[][0])
|
|
|
|
let blob2Len = sqlite3_value_bytes(ptrs[][1])
|
2022-08-19 01:24:19 +03:00
|
|
|
# sqlite3_user_data retrieves data which was pointed by 5th param to
|
2022-06-02 14:14:15 +02:00
|
|
|
# sqlite3_create_function functions, which in our case is custom function
|
|
|
|
# provided by user
|
|
|
|
let usrFun = cast[CustomFunction](sqlite3_user_data(ctx))
|
|
|
|
let s = usrFun(
|
|
|
|
toOpenArray(blob1, 0, blob1Len - 1),
|
|
|
|
toOpenArray(blob2, 0, blob2Len - 1)
|
|
|
|
)
|
|
|
|
|
2023-02-22 10:03:13 +01:00
|
|
|
if s.isOk():
|
|
|
|
let bytes = s.unsafeGet()
|
|
|
|
# try is necessary as otherwise nim marks SQLITE_TRANSIENT as throwing
|
|
|
|
# unlisted exception.
|
|
|
|
# Using SQLITE_TRANSIENT destructor type, as it inform sqlite that data
|
|
|
|
# under provided pointer may be deleted at any moment, which is the case
|
|
|
|
# for seq[byte] as it is managed by nim gc. With this flag sqlite copy bytes
|
|
|
|
# under pointer and then releases them itself.
|
|
|
|
sqlite3_result_blob(ctx, unsafeAddr bytes[0], bytes.len.cint, SQLITE_TRANSIENT)
|
|
|
|
else:
|
|
|
|
let errMsg = s.error
|
|
|
|
sqlite3_result_error(ctx, errMsg, -1)
|
2022-06-02 14:14:15 +02:00
|
|
|
|
|
|
|
proc registerCustomScalarFunction*(db: SqStoreRef, name: string, fun: CustomFunction): KvResult[void] =
|
2022-08-19 01:24:19 +03:00
|
|
|
## Register custom function inside sqlite engine. Registered function can
|
2022-06-02 14:14:15 +02:00
|
|
|
## be used in further queries by its name. Function should be side-effect
|
|
|
|
## free and depends only on provided arguments.
|
|
|
|
## Name of the function should be valid utf8 string.
|
|
|
|
|
|
|
|
# Using SQLITE_DETERMINISTIC flag to inform sqlite that provided function
|
|
|
|
# won't have any side effect this may enable additional optimisations.
|
|
|
|
let deterministicUtf8Func = cint(SQLITE_UTF8 or SQLITE_DETERMINISTIC)
|
|
|
|
|
|
|
|
let res = sqlite3_create_function(
|
|
|
|
db.env,
|
|
|
|
name,
|
|
|
|
cint(2),
|
|
|
|
deterministicUtf8Func,
|
|
|
|
cast[pointer](fun),
|
|
|
|
customScalarBlobFunction,
|
|
|
|
nil,
|
|
|
|
nil
|
|
|
|
)
|
2022-08-19 01:24:19 +03:00
|
|
|
|
2022-06-02 14:14:15 +02:00
|
|
|
if res != SQLITE_OK:
|
|
|
|
return err($sqlite3_errstr(res))
|
|
|
|
else:
|
|
|
|
return ok()
|
|
|
|
|
2020-05-09 01:30:50 +02:00
|
|
|
when defined(metrics):
|
2021-07-02 16:43:26 +02:00
|
|
|
import locks, tables, times,
|
2020-05-09 14:34:06 +02:00
|
|
|
chronicles, metrics
|
|
|
|
|
2020-05-09 01:30:50 +02:00
|
|
|
type Sqlite3Info = ref object of Gauge
|
|
|
|
|
|
|
|
proc newSqlite3Info*(name: string, help: string, registry = defaultRegistry): Sqlite3Info {.raises: [Exception].} =
|
|
|
|
validateName(name)
|
|
|
|
result = Sqlite3Info(name: name,
|
|
|
|
help: help,
|
|
|
|
typ: "gauge",
|
|
|
|
creationThreadId: getThreadId())
|
2021-07-02 16:43:26 +02:00
|
|
|
result.lock.initLock()
|
2020-05-09 01:30:50 +02:00
|
|
|
result.register(registry)
|
|
|
|
|
|
|
|
var sqlite3Info* {.global.} = newSqlite3Info("sqlite3_info", "SQLite3 info")
|
|
|
|
|
|
|
|
method collect*(collector: Sqlite3Info): Metrics =
|
|
|
|
result = initOrderedTable[Labels, seq[Metric]]()
|
|
|
|
result[@[]] = @[]
|
2020-05-09 14:34:06 +02:00
|
|
|
let timestamp = getTime().toMilliseconds()
|
|
|
|
var currentMem, highwaterMem: int64
|
2020-05-09 01:30:50 +02:00
|
|
|
|
2020-05-09 14:34:06 +02:00
|
|
|
if (let res = sqlite3_status64(SQLITE_STATUS_MEMORY_USED, currentMem.addr, highwaterMem.addr, 0); res != SQLITE_OK):
|
2020-05-09 01:30:50 +02:00
|
|
|
error "SQLite3 error", msg = sqlite3_errstr(res)
|
|
|
|
else:
|
|
|
|
result[@[]] = @[
|
|
|
|
Metric(
|
|
|
|
name: "sqlite3_memory_used_bytes",
|
|
|
|
value: currentMem.float64,
|
|
|
|
timestamp: timestamp,
|
|
|
|
),
|
|
|
|
]
|