mirror of
https://github.com/waku-org/nwaku.git
synced 2025-01-18 02:41:47 +00:00
fd6a71cdd7
* bump_dependencies.md: add nim-results dependency * change imports stew/results to results * switching to Nim 2.0.8 * waku.nimble: reflect the requirement nim 1.6.0 to 2.0.8 Adding --mm:refc as nim 2.0 enables a new garbage collector that we're not yet ready to support * adapt waku code to Nim 2.0 * gcsafe adaptations because Nim 2.0 is more strict
221 lines
6.5 KiB
Nim
221 lines
6.5 KiB
Nim
# The code in this file is an adaptation of the Sqlite KV Store found in nim-eth.
|
|
# https://github.com/status-im/nim-eth/blob/master/eth/db/kvstore_sqlite3.nim
|
|
{.push raises: [].}
|
|
|
|
import std/options, stew/byteutils, chronicles, chronos, results
|
|
import
|
|
../../../common/databases/db_sqlite,
|
|
../../../waku_core,
|
|
../../../waku_core/message/digest,
|
|
../../common,
|
|
../../driver,
|
|
./cursor,
|
|
./queries
|
|
|
|
logScope:
|
|
topics = "waku archive sqlite"
|
|
|
|
proc init(db: SqliteDatabase): ArchiveDriverResult[void] =
|
|
## Misconfiguration can lead to nil DB
|
|
if db.isNil():
|
|
return err("db not initialized")
|
|
|
|
# Create table, if doesn't exist
|
|
let resCreate = createTable(db)
|
|
if resCreate.isErr():
|
|
return err("failed to create table: " & resCreate.error())
|
|
|
|
# Create indices, if don't exist
|
|
let resRtIndex = createOldestMessageTimestampIndex(db)
|
|
if resRtIndex.isErr():
|
|
return err("failed to create i_rt index: " & resRtIndex.error())
|
|
|
|
let resMsgIndex = createHistoryQueryIndex(db)
|
|
if resMsgIndex.isErr():
|
|
return err("failed to create i_query index: " & resMsgIndex.error())
|
|
|
|
return ok()
|
|
|
|
type SqliteDriver* = ref object of ArchiveDriver
|
|
db: SqliteDatabase
|
|
insertStmt: SqliteStmt[InsertMessageParams, void]
|
|
|
|
proc new*(T: type SqliteDriver, db: SqliteDatabase): ArchiveDriverResult[T] =
|
|
# Database initialization
|
|
let resInit = init(db)
|
|
if resInit.isErr():
|
|
return err(resInit.error())
|
|
|
|
# General initialization
|
|
let insertStmt = db.prepareInsertMessageStmt()
|
|
return ok(SqliteDriver(db: db, insertStmt: insertStmt))
|
|
|
|
method put*(
|
|
s: SqliteDriver,
|
|
pubsubTopic: PubsubTopic,
|
|
message: WakuMessage,
|
|
digest: MessageDigest,
|
|
messageHash: WakuMessageHash,
|
|
receivedTime: Timestamp,
|
|
): Future[ArchiveDriverResult[void]] {.async.} =
|
|
## Inserts a message into the store
|
|
let res = s.insertStmt.exec(
|
|
(
|
|
@(digest.data), # id
|
|
@(messageHash), # messageHash
|
|
receivedTime, # storedAt
|
|
toBytes(message.contentTopic), # contentTopic
|
|
message.payload, # payload
|
|
toBytes(pubsubTopic), # pubsubTopic
|
|
int64(message.version), # version
|
|
message.timestamp, # senderTimestamp
|
|
message.meta, # meta
|
|
)
|
|
)
|
|
|
|
return res
|
|
|
|
method getAllMessages*(
|
|
s: SqliteDriver
|
|
): Future[ArchiveDriverResult[seq[ArchiveRow]]] {.async.} =
|
|
## Retrieve all messages from the store.
|
|
return s.db.selectAllMessages()
|
|
|
|
method getMessagesV2*(
|
|
s: SqliteDriver,
|
|
contentTopic = newSeq[ContentTopic](0),
|
|
pubsubTopic = none(PubsubTopic),
|
|
cursor = none(ArchiveCursor),
|
|
startTime = none(Timestamp),
|
|
endTime = none(Timestamp),
|
|
maxPageSize = DefaultPageSize,
|
|
ascendingOrder = true,
|
|
): Future[ArchiveDriverResult[seq[ArchiveRow]]] {.async, deprecated.} =
|
|
let cursor = cursor.map(toDbCursor)
|
|
|
|
let rowsRes = s.db.selectMessagesByHistoryQueryWithLimit(
|
|
contentTopic,
|
|
pubsubTopic,
|
|
cursor,
|
|
startTime,
|
|
endTime,
|
|
limit = maxPageSize,
|
|
ascending = ascendingOrder,
|
|
)
|
|
|
|
return rowsRes
|
|
|
|
method getMessages*(
|
|
s: SqliteDriver,
|
|
includeData = false,
|
|
contentTopic = newSeq[ContentTopic](0),
|
|
pubsubTopic = none(PubsubTopic),
|
|
cursor = none(ArchiveCursor),
|
|
startTime = none(Timestamp),
|
|
endTime = none(Timestamp),
|
|
hashes = newSeq[WakuMessageHash](0),
|
|
maxPageSize = DefaultPageSize,
|
|
ascendingOrder = true,
|
|
): Future[ArchiveDriverResult[seq[ArchiveRow]]] {.async.} =
|
|
let cursor =
|
|
if cursor.isSome():
|
|
some(cursor.get().hash)
|
|
else:
|
|
none(WakuMessageHash)
|
|
|
|
let rowsRes = s.db.selectMessagesByStoreQueryWithLimit(
|
|
contentTopic,
|
|
pubsubTopic,
|
|
cursor,
|
|
startTime,
|
|
endTime,
|
|
hashes,
|
|
limit = maxPageSize,
|
|
ascending = ascendingOrder,
|
|
)
|
|
|
|
return rowsRes
|
|
|
|
method getMessagesCount*(
|
|
s: SqliteDriver
|
|
): Future[ArchiveDriverResult[int64]] {.async.} =
|
|
return s.db.getMessageCount()
|
|
|
|
method getPagesCount*(s: SqliteDriver): Future[ArchiveDriverResult[int64]] {.async.} =
|
|
return s.db.getPageCount()
|
|
|
|
method getPagesSize*(s: SqliteDriver): Future[ArchiveDriverResult[int64]] {.async.} =
|
|
return s.db.getPageSize()
|
|
|
|
method getDatabaseSize*(s: SqliteDriver): Future[ArchiveDriverResult[int64]] {.async.} =
|
|
return s.db.getDatabaseSize()
|
|
|
|
method performVacuum*(s: SqliteDriver): Future[ArchiveDriverResult[void]] {.async.} =
|
|
return s.db.performSqliteVacuum()
|
|
|
|
method getOldestMessageTimestamp*(
|
|
s: SqliteDriver
|
|
): Future[ArchiveDriverResult[Timestamp]] {.async.} =
|
|
return s.db.selectOldestReceiverTimestamp()
|
|
|
|
method getNewestMessageTimestamp*(
|
|
s: SqliteDriver
|
|
): Future[ArchiveDriverResult[Timestamp]] {.async.} =
|
|
return s.db.selectnewestReceiverTimestamp()
|
|
|
|
method deleteMessagesOlderThanTimestamp*(
|
|
s: SqliteDriver, ts: Timestamp
|
|
): Future[ArchiveDriverResult[void]] {.async.} =
|
|
return s.db.deleteMessagesOlderThanTimestamp(ts)
|
|
|
|
method deleteOldestMessagesNotWithinLimit*(
|
|
s: SqliteDriver, limit: int
|
|
): Future[ArchiveDriverResult[void]] {.async.} =
|
|
return s.db.deleteOldestMessagesNotWithinLimit(limit)
|
|
|
|
method decreaseDatabaseSize*(
|
|
driver: SqliteDriver, targetSizeInBytes: int64, forceRemoval: bool = false
|
|
): Future[ArchiveDriverResult[void]] {.async.} =
|
|
## To remove 20% of the outdated data from database
|
|
const DeleteLimit = 0.80
|
|
|
|
## when db size overshoots the database limit, shread 20% of outdated messages
|
|
## get size of database
|
|
let dbSize = (await driver.getDatabaseSize()).valueOr:
|
|
return err("failed to get database size: " & $error)
|
|
|
|
## database size in bytes
|
|
let totalSizeOfDB: int64 = int64(dbSize)
|
|
|
|
if totalSizeOfDB < targetSizeInBytes:
|
|
return ok()
|
|
|
|
## to shread/delete messsges, get the total row/message count
|
|
let numMessages = (await driver.getMessagesCount()).valueOr:
|
|
return err("failed to get messages count: " & error)
|
|
|
|
## NOTE: Using SQLite vacuuming is done manually, we delete a percentage of rows
|
|
## if vacumming is done automatically then we aim to check DB size periodially for efficient
|
|
## retention policy implementation.
|
|
|
|
## 80% of the total messages are to be kept, delete others
|
|
let pageDeleteWindow = int(float(numMessages) * DeleteLimit)
|
|
|
|
(await driver.deleteOldestMessagesNotWithinLimit(limit = pageDeleteWindow)).isOkOr:
|
|
return err("deleting oldest messages failed: " & error)
|
|
|
|
return ok()
|
|
|
|
method close*(s: SqliteDriver): Future[ArchiveDriverResult[void]] {.async.} =
|
|
## Close the database connection
|
|
# Dispose statements
|
|
s.insertStmt.dispose()
|
|
# Close connection
|
|
s.db.close()
|
|
return ok()
|
|
|
|
method existsTable*(
|
|
s: SqliteDriver, tableName: string
|
|
): Future[ArchiveDriverResult[bool]] {.async.} =
|
|
return err("existsTable method not implemented in sqlite_driver")
|