2022-11-22 19:40:24 +01:00
|
|
|
# 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
|
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
else:
|
|
|
|
{.push raises: [].}
|
|
|
|
|
|
|
|
import
|
2023-01-26 10:19:58 +01:00
|
|
|
std/options,
|
2022-11-22 19:40:24 +01:00
|
|
|
stew/[byteutils, results],
|
2023-05-25 17:34:34 +02:00
|
|
|
chronicles,
|
|
|
|
chronos
|
2022-11-22 19:40:24 +01:00
|
|
|
import
|
2023-08-09 18:11:50 +01:00
|
|
|
../../../common/databases/db_sqlite,
|
2023-04-19 13:29:23 +02:00
|
|
|
../../../waku_core,
|
2023-11-22 17:32:56 +01:00
|
|
|
../../../waku_core/message/digest,
|
2022-11-22 19:40:24 +01:00
|
|
|
../../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_msg index: " & resMsgIndex.error())
|
|
|
|
|
2024-03-12 07:51:03 -04:00
|
|
|
return ok()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
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()
|
2024-03-12 07:51:03 -04:00
|
|
|
return ok(SqliteDriver(db: db, insertStmt: insertStmt))
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method put*(s: SqliteDriver,
|
|
|
|
pubsubTopic: PubsubTopic,
|
|
|
|
message: WakuMessage,
|
|
|
|
digest: MessageDigest,
|
2023-11-22 17:32:56 +01:00
|
|
|
messageHash: WakuMessageHash,
|
2023-05-25 17:34:34 +02:00
|
|
|
receivedTime: Timestamp):
|
|
|
|
Future[ArchiveDriverResult[void]] {.async.} =
|
2022-11-22 19:40:24 +01:00
|
|
|
## Inserts a message into the store
|
|
|
|
let res = s.insertStmt.exec((
|
|
|
|
@(digest.data), # id
|
2023-11-22 17:32:56 +01:00
|
|
|
@(messageHash), # messageHash
|
2022-11-22 19:40:24 +01:00
|
|
|
receivedTime, # storedAt
|
|
|
|
toBytes(message.contentTopic), # contentTopic
|
|
|
|
message.payload, # payload
|
|
|
|
toBytes(pubsubTopic), # pubsubTopic
|
|
|
|
int64(message.version), # version
|
|
|
|
message.timestamp # senderTimestamp
|
|
|
|
))
|
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
return res
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method getAllMessages*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[seq[ArchiveRow]]] {.async.} =
|
2022-11-22 19:40:24 +01:00
|
|
|
## Retrieve all messages from the store.
|
2023-05-25 17:34:34 +02:00
|
|
|
return s.db.selectAllMessages()
|
|
|
|
|
|
|
|
method getMessages*(s: SqliteDriver,
|
2024-03-12 07:51:03 -04:00
|
|
|
contentTopic = newSeq[ContentTopic](0),
|
2023-05-25 17:34:34 +02:00
|
|
|
pubsubTopic = none(PubsubTopic),
|
|
|
|
cursor = none(ArchiveCursor),
|
|
|
|
startTime = none(Timestamp),
|
|
|
|
endTime = none(Timestamp),
|
2024-03-12 07:51:03 -04:00
|
|
|
hashes = newSeq[WakuMessageHash](0),
|
2023-05-25 17:34:34 +02:00
|
|
|
maxPageSize = DefaultPageSize,
|
|
|
|
ascendingOrder = true):
|
|
|
|
Future[ArchiveDriverResult[seq[ArchiveRow]]] {.async.} =
|
|
|
|
|
2022-11-22 19:40:24 +01:00
|
|
|
let cursor = cursor.map(toDbCursor)
|
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
let rowsRes = s.db.selectMessagesByHistoryQueryWithLimit(
|
2022-11-22 19:40:24 +01:00
|
|
|
contentTopic,
|
|
|
|
pubsubTopic,
|
|
|
|
cursor,
|
|
|
|
startTime,
|
|
|
|
endTime,
|
2024-03-12 07:51:03 -04:00
|
|
|
hashes,
|
2022-11-22 19:40:24 +01:00
|
|
|
limit=maxPageSize,
|
|
|
|
ascending=ascendingOrder
|
|
|
|
)
|
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
return rowsRes
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method getMessagesCount*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[int64]] {.async.} =
|
|
|
|
return s.db.getMessageCount()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-10-10 11:59:09 +02:00
|
|
|
method getPagesCount*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[int64]] {.async.} =
|
|
|
|
return s.db.getPageCount()
|
|
|
|
|
|
|
|
method getPagesSize*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[int64]] {.async.} =
|
|
|
|
return s.db.getPageSize()
|
|
|
|
|
2023-11-24 15:43:47 +01:00
|
|
|
method getDatabaseSize*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[int64]] {.async.} =
|
|
|
|
return s.db.getDatabaseSize()
|
|
|
|
|
2023-10-10 11:59:09 +02:00
|
|
|
method performVacuum*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[void]] {.async.} =
|
|
|
|
return s.db.performSqliteVacuum()
|
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method getOldestMessageTimestamp*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[Timestamp]] {.async.} =
|
|
|
|
return s.db.selectOldestReceiverTimestamp()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method getNewestMessageTimestamp*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[Timestamp]] {.async.} =
|
|
|
|
return s.db.selectnewestReceiverTimestamp()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method deleteMessagesOlderThanTimestamp*(s: SqliteDriver,
|
|
|
|
ts: Timestamp):
|
|
|
|
Future[ArchiveDriverResult[void]] {.async.} =
|
|
|
|
return s.db.deleteMessagesOlderThanTimestamp(ts)
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method deleteOldestMessagesNotWithinLimit*(s: SqliteDriver,
|
|
|
|
limit: int):
|
|
|
|
Future[ArchiveDriverResult[void]] {.async.} =
|
|
|
|
return s.db.deleteOldestMessagesNotWithinLimit(limit)
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2024-02-22 16:55:37 +01:00
|
|
|
method decreaseDatabaseSize*(driver: SqliteDriver,
|
2024-03-06 20:50:22 +01:00
|
|
|
targetSizeInBytes: int64,
|
|
|
|
forceRemoval: bool = false):
|
2024-02-22 16:55:37 +01:00
|
|
|
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()
|
|
|
|
|
2023-05-25 17:34:34 +02:00
|
|
|
method close*(s: SqliteDriver):
|
|
|
|
Future[ArchiveDriverResult[void]] {.async.} =
|
|
|
|
## Close the database connection
|
|
|
|
# Dispose statements
|
|
|
|
s.insertStmt.dispose()
|
|
|
|
# Close connection
|
|
|
|
s.db.close()
|
|
|
|
return ok()
|
2023-10-10 11:59:09 +02:00
|
|
|
|
2024-03-01 12:05:27 +01:00
|
|
|
method existsTable*(s: SqliteDriver, tableName: string):
|
|
|
|
Future[ArchiveDriverResult[bool]] {.async.} =
|
|
|
|
return err("existsTable method not implemented in sqlite_driver")
|
|
|
|
|