@archipelagolab/lobi 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (315) hide show
  1. package/CHANGELOG.md +164 -0
  2. package/ENDOFFILE +0 -0
  3. package/EOF +0 -0
  4. package/LICENSE +21 -0
  5. package/SPEC-SUPPORT.md +116 -0
  6. package/YAMLEND +0 -0
  7. package/api.ts +18 -0
  8. package/archipelagolab-lobi-1.0.0.tgz +0 -0
  9. package/auth-presence.ts +56 -0
  10. package/channel-plugin-api.ts +3 -0
  11. package/cli-metadata.ts +11 -0
  12. package/contract-api.ts +17 -0
  13. package/docs/CHECKLIST.md +83 -0
  14. package/docs/FORK_SDK_GUIDE.md +279 -0
  15. package/helper-api.ts +3 -0
  16. package/index.test.ts +61 -0
  17. package/index.ts +65 -0
  18. package/openclaw.plugin.json +23 -0
  19. package/package.json +52 -0
  20. package/plugin-entry.handlers.runtime.ts +1 -0
  21. package/runtime-api.ts +54 -0
  22. package/runtime-heavy-api.ts +1 -0
  23. package/scripts/migrate-to-lobi.sh +72 -0
  24. package/secret-contract-api.ts +5 -0
  25. package/setup-entry.ts +13 -0
  26. package/src/account-selection.test.ts +124 -0
  27. package/src/account-selection.ts +226 -0
  28. package/src/actions.account-propagation.test.ts +251 -0
  29. package/src/actions.test.ts +251 -0
  30. package/src/actions.ts +336 -0
  31. package/src/approval-auth.test.ts +23 -0
  32. package/src/approval-auth.ts +25 -0
  33. package/src/approval-handler.runtime.test.ts +46 -0
  34. package/src/approval-handler.runtime.ts +400 -0
  35. package/src/approval-ids.ts +6 -0
  36. package/src/approval-native.test.ts +329 -0
  37. package/src/approval-native.ts +336 -0
  38. package/src/approval-reactions.test.ts +107 -0
  39. package/src/approval-reactions.ts +158 -0
  40. package/src/auth-precedence.ts +61 -0
  41. package/src/channel-account-paths.ts +92 -0
  42. package/src/channel.account-paths.test.ts +102 -0
  43. package/src/channel.directory.test.ts +601 -0
  44. package/src/channel.resolve.test.ts +38 -0
  45. package/src/channel.runtime.ts +16 -0
  46. package/src/channel.setup.test.ts +269 -0
  47. package/src/channel.ts +570 -0
  48. package/src/cli-metadata.ts +19 -0
  49. package/src/cli.test.ts +1015 -0
  50. package/src/cli.ts +1198 -0
  51. package/src/config-adapter.ts +41 -0
  52. package/src/config-schema.test.ts +90 -0
  53. package/src/config-schema.ts +114 -0
  54. package/src/directory-live.test.ts +200 -0
  55. package/src/directory-live.ts +238 -0
  56. package/src/doctor-contract.ts +287 -0
  57. package/src/doctor.test.ts +440 -0
  58. package/src/doctor.ts +262 -0
  59. package/src/env-vars.ts +92 -0
  60. package/src/exec-approval-resolver.test.ts +68 -0
  61. package/src/exec-approval-resolver.ts +23 -0
  62. package/src/exec-approvals.test.ts +483 -0
  63. package/src/exec-approvals.ts +290 -0
  64. package/src/group-mentions.ts +41 -0
  65. package/src/legacy-crypto-inspector-availability.test.ts +81 -0
  66. package/src/legacy-crypto-inspector-availability.ts +60 -0
  67. package/src/legacy-crypto.test.ts +234 -0
  68. package/src/legacy-crypto.ts +549 -0
  69. package/src/legacy-state.test.ts +86 -0
  70. package/src/legacy-state.ts +156 -0
  71. package/src/matrix/account-config.ts +150 -0
  72. package/src/matrix/accounts.readiness.test.ts +27 -0
  73. package/src/matrix/accounts.test.ts +757 -0
  74. package/src/matrix/accounts.ts +194 -0
  75. package/src/matrix/actions/client.test.ts +215 -0
  76. package/src/matrix/actions/client.ts +31 -0
  77. package/src/matrix/actions/devices.test.ts +114 -0
  78. package/src/matrix/actions/devices.ts +34 -0
  79. package/src/matrix/actions/limits.test.ts +15 -0
  80. package/src/matrix/actions/limits.ts +6 -0
  81. package/src/matrix/actions/messages.test.ts +289 -0
  82. package/src/matrix/actions/messages.ts +123 -0
  83. package/src/matrix/actions/pins.test.ts +74 -0
  84. package/src/matrix/actions/pins.ts +64 -0
  85. package/src/matrix/actions/polls.test.ts +71 -0
  86. package/src/matrix/actions/polls.ts +109 -0
  87. package/src/matrix/actions/profile.test.ts +109 -0
  88. package/src/matrix/actions/profile.ts +37 -0
  89. package/src/matrix/actions/reactions.test.ts +135 -0
  90. package/src/matrix/actions/reactions.ts +59 -0
  91. package/src/matrix/actions/room.test.ts +79 -0
  92. package/src/matrix/actions/room.ts +71 -0
  93. package/src/matrix/actions/summary.test.ts +87 -0
  94. package/src/matrix/actions/summary.ts +88 -0
  95. package/src/matrix/actions/types.ts +82 -0
  96. package/src/matrix/actions/verification.test.ts +105 -0
  97. package/src/matrix/actions/verification.ts +237 -0
  98. package/src/matrix/actions.ts +37 -0
  99. package/src/matrix/active-client.ts +26 -0
  100. package/src/matrix/async-lock.ts +18 -0
  101. package/src/matrix/backup-health.ts +115 -0
  102. package/src/matrix/client/config-runtime-api.ts +14 -0
  103. package/src/matrix/client/config-secret-input.runtime.ts +1 -0
  104. package/src/matrix/client/config.ts +982 -0
  105. package/src/matrix/client/create-client.test.ts +115 -0
  106. package/src/matrix/client/create-client.ts +101 -0
  107. package/src/matrix/client/env-auth.ts +6 -0
  108. package/src/matrix/client/file-sync-store.test.ts +265 -0
  109. package/src/matrix/client/file-sync-store.ts +289 -0
  110. package/src/matrix/client/logging.ts +123 -0
  111. package/src/matrix/client/migration-snapshot.runtime.ts +1 -0
  112. package/src/matrix/client/private-network-host.ts +56 -0
  113. package/src/matrix/client/runtime.ts +4 -0
  114. package/src/matrix/client/shared.test.ts +344 -0
  115. package/src/matrix/client/shared.ts +306 -0
  116. package/src/matrix/client/storage.test.ts +634 -0
  117. package/src/matrix/client/storage.ts +544 -0
  118. package/src/matrix/client/types.ts +50 -0
  119. package/src/matrix/client-bootstrap.test.ts +84 -0
  120. package/src/matrix/client-bootstrap.ts +164 -0
  121. package/src/matrix/client-resolver.test-helpers.ts +147 -0
  122. package/src/matrix/client.test.ts +1521 -0
  123. package/src/matrix/client.ts +23 -0
  124. package/src/matrix/config-paths.ts +31 -0
  125. package/src/matrix/config-update.test.ts +237 -0
  126. package/src/matrix/config-update.ts +291 -0
  127. package/src/matrix/credentials-read.ts +206 -0
  128. package/src/matrix/credentials-write.runtime.ts +26 -0
  129. package/src/matrix/credentials.test.ts +501 -0
  130. package/src/matrix/credentials.ts +95 -0
  131. package/src/matrix/deps.test.ts +74 -0
  132. package/src/matrix/deps.ts +225 -0
  133. package/src/matrix/device-health.test.ts +45 -0
  134. package/src/matrix/device-health.ts +31 -0
  135. package/src/matrix/direct-management.test.ts +350 -0
  136. package/src/matrix/direct-management.ts +347 -0
  137. package/src/matrix/direct-room.test.ts +61 -0
  138. package/src/matrix/direct-room.ts +128 -0
  139. package/src/matrix/draft-stream.test.ts +406 -0
  140. package/src/matrix/draft-stream.ts +216 -0
  141. package/src/matrix/encryption-guidance.ts +27 -0
  142. package/src/matrix/errors.ts +21 -0
  143. package/src/matrix/format.test.ts +340 -0
  144. package/src/matrix/format.ts +428 -0
  145. package/src/matrix/legacy-crypto-inspector.ts +95 -0
  146. package/src/matrix/media-errors.ts +20 -0
  147. package/src/matrix/media-text.ts +169 -0
  148. package/src/matrix/monitor/access-state.test.ts +45 -0
  149. package/src/matrix/monitor/access-state.ts +77 -0
  150. package/src/matrix/monitor/ack-config.test.ts +57 -0
  151. package/src/matrix/monitor/ack-config.ts +26 -0
  152. package/src/matrix/monitor/allowlist.test.ts +45 -0
  153. package/src/matrix/monitor/allowlist.ts +94 -0
  154. package/src/matrix/monitor/auto-join.test.ts +203 -0
  155. package/src/matrix/monitor/auto-join.ts +86 -0
  156. package/src/matrix/monitor/config.test.ts +197 -0
  157. package/src/matrix/monitor/config.ts +303 -0
  158. package/src/matrix/monitor/context-summary.ts +43 -0
  159. package/src/matrix/monitor/direct.test.ts +529 -0
  160. package/src/matrix/monitor/direct.ts +270 -0
  161. package/src/matrix/monitor/events.test.ts +1524 -0
  162. package/src/matrix/monitor/events.ts +213 -0
  163. package/src/matrix/monitor/handler.body-for-agent.test.ts +396 -0
  164. package/src/matrix/monitor/handler.group-history.test.ts +648 -0
  165. package/src/matrix/monitor/handler.media-failure.test.ts +267 -0
  166. package/src/matrix/monitor/handler.test-helpers.ts +308 -0
  167. package/src/matrix/monitor/handler.test.ts +2952 -0
  168. package/src/matrix/monitor/handler.thread-root-media.test.ts +82 -0
  169. package/src/matrix/monitor/handler.ts +1679 -0
  170. package/src/matrix/monitor/inbound-dedupe.test.ts +146 -0
  171. package/src/matrix/monitor/inbound-dedupe.ts +267 -0
  172. package/src/matrix/monitor/index.test.ts +920 -0
  173. package/src/matrix/monitor/index.ts +434 -0
  174. package/src/matrix/monitor/legacy-crypto-restore.test.ts +206 -0
  175. package/src/matrix/monitor/legacy-crypto-restore.ts +139 -0
  176. package/src/matrix/monitor/location.ts +100 -0
  177. package/src/matrix/monitor/media.test.ts +159 -0
  178. package/src/matrix/monitor/media.ts +119 -0
  179. package/src/matrix/monitor/mentions.test.ts +289 -0
  180. package/src/matrix/monitor/mentions.ts +177 -0
  181. package/src/matrix/monitor/reaction-events.test.ts +326 -0
  182. package/src/matrix/monitor/reaction-events.ts +187 -0
  183. package/src/matrix/monitor/recent-invite.test.ts +92 -0
  184. package/src/matrix/monitor/recent-invite.ts +30 -0
  185. package/src/matrix/monitor/replies.test.ts +265 -0
  186. package/src/matrix/monitor/replies.ts +136 -0
  187. package/src/matrix/monitor/reply-context.test.ts +276 -0
  188. package/src/matrix/monitor/reply-context.ts +92 -0
  189. package/src/matrix/monitor/room-history.test.ts +258 -0
  190. package/src/matrix/monitor/room-history.ts +301 -0
  191. package/src/matrix/monitor/room-info.test.ts +201 -0
  192. package/src/matrix/monitor/room-info.ts +126 -0
  193. package/src/matrix/monitor/rooms.test.ts +121 -0
  194. package/src/matrix/monitor/rooms.ts +52 -0
  195. package/src/matrix/monitor/route.test.ts +255 -0
  196. package/src/matrix/monitor/route.ts +178 -0
  197. package/src/matrix/monitor/runtime-api.ts +31 -0
  198. package/src/matrix/monitor/startup-verification.test.ts +294 -0
  199. package/src/matrix/monitor/startup-verification.ts +237 -0
  200. package/src/matrix/monitor/startup.test.ts +257 -0
  201. package/src/matrix/monitor/startup.ts +218 -0
  202. package/src/matrix/monitor/status.ts +111 -0
  203. package/src/matrix/monitor/sync-lifecycle.test.ts +224 -0
  204. package/src/matrix/monitor/sync-lifecycle.ts +91 -0
  205. package/src/matrix/monitor/task-runner.ts +38 -0
  206. package/src/matrix/monitor/thread-context.test.ts +149 -0
  207. package/src/matrix/monitor/thread-context.ts +108 -0
  208. package/src/matrix/monitor/threads.test.ts +68 -0
  209. package/src/matrix/monitor/threads.ts +85 -0
  210. package/src/matrix/monitor/types.ts +30 -0
  211. package/src/matrix/monitor/verification-events.ts +627 -0
  212. package/src/matrix/monitor/verification-utils.test.ts +47 -0
  213. package/src/matrix/monitor/verification-utils.ts +46 -0
  214. package/src/matrix/outbound-media-runtime.ts +1 -0
  215. package/src/matrix/poll-summary.ts +110 -0
  216. package/src/matrix/poll-types.test.ts +205 -0
  217. package/src/matrix/poll-types.ts +433 -0
  218. package/src/matrix/probe.runtime.ts +4 -0
  219. package/src/matrix/probe.test.ts +154 -0
  220. package/src/matrix/probe.ts +96 -0
  221. package/src/matrix/profile.test.ts +154 -0
  222. package/src/matrix/profile.ts +184 -0
  223. package/src/matrix/reaction-common.test.ts +96 -0
  224. package/src/matrix/reaction-common.ts +147 -0
  225. package/src/matrix/sdk/crypto-bootstrap.test.ts +505 -0
  226. package/src/matrix/sdk/crypto-bootstrap.ts +341 -0
  227. package/src/matrix/sdk/crypto-facade.test.ts +197 -0
  228. package/src/matrix/sdk/crypto-facade.ts +207 -0
  229. package/src/matrix/sdk/crypto-node.runtime.test.ts +27 -0
  230. package/src/matrix/sdk/crypto-node.runtime.ts +9 -0
  231. package/src/matrix/sdk/crypto-runtime.ts +11 -0
  232. package/src/matrix/sdk/decrypt-bridge.ts +356 -0
  233. package/src/matrix/sdk/event-helpers.test.ts +60 -0
  234. package/src/matrix/sdk/event-helpers.ts +71 -0
  235. package/src/matrix/sdk/http-client.test.ts +134 -0
  236. package/src/matrix/sdk/http-client.ts +87 -0
  237. package/src/matrix/sdk/idb-persistence-lock.ts +51 -0
  238. package/src/matrix/sdk/idb-persistence.lock-order.test.ts +108 -0
  239. package/src/matrix/sdk/idb-persistence.test-helpers.ts +88 -0
  240. package/src/matrix/sdk/idb-persistence.test.ts +149 -0
  241. package/src/matrix/sdk/idb-persistence.ts +283 -0
  242. package/src/matrix/sdk/logger.test.ts +25 -0
  243. package/src/matrix/sdk/logger.ts +108 -0
  244. package/src/matrix/sdk/read-response-with-limit.ts +19 -0
  245. package/src/matrix/sdk/recovery-key-store.test.ts +385 -0
  246. package/src/matrix/sdk/recovery-key-store.ts +430 -0
  247. package/src/matrix/sdk/transport.test.ts +161 -0
  248. package/src/matrix/sdk/transport.ts +344 -0
  249. package/src/matrix/sdk/types.ts +236 -0
  250. package/src/matrix/sdk/verification-manager.test.ts +509 -0
  251. package/src/matrix/sdk/verification-manager.ts +694 -0
  252. package/src/matrix/sdk/verification-status.ts +23 -0
  253. package/src/matrix/sdk.test.ts +2568 -0
  254. package/src/matrix/sdk.ts +1789 -0
  255. package/src/matrix/send/client.test.ts +174 -0
  256. package/src/matrix/send/client.ts +90 -0
  257. package/src/matrix/send/formatting.ts +189 -0
  258. package/src/matrix/send/media.ts +244 -0
  259. package/src/matrix/send/targets.test.ts +254 -0
  260. package/src/matrix/send/targets.ts +104 -0
  261. package/src/matrix/send/types.ts +134 -0
  262. package/src/matrix/send.test.ts +958 -0
  263. package/src/matrix/send.ts +609 -0
  264. package/src/matrix/session-store-metadata.ts +108 -0
  265. package/src/matrix/startup-abort.ts +44 -0
  266. package/src/matrix/sync-state.ts +27 -0
  267. package/src/matrix/target-ids.ts +102 -0
  268. package/src/matrix/thread-bindings-shared.ts +201 -0
  269. package/src/matrix/thread-bindings.test.ts +673 -0
  270. package/src/matrix/thread-bindings.ts +577 -0
  271. package/src/matrix-migration.runtime.ts +9 -0
  272. package/src/migration-config.test.ts +228 -0
  273. package/src/migration-config.ts +243 -0
  274. package/src/migration-snapshot-backup.ts +117 -0
  275. package/src/migration-snapshot.test.ts +184 -0
  276. package/src/migration-snapshot.ts +55 -0
  277. package/src/onboarding.resolve.test.ts +55 -0
  278. package/src/onboarding.test-harness.ts +158 -0
  279. package/src/onboarding.test.ts +665 -0
  280. package/src/onboarding.ts +773 -0
  281. package/src/outbound.test.ts +173 -0
  282. package/src/outbound.ts +78 -0
  283. package/src/plugin-entry.runtime.js +159 -0
  284. package/src/plugin-entry.runtime.test.ts +108 -0
  285. package/src/plugin-entry.runtime.ts +68 -0
  286. package/src/profile-update.ts +68 -0
  287. package/src/record-shared.ts +3 -0
  288. package/src/resolve-targets.test.ts +178 -0
  289. package/src/resolve-targets.ts +175 -0
  290. package/src/resolver.ts +21 -0
  291. package/src/runtime-api.ts +144 -0
  292. package/src/runtime.ts +7 -0
  293. package/src/secret-contract.ts +174 -0
  294. package/src/session-route.test.ts +315 -0
  295. package/src/session-route.ts +113 -0
  296. package/src/setup-bootstrap.ts +94 -0
  297. package/src/setup-config.ts +222 -0
  298. package/src/setup-contract.ts +89 -0
  299. package/src/setup-core.test.ts +326 -0
  300. package/src/setup-core.ts +50 -0
  301. package/src/setup-surface.ts +4 -0
  302. package/src/startup-maintenance.test.ts +227 -0
  303. package/src/startup-maintenance.ts +114 -0
  304. package/src/storage-paths.ts +92 -0
  305. package/src/test-helpers.ts +42 -0
  306. package/src/test-mocks.ts +55 -0
  307. package/src/test-runtime.ts +72 -0
  308. package/src/test-support/monitor-route-test-support.ts +8 -0
  309. package/src/tool-actions.runtime.ts +1 -0
  310. package/src/tool-actions.test.ts +422 -0
  311. package/src/tool-actions.ts +498 -0
  312. package/src/types.ts +230 -0
  313. package/test-api.ts +2 -0
  314. package/thread-bindings-runtime.ts +4 -0
  315. package/tsconfig.json +16 -0
