@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,267 @@
1
+ import { beforeEach, describe, expect, it, vi } from "vitest";
2
+ import { installMatrixMonitorTestRuntime } from "../../test-runtime.js";
3
+ import { MatrixMediaSizeLimitError } from "../media-errors.js";
4
+ import {
5
+ createMatrixHandlerTestHarness,
6
+ createMatrixRoomMessageEvent,
7
+ } from "./handler.test-helpers.js";
8
+
9
+ const { downloadMatrixMediaMock } = vi.hoisted(() => ({
10
+ downloadMatrixMediaMock: vi.fn(),
11
+ }));
12
+
13
+ vi.mock("./media.js", async () => {
14
+ const actual = await vi.importActual<typeof import("./media.js")>("./media.js");
15
+ return {
16
+ ...actual,
17
+ downloadMatrixMedia: (...args: unknown[]) => downloadMatrixMediaMock(...args),
18
+ };
19
+ });
20
+
21
+ function createMediaFailureHarness() {
22
+ const logger = {
23
+ info: vi.fn(),
24
+ warn: vi.fn(),
25
+ error: vi.fn(),
26
+ };
27
+ const runtime = {
28
+ error: vi.fn(),
29
+ };
30
+ const harness = createMatrixHandlerTestHarness({
31
+ logger: logger as never,
32
+ runtime: runtime as never,
33
+ shouldHandleTextCommands: () => true,
34
+ resolveMarkdownTableMode: () => "code",
35
+ resolveAgentRoute: () => ({
36
+ agentId: "main",
37
+ accountId: "ops",
38
+ sessionKey: "agent:main:matrix:channel:!room:example.org",
39
+ mainSessionKey: "agent:main:main",
40
+ channel: "matrix",
41
+ matchedBy: "binding.account",
42
+ }),
43
+ resolveStorePath: () => "/tmp/openclaw-test-session.json",
44
+ readSessionUpdatedAt: () => 123,
45
+ getRoomInfo: async () => ({
46
+ name: "Media Room",
47
+ canonicalAlias: "#media:example.org",
48
+ altAliases: [],
49
+ }),
50
+ getMemberDisplayName: async () => "Gum",
51
+ startupMs: Date.now() - 120_000,
52
+ startupGraceMs: 60_000,
53
+ textLimit: 4000,
54
+ mediaMaxBytes: 5 * 1024 * 1024,
55
+ replyToMode: "first",
56
+ });
57
+
58
+ return {
59
+ ...harness,
60
+ logger,
61
+ runtime,
62
+ };
63
+ }
64
+
65
+ function createImageEvent(content: Record<string, unknown>) {
66
+ return createMatrixRoomMessageEvent({
67
+ eventId: "$event1",
68
+ sender: "@gum:matrix.example.org",
69
+ content: {
70
+ ...content,
71
+ "m.mentions": { user_ids: ["@bot:matrix.example.org"] },
72
+ } as never,
73
+ });
74
+ }
75
+
76
+ describe("createMatrixRoomMessageHandler media failures", () => {
77
+ beforeEach(() => {
78
+ downloadMatrixMediaMock.mockReset();
79
+ installMatrixMonitorTestRuntime();
80
+ });
81
+
82
+ it("forwards the Matrix event body as originalFilename for media downloads", async () => {
83
+ downloadMatrixMediaMock.mockResolvedValue({
84
+ path: "/tmp/inbound/Screenshot-2026-03-27---uuid.png",
85
+ contentType: "image/png",
86
+ placeholder: "[matrix media]",
87
+ });
88
+ const { handler } = createMediaFailureHarness();
89
+
90
+ await handler(
91
+ "!room:example.org",
92
+ createImageEvent({
93
+ msgtype: "m.image",
94
+ body: " Screenshot 2026-03-27.png ",
95
+ url: "mxc://example/image",
96
+ }),
97
+ );
98
+
99
+ expect(downloadMatrixMediaMock).toHaveBeenCalledWith(
100
+ expect.objectContaining({
101
+ mxcUrl: "mxc://example/image",
102
+ maxBytes: 5 * 1024 * 1024,
103
+ originalFilename: "Screenshot 2026-03-27.png",
104
+ }),
105
+ );
106
+ });
107
+
108
+ it("prefers content.filename over body text when deriving originalFilename", async () => {
109
+ downloadMatrixMediaMock.mockResolvedValue({
110
+ path: "/tmp/inbound/Screenshot-2026-03-27---uuid.png",
111
+ contentType: "image/png",
112
+ placeholder: "[matrix media]",
113
+ });
114
+ const { handler } = createMediaFailureHarness();
115
+
116
+ await handler(
117
+ "!room:example.org",
118
+ createImageEvent({
119
+ msgtype: "m.image",
120
+ body: "can you review this screenshot?",
121
+ filename: "Screenshot 2026-03-27.png",
122
+ url: "mxc://example/image",
123
+ }),
124
+ );
125
+
126
+ expect(downloadMatrixMediaMock).toHaveBeenCalledWith(
127
+ expect.objectContaining({
128
+ originalFilename: "Screenshot 2026-03-27.png",
129
+ }),
130
+ );
131
+ });
132
+
133
+ it("replaces bare image filenames with an unavailable marker when unencrypted download fails", async () => {
134
+ downloadMatrixMediaMock.mockRejectedValue(new Error("download failed"));
135
+ const { handler, recordInboundSession, logger, runtime } = createMediaFailureHarness();
136
+
137
+ await handler(
138
+ "!room:example.org",
139
+ createImageEvent({
140
+ msgtype: "m.image",
141
+ body: "image.png",
142
+ url: "mxc://example/image",
143
+ }),
144
+ );
145
+
146
+ expect(recordInboundSession).toHaveBeenCalledWith(
147
+ expect.objectContaining({
148
+ ctx: expect.objectContaining({
149
+ RawBody: "[matrix image attachment unavailable]",
150
+ CommandBody: "[matrix image attachment unavailable]",
151
+ MediaPath: undefined,
152
+ }),
153
+ }),
154
+ );
155
+ expect(logger.warn).toHaveBeenCalledWith(
156
+ "matrix media download failed",
157
+ expect.objectContaining({
158
+ eventId: "$event1",
159
+ msgtype: "m.image",
160
+ encrypted: false,
161
+ }),
162
+ );
163
+ expect(runtime.error).not.toHaveBeenCalled();
164
+ });
165
+
166
+ it("replaces bare image filenames with an unavailable marker when encrypted download fails", async () => {
167
+ downloadMatrixMediaMock.mockRejectedValue(new Error("decrypt failed"));
168
+ const { handler, recordInboundSession } = createMediaFailureHarness();
169
+
170
+ await handler(
171
+ "!room:example.org",
172
+ createImageEvent({
173
+ msgtype: "m.image",
174
+ body: "photo.jpg",
175
+ file: {
176
+ url: "mxc://example/encrypted",
177
+ key: { kty: "oct", key_ops: ["encrypt"], alg: "A256CTR", k: "secret", ext: true },
178
+ iv: "iv",
179
+ hashes: { sha256: "hash" },
180
+ v: "v2",
181
+ },
182
+ }),
183
+ );
184
+
185
+ expect(recordInboundSession).toHaveBeenCalledWith(
186
+ expect.objectContaining({
187
+ ctx: expect.objectContaining({
188
+ RawBody: "[matrix image attachment unavailable]",
189
+ CommandBody: "[matrix image attachment unavailable]",
190
+ MediaPath: undefined,
191
+ }),
192
+ }),
193
+ );
194
+ });
195
+
196
+ it("preserves a real caption while marking the attachment unavailable", async () => {
197
+ downloadMatrixMediaMock.mockRejectedValue(new Error("download failed"));
198
+ const { handler, recordInboundSession } = createMediaFailureHarness();
199
+
200
+ await handler(
201
+ "!room:example.org",
202
+ createImageEvent({
203
+ msgtype: "m.image",
204
+ body: "can you see this image?",
205
+ filename: "image.png",
206
+ url: "mxc://example/image",
207
+ }),
208
+ );
209
+
210
+ expect(recordInboundSession).toHaveBeenCalledWith(
211
+ expect.objectContaining({
212
+ ctx: expect.objectContaining({
213
+ RawBody: "can you see this image?\n\n[matrix image attachment unavailable]",
214
+ CommandBody: "can you see this image?\n\n[matrix image attachment unavailable]",
215
+ }),
216
+ }),
217
+ );
218
+ });
219
+
220
+ it("shows a too-large marker when the download is rejected due to size limit", async () => {
221
+ downloadMatrixMediaMock.mockRejectedValue(new MatrixMediaSizeLimitError());
222
+ const { handler, recordInboundSession } = createMediaFailureHarness();
223
+
224
+ await handler(
225
+ "!room:example.org",
226
+ createImageEvent({
227
+ msgtype: "m.image",
228
+ body: "big-photo.jpg",
229
+ url: "mxc://example/big-image",
230
+ }),
231
+ );
232
+
233
+ expect(recordInboundSession).toHaveBeenCalledWith(
234
+ expect.objectContaining({
235
+ ctx: expect.objectContaining({
236
+ RawBody: "[matrix image attachment too large]",
237
+ CommandBody: "[matrix image attachment too large]",
238
+ MediaPath: undefined,
239
+ }),
240
+ }),
241
+ );
242
+ });
243
+
244
+ it("preserves a real caption while marking the attachment too large on size limit error", async () => {
245
+ downloadMatrixMediaMock.mockRejectedValue(new MatrixMediaSizeLimitError());
246
+ const { handler, recordInboundSession } = createMediaFailureHarness();
247
+
248
+ await handler(
249
+ "!room:example.org",
250
+ createImageEvent({
251
+ msgtype: "m.image",
252
+ body: "check this out",
253
+ filename: "large-photo.jpg",
254
+ url: "mxc://example/big-image",
255
+ }),
256
+ );
257
+
258
+ expect(recordInboundSession).toHaveBeenCalledWith(
259
+ expect.objectContaining({
260
+ ctx: expect.objectContaining({
261
+ RawBody: "check this out\n\n[matrix image attachment too large]",
262
+ CommandBody: "check this out\n\n[matrix image attachment too large]",
263
+ }),
264
+ }),
265
+ );
266
+ });
267
+ });
@@ -0,0 +1,308 @@
1
+ import { vi } from "vitest";
2
+ import type { RuntimeEnv, RuntimeLogger } from "../../runtime-api.js";
3
+ import type { MatrixRoomConfig, MatrixStreamingMode, ReplyToMode } from "../../types.js";
4
+ import type { MatrixClient } from "../sdk.js";
5
+ import { createMatrixRoomMessageHandler, type MatrixMonitorHandlerParams } from "./handler.js";
6
+ import { EventType, type MatrixRawEvent, type RoomMessageEventContent } from "./types.js";
7
+
8
+ const DEFAULT_ROUTE = {
9
+ agentId: "ops",
10
+ channel: "matrix",
11
+ accountId: "ops",
12
+ sessionKey: "agent:ops:main",
13
+ mainSessionKey: "agent:ops:main",
14
+ matchedBy: "binding.account" as const,
15
+ };
16
+
17
+ type MatrixHandlerTestHarnessOptions = {
18
+ accountId?: string;
19
+ cfg?: unknown;
20
+ client?: Partial<MatrixClient>;
21
+ runtime?: RuntimeEnv;
22
+ logger?: RuntimeLogger;
23
+ logVerboseMessage?: (message: string) => void;
24
+ allowFrom?: string[];
25
+ groupAllowFrom?: string[];
26
+ roomsConfig?: Record<string, MatrixRoomConfig>;
27
+ accountAllowBots?: boolean | "mentions";
28
+ configuredBotUserIds?: Set<string>;
29
+ mentionRegexes?: RegExp[];
30
+ groupPolicy?: "open" | "allowlist" | "disabled";
31
+ replyToMode?: ReplyToMode;
32
+ threadReplies?: "off" | "inbound" | "always";
33
+ dmThreadReplies?: "off" | "inbound" | "always";
34
+ dmSessionScope?: "per-user" | "per-room";
35
+ streaming?: MatrixStreamingMode;
36
+ blockStreamingEnabled?: boolean;
37
+ dmEnabled?: boolean;
38
+ dmPolicy?: "pairing" | "allowlist" | "open" | "disabled";
39
+ textLimit?: number;
40
+ mediaMaxBytes?: number;
41
+ startupMs?: number;
42
+ startupGraceMs?: number;
43
+ dropPreStartupMessages?: boolean;
44
+ needsRoomAliasesForConfig?: boolean;
45
+ isDirectMessage?: boolean;
46
+ historyLimit?: number;
47
+ readAllowFromStore?: MatrixMonitorHandlerParams["core"]["channel"]["pairing"]["readAllowFromStore"];
48
+ upsertPairingRequest?: MatrixMonitorHandlerParams["core"]["channel"]["pairing"]["upsertPairingRequest"];
49
+ buildPairingReply?: () => string;
50
+ shouldHandleTextCommands?: () => boolean;
51
+ hasControlCommand?: () => boolean;
52
+ resolveMarkdownTableMode?: () => string;
53
+ resolveAgentRoute?: () => typeof DEFAULT_ROUTE;
54
+ resolveStorePath?: () => string;
55
+ readSessionUpdatedAt?: () => number | undefined;
56
+ recordInboundSession?: (...args: unknown[]) => Promise<void>;
57
+ resolveEnvelopeFormatOptions?: () => Record<string, never>;
58
+ formatAgentEnvelope?: ({ body }: { body: string }) => string;
59
+ finalizeInboundContext?: (ctx: unknown) => unknown;
60
+ createReplyDispatcherWithTyping?: (params?: {
61
+ onError?: (err: unknown, info: { kind: "tool" | "block" | "final" }) => void;
62
+ }) => {
63
+ dispatcher: Record<string, unknown>;
64
+ replyOptions: Record<string, unknown>;
65
+ markDispatchIdle: () => void;
66
+ markRunComplete: () => void;
67
+ };
68
+ resolveHumanDelayConfig?: () => undefined;
69
+ dispatchReplyFromConfig?: () => Promise<{
70
+ queuedFinal: boolean;
71
+ counts: { final: number; block: number; tool: number };
72
+ }>;
73
+ withReplyDispatcher?: <T>(params: {
74
+ dispatcher: {
75
+ markComplete?: () => void;
76
+ waitForIdle?: () => Promise<void>;
77
+ };
78
+ run: () => Promise<T>;
79
+ onSettled?: () => void | Promise<void>;
80
+ }) => Promise<T>;
81
+ inboundDeduper?: MatrixMonitorHandlerParams["inboundDeduper"];
82
+ shouldAckReaction?: () => boolean;
83
+ enqueueSystemEvent?: (...args: unknown[]) => void;
84
+ getRoomInfo?: MatrixMonitorHandlerParams["getRoomInfo"];
85
+ getMemberDisplayName?: MatrixMonitorHandlerParams["getMemberDisplayName"];
86
+ };
87
+
88
+ type MatrixHandlerTestHarness = {
89
+ dispatchReplyFromConfig: () => Promise<{
90
+ queuedFinal: boolean;
91
+ counts: { final: number; block: number; tool: number };
92
+ }>;
93
+ enqueueSystemEvent: (...args: unknown[]) => void;
94
+ finalizeInboundContext: (ctx: unknown) => unknown;
95
+ handler: ReturnType<typeof createMatrixRoomMessageHandler>;
96
+ readAllowFromStore: MatrixMonitorHandlerParams["core"]["channel"]["pairing"]["readAllowFromStore"];
97
+ recordInboundSession: (...args: unknown[]) => Promise<void>;
98
+ resolveAgentRoute: () => typeof DEFAULT_ROUTE;
99
+ upsertPairingRequest: MatrixMonitorHandlerParams["core"]["channel"]["pairing"]["upsertPairingRequest"];
100
+ };
101
+
102
+ export function createMatrixHandlerTestHarness(
103
+ options: MatrixHandlerTestHarnessOptions = {},
104
+ ): MatrixHandlerTestHarness {
105
+ const readAllowFromStore = options.readAllowFromStore ?? vi.fn(async () => [] as string[]);
106
+ const upsertPairingRequest =
107
+ options.upsertPairingRequest ?? vi.fn(async () => ({ code: "ABCDEFGH", created: false }));
108
+ const resolveAgentRoute = options.resolveAgentRoute ?? vi.fn(() => DEFAULT_ROUTE);
109
+ const recordInboundSession = options.recordInboundSession ?? vi.fn(async () => {});
110
+ const finalizeInboundContext = options.finalizeInboundContext ?? vi.fn((ctx) => ctx);
111
+ const dispatchReplyFromConfig =
112
+ options.dispatchReplyFromConfig ??
113
+ (async () => ({
114
+ queuedFinal: false,
115
+ counts: { final: 0, block: 0, tool: 0 },
116
+ }));
117
+ const enqueueSystemEvent = options.enqueueSystemEvent ?? vi.fn();
118
+
119
+ const handler = createMatrixRoomMessageHandler({
120
+ client: {
121
+ getUserId: async () => "@bot:example.org",
122
+ getEvent: async () => ({ sender: "@bot:example.org" }),
123
+ ...options.client,
124
+ } as never,
125
+ core: {
126
+ channel: {
127
+ pairing: {
128
+ readAllowFromStore,
129
+ upsertPairingRequest,
130
+ buildPairingReply: options.buildPairingReply ?? (() => "pairing"),
131
+ },
132
+ commands: {
133
+ shouldHandleTextCommands: options.shouldHandleTextCommands ?? (() => false),
134
+ },
135
+ text: {
136
+ hasControlCommand: options.hasControlCommand ?? (() => false),
137
+ resolveMarkdownTableMode: options.resolveMarkdownTableMode ?? (() => "preserve"),
138
+ },
139
+ routing: {
140
+ resolveAgentRoute,
141
+ },
142
+ mentions: {
143
+ buildMentionRegexes: () => options.mentionRegexes ?? [],
144
+ },
145
+ session: {
146
+ resolveStorePath: options.resolveStorePath ?? (() => "/tmp/session-store"),
147
+ readSessionUpdatedAt: options.readSessionUpdatedAt ?? (() => undefined),
148
+ recordInboundSession,
149
+ },
150
+ reply: {
151
+ resolveEnvelopeFormatOptions: options.resolveEnvelopeFormatOptions ?? (() => ({})),
152
+ formatAgentEnvelope:
153
+ options.formatAgentEnvelope ?? (({ body }: { body: string }) => body),
154
+ finalizeInboundContext,
155
+ createReplyDispatcherWithTyping:
156
+ options.createReplyDispatcherWithTyping ??
157
+ (() => ({
158
+ dispatcher: {},
159
+ replyOptions: {},
160
+ markDispatchIdle: () => {},
161
+ markRunComplete: () => {},
162
+ })),
163
+ resolveHumanDelayConfig: options.resolveHumanDelayConfig ?? (() => undefined),
164
+ dispatchReplyFromConfig,
165
+ withReplyDispatcher:
166
+ options.withReplyDispatcher ??
167
+ (async <T>(params: {
168
+ dispatcher: {
169
+ markComplete?: () => void;
170
+ waitForIdle?: () => Promise<void>;
171
+ };
172
+ run: () => Promise<T>;
173
+ onSettled?: () => void | Promise<void>;
174
+ }) => {
175
+ const { dispatcher, run, onSettled } = params;
176
+ try {
177
+ return await run();
178
+ } finally {
179
+ dispatcher.markComplete?.();
180
+ try {
181
+ await dispatcher.waitForIdle?.();
182
+ } finally {
183
+ await onSettled?.();
184
+ }
185
+ }
186
+ }),
187
+ },
188
+ reactions: {
189
+ shouldAckReaction: options.shouldAckReaction ?? (() => false),
190
+ },
191
+ },
192
+ system: {
193
+ enqueueSystemEvent,
194
+ },
195
+ } as never,
196
+ cfg: (options.cfg ?? {}) as never,
197
+ accountId: options.accountId ?? "ops",
198
+ runtime:
199
+ options.runtime ??
200
+ ({
201
+ error: () => {},
202
+ } as RuntimeEnv),
203
+ logger:
204
+ options.logger ??
205
+ ({
206
+ info: () => {},
207
+ warn: () => {},
208
+ error: () => {},
209
+ } as RuntimeLogger),
210
+ logVerboseMessage: options.logVerboseMessage ?? (() => {}),
211
+ allowFrom: options.allowFrom ?? [],
212
+ groupAllowFrom: options.groupAllowFrom ?? [],
213
+ roomsConfig: options.roomsConfig,
214
+ accountAllowBots: options.accountAllowBots,
215
+ configuredBotUserIds: options.configuredBotUserIds,
216
+ groupPolicy: options.groupPolicy ?? "open",
217
+ replyToMode: options.replyToMode ?? "off",
218
+ threadReplies: options.threadReplies ?? "inbound",
219
+ dmThreadReplies: options.dmThreadReplies,
220
+ dmSessionScope: options.dmSessionScope,
221
+ streaming: options.streaming ?? "off",
222
+ blockStreamingEnabled: options.blockStreamingEnabled ?? false,
223
+ dmEnabled: options.dmEnabled ?? true,
224
+ dmPolicy: options.dmPolicy ?? "open",
225
+ textLimit: options.textLimit ?? 8_000,
226
+ mediaMaxBytes: options.mediaMaxBytes ?? 10_000_000,
227
+ startupMs: options.startupMs ?? 0,
228
+ startupGraceMs: options.startupGraceMs ?? 0,
229
+ dropPreStartupMessages: options.dropPreStartupMessages ?? true,
230
+ inboundDeduper: options.inboundDeduper,
231
+ directTracker: {
232
+ isDirectMessage: async () => options.isDirectMessage ?? true,
233
+ },
234
+ getRoomInfo: options.getRoomInfo ?? (async () => ({ altAliases: [] })),
235
+ getMemberDisplayName: options.getMemberDisplayName ?? (async () => "sender"),
236
+ needsRoomAliasesForConfig: options.needsRoomAliasesForConfig ?? false,
237
+ historyLimit: options.historyLimit ?? 0,
238
+ });
239
+
240
+ return {
241
+ dispatchReplyFromConfig,
242
+ enqueueSystemEvent,
243
+ finalizeInboundContext,
244
+ handler,
245
+ readAllowFromStore,
246
+ recordInboundSession,
247
+ resolveAgentRoute,
248
+ upsertPairingRequest,
249
+ };
250
+ }
251
+
252
+ export function createMatrixTextMessageEvent(params: {
253
+ eventId: string;
254
+ sender?: string;
255
+ body: string;
256
+ originServerTs?: number;
257
+ relatesTo?: RoomMessageEventContent["m.relates_to"];
258
+ mentions?: RoomMessageEventContent["m.mentions"];
259
+ }): MatrixRawEvent {
260
+ return createMatrixRoomMessageEvent({
261
+ eventId: params.eventId,
262
+ sender: params.sender,
263
+ originServerTs: params.originServerTs,
264
+ content: {
265
+ msgtype: "m.text",
266
+ body: params.body,
267
+ ...(params.relatesTo ? { "m.relates_to": params.relatesTo } : {}),
268
+ ...(params.mentions ? { "m.mentions": params.mentions } : {}),
269
+ },
270
+ });
271
+ }
272
+
273
+ export function createMatrixRoomMessageEvent(params: {
274
+ eventId: string;
275
+ sender?: string;
276
+ originServerTs?: number;
277
+ content: RoomMessageEventContent;
278
+ }): MatrixRawEvent {
279
+ return {
280
+ type: EventType.RoomMessage,
281
+ sender: params.sender ?? "@user:example.org",
282
+ event_id: params.eventId,
283
+ origin_server_ts: params.originServerTs ?? Date.now(),
284
+ content: params.content,
285
+ } as MatrixRawEvent;
286
+ }
287
+
288
+ export function createMatrixReactionEvent(params: {
289
+ eventId: string;
290
+ targetEventId: string;
291
+ key: string;
292
+ sender?: string;
293
+ originServerTs?: number;
294
+ }): MatrixRawEvent {
295
+ return {
296
+ type: EventType.Reaction,
297
+ sender: params.sender ?? "@user:example.org",
298
+ event_id: params.eventId,
299
+ origin_server_ts: params.originServerTs ?? Date.now(),
300
+ content: {
301
+ "m.relates_to": {
302
+ rel_type: "m.annotation",
303
+ event_id: params.targetEventId,
304
+ key: params.key,
305
+ },
306
+ },
307
+ } as MatrixRawEvent;
308
+ }