2024-05-21 21:03:33 +00:00
|
|
|
import
|
2024-08-21 12:54:18 +00:00
|
|
|
std/[strformat, sysrand, random, sequtils],
|
2024-05-21 21:03:33 +00:00
|
|
|
system/ansi_c,
|
|
|
|
chronicles,
|
|
|
|
chronos,
|
2024-08-21 12:54:18 +00:00
|
|
|
chronos/timer as chtimer,
|
2024-05-21 21:03:33 +00:00
|
|
|
stew/byteutils,
|
2024-07-09 11:14:28 +00:00
|
|
|
results,
|
2024-05-21 21:03:33 +00:00
|
|
|
json_serialization as js
|
|
|
|
import
|
2024-08-21 12:54:18 +00:00
|
|
|
waku/[
|
|
|
|
common/logging,
|
|
|
|
waku_node,
|
|
|
|
node/peer_manager,
|
|
|
|
waku_core,
|
|
|
|
waku_lightpush/client,
|
|
|
|
common/utils/parse_size_units,
|
|
|
|
],
|
2024-05-21 21:03:33 +00:00
|
|
|
./tester_config,
|
2024-09-04 06:35:51 +00:00
|
|
|
./tester_message,
|
|
|
|
./lpt_metrics
|
2024-05-21 21:03:33 +00:00
|
|
|
|
2024-08-21 12:54:18 +00:00
|
|
|
randomize()
|
|
|
|
|
|
|
|
type SizeRange* = tuple[min: uint64, max: uint64]
|
|
|
|
|
|
|
|
var RANDOM_PALYLOAD {.threadvar.}: seq[byte]
|
|
|
|
RANDOM_PALYLOAD = urandom(1024 * 1024)
|
|
|
|
# 1MiB of random payload to be used to extend message
|
|
|
|
|
2024-05-21 21:03:33 +00:00
|
|
|
proc prepareMessage(
|
|
|
|
sender: string,
|
|
|
|
messageIndex, numMessages: uint32,
|
|
|
|
startedAt: TimeStamp,
|
|
|
|
prevMessageAt: var Timestamp,
|
|
|
|
contentTopic: ContentTopic,
|
2024-08-21 12:54:18 +00:00
|
|
|
size: SizeRange,
|
|
|
|
): (WakuMessage, uint64) =
|
|
|
|
var renderSize = rand(size.min .. size.max)
|
2024-05-21 21:03:33 +00:00
|
|
|
let current = getNowInNanosecondTime()
|
|
|
|
let payload = ProtocolTesterMessage(
|
|
|
|
sender: sender,
|
|
|
|
index: messageIndex,
|
|
|
|
count: numMessages,
|
|
|
|
startedAt: startedAt,
|
|
|
|
sinceStart: current - startedAt,
|
|
|
|
sincePrev: current - prevMessageAt,
|
2024-08-21 12:54:18 +00:00
|
|
|
size: renderSize,
|
2024-05-21 21:03:33 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
prevMessageAt = current
|
|
|
|
|
|
|
|
let text = js.Json.encode(payload)
|
2024-08-21 12:54:18 +00:00
|
|
|
let contentPayload = toBytes(text & " \0")
|
|
|
|
|
|
|
|
if renderSize < len(contentPayload).uint64:
|
|
|
|
renderSize = len(contentPayload).uint64
|
|
|
|
|
|
|
|
let finalPayload = concat(
|
|
|
|
contentPayload, RANDOM_PALYLOAD[0 .. renderSize - len(contentPayload).uint64]
|
|
|
|
)
|
2024-05-21 21:03:33 +00:00
|
|
|
let message = WakuMessage(
|
2024-08-21 12:54:18 +00:00
|
|
|
payload: finalPayload, # content of the message
|
2024-05-21 21:03:33 +00:00
|
|
|
contentTopic: contentTopic, # content topic to publish to
|
|
|
|
ephemeral: true, # tell store nodes to not store it
|
|
|
|
timestamp: current, # current timestamp
|
|
|
|
)
|
|
|
|
|
2024-08-21 12:54:18 +00:00
|
|
|
return (message, renderSize)
|
|
|
|
|
|
|
|
var sentMessages {.threadvar.}: OrderedTable[uint32, tuple[hash: string, relayed: bool]]
|
|
|
|
var failedToSendCause {.threadvar.}: Table[string, uint32]
|
|
|
|
var failedToSendCount {.threadvar.}: uint32
|
|
|
|
var numMessagesToSend {.threadvar.}: uint32
|
|
|
|
var messagesSent {.threadvar.}: uint32
|
|
|
|
|
|
|
|
proc reportSentMessages() {.async.} =
|
|
|
|
while true:
|
|
|
|
await sleepAsync(chtimer.seconds(60))
|
|
|
|
let report = catch:
|
|
|
|
"""*----------------------------------------*
|
|
|
|
| Expected | Sent | Failed |
|
|
|
|
|{numMessagesToSend+failedToSendCount:>11} |{messagesSent:>11} |{failedToSendCount:>11} |
|
|
|
|
*----------------------------------------*""".fmt()
|
|
|
|
|
|
|
|
if report.isErr:
|
|
|
|
echo "Error while printing statistics"
|
|
|
|
else:
|
|
|
|
echo report.get()
|
|
|
|
|
|
|
|
echo "*--------------------------------------------------------------------------------------------------*"
|
|
|
|
echo "| Failur cause | count |"
|
|
|
|
for (cause, count) in failedToSendCause.pairs:
|
|
|
|
echo fmt"|{cause:<87}|{count:>10}|"
|
|
|
|
echo "*--------------------------------------------------------------------------------------------------*"
|
|
|
|
|
|
|
|
echo "*--------------------------------------------------------------------------------------------------*"
|
|
|
|
echo "| Index | Relayed | Hash |"
|
|
|
|
for (index, info) in sentMessages.pairs:
|
|
|
|
echo fmt"|{index:>10}|{info.relayed:<9}| {info.hash:<76}|"
|
|
|
|
echo "*--------------------------------------------------------------------------------------------------*"
|
|
|
|
# evere sent message hash should logged once
|
|
|
|
sentMessages.clear()
|
2024-05-21 21:03:33 +00:00
|
|
|
|
|
|
|
proc publishMessages(
|
|
|
|
wakuNode: WakuNode,
|
|
|
|
lightpushPubsubTopic: PubsubTopic,
|
|
|
|
lightpushContentTopic: ContentTopic,
|
|
|
|
numMessages: uint32,
|
2024-08-21 12:54:18 +00:00
|
|
|
messageSizeRange: SizeRange,
|
2024-05-21 21:03:33 +00:00
|
|
|
delayMessages: Duration,
|
|
|
|
) {.async.} =
|
|
|
|
let startedAt = getNowInNanosecondTime()
|
|
|
|
var prevMessageAt = startedAt
|
2024-08-21 12:54:18 +00:00
|
|
|
var renderMsgSize = messageSizeRange
|
|
|
|
# sets some default of min max message size to avoid conflict with meaningful payload size
|
|
|
|
renderMsgSize.min = max(1024.uint64, renderMsgSize.min) # do not use less than 1KB
|
|
|
|
renderMsgSize.max = max(2048.uint64, renderMsgSize.max) # minimum of max is 2KB
|
|
|
|
renderMsgSize.min = min(renderMsgSize.min, renderMsgSize.max)
|
|
|
|
renderMsgSize.max = max(renderMsgSize.min, renderMsgSize.max)
|
2024-05-21 21:03:33 +00:00
|
|
|
|
|
|
|
let selfPeerId = $wakuNode.switch.peerInfo.peerId
|
2024-08-21 12:54:18 +00:00
|
|
|
failedToSendCount = 0
|
|
|
|
numMessagesToSend = if numMessages == 0: uint32.high else: numMessages
|
|
|
|
messagesSent = 1
|
|
|
|
|
|
|
|
while numMessagesToSend >= messagesSent:
|
|
|
|
let (message, msgSize) = prepareMessage(
|
|
|
|
selfPeerId, messagesSent, numMessagesToSend, startedAt, prevMessageAt,
|
|
|
|
lightpushContentTopic, renderMsgSize,
|
2024-05-21 21:03:33 +00:00
|
|
|
)
|
|
|
|
let wlpRes = await wakuNode.lightpushPublish(some(lightpushPubsubTopic), message)
|
|
|
|
|
2024-08-21 12:54:18 +00:00
|
|
|
let msgHash = computeMessageHash(lightpushPubsubTopic, message).to0xHex
|
|
|
|
|
2024-05-21 21:03:33 +00:00
|
|
|
if wlpRes.isOk():
|
2024-08-21 12:54:18 +00:00
|
|
|
sentMessages[messagesSent] = (hash: msgHash, relayed: true)
|
|
|
|
notice "published message using lightpush",
|
|
|
|
index = messagesSent,
|
|
|
|
count = numMessagesToSend,
|
|
|
|
size = msgSize,
|
|
|
|
pubsubTopic = lightpushPubsubTopic,
|
|
|
|
hash = msgHash
|
|
|
|
inc(messagesSent)
|
2024-09-04 06:35:51 +00:00
|
|
|
lpt_publisher_sent_messages_count.inc()
|
|
|
|
lpt_publisher_sent_bytes.inc(amount = msgSize.int64)
|
2024-05-21 21:03:33 +00:00
|
|
|
else:
|
2024-08-21 12:54:18 +00:00
|
|
|
sentMessages[messagesSent] = (hash: msgHash, relayed: false)
|
|
|
|
failedToSendCause.mgetOrPut(wlpRes.error, 1).inc()
|
|
|
|
error "failed to publish message using lightpush",
|
|
|
|
err = wlpRes.error, hash = msgHash
|
2024-05-21 21:03:33 +00:00
|
|
|
inc(failedToSendCount)
|
2024-09-04 06:35:51 +00:00
|
|
|
lpt_publisher_failed_messages_count.inc(labelValues = [wlpRes.error])
|
2024-05-21 21:03:33 +00:00
|
|
|
|
2024-08-21 12:54:18 +00:00
|
|
|
await sleepAsync(delayMessages)
|
2024-05-21 21:03:33 +00:00
|
|
|
|
2024-08-21 12:54:18 +00:00
|
|
|
waitFor reportSentMessages()
|
2024-05-21 21:03:33 +00:00
|
|
|
|
|
|
|
discard c_raise(ansi_c.SIGTERM)
|
|
|
|
|
|
|
|
proc setupAndPublish*(wakuNode: WakuNode, conf: LiteProtocolTesterConf) =
|
|
|
|
if isNil(wakuNode.wakuLightpushClient):
|
|
|
|
error "WakuFilterClient not initialized"
|
|
|
|
return
|
|
|
|
|
|
|
|
# give some time to receiver side to set up
|
2024-08-21 12:54:18 +00:00
|
|
|
let waitTillStartTesting = conf.startPublishingAfter.seconds
|
|
|
|
|
|
|
|
let parsedMinMsgSize = parseMsgSize(conf.minTestMessageSize).valueOr:
|
|
|
|
error "failed to parse 'min-test-msg-size' param: ", error = error
|
|
|
|
return
|
|
|
|
|
|
|
|
let parsedMaxMsgSize = parseMsgSize(conf.maxTestMessageSize).valueOr:
|
|
|
|
error "failed to parse 'max-test-msg-size' param: ", error = error
|
|
|
|
return
|
2024-05-21 21:03:33 +00:00
|
|
|
|
|
|
|
info "Sending test messages in", wait = waitTillStartTesting
|
|
|
|
waitFor sleepAsync(waitTillStartTesting)
|
|
|
|
|
|
|
|
info "Start sending messages to service node using lightpush"
|
|
|
|
|
2024-08-21 12:54:18 +00:00
|
|
|
sentMessages.sort(system.cmp)
|
2024-05-21 21:03:33 +00:00
|
|
|
# Start maintaining subscription
|
|
|
|
asyncSpawn publishMessages(
|
|
|
|
wakuNode,
|
|
|
|
conf.pubsubTopics[0],
|
|
|
|
conf.contentTopics[0],
|
|
|
|
conf.numMessages,
|
2024-08-21 12:54:18 +00:00
|
|
|
(min: parsedMinMsgSize, max: parsedMaxMsgSize),
|
2024-05-21 21:03:33 +00:00
|
|
|
conf.delayMessages.milliseconds,
|
|
|
|
)
|
2024-08-21 12:54:18 +00:00
|
|
|
|
|
|
|
asyncSpawn reportSentMessages()
|