@@ -0,0 +1,529 @@
1
+ import { afterEach, describe, expect, it, vi } from "vitest";
2
+ import type { MatrixClient } from "../sdk.js";
3
+ import { EventType } from "../send/types.js";
4
+ import { createDirectRoomTracker } from "./direct.js";
5
+
6
+ type MockStateEvents = Record<string, Record<string, unknown>>;
7
+
8
+ function createMockClient(params: {
9
+ isDm?: boolean;
10
+ members?: string[];
11
+ stateEvents?: MockStateEvents;
12
+ dmCacheAvailable?: boolean;
13
+ directAccountData?: Record<string, string[]>;
14
+ setAccountDataError?: Error;
15
+ }) {
16
+ let members = params.members ?? ["@alice:example.org", "@bot:example.org"];
17
+ const stateEvents = params.stateEvents ?? {};
18
+ let directAccountData = params.directAccountData ?? {};
19
+ const dmRoomIds = new Set<string>();
20
+ if (params.isDm === true) {
21
+ dmRoomIds.add("!room:example.org");
22
+ }
23
+ return {
24
+ dms: {
25
+ update: vi.fn().mockResolvedValue(params.dmCacheAvailable !== false),
26
+ isDm: vi.fn().mockImplementation((roomId: string) => dmRoomIds.has(roomId)),
27
+ },
28
+ getUserId: vi.fn().mockResolvedValue("@bot:example.org"),
29
+ getAccountData: vi
30
+ .fn()
31
+ .mockImplementation(async (eventType: string) =>
32
+ eventType === EventType.Direct ? directAccountData : undefined,
33
+ ),
34
+ getJoinedRoomMembers: vi.fn().mockImplementation(async () => members),
35
+ getRoomStateEvent: vi
36
+ .fn()
37
+ .mockImplementation(async (roomId: string, eventType: string, stateKey = "") => {
38
+ const key = `${roomId}|${eventType}|${stateKey}`;
39
+ const state = stateEvents[key];
40
+ if (state === undefined) {
41
+ throw new Error(`State event not found: ${key}`);
42
+ }
43
+ return state;
44
+ }),
45
+ setAccountData: vi.fn().mockImplementation(async (eventType: string, content: unknown) => {
46
+ if (params.setAccountDataError) {
47
+ throw params.setAccountDataError;
48
+ }
49
+ if (eventType !== EventType.Direct) {
50
+ return;
51
+ }
52
+ directAccountData = (content as Record<string, string[]>) ?? {};
53
+ dmRoomIds.clear();
54
+ for (const value of Object.values(directAccountData)) {
55
+ if (!Array.isArray(value)) {
56
+ continue;
57
+ }
58
+ for (const roomId of value) {
59
+ if (typeof roomId === "string" && roomId.trim()) {
60
+ dmRoomIds.add(roomId);
61
+ }
62
+ }
63
+ }
64
+ }),
65
+ __setMembers(next: string[]) {
66
+ members = next;
67
+ },
68
+ } as unknown as MatrixClient & {
69
+ dms: {
70
+ update: ReturnType<typeof vi.fn>;
71
+ isDm: ReturnType<typeof vi.fn>;
72
+ };
73
+ getAccountData: ReturnType<typeof vi.fn>;
74
+ getJoinedRoomMembers: ReturnType<typeof vi.fn>;
75
+ getRoomStateEvent: ReturnType<typeof vi.fn>;
76
+ setAccountData: ReturnType<typeof vi.fn>;
77
+ __setMembers: (members: string[]) => void;
78
+ };
79
+ }
80
+
81
+ describe("createDirectRoomTracker", () => {
82
+ afterEach(() => {
83
+ vi.useRealTimers();
84
+ });
85
+
86
+ it("treats m.direct rooms as DMs", async () => {
87
+ const client = createMockClient({ isDm: true });
88
+ const tracker = createDirectRoomTracker(client);
89
+
90
+ await expect(
91
+ tracker.isDirectMessage({
92
+ roomId: "!room:example.org",
93
+ senderId: "@alice:example.org",
94
+ }),
95
+ ).resolves.toBe(true);
96
+
97
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledWith("!room:example.org");
98
+ });
99
+
100
+ it("does not trust stale m.direct classifications for shared rooms", async () => {
101
+ const client = createMockClient({
102
+ isDm: true,
103
+ members: ["@alice:example.org", "@bot:example.org", "@extra:example.org"],
104
+ });
105
+ const tracker = createDirectRoomTracker(client);
106
+
107
+ await expect(
108
+ tracker.isDirectMessage({
109
+ roomId: "!room:example.org",
110
+ senderId: "@alice:example.org",
111
+ }),
112
+ ).resolves.toBe(false);
113
+
114
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledWith("!room:example.org");
115
+ });
116
+
117
+ it("does not classify 2-member rooms as DMs when the dm cache refresh succeeds", async () => {
118
+ const client = createMockClient({ isDm: false, dmCacheAvailable: true });
119
+ const tracker = createDirectRoomTracker(client);
120
+
121
+ await expect(
122
+ tracker.isDirectMessage({
123
+ roomId: "!room:example.org",
124
+ senderId: "@alice:example.org",
125
+ }),
126
+ ).resolves.toBe(false);
127
+
128
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledWith("!room:example.org");
129
+ });
130
+
131
+ it("falls back to strict 2-member membership before m.direct account data is available", async () => {
132
+ const client = createMockClient({ isDm: false, dmCacheAvailable: false });
133
+ const tracker = createDirectRoomTracker(client);
134
+
135
+ await expect(
136
+ tracker.isDirectMessage({
137
+ roomId: "!room:example.org",
138
+ senderId: "@alice:example.org",
139
+ }),
140
+ ).resolves.toBe(true);
141
+
142
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledWith("!room:example.org");
143
+ });
144
+
145
+ it("keeps using the strict 2-member fallback until the dm cache seeds successfully", async () => {
146
+ const client = createMockClient({ isDm: false, dmCacheAvailable: false });
147
+ const tracker = createDirectRoomTracker(client);
148
+
149
+ await expect(
150
+ tracker.isDirectMessage({
151
+ roomId: "!room:example.org",
152
+ senderId: "@alice:example.org",
153
+ }),
154
+ ).resolves.toBe(true);
155
+ await expect(
156
+ tracker.isDirectMessage({
157
+ roomId: "!room:example.org",
158
+ senderId: "@alice:example.org",
159
+ }),
160
+ ).resolves.toBe(true);
161
+
162
+ expect(client.dms.update).toHaveBeenCalledTimes(1);
163
+ });
164
+
165
+ it("does not classify rooms with extra members as DMs when falling back", async () => {
166
+ const client = createMockClient({
167
+ isDm: false,
168
+ members: ["@alice:example.org", "@bot:example.org", "@observer:example.org"],
169
+ dmCacheAvailable: false,
170
+ });
171
+ const tracker = createDirectRoomTracker(client);
172
+
173
+ await expect(
174
+ tracker.isDirectMessage({
175
+ roomId: "!room:example.org",
176
+ senderId: "@alice:example.org",
177
+ }),
178
+ ).resolves.toBe(false);
179
+
180
+ expect(client.getRoomStateEvent).not.toHaveBeenCalled();
181
+ });
182
+
183
+ it("does not treat sender is_direct member state as a DM signal", async () => {
184
+ const client = createMockClient({
185
+ isDm: false,
186
+ dmCacheAvailable: true,
187
+ stateEvents: {
188
+ "!room:example.org|m.room.member|@alice:example.org": { is_direct: true },
189
+ },
190
+ });
191
+ const tracker = createDirectRoomTracker(client);
192
+
193
+ await expect(
194
+ tracker.isDirectMessage({
195
+ roomId: "!room:example.org",
196
+ senderId: "@alice:example.org",
197
+ }),
198
+ ).resolves.toBe(false);
199
+ });
200
+
201
+ it("treats self is_direct member state as a DM signal", async () => {
202
+ const client = createMockClient({
203
+ isDm: false,
204
+ stateEvents: {
205
+ "!room:example.org|m.room.member|@bot:example.org": { is_direct: true },
206
+ },
207
+ });
208
+ const tracker = createDirectRoomTracker(client);
209
+
210
+ await expect(
211
+ tracker.isDirectMessage({
212
+ roomId: "!room:example.org",
213
+ senderId: "@alice:example.org",
214
+ }),
215
+ ).resolves.toBe(true);
216
+ });
217
+
218
+ it("treats self is_direct false member state as a non-DM signal", async () => {
219
+ const client = createMockClient({
220
+ isDm: false,
221
+ dmCacheAvailable: false,
222
+ stateEvents: {
223
+ "!room:example.org|m.room.member|@bot:example.org": { is_direct: false },
224
+ },
225
+ });
226
+ const tracker = createDirectRoomTracker(client);
227
+
228
+ await expect(
229
+ tracker.isDirectMessage({
230
+ roomId: "!room:example.org",
231
+ senderId: "@alice:example.org",
232
+ }),
233
+ ).resolves.toBe(false);
234
+ });
235
+
236
+ it("treats strict rooms from recent invites as DMs after the dm cache has seeded", async () => {
237
+ const client = createMockClient({ isDm: false, dmCacheAvailable: true });
238
+ const tracker = createDirectRoomTracker(client);
239
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
240
+
241
+ await expect(
242
+ tracker.isDirectMessage({
243
+ roomId: "!room:example.org",
244
+ senderId: "@alice:example.org",
245
+ }),
246
+ ).resolves.toBe(true);
247
+
248
+ expect(client.setAccountData).toHaveBeenCalledWith(
249
+ EventType.Direct,
250
+ expect.objectContaining({
251
+ "@alice:example.org": ["!room:example.org"],
252
+ }),
253
+ );
254
+ });
255
+
256
+ it("keeps recent invite candidates across room invalidation", async () => {
257
+ const client = createMockClient({ isDm: false, dmCacheAvailable: true });
258
+ const tracker = createDirectRoomTracker(client);
259
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
260
+ tracker.invalidateRoom("!room:example.org");
261
+
262
+ await expect(
263
+ tracker.isDirectMessage({
264
+ roomId: "!room:example.org",
265
+ senderId: "@alice:example.org",
266
+ }),
267
+ ).resolves.toBe(true);
268
+ });
269
+
270
+ it("still rejects recent invite candidates when self member state is_direct is false", async () => {
271
+ const client = createMockClient({
272
+ isDm: false,
273
+ dmCacheAvailable: true,
274
+ stateEvents: {
275
+ "!room:example.org|m.room.member|@bot:example.org": { is_direct: false },
276
+ },
277
+ });
278
+ const tracker = createDirectRoomTracker(client);
279
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
280
+
281
+ await expect(
282
+ tracker.isDirectMessage({
283
+ roomId: "!room:example.org",
284
+ senderId: "@alice:example.org",
285
+ }),
286
+ ).resolves.toBe(false);
287
+ });
288
+
289
+ it("does not promote recent invite candidates when local vetoes mark the room as non-DM", async () => {
290
+ const client = createMockClient({
291
+ isDm: false,
292
+ dmCacheAvailable: true,
293
+ });
294
+ const tracker = createDirectRoomTracker(client, {
295
+ canPromoteRecentInvite: () => false,
296
+ });
297
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
298
+
299
+ await expect(
300
+ tracker.isDirectMessage({
301
+ roomId: "!room:example.org",
302
+ senderId: "@alice:example.org",
303
+ }),
304
+ ).resolves.toBe(false);
305
+
306
+ expect(client.setAccountData).not.toHaveBeenCalled();
307
+ });
308
+
309
+ it("still treats recent invite candidates as DMs when m.direct repair fails", async () => {
310
+ const client = createMockClient({
311
+ isDm: false,
312
+ dmCacheAvailable: true,
313
+ setAccountDataError: new Error("account data unavailable"),
314
+ });
315
+ const tracker = createDirectRoomTracker(client);
316
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
317
+
318
+ await expect(
319
+ tracker.isDirectMessage({
320
+ roomId: "!room:example.org",
321
+ senderId: "@alice:example.org",
322
+ }),
323
+ ).resolves.toBe(true);
324
+ });
325
+
326
+ it("keeps locally promoted direct rooms stable after repair failures", async () => {
327
+ vi.useFakeTimers();
328
+ vi.setSystemTime(new Date("2026-03-30T23:00:00Z"));
329
+ const client = createMockClient({
330
+ isDm: false,
331
+ dmCacheAvailable: true,
332
+ setAccountDataError: new Error("account data unavailable"),
333
+ });
334
+ const tracker = createDirectRoomTracker(client);
335
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
336
+
337
+ await expect(
338
+ tracker.isDirectMessage({
339
+ roomId: "!room:example.org",
340
+ senderId: "@alice:example.org",
341
+ }),
342
+ ).resolves.toBe(true);
343
+
344
+ tracker.invalidateRoom("!room:example.org");
345
+
346
+ vi.setSystemTime(new Date("2026-03-30T23:01:00Z"));
347
+
348
+ await expect(
349
+ tracker.isDirectMessage({
350
+ roomId: "!room:example.org",
351
+ senderId: "@alice:example.org",
352
+ }),
353
+ ).resolves.toBe(true);
354
+ });
355
+
356
+ it("drops locally promoted direct rooms when room metadata later vetoes promotion", async () => {
357
+ vi.useFakeTimers();
358
+ vi.setSystemTime(new Date("2026-03-30T23:00:00Z"));
359
+ let keepLocalPromotion = true;
360
+ const client = createMockClient({
361
+ isDm: false,
362
+ dmCacheAvailable: true,
363
+ setAccountDataError: new Error("account data unavailable"),
364
+ });
365
+ const tracker = createDirectRoomTracker(client, {
366
+ canPromoteRecentInvite: () => true,
367
+ shouldKeepLocallyPromotedDirectRoom: () => keepLocalPromotion,
368
+ });
369
+ tracker.rememberInvite("!room:example.org", "@alice:example.org");
370
+
371
+ await expect(
372
+ tracker.isDirectMessage({
373
+ roomId: "!room:example.org",
374
+ senderId: "@alice:example.org",
375
+ }),
376
+ ).resolves.toBe(true);
377
+
378
+ keepLocalPromotion = false;
379
+ vi.setSystemTime(new Date("2026-03-30T23:01:00Z"));
380
+
381
+ await expect(
382
+ tracker.isDirectMessage({
383
+ roomId: "!room:example.org",
384
+ senderId: "@alice:example.org",
385
+ }),
386
+ ).resolves.toBe(false);
387
+ });
388
+
389
+ it("does not classify 2-member rooms whose sender is not a joined member when falling back", async () => {
390
+ const client = createMockClient({
391
+ isDm: false,
392
+ members: ["@mallory:example.org", "@bot:example.org"],
393
+ dmCacheAvailable: false,
394
+ });
395
+ const tracker = createDirectRoomTracker(client);
396
+
397
+ await expect(
398
+ tracker.isDirectMessage({
399
+ roomId: "!room:example.org",
400
+ senderId: "@alice:example.org",
401
+ }),
402
+ ).resolves.toBe(false);
403
+ });
404
+
405
+ it("does not re-enable the strict 2-member fallback after the dm cache has seeded", async () => {
406
+ const client = createMockClient({ isDm: false, dmCacheAvailable: true });
407
+ const tracker = createDirectRoomTracker(client);
408
+
409
+ await expect(
410
+ tracker.isDirectMessage({
411
+ roomId: "!room:example.org",
412
+ senderId: "@alice:example.org",
413
+ }),
414
+ ).resolves.toBe(false);
415
+
416
+ client.dms.update.mockResolvedValue(false);
417
+ tracker.invalidateRoom("!room:example.org");
418
+
419
+ await expect(
420
+ tracker.isDirectMessage({
421
+ roomId: "!room:example.org",
422
+ senderId: "@alice:example.org",
423
+ }),
424
+ ).resolves.toBe(false);
425
+ });
426
+
427
+ it("re-checks room membership after invalidation when fallback membership changes", async () => {
428
+ const client = createMockClient({ isDm: false, dmCacheAvailable: false });
429
+ const tracker = createDirectRoomTracker(client);
430
+
431
+ await expect(
432
+ tracker.isDirectMessage({
433
+ roomId: "!room:example.org",
434
+ senderId: "@alice:example.org",
435
+ }),
436
+ ).resolves.toBe(true);
437
+
438
+ client.__setMembers(["@alice:example.org", "@bot:example.org", "@mallory:example.org"]);
439
+ tracker.invalidateRoom("!room:example.org");
440
+
441
+ await expect(
442
+ tracker.isDirectMessage({
443
+ roomId: "!room:example.org",
444
+ senderId: "@alice:example.org",
445
+ }),
446
+ ).resolves.toBe(false);
447
+ });
448
+
449
+ it("bounds joined-room membership cache size", async () => {
450
+ const client = createMockClient({ isDm: false, dmCacheAvailable: false });
451
+ const tracker = createDirectRoomTracker(client);
452
+
453
+ for (let i = 0; i <= 1024; i += 1) {
454
+ await tracker.isDirectMessage({
455
+ roomId: `!room-${i}:example.org`,
456
+ senderId: "@alice:example.org",
457
+ });
458
+ }
459
+
460
+ await tracker.isDirectMessage({
461
+ roomId: "!room-0:example.org",
462
+ senderId: "@alice:example.org",
463
+ });
464
+
465
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledTimes(1026);
466
+ });
467
+
468
+ it("refreshes dm and membership caches after the ttl expires", async () => {
469
+ vi.useFakeTimers();
470
+ vi.setSystemTime(new Date("2026-03-12T10:00:00Z"));
471
+ const client = createMockClient({ isDm: true });
472
+ const tracker = createDirectRoomTracker(client);
473
+
474
+ await tracker.isDirectMessage({
475
+ roomId: "!room:example.org",
476
+ senderId: "@alice:example.org",
477
+ });
478
+ await tracker.isDirectMessage({
479
+ roomId: "!room:example.org",
480
+ senderId: "@alice:example.org",
481
+ });
482
+
483
+ expect(client.dms.update).toHaveBeenCalledTimes(1);
484
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledTimes(1);
485
+
486
+ vi.setSystemTime(new Date("2026-03-12T10:00:31Z"));
487
+
488
+ await tracker.isDirectMessage({
489
+ roomId: "!room:example.org",
490
+ senderId: "@alice:example.org",
491
+ });
492
+
493
+ expect(client.dms.update).toHaveBeenCalledTimes(2);
494
+ expect(client.getJoinedRoomMembers).toHaveBeenCalledTimes(2);
495
+ });
496
+
497
+ it("caches member-state direct flag lookups until the ttl expires", async () => {
498
+ vi.useFakeTimers();
499
+ vi.setSystemTime(new Date("2026-03-12T10:00:00Z"));
500
+ const client = createMockClient({
501
+ isDm: false,
502
+ dmCacheAvailable: true,
503
+ stateEvents: {
504
+ "!room:example.org|m.room.member|@alice:example.org": { is_direct: true },
505
+ },
506
+ });
507
+ const tracker = createDirectRoomTracker(client);
508
+
509
+ await tracker.isDirectMessage({
510
+ roomId: "!room:example.org",
511
+ senderId: "@alice:example.org",
512
+ });
513
+ await tracker.isDirectMessage({
514
+ roomId: "!room:example.org",
515
+ senderId: "@alice:example.org",
516
+ });
517
+
518
+ expect(client.getRoomStateEvent).toHaveBeenCalledTimes(1);
519
+
520
+ vi.setSystemTime(new Date("2026-03-12T10:00:31Z"));
521
+
522
+ await tracker.isDirectMessage({
523
+ roomId: "!room:example.org",
524
+ senderId: "@alice:example.org",
525
+ });
526
+
527
+ expect(client.getRoomStateEvent).toHaveBeenCalledTimes(2);
528
+ });
529
+ });