2026-07-01 21:19:02 -04:00
# Lotus Chat — E2EE Investigation Runbook (KE-1 → KE-4)
> **Scope:** evidence-gathering only. Do **not** apply fixes from this document
> without a cross-system planning session (client rust-crypto ↔ Synapse ↔
> Element Call MatrixRTC). Symptom source: `LOTUS_BUGS.md` §"Encryption / E2EE"
> (KE-1..KE-4), observed live 2026-06-30 on `chat.lotusguild.org` during a
> 2-person Element Call.
>
> **Client:** Lotus Cinny fork, `matrix-js-sdk@41.6.0-rc.0`, rust-crypto.
> **Server:** Synapse `1.155.0` on **LXC 151** (`10.10.10.29`), PostgreSQL 17.9
> on **LXC 109** (`10.10.10.44`). Facts below are copy-pasteable against that
> deployment (paths/IPs from `/root/code/matrix/README.md`).
---
## 0. Deployment facts used by this runbook
From the matrix infra README (`/root/code/matrix/README.md` ):
2026-07-02 11:45:22 -04:00
| Thing | Value |
| ------------------------ | ------------------------------------------------------------- |
| Synapse host | LXC **151 ** , `10.10.10.29` (Synapse 1.155.0) |
| Synapse log | `/var/log/matrix-synapse/homeserver.log` |
| Synapse config | `/etc/matrix-synapse/homeserver.yaml` (+ `conf.d/` ) |
| Synapse HTTP | `10.10.10.29:8008` |
| PostgreSQL host | LXC **109 ** , `10.10.10.44` (PG 17.9), db `synapse` |
| synapse-admin UI | `http://10.10.10.29:8080` |
2026-07-01 21:19:02 -04:00
| LiveKit / lk-jwt / guard | LXC 151: LiveKit `:7880/:7881` , guard `:8070` , lk-jwt `:8071` |
2026-07-02 11:45:22 -04:00
| SSH path to Synapse | `ssh root@10.10.10.4` then `pct enter 151` |
| SSH path to PG | `ssh root@10.10.10.4` then `pct enter 109` |
2026-07-01 21:19:02 -04:00
**Getting a psql shell ** (run on LXC 109, or from 151 over the network):
``` bash
# On LXC 109:
sudo -u postgres psql synapse
# From LXC 151 (pg_hba allows 10.10.10.29):
psql "host=10.10.10.44 user=synapse dbname=synapse"
```
**Tailing Synapse during a call ** (on LXC 151):
``` bash
tail -F /var/log/matrix-synapse/homeserver.log | tee /tmp/lotus-call-$( date +%s) .log
```
Synapse E2EE/to-device logging is chatty at `INFO` ; if a category is silent,
temporarily raise it in `/etc/matrix-synapse/conf.d/log.yaml` (or the
`log_config` file referenced by `homeserver.yaml` ):
``` yaml
loggers :
2026-07-02 11:45:22 -04:00
synapse.rest.client.keys : { level : DEBUG }
synapse.handlers.e2e_keys : { level : DEBUG }
2026-07-01 21:19:02 -04:00
synapse.storage.databases.main.end_to_end_keys : { level : DEBUG }
2026-07-02 11:45:22 -04:00
synapse.handlers.devicemessage : { level : DEBUG } # to-device
2026-07-01 21:19:02 -04:00
```
Then `systemctl reload matrix-synapse` (reload re-reads log config without a
full restart). **Revert to `INFO` after the capture ** — DEBUG is very verbose.
---
## 1. Per-KE evidence matrix
Client greps assume Chrome/Firefox DevTools console (filter box or, better,
"Preserve log" + save-as). The **Crypto Diagnostics ** card (Settings →
Developer Tools) auto-captures every signature below into a downloadable JSON —
use it as the primary client artifact and DevTools as the raw backup.
### KE-1 — OTK upload conflict storm (root-cause candidate)
- **Console signature (grep):**
- `already exists`
- full: `POST /_matrix/client/v3/keys/upload … 400 M_UNKNOWN: One time key signed_curve25519:<id> already exists. Old key: {…} new key: {…}`
- **Capture client-side:**
- Timestamp (first occurrence + rate — "N/sec"), **device id ** , **user id ** .
- DevTools → **Network ** → filter `keys/upload` : for a failing call save the
**request body ** (the `one_time_keys` map — note the exact `signed_curve25519:<id>` )
and the **response body ** (the `Old key` / `new key` JSON). This diff is the
smoking gun: same key-id, different value ⇒ store vs server divergence.
- Whether it self-heals or loops forever (KE-1 loops).
- **Synapse log grep (LXC 151):**
```bash
grep -E "keys/upload|One time key .* already exists|OneTimeKey" \
/var/log/matrix-synapse/homeserver.log | grep "<user_id>"
` ``
- **Synapse SQL (LXC 109) — what the server thinks it holds:**
2026-07-02 11:45:22 -04:00
2026-07-01 21:19:02 -04:00
` ``sql
-- Current OTK inventory for the device (compare key_id set against the
-- request body the client keeps retrying).
SELECT algorithm, key_id, ts_added_ms
FROM e2e_one_time_keys_json
WHERE user_id = '@user:matrix.lotusguild.org'
AND device_id = '<DEVICE_ID>'
ORDER BY algorithm, key_id;
-- Server's advertised counts (this is what /sync tells the client it has,
-- and drives whether the client decides to upload more).
SELECT algorithm, count(*) FROM e2e_one_time_keys_json
WHERE user_id = '@user:matrix.lotusguild.org' AND device_id = '<DEVICE_ID>'
GROUP BY algorithm;
-- Fallback key state (used when OTKs are exhausted).
SELECT algorithm, key_id, used, ts_added_ms
FROM e2e_fallback_keys_json
WHERE user_id = '@user:matrix.lotusguild.org' AND device_id = '<DEVICE_ID>';
` ``
2026-07-02 11:45:22 -04:00
2026-07-01 21:19:02 -04:00
> Table names are Synapse 1.155 (` e2e_one_time_keys_json`,
> ` e2e_fallback_keys_json`). If a name is absent, list with ` \dt e2e*` in psql.
2026-07-02 11:45:22 -04:00
2026-07-01 21:19:02 -04:00
- **Confirms:** if the offending ` key_id` (from the 400) is **present** in
` e2e_one_time_keys_json` with a **different** stored value than the client's
request body → OTK state has diverged (rust-crypto store vs Synapse). That is
the KE-1 root condition.
### KE-2 — EC media keys not arriving/decrypting (audio/video cutouts)
- **Console signature (grep):**
- ` MissingKey`
- ` missing key at index` (e.g. ` MissingKey: missing key at index N for participant @user `)
- ` key set not found`
- ` io.element.call.encryption_keys` (rust-crypto: ` WARN … Received an unexpected encrypted to-device event … event_type="io.element.call.encryption_keys"`)
- **Capture client-side:**
- Timestamp windows where a participant's audio/video cut out, and the
` @participant ` + ` index N` from the message.
- The ` io.element.call.encryption_keys` warnings (these are the media-key
to-device events failing to decrypt) with their timestamps.
- Own device id + user id (to correlate with the sender's Olm session).
- **Synapse log grep (LXC 151) — to-device delivery of the media keys:**
` ``bash
grep -E "io.element.call.encryption_keys|m.room.encrypted|/sendToDevice|to_device" \
/var/log/matrix-synapse/homeserver.log | grep -E "<user_id>|<participant_id>"
` ``
- **Synapse SQL (LXC 109) — undelivered / queued to-device events:**
2026-07-02 11:45:22 -04:00
2026-07-01 21:19:02 -04:00
` ``sql
-- Backlog of to-device messages queued for the affected device. A growing
-- count here = the HS has the media-key events but the device isn't draining
-- them via /sync (or they were sent to a stale device id).
SELECT user_id, device_id, count(*) AS pending
FROM device_inbox
WHERE user_id = '@user:matrix.lotusguild.org'
GROUP BY user_id, device_id;
-- Cross-check the device id the sender is targeting actually exists / is current.
SELECT device_id, display_name, last_seen, ts
FROM devices WHERE user_id = '@user:matrix.lotusguild.org';
` ``
2026-07-02 11:45:22 -04:00
2026-07-01 21:19:02 -04:00
- **Confirms:** to-device events present but undecryptable (client shows the
` io.element.call.encryption_keys` "unexpected encrypted" warning) ⇒ there is
**no valid Olm session** to decrypt them — the expected downstream of KE-1.
### KE-3 — Timeline decryption error: missing ` algorithm` field
- **Console signature (grep):**
- ` DecryptionError`
- full: ` Error decrypting event (… type=m.room.encrypted …): DecryptionError[msg: missing field 'algorithm' at line 1 column 138 …]`
- **Capture client-side:**
- The **event id** (` $SASBBzoqj…` was one) and the **room id**.
- Pull the raw event JSON via DevTools or the Developer Tools account-data/event
viewer, or directly:
` ``
GET https://matrix.lotusguild.org/_matrix/client/v3/rooms/<roomId>/event/<eventId>
` ``
Inspect ` content` — confirm whether ` algorithm` (should be
` m.megolm.v1.aes-sha2`) is truly absent vs a serialization mismatch.
- **Synapse log grep (LXC 151):**
` ``bash
grep -E "<eventId>" /var/log/matrix-synapse/homeserver.log
` ``
- **Synapse SQL (LXC 109) — the stored event content as the HS holds it:**
` ``sql
SELECT ej.event_id, e.type, e.sender, e.origin_server_ts,
(ej.json::json -> 'content' -> 'algorithm') AS algorithm
FROM event_json ej
JOIN events e USING (event_id)
WHERE ej.event_id = '$SASBBzoqj...';
` ``
- **Confirms:** if the stored ` content.algorithm` is **NULL/absent** on the HS →
a malformed/legacy event was persisted (sender-side or federation). If it is
**present** on the HS but the client throws → an RC-SDK deserialization bug.
This distinction decides whether KE-3 is a data problem or a client problem.
### KE-4 — MatrixRTC delayed-event / membership timeouts
- **Console signature (grep):**
- ` update_delayed_event` (` org.matrix.msc4157.update_delayed_event`)
- ` delayed event` / ` Restart delayed event timed out`
- full: ` [MembershipManager] Network local timeout error while sending event, immediate retry … AbortError: Restart delayed event timed out before the HS responded`
- **Capture client-side:**
- Timestamps of each timeout; whether they correlate with call join/leave or
with general sync slowness.
- DevTools → Network: the ` …/delayed_events…` / ` update_delayed_event`
requests — their **HTTP status and latency** (timed-out vs slow-200).
- **Synapse log grep (LXC 151):**
` ``bash
grep -E "delayed_event|msc4140|msc4157|update_delayed" \
/var/log/matrix-synapse/homeserver.log | grep "<user_id>"
# HS responsiveness in the same window (KE-4 may be pure latency):
grep -E "Processed request|/sync" /var/log/matrix-synapse/homeserver.log | tail -50
` ``
- **Server-side corroboration (Grafana, ` dashboard.lotusguild.org`):** Synapse
p99 response time (excl. ` /sync`), event-processing lag, DB query latency for
the call window. High latency here ⇒ KE-4 is (partly) homeserver
responsiveness, not a client bug.
- **Confirms:** timeouts that line up with HS latency spikes → reliability/load;
timeouts with a healthy HS → client MembershipManager retry logic.
---
## 2. Causality hypothesis
` ``
KE-1 OTK upload conflict storm
(rust-crypto store ↔ Synapse OTK state DIVERGED; server rejects re-uploads)
│ no fresh OTKs can be published/claimed
▼
No new Olm (1:1) sessions can be established with this device
│
▼
KE-2 EC media-key to-device events (io.element.call.encryption_keys)
arrive but cannot be decrypted ⇒ MissingKey at index N
⇒ friend's audio/video cuts out
` ``
KE-3 (missing ` algorithm`) and KE-4 (delayed-event timeouts) are **likely
independent** of the KE-1→KE-2 chain: KE-3 is a decode/serialization path,
KE-4 is a MatrixRTC-vs-HS reliability path. Confirm/refute independence with the
decision tree below.
### Decision tree — which capture confirms/refutes each link
` ``
Q1. Does the KE-1 offending key_id from the 400 response exist in
e2e_one_time_keys_json with a DIFFERENT value than the client request body?
├─ YES → OTK divergence CONFIRMED (KE-1 root). Go to Q2.
└─ NO → Not divergence. Check: are OTK counts at 0 with fallback key ` used=true`?
├─ YES → OTK exhaustion, not divergence — different remediation.
└─ NO → Suspect RC-SDK 41.6.0-rc.0 upload-loop regression (see §3).
Q2. During the same call, are io.element.call.encryption_keys to-device events
present in device_inbox / Synapse to-device logs for our device id?
├─ YES + client shows "unexpected encrypted"/MissingKey
│ → KE-1 ⇒ KE-2 LINK CONFIRMED (events delivered, no Olm session to open them).
├─ YES + client decrypts fine, but LiveKit still silent
│ → KE-2 is downstream of LiveKit/SFU, NOT KE-1. Decouple from crypto.
└─ NO (nothing queued/targeted our device)
→ media keys never sent to us: stale device id / membership (see KE-4)
→ KE-2 is a device-targeting problem, weakly linked to KE-1.
Q3. KE-3: is content.algorithm NULL in event_json on the HS?
├─ YES → malformed persisted event (sender/federation). Independent of KE-1.
└─ NO → client-side RC-SDK deserialization bug. Independent of KE-1.
Q4. KE-4: do delayed-event timeouts coincide with Synapse p99 latency spikes
(Grafana) in the same minute?
├─ YES → homeserver responsiveness/load. Independent of KE-1..KE-3.
└─ NO → client MembershipManager retry behavior. Independent.
` ``
---
## 3. Ranked remediation options (with blast radius)
> Ordered least-destructive → most-destructive. **Do not run any of these as a
> "fix" before the planning session** — they are listed so evidence collection
> can be paired with a recovery plan. Confirm the root condition (Q1/Q2) first.
2026-07-02 11:45:22 -04:00
1. **Per-device logout + re-login of the affected device** _(lowest blast radius)_
2026-07-01 21:19:02 -04:00
- **What:** log the one glitching device out and back in. Forces a fresh
device id, fresh device keys, and a clean OTK batch — sidesteps a diverged
OTK store without touching other sessions.
- **Blast radius:** that device only. Other sessions/devices untouched.
- **Cost:** the new device must be re-verified (cross-signing) and will need
to restore room keys from **key backup** to read old encrypted history.
- **Confirms/uses:** if KE-1 stops after this, OTK-store divergence (Q1) was
the cause.
2026-07-02 11:45:22 -04:00
2. **Client crypto-store reset (` clearLoginData` path)** _(medium)_
2026-07-01 21:19:02 -04:00
- **What:** ` clearLoginData()` in ` src/client/initMatrix.ts` (coordinator's
file — do not edit) **deletes ALL IndexedDB databases** (incl.
` web-sync-store` and the rust-crypto store ` crypto-store`), **unregisters
service workers**, **clears all Cache Storage**, and **` localStorage.clear()`**,
then reloads. ` clearCacheAndReload()` is lighter — it only calls
` mx.store.deleteAllData()` (sync cache) and does **not** wipe crypto.
- **Blast radius:** this browser profile only, but total: you are logged out,
lose all cached sync state, drafts, settings, and **the local
megolm/room-key store**.
- **⚠️ Message-history / backup implication:** wiping ` crypto-store` destroys
locally-held **room keys (megolm inbound sessions)**. Any history **not
backed up to server-side Key Backup** becomes **permanently undecryptable
on this device**. Before doing this: verify Key Backup is enabled and the
recovery key / passphrase is available (Settings → Security), or the user
loses readable history. Cross-signing must be re-established too.
- **Use when:** the rust-crypto store itself is corrupt/diverged and option 1
didn't clear it.
2026-07-02 11:45:22 -04:00
3. **SDK pin change off the RC** _(medium — codebase change, needs rebuild)_
2026-07-01 21:19:02 -04:00
- **Current pin:** ` package.json` → ` "matrix-js-sdk": "41.6.0-rc.0"` (a
release candidate).
- **Finding (npm / GitHub changelog, checked 2026-07):** stable **` 41.6.0`**
2026-07-02 11:45:22 -04:00
was released **2026-05-26**. Its only changelog line is _"Throw sane error
on completeLoginOnNewDevice IdP rejection"_ — **no OTK / keys-upload / Olm /
2026-07-01 21:19:02 -04:00
to-device fix** relative to the RC. Later stable lines exist
(` 41.7.0`, ` 41.8.0`; ` 41.7.0-rc.3` / ` 41.9.0-rc.0` seen as pre-releases).
2026-07-02 11:45:22 -04:00
Nearby crypto-relevant entries: ` 41.5.0` _"Enable encrypted history sharing
by default"_; ` 41.4.0` key-backup handling. **No changelog entry directly
2026-07-01 21:19:02 -04:00
addresses the KE-1 OTK-conflict symptom** in the immediate range — so
moving RC→` 41.6.0` stable is a low-risk hygiene step but is **not expected
to fix KE-1 by itself**. Before pinning, re-read the CHANGELOG for any
` 41.7.x`/` 41.8.x` OTK/one-time-key/olm entry that post-dates this note.
- **Blast radius:** all users after the next ` cinny-build.sh` deploy. Test the
rust-crypto IndexedDB schema — a downgrade triggers the ` IDB_VERSION_CONFLICT`
path in ` initMatrix.ts`.
2026-07-02 11:45:22 -04:00
4. **Synapse-side OTK row surgery** _(LAST RESORT — highest danger)_
2026-07-01 21:19:02 -04:00
- **What:** deleting/rewriting rows in ` e2e_one_time_keys_json` (and/or
` e2e_fallback_keys_json`, ` device_inbox`) for the affected device to force
the client to re-upload a clean batch.
- **⚠️ Danger:** direct writes to Synapse crypto tables can **desync every
device of that user**, break Olm sessions **for everyone who has claimed one
of those keys**, and are easy to get wrong (wrong ` key_id`, cache not
invalidated). Synapse caches OTK counts — a raw DELETE without a restart can
leave the advertised count wrong, **worsening** the KE-1 loop.
- **Guardrails if ever done (planning session + HS owner only):** full
` pg_dump` of ` synapse` first; do it during **zero active calls**; delete only
the exact diverged ` key_id` for the exact ` device_id`; ` systemctl restart
2026-07-02 11:45:22 -04:00
matrix-synapse` to flush caches; then log the device out/in (option 1) so it
2026-07-01 21:19:02 -04:00
republishes. **Never** run this speculatively.
---
## 4. "Capture session" checklist (run during the next call)
Do these **in order**. Aim to have client + server capturing the **same call**.
1. **Prep server tail (LXC 151):** SSH in, start
` tail -F /var/log/matrix-synapse/homeserver.log | tee /tmp/lotus-call-$(date +%s).log`.
(Optionally raise the ` synapse.rest.client.keys` / ` handlers.e2e_keys` /
` handlers.devicemessage` loggers to DEBUG per §0 and ` systemctl reload
2026-07-02 11:45:22 -04:00
matrix-synapse` — remember to revert after.)
2026-07-01 21:19:02 -04:00
2. **Prep client:** open Lotus Chat → Settings → Developer Tools → **enable
Developer Tools** so the **Crypto Diagnostics** card is visible; note its
entry count starts at (or reset by reload to) 0.
3. **Open DevTools** (F12) → Console: enable **Preserve log**; Network tab:
enable **Preserve log** + **Record**. Note your **device id** and **user id**
(Settings → Devices / Developer Tools → Copy access token page shows ids).
4. **Note wall-clock start time** (ISO/UTC) on both machines so logs align.
5. **Join the Element Call** with the second participant; reproduce the fault
(wait for the audio/video cutouts and let KE-1 storm run ~30– 60s).
6. **When a fault occurs, note the wall-clock timestamp** and which symptom
(audio cut / video freeze / etc.) — this bounds the log window.
7. **Client artifacts:** in the Crypto Diagnostics card click **Download report**
(` lotus-crypto-diag-<ts>.json`); in DevTools Network, save the failing
` keys/upload` request+response (right-click → Save/Copy), and the raw HAR
(Network → Save all as HAR) for the call window.
8. **Grab KE-3 event id / KE-2 participant+index** from the console (or the
diag JSON ` entries[]`) for the SQL lookups.
9. **Server artifacts:** stop the tail; run the per-KE greps and SQL from §1
against the noted device id / user id / event id, saving output alongside the
client JSON. Screenshot the Grafana Synapse latency panels for the window
(for KE-4).
10. **Bundle & label:** put client JSON + HAR + server log slice + SQL output in
one folder named with the call's UTC start time. Revert any DEBUG log config
(` systemctl reload matrix-synapse`). Hand off to the planning session — **do
not apply §3 remediations yet.**
---
## 5. Client diagnostics helper (this kit)
- **` src/app/utils/cryptoDiagLog.ts`** — capture-only console instrumentation.
- ` installCryptoDiagLog()` — idempotent; wraps ` console.warn`/` console.error`
with pass-through wrappers (originals always called) that ring-buffer (max
**200**) any line matching the KE signatures. No network, no timers.
- ` getCryptoDiagEntries()` — snapshot copy of the buffer (` { ts, level, ke,
2026-07-02 11:45:22 -04:00
signature, message }`, most-recent-last).
2026-07-01 21:19:02 -04:00
- ` buildCryptoDiagReport(mx)` — JSON string: SDK version, device id, user id,
sync state, ` cryptoReady` (` mx.getCrypto()` presence), per-KE counts, and the
entry buffer. No tokens/PII beyond those ids; captured log lines are retained
verbatim as evidence.
- **Signatures → KE mapping:** ` already exists`→KE-1; ` missing key at index` /
` io.element.call.encryption_keys` / ` MissingKey`→KE-2; ` DecryptionError`→KE-3;
` update_delayed_event` / ` delayed event`→KE-4.
- **` src/app/features/settings/developer/CryptoDiagnostics.tsx`** — a folds
` SequenceCard`/` SettingTile` card (mirrors ` developer-tools/DevelopTools.tsx`)
showing the live matched-entry count (Badge) and a **Download report** button
(Blob → ` lotus-crypto-diag-<ts>.json`, same download idiom as
` room-settings/ExportRoomHistory.tsx`).
### Recommended mount points (coordinator)
- **Install call:** call ` installCryptoDiagLog()` **as early as possible during
boot** so it captures crypto errors from first sync — ideally at the top of
the client entry module or inside ` ClientRoot` before/around ` initClient`
(e.g. ` src/app/pages/client/ClientRoot.tsx`). It is idempotent, side-effect
only, and needs no ` mx`, so a module-scope call at app entry is safe. (Do
**not** put it in ` initMatrix.ts` — that file is off-limits.)
- **Settings card:** render ` <CryptoDiagnostics />` inside the Developer Tools
page — in ` src/app/features/settings/developer-tools/DevelopTools.tsx`, add it
to the ` Box direction="Column" gap="700"` list (guarded by the existing
` developerTools` flag), right after the "Access Token" card. It pulls ` mx`
from ` useMatrixClient()` itself, so it just needs to be placed in the tree.
2026-07-02 15:14:46 -04:00
---
## 6. 2026-07 investigation update — 41.7.0 delta + web-specific root cause
New findings this session (code-read + upstream issue triage). These **sharpen
KE-1's root cause and close the "just upgrade the SDK" lever**.
### 6.1 The 41.7.0 upgrade does NOT fix KE-1 (lever closed)
We are now on **` matrix-js-sdk@41 .7.0`** → **` @matrix -org/matrix-sdk-crypto-wasm@18 .3.1`**
(was ` 41.6.0-rc.0` when KE-1/2 were observed). Checked both changelogs:
- 41.7.0's only crypto line is the **security bump to crypto-wasm 18.3.1**. No
OTK / keys-upload / Olm-session change.
- crypto-wasm 17.0 → 18.3.1: **no entry** for one-time-keys, keys/upload,
"already exists", or upload conflicts. The 18.3.x work was **to-device
security hardening** (vodozemac 0.10; sender-spoofing check via
` sender_device_keys`; MSC4147 validation) — unrelated to the OTK loop.
- Upstream **` matrix-rust-sdk#5200 `** ("OlmMachine constantly tries to upload
keys when restoring session") is **still OPEN** (as of mid-2025). The loop
mechanism is confirmed there: on the 400, ` mark_request_as_sent()` never
fires, so the keys stay "unshared" and the SDK re-issues the identical failing
upload every cycle → the storm.
⇒ **Remediation option 3 (SDK pin) is exhausted for KE-1.** Do not expect a
version bump to help; the fix is store-hygiene, below.
### 6.2 Confirmed root cause + the web-specific trigger we can act on
Upstream ` #5200 ` + ` #1415 ` pin the root condition to **rust-crypto store ↔
server OTK divergence**, from one of:
1. **Crypto store reset/restore without deregistering the device server-side**
— the store forgets OTKs it already published; the server still holds them.
2. **Unsafe concurrent access to the crypto store** — e.g. the **same session
open in multiple browser tabs**, each running its own OlmMachine against the
one IndexedDB crypto store.
3. A store that isn't durably persisted, so a restore can't track what was sent.
**Cinny is a web client and hits two of these by construction (verified in code):**
- **No ` navigator.storage.persist()` anywhere** (` grep` clean). The rust-crypto
IndexedDB store is therefore **evictable under storage pressure** — while the
**access token + device id live in ` localStorage`** (N97), which browsers evict
_less_ aggressively. Partial eviction ⇒ the device **resurrects with a blank
crypto store but the SAME device id** ⇒ it re-uploads OTKs the server still
holds ⇒ the **exact KE-1 "already exists" divergence**, with **no user action**
and no visible cause. This is the leading hypothesis for a self-hosted web
deployment.
- **No multi-tab crypto guard** (no ` navigator.locks` / ` BroadcastChannel`
leader election in ` src/`). ` initMatrix.ts` calls ` mx.initRustCrypto()` with no
single-writer coordination, so 2+ tabs = concurrent store access = trigger #2.
### 6.3 Concrete PREVENTIVE client mitigations (new — buildable, don't need a call)
Ordered by value/effort. These reduce the _recurrence_ of KE-1; they don't heal
an already-diverged device (that still needs remediation option 1: clean
logout+login).
1. **Request persistent storage on login — ` navigator.storage.persist()`**
_(cheapest, highest value)_. Idempotent, side-effect only, no behavior change
if the browser denies it. Directly prevents the eviction-induced divergence in
6.2. Best placed at app entry alongside the other module-scope calls (NOT in
` initMatrix.ts`, which is off-limits) — e.g. a one-liner in ` ClientRoot`/app
bootstrap: ` if (navigator.storage?.persist) navigator.storage.persist();`
Optionally surface ` navigator.storage.persisted()` in the Crypto Diagnostics
card so a capture records whether the store was evictable.
2. **Multi-tab guard** _(medium)_. Detect a second tab of the same session
(BroadcastChannel or the Web Locks API) and either (a) warn "Lotus is open in
another tab — encryption may misbehave", or (b) make secondary tabs read-only
for crypto. Prevents trigger #2.
3. **Loop detection → recovery prompt** _(medium)_. Watch for repeated
` keys/upload` 400 ` M_UNKNOWN … already exists` (the client sees the rejection);
after N in a window, stop hammering and surface a "Reset encryption on this
device (log out & back in)" prompt instead of looping silently.
### 6.4 Secondary KE-2 hypothesis to test in the capture
crypto-wasm **18.3.0 tightened Olm to-device validation** (sender-spoof check +
MSC4147). It's therefore possible KE-2's ` WARN … unexpected encrypted to-device
event … io.element.call.encryption_keys` is **partly** the new validation
rejecting EC's media-key events, not _only_ the missing-Olm-session downstream of
KE-1. **Capture discriminator:** if KE-2 still occurs in a call where OTK counts
are healthy and no KE-1 storm is present (Q1 = NO), suspect the to-device
validation path (EC ↔ rust-crypto 18.3.x), not KE-1. If KE-2 only ever co-occurs
with the KE-1 storm, the original KE-1⇒KE-2 chain stands.
### 6.5 What to do now vs. at capture
- **Now (no call needed):** ship 6.3.1 (` persist()`) — it's safe and preventive.
Consider 6.3.3 (loop detection) as a follow-up.
- **At the next glitchy call:** run the §4 capture; answer Q1 (divergence?) and
6.4's discriminator. For any _currently_ stuck device, remediation option 1
(clean **logout + login**, not just "clear storage" — clearing storage without
` mx.logout()` leaves the server device + its OTKs and can re-trigger the
divergence).