mirror of
https://github.com/logos-messaging/logos-messaging-nim.git
synced 2026-06-30 21:39:30 +00:00
* Reshape per-layer API into api/ folders and thin the FFI over them
Each layer now separates its constructible core from its public surface:
- core module (waku.nim / messaging_client.nim /
reliable_channel_manager.nim): the type plus new/start/stop and the
private construction helpers.
- api/ folder: one module per differentiated set of operations
(waku: topics/relay/filter/lightpush/store/peer_manager/discovery/
debug/health) plus an events surface.
The waku api is reshaped to be the complete operation surface the C
bindings need, so the library no longer reaches into node internals:
relayPublish returns the message hash, relaySubscribe takes an optional
handler, filter/lightpush auto-select the service peer, connectedPeersInfo
returns structured data, pingPeer honours the timeout, plus
relayNumPeersInMesh / relayNumConnectedPeers / isOnline. library/ is now a
thin C-ABI shim: each {.ffi.} proc only marshals cstring/JSON/callbacks and
delegates to ctx.myLib[].waku.<op> (or messagingClient.<op>).
app_callbacks re-exports the modules defining its handler types, which the
included FFI files previously relied on by leakage.
Events move next to the surface that owns them, with each dependency kept
pointing the right way:
- waku/events/ relocated under waku/api/events/.
- channel events live in channels/api/events.nim.
- the four messaging-level message events move to messaging/api/events;
MessageSeenEvent stays in waku because it is emitted by waku core, so
moving it would make waku depend on the messaging layer.
- delivery_events renamed to filter_subscribe_events to match the
OnFilterSubscribe/Unsubscribe events it actually declares.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Add reliable-channel FFI ops + events (nim-ffi v0.1.3)
Expose the reliable-channel layer through the v0.1.3 FFI:
- channel_create / channel_send / channel_close call the
ReliableChannelManager api (createReliableChannel / send / closeChannel),
marshalling channel id + base64 payload + ephemeral by hand
- channel message received / sent / errored are surfaced by listening to the
channel-layer broker events in start_node and forwarding them through
callEventCallback (received payload base64-encoded), dropped in stop_node
Stays on nim-ffi v0.1.3 (no typed/CBOR rewrite).
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Expose reliable-channel ops in the stable C header (#3851)
The library already ships as a single .so with a tiered header surface
(liblogosdelivery.h = stable Messaging/Reliable-Channels, liblogosdelivery_kernel.h
= advanced Kernel). Per that tiering, the reliable-channel ops belong on the
stable surface, so declare channel_create / channel_send / channel_close in
liblogosdelivery.h and document the channel lifecycle events delivered through
the event callback.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Graft PR#3975 interface layer onto decomposed foundation (events deduped)
Add IKernel/IMessagingClient/IReliableChannelManager/ILogosDelivery interface
classes under logos_delivery/api/. The EventBroker types PR#3975 hoisted into
these files already exist in PR#3989's decomposed */api/events/ modules, so the
interface files re-export those modules instead of redefining the types
(avoids 8 duplicate EventBroker definitions). api/types.nim kept at the
foundation version (ChannelId stays in channels/types.nim, which the decomposed
modules import).
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Wire impl classes to interfaces (inherit; relocate SendHandler)
- Waku : IKernel, MessagingClient : IMessagingClient,
ReliableChannelManager : IReliableChannelManager.
- The operation procs already live in PR#3989's decomposed */api/ modules and
stay as plain procs (nothing dispatches through the interface types, so no
method-ization is needed).
- SendHandler now lives in reliable_channel_manager_api.nim (its PR#3975 home);
removed the duplicate from reliable_channel.nim, which re-exports the
interface module so channels/api/{channel_lifecycle,send} still see it.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Wire LogosDelivery to ILogosDelivery orchestrator interface
LogosDelivery : ILogosDelivery; start/stop/isOnline become method overrides.
Peripheral PR#3975 edits (lightpush/store clients, self_req_handlers,
statistics) are import-reorg artifacts of deleting waku/utils/requests.nim,
which the decomposed structure keeps -- so they are intentionally not ported.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Dedup EventConnectionStatusChange (re-export from health_events)
9th duplicate EventBroker type: defined in both logos_delivery_api.nim and the
decomposed waku/api/events/health_events.nim. The interface file now re-exports
it. liblogosdelivery builds clean.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* Move events back into interface-class source files (restore #3975 placement)
Reverses the earlier dedup-by-re-export: event TYPE definitions now live in the
interface classes, and the emptied decomposed event files are removed.
- MessageSeenEvent -> logos_delivery/api/kernel_api.nim
- Message{Sent,Error,Propagated,Received}Event -> api/messaging_client_api.nim
- ChannelMessage{Received,Sent,Error}Event -> api/reliable_channel_manager_api.nim
- EventConnectionStatusChange -> api/logos_delivery_api.nim
Deleted (became empty after the move):
- logos_delivery/waku/api/events/message_events.nim
- logos_delivery/messaging/api/events.nim
- logos_delivery/channels/api/events.nim
health_events.nim keeps its two remaining events (content/shard topic health).
Rewiring: each layer re-exports its interface module (waku->kernel_api,
messaging_client->messaging_client_api, reliable_channel->reliable_channel_manager_api,
which also re-exports messaging_client_api). Deep emitters/listeners
(subscription_manager, waku_node, waku_node/relay, node_health_monitor,
recv_service, send_service) import the owning interface module directly.
kernel_api stays below node level (types/topics/message/store-common) so the
node->kernel_api imports are acyclic. liblogosdelivery builds.
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
* nph formatting
---------
Co-authored-by: Ivan FB <ivansete@status.im>
Co-authored-by: Claude Opus 4.8 <noreply@anthropic.com>
252 lines
9.0 KiB
Nim
252 lines
9.0 KiB
Nim
{.used.}
|
|
|
|
import std/[options, sequtils, net, sets]
|
|
import chronos, testutils/unittests, stew/byteutils
|
|
import libp2p/[peerid, peerinfo, crypto/crypto]
|
|
import brokers/broker_context
|
|
import ../testlib/[common, wakucore, wakunode, testasync]
|
|
import ../waku_archive/archive_utils
|
|
import logos_delivery/messaging/messaging_client
|
|
import logos_delivery/messaging/delivery_service/recv_service
|
|
|
|
import
|
|
logos_delivery,
|
|
logos_delivery/waku/[
|
|
waku_node,
|
|
waku_core,
|
|
api/events/health_events,
|
|
waku_relay/protocol,
|
|
waku_archive,
|
|
waku_archive/common as archive_common,
|
|
]
|
|
import logos_delivery/waku/factory/waku_conf
|
|
import tools/confutils/cli_args
|
|
|
|
const TestTimeout = chronos.seconds(60)
|
|
|
|
type ReceiveEventListenerManager = ref object
|
|
brokerCtx: BrokerContext
|
|
receivedListener: MessageReceivedEventListener
|
|
receivedEvent: AsyncEvent
|
|
receivedMessages: seq[WakuMessage]
|
|
targetCount: int
|
|
|
|
proc newReceiveEventListenerManager(
|
|
brokerCtx: BrokerContext, expectedCount: int = 1
|
|
): ReceiveEventListenerManager =
|
|
let manager = ReceiveEventListenerManager(
|
|
brokerCtx: brokerCtx, receivedMessages: @[], targetCount: expectedCount
|
|
)
|
|
manager.receivedEvent = newAsyncEvent()
|
|
|
|
manager.receivedListener = MessageReceivedEvent
|
|
.listen(
|
|
brokerCtx,
|
|
proc(event: MessageReceivedEvent) {.async: (raises: []).} =
|
|
manager.receivedMessages.add(event.message)
|
|
if manager.receivedMessages.len >= manager.targetCount:
|
|
manager.receivedEvent.fire()
|
|
,
|
|
)
|
|
.expect("Failed to listen to MessageReceivedEvent")
|
|
|
|
return manager
|
|
|
|
proc teardown(manager: ReceiveEventListenerManager) {.async.} =
|
|
await MessageReceivedEvent.dropListener(manager.brokerCtx, manager.receivedListener)
|
|
|
|
proc waitForEvents(
|
|
manager: ReceiveEventListenerManager, timeout: Duration
|
|
): Future[bool] {.async.} =
|
|
return await manager.receivedEvent.wait().withTimeout(timeout)
|
|
|
|
proc waitForConnectionStatus(
|
|
brokerCtx: BrokerContext, expected: ConnectionStatus
|
|
) {.async.} =
|
|
## Completes when the node reports `expected`.
|
|
var future = newFuture[void]("waitForConnectionStatus")
|
|
|
|
let handler: EventConnectionStatusChangeListenerProc = proc(
|
|
e: EventConnectionStatusChange
|
|
) {.async: (raises: []), gcsafe.} =
|
|
if not future.finished and e.connectionStatus == expected:
|
|
future.complete()
|
|
|
|
let handle = EventConnectionStatusChange.listen(brokerCtx, handler).valueOr:
|
|
raiseAssert error
|
|
|
|
try:
|
|
if not await future.withTimeout(TestTimeout):
|
|
raiseAssert "Timeout waiting for status: " & $expected
|
|
finally:
|
|
await EventConnectionStatusChange.dropListener(brokerCtx, handle)
|
|
|
|
proc createApiNodeConf(numShards: uint16 = 1): WakuNodeConf =
|
|
var conf = defaultWakuNodeConf().valueOr:
|
|
raiseAssert error
|
|
conf.mode = cli_args.WakuMode.Core
|
|
conf.listenAddress = parseIpAddress("0.0.0.0")
|
|
conf.tcpPort = Port(0)
|
|
conf.discv5UdpPort = Port(0)
|
|
conf.clusterId = some(3'u16)
|
|
conf.numShardsInNetwork = numShards
|
|
conf.reliabilityEnabled = some(true)
|
|
conf.rest = false
|
|
result = conf
|
|
|
|
type TestNetwork = ref object
|
|
storeNode: WakuNode
|
|
publisher: WakuNode
|
|
subscriber: LogosDelivery
|
|
storeNodePeerInfo: RemotePeerInfo
|
|
missedPayload: seq[byte]
|
|
|
|
proc setupNetwork(testTopic: ContentTopic): Future[TestNetwork] {.async.} =
|
|
## Returns a started subscriber subscribed to `testTopic` but not yet connected
|
|
## to the store, with a message sitting in the store it never saw live.
|
|
const numShards: uint16 = 1
|
|
let shard = PubsubTopic("/waku/2/rs/3/0")
|
|
|
|
proc dummyHandler(topic: PubsubTopic, msg: WakuMessage) {.async, gcsafe.} =
|
|
discard
|
|
|
|
# store node: archive + store + relay, subscribed to the shard
|
|
var storeNode: WakuNode
|
|
lockNewGlobalBrokerContext:
|
|
storeNode =
|
|
newTestWakuNode(generateSecp256k1Key(), parseIpAddress("0.0.0.0"), Port(0))
|
|
storeNode.mountMetadata(3, toSeq(0'u16 ..< numShards)).expect(
|
|
"Failed to mount metadata on storeNode"
|
|
)
|
|
(await storeNode.mountRelay()).expect("Failed to mount relay on storeNode")
|
|
storeNode.mountArchive(newSqliteArchiveDriver()).expect("Failed to mount archive")
|
|
await storeNode.mountStore()
|
|
await storeNode.mountLibp2pPing()
|
|
await storeNode.start()
|
|
storeNode.subscribe((kind: PubsubSub, topic: shard), dummyHandler).expect(
|
|
"Failed to sub storeNode"
|
|
)
|
|
|
|
let storeNodePeerInfo = storeNode.peerInfo.toRemotePeerInfo()
|
|
|
|
# publisher: relay, connected to the store so its messages get archived
|
|
var publisher: WakuNode
|
|
lockNewGlobalBrokerContext:
|
|
publisher =
|
|
newTestWakuNode(generateSecp256k1Key(), parseIpAddress("0.0.0.0"), Port(0))
|
|
publisher.mountMetadata(3, toSeq(0'u16 ..< numShards)).expect(
|
|
"Failed to mount metadata on publisher"
|
|
)
|
|
(await publisher.mountRelay()).expect("Failed to mount relay on publisher")
|
|
await publisher.mountLibp2pPing()
|
|
await publisher.start()
|
|
publisher.subscribe((kind: PubsubSub, topic: shard), dummyHandler).expect(
|
|
"Failed to sub publisher"
|
|
)
|
|
|
|
await publisher.connectToNodes(@[storeNodePeerInfo])
|
|
|
|
var meshFormed = false
|
|
for _ in 0 ..< 50:
|
|
if publisher.wakuRelay.getNumPeersInMesh(shard).valueOr(0) > 0:
|
|
meshFormed = true
|
|
break
|
|
await sleepAsync(100.milliseconds)
|
|
if not meshFormed:
|
|
raiseAssert "publisher<->store relay mesh did not form in time"
|
|
|
|
# subscriber: created before the publish so the message timestamp lands after
|
|
# its RecvService startTimeToCheck watermark
|
|
var subscriber: LogosDelivery
|
|
lockNewGlobalBrokerContext:
|
|
subscriber = (await LogosDelivery.new(createApiNodeConf(numShards))).expect(
|
|
"Failed to create subscriber"
|
|
)
|
|
(await subscriber.start()).expect("Failed to start subscriber")
|
|
|
|
# publish while the subscriber is offline: the message reaches the archive but
|
|
# the subscriber never sees it via live relay
|
|
let missedPayload = "This message was missed".toBytes()
|
|
let missedMsg = WakuMessage(
|
|
payload: missedPayload, contentTopic: testTopic, version: 0, timestamp: now()
|
|
)
|
|
discard (await publisher.publish(some(shard), missedMsg)).expect(
|
|
"Publish missed msg failed"
|
|
)
|
|
|
|
block waitArchive:
|
|
for _ in 0 ..< 50:
|
|
let query = archive_common.ArchiveQuery(
|
|
includeData: false, contentTopics: @[testTopic], pubsubTopic: some(shard)
|
|
)
|
|
let res = await storeNode.wakuArchive.findMessages(query)
|
|
if res.isOk() and res.get().hashes.len > 0:
|
|
break waitArchive
|
|
await sleepAsync(100.milliseconds)
|
|
raiseAssert "Message was not archived in time"
|
|
|
|
# subscribe to the content topic; with no peers yet the subscriber stays offline
|
|
(await subscriber.messagingClient.subscribe(testTopic)).expect("Failed to subscribe")
|
|
|
|
return TestNetwork(
|
|
storeNode: storeNode,
|
|
publisher: publisher,
|
|
subscriber: subscriber,
|
|
storeNodePeerInfo: storeNodePeerInfo,
|
|
missedPayload: missedPayload,
|
|
)
|
|
|
|
proc teardown(net: TestNetwork) {.async.} =
|
|
if not isNil(net.subscriber):
|
|
(await net.subscriber.stop()).expect("Failed to stop subscriber")
|
|
net.subscriber = nil
|
|
if not isNil(net.publisher):
|
|
await net.publisher.stop()
|
|
net.publisher = nil
|
|
if not isNil(net.storeNode):
|
|
await net.storeNode.stop()
|
|
net.storeNode = nil
|
|
|
|
suite "Messaging API, Receive Service (store recovery)":
|
|
asyncTest "recv_service delivers store-recovered messages via MessageReceivedEvent":
|
|
## Regression: a message archived before the subscriber connects is recovered
|
|
## by an explicit checkStore() and delivered via MessageReceivedEvent.
|
|
let net = await setupNetwork(ContentTopic("/waku/2/recv-test/proto"))
|
|
defer:
|
|
await net.teardown()
|
|
|
|
let eventManager = newReceiveEventListenerManager(net.subscriber.waku.brokerCtx, 1)
|
|
defer:
|
|
await eventManager.teardown()
|
|
|
|
await net.subscriber.waku.node.connectToNodes(@[net.storeNodePeerInfo])
|
|
await net.subscriber.messagingClient.recvService.checkStore()
|
|
|
|
check await eventManager.waitForEvents(TestTimeout)
|
|
check eventManager.receivedMessages.len == 1
|
|
if eventManager.receivedMessages.len > 0:
|
|
check eventManager.receivedMessages[0].payload == net.missedPayload
|
|
|
|
asyncTest "recv_service backfills missed messages when it comes back online":
|
|
## Connecting a peer brings the subscriber online, firing the backfill that
|
|
## recovers a message archived while it was offline.
|
|
let net = await setupNetwork(ContentTopic("/waku/2/recv-reconnect-test/proto"))
|
|
defer:
|
|
await net.teardown()
|
|
|
|
let eventManager = newReceiveEventListenerManager(net.subscriber.waku.brokerCtx, 1)
|
|
defer:
|
|
await eventManager.teardown()
|
|
|
|
# sync on coming online (the transition that fires the backfill) before asserting
|
|
let onlineFut = waitForConnectionStatus(
|
|
net.subscriber.waku.brokerCtx, ConnectionStatus.PartiallyConnected
|
|
)
|
|
await net.subscriber.waku.node.connectToNodes(@[net.storeNodePeerInfo])
|
|
await onlineFut
|
|
|
|
check await eventManager.waitForEvents(TestTimeout)
|
|
check eventManager.receivedMessages.len == 1
|
|
if eventManager.receivedMessages.len > 0:
|
|
check eventManager.receivedMessages[0].payload == net.missedPayload
|