2022-08-01 18:21:11 +02:00
{. used . }
import
2022-10-28 20:13:41 +02:00
std / [ unittest , options , sequtils ] ,
2022-08-01 18:21:11 +02:00
stew / byteutils ,
2022-10-28 20:13:41 +02:00
chronos
2022-08-01 18:21:11 +02:00
import
2022-10-28 20:13:41 +02:00
.. / .. / waku / common / sqlite ,
2022-11-04 19:48:22 +01:00
.. / .. / waku / v2 / node / message_store / sqlite_store ,
.. / .. / waku / v2 / node / message_store / message_retention_policy ,
.. / .. / waku / v2 / node / message_store / message_retention_policy_capacity ,
2022-08-01 18:21:11 +02:00
.. / .. / waku / v2 / protocol / waku_message ,
2022-11-09 18:50:18 +01:00
.. / .. / waku / v2 / protocol / waku_store ,
2022-08-01 18:21:11 +02:00
.. / .. / waku / v2 / utils / time ,
2022-10-21 15:01:39 +02:00
. / utils ,
. / testlib / common
2022-08-01 18:21:11 +02:00
proc newTestDatabase ( ) : SqliteDatabase =
2022-10-28 00:05:02 +02:00
SqliteDatabase . new ( " :memory: " ) . tryGet ( )
2022-08-01 18:21:11 +02:00
suite " SQLite message store - init store " :
test " init store " :
## Given
let database = newTestDatabase ( )
## When
2022-09-16 12:55:22 +02:00
let resStore = SqliteStore . init ( database )
2022-08-01 18:21:11 +02:00
## Then
check :
resStore . isOk ( )
let store = resStore . tryGet ( )
check :
not store . isNil ( )
## Teardown
store . close ( )
test " init store with prepopulated database with messages older than retention policy " :
# TODO: Implement initialization test cases
discard
test " init store with prepopulated database with messsage count greater than max capacity " :
# TODO: Implement initialization test cases
discard
# TODO: Add test cases to cover the store retention time fucntionality
suite " SQLite message store - insert messages " :
test " insert a message " :
## Given
const contentTopic = " test-content-topic "
let
database = newTestDatabase ( )
2022-09-13 13:36:04 +02:00
store = SqliteStore . init ( database ) . tryGet ( )
2022-08-01 18:21:11 +02:00
let message = fakeWakuMessage ( contentTopic = contentTopic )
## When
2022-09-26 11:50:15 +02:00
let resPut = store . put ( DefaultPubsubTopic , message )
2022-08-01 18:21:11 +02:00
## Then
check :
resPut . isOk ( )
let storedMsg = store . getAllMessages ( ) . tryGet ( )
check :
storedMsg . len = = 1
storedMsg . all do ( item : auto ) - > bool :
2022-10-03 17:36:17 +02:00
let ( pubsubTopic , msg , digest , storeTimestamp ) = item
2022-08-01 18:21:11 +02:00
msg . contentTopic = = contentTopic and
pubsubTopic = = DefaultPubsubTopic
## Teardown
store . close ( )
test " store capacity should be limited " :
## Given
const storeCapacity = 5
const contentTopic = " test-content-topic "
let
database = newTestDatabase ( )
2022-09-16 12:55:22 +02:00
store = SqliteStore . init ( database ) . tryGet ( )
2022-09-13 13:36:04 +02:00
retentionPolicy : MessageRetentionPolicy = CapacityRetentionPolicy . init ( capacity = storeCapacity )
2022-08-01 18:21:11 +02:00
let messages = @ [
2022-10-03 15:23:39 +02:00
fakeWakuMessage ( ts = ts ( 0 ) ) ,
fakeWakuMessage ( 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-08-01 18:21:11 +02:00
]
## When
for msg in messages :
2022-10-03 15:23:39 +02:00
require store . put ( DefaultPubsubTopic , msg , computeDigest ( msg ) , msg . timestamp ) . isOk ( )
2022-09-16 12:55:22 +02:00
require retentionPolicy . execute ( store ) . isOk ( )
2022-08-01 18:21:11 +02:00
## Then
let storedMsg = store . getAllMessages ( ) . tryGet ( )
check :
storedMsg . len = = storeCapacity
storedMsg . all do ( item : auto ) - > bool :
2022-10-03 17:36:17 +02:00
let ( pubsubTopic , msg , digest , storeTimestamp ) = item
2022-08-01 18:21:11 +02:00
msg . contentTopic = = contentTopic and
pubsubTopic = = DefaultPubsubTopic
## Teardown
store . close ( )
# TODO: Review the following suite test cases
suite " Message Store " :
test " set and get works " :
## Given
let
database = newTestDatabase ( )
2022-09-13 13:36:04 +02:00
store = SqliteStore . init ( database ) . get ( )
2022-08-01 18:21:11 +02:00
2022-09-26 11:50:15 +02:00
let
2022-10-03 15:23:39 +02:00
t1 = ts ( 0 )
t2 = ts ( 1 )
2022-08-01 18:21:11 +02:00
t3 = high ( int64 )
var msgs = @ [
2022-09-26 11:50:15 +02:00
WakuMessage ( payload : @ [ byte 1 , 2 , 3 ] , contentTopic : DefaultContentTopic , version : uint32 ( 0 ) , timestamp : t1 ) ,
WakuMessage ( payload : @ [ byte 1 , 2 , 3 , 4 ] , contentTopic : DefaultContentTopic , version : uint32 ( 1 ) , timestamp : t2 ) ,
2022-08-01 18:21:11 +02:00
# high(uint32) is the largest value that fits in uint32, this is to make sure there is no overflow in the storage
2022-09-26 11:50:15 +02:00
WakuMessage ( payload : @ [ byte 1 , 2 , 3 , 4 , 5 ] , contentTopic : DefaultContentTopic , version : high ( uint32 ) , timestamp : t3 ) ,
2022-08-01 18:21:11 +02:00
]
2022-11-09 18:50:18 +01:00
var indexes : seq [ HistoryCursor ] = @ [ ]
2022-08-01 18:21:11 +02:00
for msg in msgs :
2022-09-26 11:50:15 +02:00
require store . put ( DefaultPubsubTopic , msg , computeDigest ( msg ) , msg . timestamp ) . isOk ( )
2022-11-09 18:50:18 +01:00
let cursor = HistoryCursor (
pubsubTopic : DefaultPubsubTopic ,
senderTime : msg . timestamp ,
storeTime : msg . timestamp ,
digest : computeDigest ( msg )
)
indexes . add ( cursor )
2022-08-01 18:21:11 +02:00
## When
let res = store . getAllMessages ( )
## Then
check :
res . isOk ( )
let result = res . value
check :
result . len = = 3
# flags for version
var v0Flag , v1Flag , vMaxFlag : bool = false
# flags for sender timestamp
var t1Flag , t2Flag , t3Flag : bool = false
# flags for receiver timestamp
var rt1Flag , rt2Flag , rt3Flag : bool = false
2022-10-03 17:36:17 +02:00
for ( pubsubTopic , msg , digest , receiverTimestamp ) in result :
2022-09-26 11:50:15 +02:00
check :
pubsubTopic = = DefaultPubsubTopic
2022-08-01 18:21:11 +02:00
# check correct retrieval of receiver timestamps
2022-11-09 18:50:18 +01:00
if receiverTimestamp = = indexes [ 0 ] . storeTime : rt1Flag = true
if receiverTimestamp = = indexes [ 1 ] . storeTime : rt2Flag = true
if receiverTimestamp = = indexes [ 2 ] . storeTime : rt3Flag = true
2022-08-01 18:21:11 +02:00
check :
msg in msgs
# check the correct retrieval of versions
if msg . version = = uint32 ( 0 ) : v0Flag = true
if msg . version = = uint32 ( 1 ) : v1Flag = true
if msg . version = = high ( uint32 ) : vMaxFlag = true
# check correct retrieval of sender timestamps
if msg . timestamp = = t1 : t1Flag = true
if msg . timestamp = = t2 : t2Flag = true
if msg . timestamp = = t3 : t3Flag = true
check :
# check version
v0Flag = = true
v1Flag = = true
vMaxFlag = = true
# check sender timestamp
t1Flag = = true
t2Flag = = true
t3Flag = = true
# check receiver timestamp
rt1Flag = = true
rt2Flag = = true
rt3Flag = = true
## Cleanup
store . close ( )
test " set and get user version " :
## Given
let
database = newTestDatabase ( )
2022-09-13 13:36:04 +02:00
store = SqliteStore . init ( database ) . get ( )
2022-08-01 18:21:11 +02:00
## When
let resSetVersion = database . setUserVersion ( 5 )
let resGetVersion = database . getUserVersion ( )
## Then
check :
resSetVersion . isOk ( )
resGetVersion . isOk ( )
let version = resGetVersion . tryGet ( )
check :
version = = 5
## Cleanup
store . close ( )
2022-10-03 15:23:39 +02:00
# TODO: Move this test case to retention policy test suite
2022-08-01 18:21:11 +02:00
test " number of messages retrieved by getAll is bounded by storeCapacity " :
2022-10-03 15:23:39 +02:00
let capacity = 10
2022-09-13 13:36:04 +02:00
let
2022-09-16 12:55:22 +02:00
database = newTestDatabase ( )
store = SqliteStore . init ( database ) . tryGet ( )
2022-09-13 13:36:04 +02:00
retentionPolicy : MessageRetentionPolicy = CapacityRetentionPolicy . init ( capacity = capacity )
2022-08-01 18:21:11 +02:00
for i in 1 .. capacity :
2022-10-03 15:23:39 +02:00
let msg = WakuMessage ( payload : @ [ byte i ] , contentTopic : DefaultContentTopic , version : 0 , timestamp : Timestamp ( i ) )
require store . put ( DefaultPubsubTopic , msg , computeDigest ( msg ) , msg . timestamp ) . isOk ( )
2022-09-16 12:55:22 +02:00
require retentionPolicy . execute ( store ) . isOk ( )
## Then
2022-08-01 18:21:11 +02:00
# Test limited getAll function when store is at capacity
let resMax = store . getAllMessages ( )
check :
resMax . isOk ( )
let response = resMax . tryGet ( )
let lastMessageTimestamp = response [ ^ 1 ] [ 1 ] . timestamp
check :
response . len = = capacity # We retrieved all items
lastMessageTimestamp = = Timestamp ( capacity ) # Returned rows were ordered correctly # TODO: not representative because the timestamp only has second resolution
## Cleanup
store . close ( )
2022-10-03 15:23:39 +02:00
# TODO: Move this test case to retention policy test suite
2022-08-01 18:21:11 +02:00
test " DB store capacity " :
let
contentTopic = ContentTopic ( " /waku/2/default-content/proto " )
pubsubTopic = " /waku/2/default-waku/proto "
capacity = 100
overload = 65
2022-09-13 13:36:04 +02:00
let
2022-10-28 00:05:02 +02:00
database = newTestDatabase ( )
2022-09-16 12:55:22 +02:00
store = SqliteStore . init ( database ) . tryGet ( )
2022-09-13 13:36:04 +02:00
retentionPolicy : MessageRetentionPolicy = CapacityRetentionPolicy . init ( capacity = capacity )
2022-08-01 18:21:11 +02:00
for i in 1 .. capacity + overload :
2022-09-26 11:50:15 +02:00
let msg = WakuMessage ( payload : ( $ i ) . toBytes ( ) , contentTopic : contentTopic , version : uint32 ( 0 ) , timestamp : Timestamp ( i ) )
require store . put ( pubsubTopic , msg ) . isOk ( )
2022-09-16 12:55:22 +02:00
require retentionPolicy . execute ( store ) . isOk ( )
2022-08-01 18:21:11 +02:00
# count messages in DB
2022-09-26 11:50:15 +02:00
let numMessages = store . getMessagesCount ( ) . tryGet ( )
2022-08-01 18:21:11 +02: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)
2022-09-13 13:36:04 +02:00
# the window size changes when changing `const maxStoreOverflow = 1.3 in sqlite_store
2022-09-16 12:55:22 +02:00
numMessages = = 120
## Teardown
store . close ( )