2023-05-31 08:28:48 +00:00
|
|
|
{.used.}
|
|
|
|
|
|
|
|
import
|
|
|
|
std/[sequtils,times,options],
|
|
|
|
testutils/unittests,
|
|
|
|
chronos
|
|
|
|
import
|
2023-08-09 17:11:50 +00:00
|
|
|
../../../waku/waku_archive,
|
|
|
|
../../../waku/waku_archive/driver/postgres_driver,
|
|
|
|
../../../waku/waku_core,
|
2023-11-22 16:32:56 +00:00
|
|
|
../../../waku/waku_core/message/digest,
|
2023-05-31 08:28:48 +00:00
|
|
|
../testlib/wakucore
|
|
|
|
|
|
|
|
proc now():int64 = getTime().toUnix()
|
|
|
|
|
|
|
|
proc computeTestCursor(pubsubTopic: PubsubTopic,
|
|
|
|
message: WakuMessage):
|
|
|
|
ArchiveCursor =
|
|
|
|
ArchiveCursor(
|
|
|
|
pubsubTopic: pubsubTopic,
|
|
|
|
senderTime: message.timestamp,
|
|
|
|
storeTime: message.timestamp,
|
2023-11-08 00:41:23 +00:00
|
|
|
digest: computeDigest(message)
|
2023-05-31 08:28:48 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
suite "Postgres driver":
|
|
|
|
|
|
|
|
const storeMessageDbUrl = "postgres://postgres:test123@localhost:5432/postgres"
|
|
|
|
|
|
|
|
asyncTest "Asynchronous queries":
|
2023-06-07 08:08:43 +00:00
|
|
|
let driverRes = PostgresDriver.new(dbUrl = storeMessageDbUrl,
|
|
|
|
maxConnections = 100)
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
assert driverRes.isOk(), driverRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let driver = driverRes.value
|
|
|
|
discard await driver.reset()
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
var futures = newSeq[Future[ArchiveDriverResult[void]]](0)
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let beforeSleep = now()
|
2023-06-07 08:08:43 +00:00
|
|
|
for _ in 1 .. 100:
|
|
|
|
futures.add(driver.sleep(1))
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
await allFutures(futures)
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let diff = now() - beforeSleep
|
|
|
|
# Actually, the diff randomly goes between 1 and 2 seconds.
|
|
|
|
# although in theory it should spend 1s because we establish 100
|
|
|
|
# connections and we spawn 100 tasks that spend ~1s each.
|
|
|
|
require diff < 20
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
(await driver.close()).expect("driver to close")
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
asyncTest "Init database":
|
2023-05-31 08:28:48 +00:00
|
|
|
let driverRes = PostgresDriver.new(storeMessageDbUrl)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert driverRes.isOk(), driverRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let driver = driverRes.value
|
|
|
|
discard await driver.reset()
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let initRes = await driver.init()
|
|
|
|
assert initRes.isOk(), initRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
(await driver.close()).expect("driver to close")
|
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
asyncTest "Insert a message":
|
2023-05-31 08:28:48 +00:00
|
|
|
const contentTopic = "test-content-topic"
|
|
|
|
|
|
|
|
let driverRes = PostgresDriver.new(storeMessageDbUrl)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert driverRes.isOk(), driverRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let driver = driverRes.get()
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
discard await driver.reset()
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let initRes = await driver.init()
|
|
|
|
assert initRes.isOk(), initRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let msg = fakeWakuMessage(contentTopic=contentTopic)
|
|
|
|
|
2023-11-08 00:41:23 +00:00
|
|
|
let computedDigest = computeDigest(msg)
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-11-22 16:32:56 +00:00
|
|
|
let putRes = await driver.put(DefaultPubsubTopic, msg, computedDigest, computeMessageHash(DefaultPubsubTopic, msg), msg.timestamp)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert putRes.isOk(), putRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let storedMsg = (await driver.getAllMessages()).tryGet()
|
|
|
|
require:
|
|
|
|
storedMsg.len == 1
|
|
|
|
storedMsg.all do (item: auto) -> bool:
|
2023-10-24 14:05:39 +00:00
|
|
|
let (pubsubTopic, actualMsg, digest, storeTimestamp) = item
|
2023-05-31 08:28:48 +00:00
|
|
|
actualMsg.contentTopic == contentTopic and
|
|
|
|
pubsubTopic == DefaultPubsubTopic and
|
|
|
|
toHex(computedDigest.data) == toHex(digest) and
|
|
|
|
toHex(actualMsg.payload) == toHex(msg.payload)
|
|
|
|
|
|
|
|
(await driver.close()).expect("driver to close")
|
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
asyncTest "Insert and query message":
|
2023-05-31 08:28:48 +00:00
|
|
|
const contentTopic1 = "test-content-topic-1"
|
|
|
|
const contentTopic2 = "test-content-topic-2"
|
|
|
|
const pubsubTopic1 = "pubsubtopic-1"
|
|
|
|
const pubsubTopic2 = "pubsubtopic-2"
|
|
|
|
|
|
|
|
let driverRes = PostgresDriver.new(storeMessageDbUrl)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert driverRes.isOk(), driverRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let driver = driverRes.value
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
discard await driver.reset()
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let initRes = await driver.init()
|
|
|
|
assert initRes.isOk(), initRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let msg1 = fakeWakuMessage(contentTopic=contentTopic1)
|
|
|
|
|
2023-11-22 16:32:56 +00:00
|
|
|
var putRes = await driver.put(pubsubTopic1, msg1, computeDigest(msg1), computeMessageHash(pubsubTopic1, msg1), msg1.timestamp)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert putRes.isOk(), putRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let msg2 = fakeWakuMessage(contentTopic=contentTopic2)
|
|
|
|
|
2023-11-22 16:32:56 +00:00
|
|
|
putRes = await driver.put(pubsubTopic2, msg2, computeDigest(msg2), computeMessageHash(pubsubTopic2, msg2), msg2.timestamp)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert putRes.isOk(), putRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let countMessagesRes = await driver.getMessagesCount()
|
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
require countMessagesRes.isOk() and countMessagesRes.get() == 2
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
var messagesRes = await driver.getMessages(contentTopic = @[contentTopic1])
|
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
require messagesRes.isOk()
|
|
|
|
require messagesRes.get().len == 1
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
# Get both content topics, check ordering
|
|
|
|
messagesRes = await driver.getMessages(contentTopic = @[contentTopic1,
|
|
|
|
contentTopic2])
|
2023-06-07 08:08:43 +00:00
|
|
|
assert messagesRes.isOk(), messagesRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
require:
|
|
|
|
messagesRes.get().len == 2 and
|
2023-06-07 08:08:43 +00:00
|
|
|
messagesRes.get()[0][1].contentTopic == contentTopic1
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
# Descending order
|
|
|
|
messagesRes = await driver.getMessages(contentTopic = @[contentTopic1,
|
|
|
|
contentTopic2],
|
|
|
|
ascendingOrder = false)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert messagesRes.isOk(), messagesRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
require:
|
|
|
|
messagesRes.get().len == 2 and
|
2023-06-07 08:08:43 +00:00
|
|
|
messagesRes.get()[0][1].contentTopic == contentTopic2
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
# cursor
|
|
|
|
# Get both content topics
|
|
|
|
messagesRes =
|
|
|
|
await driver.getMessages(contentTopic = @[contentTopic1,
|
|
|
|
contentTopic2],
|
|
|
|
cursor = some(
|
|
|
|
computeTestCursor(pubsubTopic1,
|
2023-06-09 10:42:33 +00:00
|
|
|
messagesRes.get()[1][1])))
|
2023-06-07 08:08:43 +00:00
|
|
|
require messagesRes.isOk()
|
|
|
|
require messagesRes.get().len == 1
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
# Get both content topics but one pubsub topic
|
|
|
|
messagesRes = await driver.getMessages(contentTopic = @[contentTopic1,
|
|
|
|
contentTopic2],
|
|
|
|
pubsubTopic = some(pubsubTopic1))
|
2023-06-07 08:08:43 +00:00
|
|
|
assert messagesRes.isOk(), messagesRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
require:
|
|
|
|
messagesRes.get().len == 1 and
|
2023-06-07 08:08:43 +00:00
|
|
|
messagesRes.get()[0][1].contentTopic == contentTopic1
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
# Limit
|
|
|
|
messagesRes = await driver.getMessages(contentTopic = @[contentTopic1,
|
|
|
|
contentTopic2],
|
|
|
|
maxPageSize = 1)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert messagesRes.isOk(), messagesRes.error
|
|
|
|
require messagesRes.get().len == 1
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
(await driver.close()).expect("driver to close")
|
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
asyncTest "Insert true duplicated messages":
|
2023-05-31 08:28:48 +00:00
|
|
|
# Validates that two completely equal messages can not be stored.
|
|
|
|
let driverRes = PostgresDriver.new(storeMessageDbUrl)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert driverRes.isOk(), driverRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let driver = driverRes.value
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
discard await driver.reset()
|
2023-05-31 08:28:48 +00:00
|
|
|
|
2023-06-07 08:08:43 +00:00
|
|
|
let initRes = await driver.init()
|
|
|
|
assert initRes.isOk(), initRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
let now = now()
|
|
|
|
|
|
|
|
let msg1 = fakeWakuMessage(ts = now)
|
|
|
|
let msg2 = fakeWakuMessage(ts = now)
|
|
|
|
|
|
|
|
var putRes = await driver.put(DefaultPubsubTopic,
|
2023-11-22 16:32:56 +00:00
|
|
|
msg1, computeDigest(msg1), computeMessageHash(DefaultPubsubTopic, msg1), msg1.timestamp)
|
2023-06-07 08:08:43 +00:00
|
|
|
assert putRes.isOk(), putRes.error
|
2023-05-31 08:28:48 +00:00
|
|
|
|
|
|
|
putRes = await driver.put(DefaultPubsubTopic,
|
2023-11-22 16:32:56 +00:00
|
|
|
msg2, computeDigest(msg2), computeMessageHash(DefaultPubsubTopic, msg2), msg2.timestamp)
|
2023-06-07 08:08:43 +00:00
|
|
|
require not putRes.isOk()
|
2023-11-07 12:38:37 +00:00
|
|
|
|
|
|
|
(await driver.close()).expect("driver to close")
|