Upstream RNS 1.2.4 (2026-05-07) announces it is "probably the last release that is also published to GitHub" — pip continues until rnpkg is complete and RNS is self-hosting. All 13 verifiers pass against 1.2.4 / 0.9.7; no wire-format, signing, or protocol behavior changed between 1.2.0 and 1.2.4, so the changes here are purely currency: - Pin tools/requirements.txt to rns==1.2.4 / lxmf==0.9.7 so the verifier stays reproducible if upstream stops mirroring to PyPI before the migration is ready. - Add an "Upstream distribution shift" watch-list to todo.md (local Reticulum node, repo destination hash, rnpkg install/upgrade commands, rsg signature verification, mirroring source citations). - Bump SPEC.md frontmatter and re-anchor ~50 line citations across Identity.py, Transport.py, Resource.py, Link.py, Reticulum.py, Packet.py, and LXMF/* (Identity.py drift was the heaviest at +13 to +31 lines; Transport.py was variable). Fix one numeric (MAX_RANDOM_BLOBS = 32 → 64) and one semantic (§6.6.3 LRPROOF MTU clamp citation pointed at the wrong location — corrected to point at the transit-relay clamp at Transport.py:1539-1556). - Update §10.4 decompression-bomb hazard to note upstream's 1.1.9 cap adoption, with citations to Resource.py:686-691 and Buffer.py:95-97 plus a "do not use one-shot bz2.decompress()" warning. - Re-anchor 11 flows/ files (version pins + ~30 line citations). - Bump version labels in tools/README.md, test-vectors/README.md, and 4 verifier docstrings + 2 hardcoded print strings. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
6.1 KiB
Flow: receive a propagated LXMF message (recipient pulls via /get)
The closing half of send-propagated-lxmf.md: how a recipient client retrieves messages that were store-and-forwarded for it by a propagation node. Pinned against RNS 1.2.4 / LXMF 0.9.7; cross-references ../SPEC.md §5.8 (propagation protocol), §11 (REQUEST/RESPONSE).
This is the inverse-side flow that turns "the message was queued at a propagation node" (send-propagated-lxmf.md step 9) into "the message arrives in the recipient's inbox".
Preconditions
- Recipient has discovered at least one propagation node via its
lxmf.propagationannounce. The recipient'sLXMRouter.outbound_propagation_noderecords which one to use. - Recipient has a path to the propagation node in
Transport.path_table.
Sequence
1. Recipient initiates retrieval
LXMRouter.request_messages_from_propagation_node(identity, max_messages) (LXMF/LXMRouter.py:484+). Triggered by:
- Manual user action (Sideband "Refresh inbox" button).
- Periodic background poll (every few minutes by default in long-running clients).
- An incoming
lxmf.propagationannounce from the configured PN, signalling availability.
2. Open a Link to the propagation node
If Transport.has_path(propagation_node_dest) is False, request_path first and defer (same pattern as opportunistic LXMF send). Otherwise:
self.outbound_propagation_link = RNS.Link(
propagation_node_destination,
established_callback=msg_request_established_callback,
)
(LXMF/LXMRouter.py:513). Standard Link establishment per flows/send-link-lxmf.md steps 3-4.
3. Identify on the link
Once the link is ACTIVE, the recipient calls link.identify(my_lxmf_delivery_identity) so the propagation node knows whose mail to deliver. Without this, the /get request handler returns LXMPeer.ERROR_NO_IDENTITY (per §5.8.3).
4. Listing query — /get with [None, None]
data = [None, None] # [wanted, have]
link.request("/get", data, response_callback=on_message_list)
The propagation node's message_get_request handler at LXMF/LXMRouter.py:1426-1450 walks propagation_entries for messages keyed to the requester's destination_hash and returns:
[ [transient_id_1(16), size_1(int)],
[transient_id_2(16), size_2(int)],
... ] # sorted by size ascending
For [None, None], the response after the propagation node strips its internal (transient_id, size) tuples to just transient_ids:
return [transient_id_1, transient_id_2, ...]
5. Recipient picks which messages to fetch
Application logic decides. Common heuristic: fetch all transient_ids the recipient doesn't already have stored locally, prioritising smaller messages first. Build:
data = [wanted_ids, have_ids, transfer_limit_kb]
link.request("/get", data, response_callback=on_message_batch)
wanted_ids— list of 16-byte transient_ids to deliver.have_ids— list of 16-byte transient_ids the recipient already has stored locally; the propagation node deletes these from its store as a side effect (§5.8.3 "ack and purge").transfer_limit_kb— optional cap on total bytes the recipient is willing to receive in one batch.
6. Propagation node returns a message bundle
message_get_request builds response_messages = [] of the matching LXMF bodies, packs them as:
data = msgpack.packb([time.time(), [lxmf_data_1, lxmf_data_2, ...]])
Returns this as a §11 RESPONSE. If the bundle fits in link.mdu it's a single Link DATA packet; otherwise it's a Resource (per flows/send-resource.md).
7. Recipient unpacks the bundle and processes each message
The recipient's propagation_resource_concluded handler (or its single-packet equivalent) at LXMF/LXMRouter.py:2200+ walks the bundle:
data = msgpack.unpackb(resource.data.read())
remote_timebase = data[0]
messages = data[1]
for lxmf_data in messages:
self.lxmf_delivery(lxmf_data, destination_type=SINGLE)
lxmf_delivery is the same path used for opportunistic and direct receive (flows/receive-opportunistic-lxmf.md step 11+) — it calls LXMessage.unpack_from_bytes, validates the signature against the sender's known identity, runs ticket / stamp / dedup checks, and fires the application's delivery callback. The LXMF body bytes are identical regardless of how they arrived — opportunistic, direct over a Link, or propagated. The propagation node never touched the encrypted body.
8. (Optional) Acknowledge and purge
In the next /get request, the recipient passes the just-fetched transient_ids in the have_ids slot per step 5. The propagation node deletes those entries on receipt. This caps the propagation node's storage growth — without it, every message would accumulate forever until the operator manually purged.
A clean-room recipient that doesn't implement the purge handshake works correctly (gets messages delivered) but contributes to long-term storage growth on shared propagation nodes. Implement the purge as a courtesy.
9. Link teardown or reuse
After the bundle is processed, the recipient either tears down the link (link.teardown() per §6.7) or keeps it for another /get round if more messages are expected. Most clients tear down after each successful batch — the propagation node's lxmf.propagation destination is ALLOW_ALL for /offer and /get so reopening a fresh link has no auth cost.
Source map
| Step | File | Function / line |
|---|---|---|
| 1 | LXMF/LXMRouter.py |
request_messages_from_propagation_node, line 485 |
| 2 | LXMF/LXMRouter.py |
outbound link establishment, line 505-520 |
| 3 | RNS/Link.py |
Link.identify, line ~1010 |
| 4-6 | LXMF/LXMRouter.py |
message_get_request handler, line 1427-1500 |
| 7 | LXMF/LXMRouter.py |
propagation_resource_concluded, line 2194+ |
| 7 | LXMF/LXMRouter.py |
lxmf_delivery, line 1732 |
| 8 | LXMF/LXMRouter.py |
purge via /get have_ids slot, line 1453-1465 |
| 9 | RNS/Link.py |
teardown, line 699 |