2021-03-23 08:24:09 +00:00
{. used . }
2020-12-21 11:45:07 +00:00
import
2021-06-16 20:46:57 +00:00
std / [ unittest , options , tables , sets , times , os , strutils ] ,
chronos ,
2022-03-28 19:09:46 +00:00
sqlite3_abi ,
stew / byteutils ,
2021-03-25 09:03:17 +00:00
.. / .. / waku / v2 / node / storage / message / waku_message_store ,
2021-06-16 20:46:57 +00:00
.. / .. / waku / v2 / node / storage / sqlite ,
2020-12-21 11:45:07 +00:00
.. / .. / waku / v2 / protocol / waku_store / waku_store ,
2022-02-17 15:26:49 +00:00
.. / .. / waku / v2 / utils / time ,
2021-01-22 09:50:50 +00:00
. / utils
2020-12-21 11:45:07 +00:00
2022-03-28 19:09:46 +00:00
2020-12-21 11:45:07 +00:00
suite " Message Store " :
test " set and get works " :
let
database = SqliteDatabase . init ( " " , inMemory = true ) [ ]
2021-01-22 09:50:50 +00:00
store = WakuMessageStore . init ( database ) [ ]
2021-04-08 10:15:29 +00:00
topic = ContentTopic ( " /waku/2/default-content/proto " )
2021-04-28 00:14:55 +00:00
pubsubTopic = " /waku/2/default-waku/proto "
2020-12-21 11:45:07 +00:00
2022-02-17 15:26:49 +00:00
t1 = getNanosecondTime ( epochTime ( ) )
t2 = getNanosecondTime ( epochTime ( ) )
2022-03-18 10:36:27 +00:00
t3 = high ( int64 )
2020-12-21 11:45:07 +00:00
var msgs = @ [
2021-06-16 20:46:57 +00:00
WakuMessage ( payload : @ [ byte 1 , 2 , 3 ] , contentTopic : topic , version : uint32 ( 0 ) , timestamp : t1 ) ,
WakuMessage ( payload : @ [ byte 1 , 2 , 3 , 4 ] , contentTopic : topic , version : uint32 ( 1 ) , timestamp : t2 ) ,
WakuMessage ( payload : @ [ byte 1 , 2 , 3 , 4 , 5 ] , contentTopic : topic , version : high ( uint32 ) , timestamp : t3 ) ,
2020-12-21 11:45:07 +00:00
]
defer : store . close ( )
2021-06-16 20:46:57 +00:00
var indexes : seq [ Index ] = @ [ ]
2020-12-21 11:45:07 +00:00
for msg in msgs :
2021-06-16 20:46:57 +00:00
var index = computeIndex ( msg )
let output = store . put ( index , msg , pubsubTopic )
2021-04-28 00:14:55 +00:00
check output . isOk
2021-06-16 20:46:57 +00:00
indexes . add ( index )
2020-12-21 11:45:07 +00:00
2021-06-16 20:46:57 +00:00
# flags for version
2021-05-05 23:14:50 +00:00
var v0Flag , v1Flag , vMaxFlag : bool = false
2021-06-16 20:46:57 +00:00
# flags for sender timestamp
var t1Flag , t2Flag , t3Flag : bool = false
# flags for receiver timestamp
var rt1Flag , rt2Flag , rt3Flag : bool = false
2021-07-14 18:17:36 +00:00
# flags for message/pubsubTopic (default true)
var msgFlag , psTopicFlag = true
2021-06-16 20:46:57 +00:00
2020-12-21 11:45:07 +00:00
var responseCount = 0
2022-02-17 15:26:49 +00:00
proc data ( receiverTimestamp : Timestamp , msg : WakuMessage , psTopic : string ) {. raises : [ Defect ] . } =
2020-12-21 11:45:07 +00:00
responseCount + = 1
2021-07-14 18:17:36 +00:00
# Note: cannot use `check` within `{.raises: [Defect].}` block:
# @TODO: /Nim/lib/pure/unittest.nim(577, 16) Error: can raise an unlisted exception: Exception
if msg notin msgs :
msgFlag = false
if psTopic ! = pubsubTopic :
psTopicFlag = false
2021-06-16 20:46:57 +00:00
2021-05-05 23:14:50 +00:00
# check the correct retrieval of versions
if msg . version = = uint32 ( 0 ) : v0Flag = true
if msg . version = = uint32 ( 1 ) : v1Flag = true
# high(uint32) is the largest value that fits in uint32, this is to make sure there is no overflow in the storage
if msg . version = = high ( uint32 ) : vMaxFlag = true
2021-06-16 20:46:57 +00:00
# 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 correct retrieval of receiver timestamps
2021-07-08 00:18:43 +00:00
if receiverTimestamp = = indexes [ 0 ] . receiverTime : rt1Flag = true
if receiverTimestamp = = indexes [ 1 ] . receiverTime : rt2Flag = true
if receiverTimestamp = = indexes [ 2 ] . receiverTime : rt3Flag = true
2021-06-16 20:46:57 +00:00
2021-05-05 23:14:50 +00:00
2020-12-21 11:45:07 +00:00
let res = store . getAll ( data )
check :
res . isErr = = false
responseCount = = 3
2021-06-16 20:46:57 +00:00
# check version
2021-05-05 23:14:50 +00:00
v0Flag = = true
v1Flag = = true
vMaxFlag = = true
2021-06-16 20:46:57 +00:00
# check sender timestamp
t1Flag = = true
t2Flag = = true
t3Flag = = true
# check receiver timestamp
rt1Flag = = true
rt2Flag = = true
rt3Flag = = true
2021-07-14 18:17:36 +00:00
# check messages and pubsubTopic
msgFlag = = true
psTopicFlag = = true
2021-06-16 20:46:57 +00:00
test " set and get user version " :
let
database = SqliteDatabase . init ( " " , inMemory = true ) [ ]
store = WakuMessageStore . init ( database ) [ ]
defer : store . close ( )
let res = database . setUserVersion ( 5 )
check res . isErr = = false
2021-05-05 23:14:50 +00:00
2021-06-16 20:46:57 +00:00
let ver = database . getUserVersion ( )
check :
ver . isErr = = false
ver . value = = 5
test " migration " :
let
database = SqliteDatabase . init ( " " , inMemory = true ) [ ]
store = WakuMessageStore . init ( database ) [ ]
defer : store . close ( )
2021-05-05 23:14:50 +00:00
2021-06-16 20:46:57 +00:00
template sourceDir : string = currentSourcePath . rsplit ( DirSep , 1 ) [ 0 ]
let migrationPath = sourceDir
let res = database . migrate ( migrationPath , 10 )
check :
res . isErr = = false
let ver = database . getUserVersion ( )
check :
ver . isErr = = false
ver . value = = 10
2021-11-03 11:27:13 +00:00
2022-03-28 19:09:46 +00:00
test " number of messages retrieved by getAll is bounded by storeCapacity " :
2021-11-03 11:27:13 +00:00
let
2022-03-28 19:09:46 +00:00
database = SqliteDatabase . init ( " " , inMemory = true ) [ ]
contentTopic = ContentTopic ( " /waku/2/default-content/proto " )
pubsubTopic = " /waku/2/default-waku/proto "
capacity = 10
store = WakuMessageStore . init ( database , capacity ) [ ]
defer : store . close ( )
for i in 1 .. capacity :
let
msg = WakuMessage ( payload : @ [ byte i ] , contentTopic : contentTopic , version : uint32 ( 0 ) , timestamp : Timestamp ( i ) )
index = computeIndex ( msg )
output = store . put ( index , msg , pubsubTopic )
check output . isOk
var
responseCount = 0
lastMessageTimestamp = Timestamp ( 0 )
proc data ( receiverTimestamp : Timestamp , msg : WakuMessage , psTopic : string ) {. raises : [ Defect ] . } =
responseCount + = 1
lastMessageTimestamp = msg . timestamp
# Test limited getAll function when store is at capacity
let resMax = store . getAll ( data )
check :
resMax . isOk
responseCount = = capacity # We retrieved all items
lastMessageTimestamp = = Timestamp ( capacity ) # Returned rows were ordered correctly # TODO: not representative because the timestamp only has second resolution
test " DB store capacity " :
let
database = SqliteDatabase . init ( " " , inMemory = true ) [ ]
contentTopic = ContentTopic ( " /waku/2/default-content/proto " )
pubsubTopic = " /waku/2/default-waku/proto "
capacity = 100
overload = 65
store = WakuMessageStore . init ( database , capacity ) [ ]
defer : store . close ( )
for i in 1 .. capacity + overload :
let
msg = WakuMessage ( payload : ( $ i ) . toBytes ( ) , contentTopic : contentTopic , version : uint32 ( 0 ) , timestamp : Timestamp ( i ) )
index = computeIndex ( msg )
output = store . put ( index , msg , pubsubTopic )
check output . isOk
# count messages in DB
var numMessages : int64
proc handler ( s : ptr sqlite3_stmt ) =
numMessages = sqlite3_column_int64 ( s , 0 )
let countQuery = " SELECT COUNT(*) FROM message " # the table name is set in a const in waku_message_store
discard store . database . query ( countQuery , handler )
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 waku_message_store
numMessages = = 120
2021-11-03 11:27:13 +00:00