@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,255 @@
1
+ import { beforeEach, describe, expect, it, vi } from "vitest";
2
+ import { matrixPlugin } from "../../channel.js";
3
+ import {
4
+ __testing as sessionBindingTesting,
5
+ createTestRegistry,
6
+ registerSessionBindingAdapter,
7
+ resolveAgentRoute,
8
+ setActivePluginRegistry,
9
+ type OpenClawConfig,
10
+ } from "../../test-support/monitor-route-test-support.js";
11
+ import { resolveMatrixInboundRoute } from "./route.js";
12
+
13
+ const baseCfg = {
14
+ session: { mainKey: "main" },
15
+ agents: {
16
+ list: [{ id: "main" }, { id: "sender-agent" }, { id: "room-agent" }, { id: "acp-agent" }],
17
+ },
18
+ } satisfies OpenClawConfig;
19
+
20
+ type RouteBinding = NonNullable<OpenClawConfig["bindings"]>[number];
21
+ type RoutePeer = NonNullable<RouteBinding["match"]["peer"]>;
22
+
23
+ function matrixBinding(
24
+ agentId: string,
25
+ peer?: RoutePeer,
26
+ type?: RouteBinding["type"],
27
+ ): RouteBinding {
28
+ return {
29
+ ...(type ? { type } : {}),
30
+ agentId,
31
+ match: {
32
+ channel: "matrix",
33
+ accountId: "ops",
34
+ ...(peer ? { peer } : {}),
35
+ },
36
+ } as RouteBinding;
37
+ }
38
+
39
+ function senderPeer(id = "@alice:example.org"): RoutePeer {
40
+ return { kind: "direct", id };
41
+ }
42
+
43
+ function dmRoomPeer(id = "!dm:example.org"): RoutePeer {
44
+ return { kind: "channel", id };
45
+ }
46
+
47
+ function resolveDmRoute(
48
+ cfg: OpenClawConfig,
49
+ opts: {
50
+ dmSessionScope?: "per-user" | "per-room";
51
+ } = {},
52
+ ) {
53
+ return resolveMatrixInboundRoute({
54
+ cfg,
55
+ accountId: "ops",
56
+ roomId: "!dm:example.org",
57
+ senderId: "@alice:example.org",
58
+ isDirectMessage: true,
59
+ dmSessionScope: opts.dmSessionScope,
60
+ resolveAgentRoute,
61
+ });
62
+ }
63
+
64
+ describe("resolveMatrixInboundRoute", () => {
65
+ beforeEach(() => {
66
+ sessionBindingTesting.resetSessionBindingAdaptersForTests();
67
+ setActivePluginRegistry(
68
+ createTestRegistry([{ pluginId: "matrix", source: "test", plugin: matrixPlugin }]),
69
+ );
70
+ });
71
+
72
+ it("prefers sender-bound DM routing over DM room fallback bindings", () => {
73
+ const cfg = {
74
+ ...baseCfg,
75
+ bindings: [
76
+ matrixBinding("room-agent", dmRoomPeer()),
77
+ matrixBinding("sender-agent", senderPeer()),
78
+ ],
79
+ } satisfies OpenClawConfig;
80
+
81
+ const { route, configuredBinding } = resolveDmRoute(cfg);
82
+
83
+ expect(configuredBinding).toBeNull();
84
+ expect(route.agentId).toBe("sender-agent");
85
+ expect(route.matchedBy).toBe("binding.peer");
86
+ expect(route.sessionKey).toBe("agent:sender-agent:main");
87
+ });
88
+
89
+ it("uses the DM room as a parent-peer fallback before account-level bindings", () => {
90
+ const cfg = {
91
+ ...baseCfg,
92
+ bindings: [matrixBinding("acp-agent"), matrixBinding("room-agent", dmRoomPeer())],
93
+ } satisfies OpenClawConfig;
94
+
95
+ const { route, configuredBinding } = resolveDmRoute(cfg);
96
+
97
+ expect(configuredBinding).toBeNull();
98
+ expect(route.agentId).toBe("room-agent");
99
+ expect(route.matchedBy).toBe("binding.peer.parent");
100
+ expect(route.sessionKey).toBe("agent:room-agent:main");
101
+ });
102
+
103
+ it("can isolate Matrix DMs per room without changing agent selection", () => {
104
+ const cfg = {
105
+ ...baseCfg,
106
+ bindings: [matrixBinding("sender-agent", senderPeer())],
107
+ } satisfies OpenClawConfig;
108
+
109
+ const { route, configuredBinding } = resolveDmRoute(cfg, {
110
+ dmSessionScope: "per-room",
111
+ });
112
+
113
+ expect(configuredBinding).toBeNull();
114
+ expect(route.agentId).toBe("sender-agent");
115
+ expect(route.matchedBy).toBe("binding.peer");
116
+ expect(route.sessionKey).toBe("agent:sender-agent:matrix:channel:!dm:example.org");
117
+ expect(route.mainSessionKey).toBe("agent:sender-agent:main");
118
+ expect(route.lastRoutePolicy).toBe("session");
119
+ });
120
+
121
+ it("lets configured ACP room bindings override DM parent-peer routing", () => {
122
+ const cfg = {
123
+ ...baseCfg,
124
+ bindings: [
125
+ matrixBinding("room-agent", dmRoomPeer()),
126
+ matrixBinding("acp-agent", dmRoomPeer(), "acp"),
127
+ ],
128
+ } satisfies OpenClawConfig;
129
+
130
+ const { route, configuredBinding } = resolveDmRoute(cfg);
131
+
132
+ expect(configuredBinding?.spec.agentId).toBe("acp-agent");
133
+ expect(route.agentId).toBe("acp-agent");
134
+ expect(route.matchedBy).toBe("binding.channel");
135
+ expect(route.sessionKey).toContain("agent:acp-agent:acp:binding:matrix:ops:");
136
+ expect(route.lastRoutePolicy).toBe("session");
137
+ });
138
+
139
+ it("keeps configured ACP room bindings ahead of per-room DM session scope", () => {
140
+ const cfg = {
141
+ ...baseCfg,
142
+ bindings: [
143
+ matrixBinding("room-agent", dmRoomPeer()),
144
+ matrixBinding("acp-agent", dmRoomPeer(), "acp"),
145
+ ],
146
+ } satisfies OpenClawConfig;
147
+
148
+ const { route, configuredBinding } = resolveDmRoute(cfg, {
149
+ dmSessionScope: "per-room",
150
+ });
151
+
152
+ expect(configuredBinding?.spec.agentId).toBe("acp-agent");
153
+ expect(route.agentId).toBe("acp-agent");
154
+ expect(route.matchedBy).toBe("binding.channel");
155
+ expect(route.sessionKey).toContain("agent:acp-agent:acp:binding:matrix:ops:");
156
+ expect(route.sessionKey).not.toBe("agent:acp-agent:matrix:channel:!dm:example.org");
157
+ expect(route.lastRoutePolicy).toBe("session");
158
+ });
159
+
160
+ it("lets runtime conversation bindings override both sender and room route matches", () => {
161
+ const touch = vi.fn();
162
+ registerSessionBindingAdapter({
163
+ channel: "matrix",
164
+ accountId: "ops",
165
+ listBySession: () => [],
166
+ resolveByConversation: (ref) =>
167
+ ref.conversationId === "!dm:example.org"
168
+ ? {
169
+ bindingId: "ops:!dm:example.org",
170
+ targetSessionKey: "agent:bound:session-1",
171
+ targetKind: "session",
172
+ conversation: {
173
+ channel: "matrix",
174
+ accountId: "ops",
175
+ conversationId: "!dm:example.org",
176
+ },
177
+ status: "active",
178
+ boundAt: Date.now(),
179
+ metadata: { boundBy: "user-1" },
180
+ }
181
+ : null,
182
+ touch,
183
+ });
184
+
185
+ const cfg = {
186
+ ...baseCfg,
187
+ bindings: [
188
+ matrixBinding("sender-agent", senderPeer()),
189
+ matrixBinding("room-agent", dmRoomPeer()),
190
+ ],
191
+ } satisfies OpenClawConfig;
192
+
193
+ const { route, configuredBinding, runtimeBindingId } = resolveDmRoute(cfg);
194
+
195
+ expect(configuredBinding).toBeNull();
196
+ expect(runtimeBindingId).toBe("ops:!dm:example.org");
197
+ expect(route.agentId).toBe("bound");
198
+ expect(route.matchedBy).toBe("binding.channel");
199
+ expect(route.sessionKey).toBe("agent:bound:session-1");
200
+ expect(route.lastRoutePolicy).toBe("session");
201
+ expect(touch).not.toHaveBeenCalled();
202
+ });
203
+ });
204
+
205
+ describe("resolveMatrixInboundRoute thread-isolated sessions", () => {
206
+ beforeEach(() => {
207
+ sessionBindingTesting.resetSessionBindingAdaptersForTests();
208
+ setActivePluginRegistry(
209
+ createTestRegistry([{ pluginId: "matrix", source: "test", plugin: matrixPlugin }]),
210
+ );
211
+ });
212
+
213
+ it("scopes session key to thread when a thread id is provided", () => {
214
+ const { route } = resolveMatrixInboundRoute({
215
+ cfg: baseCfg as never,
216
+ accountId: "ops",
217
+ roomId: "!room:example.org",
218
+ senderId: "@alice:example.org",
219
+ isDirectMessage: false,
220
+ threadId: "$thread-root",
221
+ resolveAgentRoute,
222
+ });
223
+
224
+ expect(route.sessionKey).toContain(":thread:$thread-root");
225
+ expect(route.mainSessionKey).not.toContain(":thread:");
226
+ expect(route.lastRoutePolicy).toBe("session");
227
+ });
228
+
229
+ it("preserves mixed-case matrix thread ids in session keys", () => {
230
+ const { route } = resolveMatrixInboundRoute({
231
+ cfg: baseCfg as never,
232
+ accountId: "ops",
233
+ roomId: "!room:example.org",
234
+ senderId: "@alice:example.org",
235
+ isDirectMessage: false,
236
+ threadId: "$AbC123:example.org",
237
+ resolveAgentRoute,
238
+ });
239
+
240
+ expect(route.sessionKey).toContain(":thread:$AbC123:example.org");
241
+ });
242
+
243
+ it("does not scope session key when thread id is absent", () => {
244
+ const { route } = resolveMatrixInboundRoute({
245
+ cfg: baseCfg as never,
246
+ accountId: "ops",
247
+ roomId: "!room:example.org",
248
+ senderId: "@alice:example.org",
249
+ isDirectMessage: false,
250
+ resolveAgentRoute,
251
+ });
252
+
253
+ expect(route.sessionKey).not.toContain(":thread:");
254
+ });
255
+ });
@@ -0,0 +1,178 @@
1
+ import { buildAgentSessionKey, deriveLastRoutePolicy } from "openclaw/plugin-sdk/routing";
2
+ import {
3
+ getSessionBindingService,
4
+ resolveAgentIdFromSessionKey,
5
+ resolveConfiguredAcpBindingRecord,
6
+ type PluginRuntime,
7
+ } from "../../runtime-api.js";
8
+ import type { CoreConfig } from "../../types.js";
9
+ import { resolveMatrixThreadSessionKeys } from "./threads.js";
10
+
11
+ type MatrixResolvedRoute = ReturnType<PluginRuntime["channel"]["routing"]["resolveAgentRoute"]>;
12
+
13
+ function resolveMatrixDmSessionKey(params: {
14
+ accountId: string;
15
+ agentId: string;
16
+ roomId: string;
17
+ dmSessionScope?: "per-user" | "per-room";
18
+ fallbackSessionKey: string;
19
+ }): string {
20
+ if (params.dmSessionScope !== "per-room") {
21
+ return params.fallbackSessionKey;
22
+ }
23
+ return buildAgentSessionKey({
24
+ agentId: params.agentId,
25
+ channel: "matrix",
26
+ accountId: params.accountId,
27
+ peer: {
28
+ kind: "channel",
29
+ id: params.roomId,
30
+ },
31
+ });
32
+ }
33
+
34
+ function shouldApplyMatrixPerRoomDmSessionScope(params: {
35
+ isDirectMessage: boolean;
36
+ configuredSessionKey?: string;
37
+ }): boolean {
38
+ return params.isDirectMessage && !params.configuredSessionKey;
39
+ }
40
+
41
+ export function resolveMatrixInboundRoute(params: {
42
+ cfg: CoreConfig;
43
+ accountId: string;
44
+ roomId: string;
45
+ senderId: string;
46
+ isDirectMessage: boolean;
47
+ dmSessionScope?: "per-user" | "per-room";
48
+ threadId?: string;
49
+ eventTs?: number;
50
+ resolveAgentRoute: PluginRuntime["channel"]["routing"]["resolveAgentRoute"];
51
+ }): {
52
+ route: MatrixResolvedRoute;
53
+ configuredBinding: ReturnType<typeof resolveConfiguredAcpBindingRecord>;
54
+ runtimeBindingId: string | null;
55
+ } {
56
+ const baseRoute = params.resolveAgentRoute({
57
+ cfg: params.cfg,
58
+ channel: "matrix",
59
+ accountId: params.accountId,
60
+ peer: {
61
+ kind: params.isDirectMessage ? "direct" : "channel",
62
+ id: params.isDirectMessage ? params.senderId : params.roomId,
63
+ },
64
+ // Matrix DMs are still sender-addressed first, but the room ID remains a
65
+ // useful fallback binding key for generic route matching.
66
+ parentPeer: params.isDirectMessage
67
+ ? {
68
+ kind: "channel",
69
+ id: params.roomId,
70
+ }
71
+ : undefined,
72
+ });
73
+ const bindingConversationId = params.threadId ?? params.roomId;
74
+ const bindingParentConversationId = params.threadId ? params.roomId : undefined;
75
+ const sessionBindingService = getSessionBindingService();
76
+ const runtimeBinding = sessionBindingService.resolveByConversation({
77
+ channel: "matrix",
78
+ accountId: params.accountId,
79
+ conversationId: bindingConversationId,
80
+ parentConversationId: bindingParentConversationId,
81
+ });
82
+ const boundSessionKey = runtimeBinding?.targetSessionKey?.trim();
83
+
84
+ if (runtimeBinding && boundSessionKey) {
85
+ return {
86
+ route: {
87
+ ...baseRoute,
88
+ sessionKey: boundSessionKey,
89
+ agentId: resolveAgentIdFromSessionKey(boundSessionKey) || baseRoute.agentId,
90
+ lastRoutePolicy: deriveLastRoutePolicy({
91
+ sessionKey: boundSessionKey,
92
+ mainSessionKey: baseRoute.mainSessionKey,
93
+ }),
94
+ matchedBy: "binding.channel",
95
+ },
96
+ configuredBinding: null,
97
+ runtimeBindingId: runtimeBinding.bindingId,
98
+ };
99
+ }
100
+
101
+ const configuredBinding =
102
+ runtimeBinding == null
103
+ ? resolveConfiguredAcpBindingRecord({
104
+ cfg: params.cfg,
105
+ channel: "matrix",
106
+ accountId: params.accountId,
107
+ conversationId: bindingConversationId,
108
+ parentConversationId: bindingParentConversationId,
109
+ })
110
+ : null;
111
+ const configuredSessionKey = configuredBinding?.record.targetSessionKey?.trim();
112
+
113
+ const effectiveRoute =
114
+ configuredBinding && configuredSessionKey
115
+ ? {
116
+ ...baseRoute,
117
+ sessionKey: configuredSessionKey,
118
+ agentId:
119
+ resolveAgentIdFromSessionKey(configuredSessionKey) ||
120
+ configuredBinding.spec.agentId ||
121
+ baseRoute.agentId,
122
+ lastRoutePolicy: deriveLastRoutePolicy({
123
+ sessionKey: configuredSessionKey,
124
+ mainSessionKey: baseRoute.mainSessionKey,
125
+ }),
126
+ matchedBy: "binding.channel" as const,
127
+ }
128
+ : baseRoute;
129
+
130
+ const dmSessionKey = shouldApplyMatrixPerRoomDmSessionScope({
131
+ isDirectMessage: params.isDirectMessage,
132
+ configuredSessionKey,
133
+ })
134
+ ? resolveMatrixDmSessionKey({
135
+ accountId: params.accountId,
136
+ agentId: effectiveRoute.agentId,
137
+ roomId: params.roomId,
138
+ dmSessionScope: params.dmSessionScope,
139
+ fallbackSessionKey: effectiveRoute.sessionKey,
140
+ })
141
+ : effectiveRoute.sessionKey;
142
+ const routeWithDmScope =
143
+ dmSessionKey === effectiveRoute.sessionKey
144
+ ? effectiveRoute
145
+ : {
146
+ ...effectiveRoute,
147
+ sessionKey: dmSessionKey,
148
+ lastRoutePolicy: "session" as const,
149
+ };
150
+
151
+ // When no binding overrides the session key, isolate threads into their own sessions.
152
+ if (!configuredBinding && !configuredSessionKey && params.threadId) {
153
+ const threadKeys = resolveMatrixThreadSessionKeys({
154
+ baseSessionKey: routeWithDmScope.sessionKey,
155
+ threadId: params.threadId,
156
+ parentSessionKey: routeWithDmScope.sessionKey,
157
+ });
158
+ return {
159
+ route: {
160
+ ...routeWithDmScope,
161
+ sessionKey: threadKeys.sessionKey,
162
+ mainSessionKey: threadKeys.parentSessionKey ?? routeWithDmScope.sessionKey,
163
+ lastRoutePolicy: deriveLastRoutePolicy({
164
+ sessionKey: threadKeys.sessionKey,
165
+ mainSessionKey: threadKeys.parentSessionKey ?? routeWithDmScope.sessionKey,
166
+ }),
167
+ },
168
+ configuredBinding,
169
+ runtimeBindingId: null,
170
+ };
171
+ }
172
+
173
+ return {
174
+ route: routeWithDmScope,
175
+ configuredBinding,
176
+ runtimeBindingId: null,
177
+ };
178
+ }
@@ -0,0 +1,31 @@
1
+ // Narrow Matrix monitor helper seam.
2
+ // Keep monitor internals off the broad package runtime-api barrel so monitor
3
+ // tests and shared workers do not pull unrelated Matrix helper surfaces.
4
+
5
+ export { ensureConfiguredAcpBindingReady } from "openclaw/plugin-sdk/acp-binding-runtime";
6
+ export type { NormalizedLocation } from "openclaw/plugin-sdk/channel-inbound";
7
+ export type { PluginRuntime, RuntimeLogger } from "openclaw/plugin-sdk/plugin-runtime";
8
+ export type { BlockReplyContext, ReplyPayload } from "openclaw/plugin-sdk/reply-runtime";
9
+ export type { MarkdownTableMode, OpenClawConfig } from "openclaw/plugin-sdk/config-runtime";
10
+ export type { RuntimeEnv } from "openclaw/plugin-sdk/runtime";
11
+ export {
12
+ addAllowlistUserEntriesFromConfigEntry,
13
+ buildAllowlistResolutionSummary,
14
+ canonicalizeAllowlistWithResolvedIds,
15
+ formatAllowlistMatchMeta,
16
+ patchAllowlistUsersInConfigEntries,
17
+ summarizeMapping,
18
+ } from "openclaw/plugin-sdk/allow-from";
19
+ export { createReplyPrefixOptions } from "openclaw/plugin-sdk/channel-reply-pipeline";
20
+ export { createTypingCallbacks } from "openclaw/plugin-sdk/channel-reply-pipeline";
21
+ export {
22
+ formatLocationText,
23
+ logInboundDrop,
24
+ toLocationContext,
25
+ } from "openclaw/plugin-sdk/channel-inbound";
26
+ export { getAgentScopedMediaLocalRoots } from "openclaw/plugin-sdk/agent-media-payload";
27
+ export { logTypingFailure, resolveAckReaction } from "openclaw/plugin-sdk/channel-feedback";
28
+ export {
29
+ buildChannelKeyCandidates,
30
+ resolveChannelEntryMatch,
31
+ } from "openclaw/plugin-sdk/channel-targets";