2022-11-22 19:40:24 +01:00
|
|
|
{.used.}
|
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
import std/[sequtils, times], stew/results, testutils/unittests, chronos
|
2022-11-22 19:40:24 +01:00
|
|
|
import
|
2023-06-22 11:27:40 +02:00
|
|
|
../../../waku/common/databases/db_sqlite,
|
2023-08-09 18:11:50 +01:00
|
|
|
../../../waku/waku_core,
|
2023-11-22 17:32:56 +01:00
|
|
|
../../../waku/waku_core/message/digest,
|
2023-08-09 18:11:50 +01:00
|
|
|
../../../waku/waku_archive,
|
|
|
|
../../../waku/waku_archive/driver/sqlite_driver,
|
|
|
|
../../../waku/waku_archive/retention_policy,
|
|
|
|
../../../waku/waku_archive/retention_policy/retention_policy_capacity,
|
2023-10-10 11:59:09 +02:00
|
|
|
../../../waku/waku_archive/retention_policy/retention_policy_size,
|
2023-11-27 18:33:27 +01:00
|
|
|
../waku_archive/archive_utils,
|
2023-02-13 11:43:49 +01:00
|
|
|
../testlib/common,
|
2023-04-05 16:01:51 +02:00
|
|
|
../testlib/wakucore
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
suite "Waku Archive - Retention policy":
|
|
|
|
test "capacity retention policy - windowed message deletion":
|
|
|
|
## Given
|
|
|
|
let
|
|
|
|
capacity = 100
|
2023-10-10 11:59:09 +02:00
|
|
|
excess = 60
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-11-27 18:33:27 +01:00
|
|
|
let driver = newSqliteArchiveDriver()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
let retentionPolicy: RetentionPolicy =
|
|
|
|
CapacityRetentionPolicy.new(capacity = capacity)
|
2023-10-10 11:59:09 +02:00
|
|
|
var putFutures = newSeq[Future[ArchiveDriverResult[void]]]()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
## When
|
2024-03-16 00:08:47 +01:00
|
|
|
for i in 1 .. capacity + excess:
|
|
|
|
let msg = fakeWakuMessage(
|
|
|
|
payload = @[byte i], contentTopic = DefaultContentTopic, ts = Timestamp(i)
|
|
|
|
)
|
|
|
|
putFutures.add(
|
|
|
|
driver.put(
|
|
|
|
DefaultPubsubTopic,
|
|
|
|
msg,
|
|
|
|
computeDigest(msg),
|
|
|
|
computeMessageHash(DefaultPubsubTopic, msg),
|
|
|
|
msg.timestamp,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2023-10-10 11:59:09 +02:00
|
|
|
discard waitFor allFinished(putFutures)
|
2022-11-22 19:40:24 +01:00
|
|
|
|
2023-10-10 11:59:09 +02:00
|
|
|
require (waitFor retentionPolicy.execute(driver)).isOk()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
## Then
|
2023-05-25 17:34:34 +02:00
|
|
|
let numMessages = (waitFor driver.getMessagesCount()).tryGet()
|
2022-11-22 19:40:24 +01:00
|
|
|
check:
|
|
|
|
# Expected number of messages is 120 because
|
|
|
|
# (capacity = 100) + (half of the overflow window = 15) + (5 messages added after after the last delete)
|
|
|
|
# the window size changes when changing `const maxStoreOverflow = 1.3 in sqlite_store
|
2023-10-10 11:59:09 +02:00
|
|
|
numMessages == 115
|
|
|
|
|
|
|
|
## Cleanup
|
|
|
|
(waitFor driver.close()).expect("driver to close")
|
2024-03-16 00:08:47 +01:00
|
|
|
|
2023-10-10 11:59:09 +02:00
|
|
|
test "size retention policy - windowed message deletion":
|
|
|
|
## Given
|
|
|
|
let
|
2023-11-24 15:43:47 +01:00
|
|
|
# in bytes
|
2024-03-16 00:08:47 +01:00
|
|
|
sizeLimit: int64 = 52428
|
2023-10-10 11:59:09 +02:00
|
|
|
excess = 325
|
|
|
|
|
2023-11-27 18:33:27 +01:00
|
|
|
let driver = newSqliteArchiveDriver()
|
2023-10-10 11:59:09 +02:00
|
|
|
|
2024-03-16 00:08:47 +01:00
|
|
|
let retentionPolicy: RetentionPolicy = SizeRetentionPolicy.new(size = sizeLimit)
|
2023-10-10 11:59:09 +02:00
|
|
|
var putFutures = newSeq[Future[ArchiveDriverResult[void]]]()
|
|
|
|
|
|
|
|
# make sure that the db is empty to before test begins
|
|
|
|
let storedMsg = (waitFor driver.getAllMessages()).tryGet()
|
|
|
|
# if there are messages in db, empty them
|
|
|
|
if storedMsg.len > 0:
|
|
|
|
let now = getNanosecondTime(getTime().toUnixFloat())
|
2024-03-16 00:08:47 +01:00
|
|
|
require (waitFor driver.deleteMessagesOlderThanTimestamp(ts = now)).isOk()
|
2023-10-10 11:59:09 +02:00
|
|
|
require (waitFor driver.performVacuum()).isOk()
|
|
|
|
|
|
|
|
## When
|
2023-11-22 17:32:56 +01:00
|
|
|
##
|
2023-10-10 11:59:09 +02:00
|
|
|
|
|
|
|
# create a number of messages so that the size of the DB overshoots
|
2024-03-16 00:08:47 +01:00
|
|
|
for i in 1 .. excess:
|
|
|
|
let msg = fakeWakuMessage(
|
|
|
|
payload = @[byte i], contentTopic = DefaultContentTopic, ts = Timestamp(i)
|
|
|
|
)
|
|
|
|
putFutures.add(
|
|
|
|
driver.put(
|
|
|
|
DefaultPubsubTopic,
|
|
|
|
msg,
|
|
|
|
computeDigest(msg),
|
|
|
|
computeMessageHash(DefaultPubsubTopic, msg),
|
|
|
|
msg.timestamp,
|
|
|
|
)
|
|
|
|
)
|
2023-10-10 11:59:09 +02:00
|
|
|
|
|
|
|
# waitFor is used to synchronously wait for the futures to complete.
|
|
|
|
discard waitFor allFinished(putFutures)
|
|
|
|
|
|
|
|
## Then
|
|
|
|
# calculate the current database size
|
2023-11-24 15:43:47 +01:00
|
|
|
let sizeDB = int64((waitFor driver.getDatabaseSize()).tryGet())
|
2023-11-21 11:27:50 +01:00
|
|
|
|
|
|
|
# NOTE: since vacuumin is done manually, this needs to be revisited if vacuuming done automatically
|
|
|
|
|
|
|
|
# get the rows count pre-deletion
|
|
|
|
let rowsCountBeforeDeletion = (waitFor driver.getMessagesCount()).tryGet()
|
2023-10-10 11:59:09 +02:00
|
|
|
|
2023-11-21 11:27:50 +01:00
|
|
|
# execute policy provided the current db size oveflows, results in rows deletion
|
2023-10-10 11:59:09 +02:00
|
|
|
require (sizeDB >= sizeLimit)
|
|
|
|
require (waitFor retentionPolicy.execute(driver)).isOk()
|
2024-03-16 00:08:47 +01:00
|
|
|
|
2023-11-24 15:43:47 +01:00
|
|
|
# get the number or rows from database
|
2023-11-21 11:27:50 +01:00
|
|
|
let rowCountAfterDeletion = (waitFor driver.getMessagesCount()).tryGet()
|
2023-10-10 11:59:09 +02:00
|
|
|
|
|
|
|
check:
|
|
|
|
# size of the database is used to check if the storage limit has been preserved
|
|
|
|
# check the current database size with the limitSize provided by the user
|
|
|
|
# it should be lower
|
2023-11-21 11:27:50 +01:00
|
|
|
rowCountAfterDeletion <= rowsCountBeforeDeletion
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
## Cleanup
|
2023-05-25 17:34:34 +02:00
|
|
|
(waitFor driver.close()).expect("driver to close")
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
test "store capacity should be limited":
|
|
|
|
## Given
|
|
|
|
const capacity = 5
|
|
|
|
const contentTopic = "test-content-topic"
|
|
|
|
|
|
|
|
let
|
2023-11-27 18:33:27 +01:00
|
|
|
driver = newSqliteArchiveDriver()
|
2024-03-16 00:08:47 +01:00
|
|
|
retentionPolicy: RetentionPolicy =
|
|
|
|
CapacityRetentionPolicy.new(capacity = capacity)
|
|
|
|
|
|
|
|
let messages =
|
|
|
|
@[
|
|
|
|
fakeWakuMessage(contentTopic = DefaultContentTopic, ts = ts(0)),
|
|
|
|
fakeWakuMessage(contentTopic = DefaultContentTopic, ts = ts(1)),
|
|
|
|
fakeWakuMessage(contentTopic = contentTopic, ts = ts(2)),
|
|
|
|
fakeWakuMessage(contentTopic = contentTopic, ts = ts(3)),
|
|
|
|
fakeWakuMessage(contentTopic = contentTopic, ts = ts(4)),
|
|
|
|
fakeWakuMessage(contentTopic = contentTopic, ts = ts(5)),
|
|
|
|
fakeWakuMessage(contentTopic = contentTopic, ts = ts(6)),
|
|
|
|
]
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
## When
|
|
|
|
for msg in messages:
|
2024-03-16 00:08:47 +01:00
|
|
|
require (
|
|
|
|
waitFor driver.put(
|
|
|
|
DefaultPubsubTopic,
|
|
|
|
msg,
|
|
|
|
computeDigest(msg),
|
|
|
|
computeMessageHash(DefaultPubsubTopic, msg),
|
|
|
|
msg.timestamp,
|
|
|
|
)
|
|
|
|
).isOk()
|
2023-05-25 17:34:34 +02:00
|
|
|
require (waitFor retentionPolicy.execute(driver)).isOk()
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
## Then
|
2023-05-25 17:34:34 +02:00
|
|
|
let storedMsg = (waitFor driver.getAllMessages()).tryGet()
|
2022-11-22 19:40:24 +01:00
|
|
|
check:
|
|
|
|
storedMsg.len == capacity
|
2024-03-16 00:08:47 +01:00
|
|
|
storedMsg.all do(item: auto) -> bool:
|
2024-03-12 07:51:03 -04:00
|
|
|
let (pubsubTopic, msg, _, _, _) = item
|
2024-03-16 00:08:47 +01:00
|
|
|
msg.contentTopic == contentTopic and pubsubTopic == DefaultPubsubTopic
|
2022-11-22 19:40:24 +01:00
|
|
|
|
|
|
|
## Cleanup
|
2023-05-25 17:34:34 +02:00
|
|
|
(waitFor driver.close()).expect("driver to close")
|