@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,289 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+ import { setMatrixRuntime } from "../../runtime.js";
3
+ import type { MatrixClient } from "../sdk.js";
4
+ import * as sendModule from "../send.js";
5
+ import { editMatrixMessage, readMatrixMessages } from "./messages.js";
6
+
7
+ function installMatrixActionTestRuntime(): void {
8
+ setMatrixRuntime({
9
+ config: {
10
+ loadConfig: () => ({}),
11
+ },
12
+ channel: {
13
+ text: {
14
+ resolveMarkdownTableMode: () => "code",
15
+ convertMarkdownTables: (text: string) => text,
16
+ },
17
+ },
18
+ } as unknown as import("../../runtime-api.js").PluginRuntime);
19
+ }
20
+
21
+ function createPollResponseEvent(): Record<string, unknown> {
22
+ return {
23
+ event_id: "$vote",
24
+ sender: "@bob:example.org",
25
+ type: "m.poll.response",
26
+ origin_server_ts: 20,
27
+ content: {
28
+ "m.poll.response": { answers: ["a1"] },
29
+ "m.relates_to": { rel_type: "m.reference", event_id: "$poll" },
30
+ },
31
+ };
32
+ }
33
+
34
+ function createPollStartEvent(params?: {
35
+ answers?: Array<Record<string, unknown>>;
36
+ includeDisclosedKind?: boolean;
37
+ maxSelections?: number;
38
+ }): Record<string, unknown> {
39
+ return {
40
+ event_id: "$poll",
41
+ sender: "@alice:example.org",
42
+ type: "m.poll.start",
43
+ origin_server_ts: 1,
44
+ content: {
45
+ "m.poll.start": {
46
+ question: { "m.text": "Favorite fruit?" },
47
+ ...(params?.includeDisclosedKind ? { kind: "m.poll.disclosed" } : {}),
48
+ ...(params?.maxSelections !== undefined ? { max_selections: params.maxSelections } : {}),
49
+ answers: params?.answers ?? [{ id: "a1", "m.text": "Apple" }],
50
+ },
51
+ },
52
+ };
53
+ }
54
+
55
+ function createMessagesClient(params: {
56
+ chunk: Array<Record<string, unknown>>;
57
+ hydratedChunk?: Array<Record<string, unknown>>;
58
+ pollRoot?: Record<string, unknown>;
59
+ pollRelations?: Array<Record<string, unknown>>;
60
+ }) {
61
+ const doRequest = vi.fn(async () => ({
62
+ chunk: params.chunk,
63
+ start: "start-token",
64
+ end: "end-token",
65
+ }));
66
+ const hydrateEvents = vi.fn(
67
+ async (_roomId: string, _events: Array<Record<string, unknown>>) =>
68
+ (params.hydratedChunk ?? params.chunk) as unknown,
69
+ );
70
+ const getEvent = vi.fn(async () => params.pollRoot ?? null);
71
+ const getRelations = vi.fn(async () => ({
72
+ events: params.pollRelations ?? [],
73
+ nextBatch: null,
74
+ prevBatch: null,
75
+ }));
76
+
77
+ return {
78
+ client: {
79
+ doRequest,
80
+ hydrateEvents,
81
+ getEvent,
82
+ getRelations,
83
+ stop: vi.fn(),
84
+ } as unknown as MatrixClient,
85
+ doRequest,
86
+ hydrateEvents,
87
+ getEvent,
88
+ getRelations,
89
+ };
90
+ }
91
+
92
+ describe("matrix message actions", () => {
93
+ it("forwards timeoutMs to the shared Matrix edit helper", async () => {
94
+ const editSpy = vi.spyOn(sendModule, "editMessageMatrix").mockResolvedValue("evt-edit");
95
+
96
+ try {
97
+ const result = await editMatrixMessage("!room:example.org", "$original", "hello", {
98
+ timeoutMs: 12_345,
99
+ });
100
+
101
+ expect(result).toEqual({ eventId: "evt-edit" });
102
+ expect(editSpy).toHaveBeenCalledWith("!room:example.org", "$original", "hello", {
103
+ cfg: undefined,
104
+ accountId: undefined,
105
+ client: undefined,
106
+ timeoutMs: 12_345,
107
+ });
108
+ } finally {
109
+ editSpy.mockRestore();
110
+ }
111
+ });
112
+
113
+ it("routes edits through the shared Matrix edit helper so mentions are preserved", async () => {
114
+ installMatrixActionTestRuntime();
115
+ const sendMessage = vi.fn().mockResolvedValue("evt-edit");
116
+ const client = {
117
+ getEvent: vi.fn().mockResolvedValue({
118
+ content: {
119
+ body: "hello @alice:example.org",
120
+ "m.mentions": { user_ids: ["@alice:example.org"] },
121
+ },
122
+ }),
123
+ getJoinedRoomMembers: vi.fn().mockResolvedValue([]),
124
+ getUserId: vi.fn().mockResolvedValue("@bot:example.org"),
125
+ sendMessage,
126
+ prepareForOneOff: vi.fn(async () => undefined),
127
+ start: vi.fn(async () => undefined),
128
+ stop: vi.fn(() => undefined),
129
+ stopAndPersist: vi.fn(async () => undefined),
130
+ } as unknown as MatrixClient;
131
+
132
+ const result = await editMatrixMessage(
133
+ "!room:example.org",
134
+ "$original",
135
+ "hello @alice:example.org and @bob:example.org",
136
+ { client },
137
+ );
138
+
139
+ expect(result).toEqual({ eventId: "evt-edit" });
140
+ expect(sendMessage).toHaveBeenCalledWith(
141
+ "!room:example.org",
142
+ expect.objectContaining({
143
+ "m.mentions": { user_ids: ["@bob:example.org"] },
144
+ "m.new_content": expect.objectContaining({
145
+ "m.mentions": { user_ids: ["@alice:example.org", "@bob:example.org"] },
146
+ }),
147
+ }),
148
+ );
149
+ });
150
+
151
+ it("does not re-notify legacy mentions when action edits target pre-m.mentions messages", async () => {
152
+ installMatrixActionTestRuntime();
153
+ const sendMessage = vi.fn().mockResolvedValue("evt-edit");
154
+ const client = {
155
+ getEvent: vi.fn().mockResolvedValue({
156
+ content: {
157
+ body: "hello @alice:example.org",
158
+ },
159
+ }),
160
+ getJoinedRoomMembers: vi.fn().mockResolvedValue([]),
161
+ getUserId: vi.fn().mockResolvedValue("@bot:example.org"),
162
+ sendMessage,
163
+ prepareForOneOff: vi.fn(async () => undefined),
164
+ start: vi.fn(async () => undefined),
165
+ stop: vi.fn(() => undefined),
166
+ stopAndPersist: vi.fn(async () => undefined),
167
+ } as unknown as MatrixClient;
168
+
169
+ const result = await editMatrixMessage(
170
+ "!room:example.org",
171
+ "$original",
172
+ "hello again @alice:example.org",
173
+ { client },
174
+ );
175
+
176
+ expect(result).toEqual({ eventId: "evt-edit" });
177
+ expect(sendMessage).toHaveBeenCalledWith(
178
+ "!room:example.org",
179
+ expect.objectContaining({
180
+ "m.mentions": {},
181
+ "m.new_content": expect.objectContaining({
182
+ body: "hello again @alice:example.org",
183
+ "m.mentions": { user_ids: ["@alice:example.org"] },
184
+ }),
185
+ }),
186
+ );
187
+ });
188
+
189
+ it("includes poll snapshots when reading message history", async () => {
190
+ const { client, doRequest, getEvent, getRelations } = createMessagesClient({
191
+ chunk: [
192
+ createPollResponseEvent(),
193
+ {
194
+ event_id: "$msg",
195
+ sender: "@alice:example.org",
196
+ type: "m.room.message",
197
+ origin_server_ts: 10,
198
+ content: {
199
+ msgtype: "m.text",
200
+ body: "hello",
201
+ },
202
+ },
203
+ ],
204
+ pollRoot: createPollStartEvent({
205
+ includeDisclosedKind: true,
206
+ maxSelections: 1,
207
+ answers: [
208
+ { id: "a1", "m.text": "Apple" },
209
+ { id: "a2", "m.text": "Strawberry" },
210
+ ],
211
+ }),
212
+ pollRelations: [createPollResponseEvent()],
213
+ });
214
+
215
+ const result = await readMatrixMessages("room:!room:example.org", { client, limit: 2.9 });
216
+
217
+ expect(doRequest).toHaveBeenCalledWith(
218
+ "GET",
219
+ expect.stringContaining("/rooms/!room%3Aexample.org/messages"),
220
+ expect.objectContaining({ limit: 2 }),
221
+ );
222
+ expect(getEvent).toHaveBeenCalledWith("!room:example.org", "$poll");
223
+ expect(getRelations).toHaveBeenCalledWith(
224
+ "!room:example.org",
225
+ "$poll",
226
+ "m.reference",
227
+ undefined,
228
+ {
229
+ from: undefined,
230
+ },
231
+ );
232
+ expect(result.messages).toEqual([
233
+ expect.objectContaining({
234
+ eventId: "$poll",
235
+ body: expect.stringContaining("1. Apple (1 vote)"),
236
+ msgtype: "m.text",
237
+ }),
238
+ expect.objectContaining({
239
+ eventId: "$msg",
240
+ body: "hello",
241
+ }),
242
+ ]);
243
+ });
244
+
245
+ it("dedupes multiple poll events for the same poll within one read page", async () => {
246
+ const { client, getEvent } = createMessagesClient({
247
+ chunk: [createPollResponseEvent(), createPollStartEvent()],
248
+ pollRoot: createPollStartEvent(),
249
+ pollRelations: [],
250
+ });
251
+
252
+ const result = await readMatrixMessages("room:!room:example.org", { client });
253
+
254
+ expect(result.messages).toHaveLength(1);
255
+ expect(result.messages[0]).toEqual(
256
+ expect.objectContaining({
257
+ eventId: "$poll",
258
+ body: expect.stringContaining("[Poll]"),
259
+ }),
260
+ );
261
+ expect(getEvent).toHaveBeenCalledTimes(1);
262
+ });
263
+
264
+ it("uses hydrated history events so encrypted poll entries can be read", async () => {
265
+ const { client, hydrateEvents } = createMessagesClient({
266
+ chunk: [
267
+ {
268
+ event_id: "$enc",
269
+ sender: "@bob:example.org",
270
+ type: "m.room.encrypted",
271
+ origin_server_ts: 20,
272
+ content: {},
273
+ },
274
+ ],
275
+ hydratedChunk: [createPollResponseEvent()],
276
+ pollRoot: createPollStartEvent(),
277
+ pollRelations: [],
278
+ });
279
+
280
+ const result = await readMatrixMessages("room:!room:example.org", { client });
281
+
282
+ expect(hydrateEvents).toHaveBeenCalledWith(
283
+ "!room:example.org",
284
+ expect.arrayContaining([expect.objectContaining({ event_id: "$enc" })]),
285
+ );
286
+ expect(result.messages).toHaveLength(1);
287
+ expect(result.messages[0]?.eventId).toBe("$poll");
288
+ });
289
+ });
@@ -0,0 +1,123 @@
1
+ import { normalizeOptionalString } from "openclaw/plugin-sdk/text-runtime";
2
+ import { fetchMatrixPollMessageSummary, resolveMatrixPollRootEventId } from "../poll-summary.js";
3
+ import { isPollEventType } from "../poll-types.js";
4
+ import { editMessageMatrix, sendMessageMatrix } from "../send.js";
5
+ import { withResolvedRoomAction } from "./client.js";
6
+ import { resolveMatrixActionLimit } from "./limits.js";
7
+ import { summarizeMatrixRawEvent } from "./summary.js";
8
+ import {
9
+ EventType,
10
+ type MatrixActionClientOpts,
11
+ type MatrixMessageSummary,
12
+ type MatrixRawEvent,
13
+ } from "./types.js";
14
+
15
+ export async function sendMatrixMessage(
16
+ to: string,
17
+ content: string | undefined,
18
+ opts: MatrixActionClientOpts & {
19
+ mediaUrl?: string;
20
+ replyToId?: string;
21
+ threadId?: string;
22
+ audioAsVoice?: boolean;
23
+ } = {},
24
+ ) {
25
+ return await sendMessageMatrix(to, content, {
26
+ cfg: opts.cfg,
27
+ mediaUrl: opts.mediaUrl,
28
+ mediaLocalRoots: opts.mediaLocalRoots,
29
+ replyToId: opts.replyToId,
30
+ threadId: opts.threadId,
31
+ audioAsVoice: opts.audioAsVoice,
32
+ accountId: opts.accountId ?? undefined,
33
+ client: opts.client,
34
+ timeoutMs: opts.timeoutMs,
35
+ });
36
+ }
37
+
38
+ export async function editMatrixMessage(
39
+ roomId: string,
40
+ messageId: string,
41
+ content: string,
42
+ opts: MatrixActionClientOpts = {},
43
+ ) {
44
+ const trimmed = content.trim();
45
+ if (!trimmed) {
46
+ throw new Error("Matrix edit requires content");
47
+ }
48
+ const eventId = await editMessageMatrix(roomId, messageId, trimmed, {
49
+ cfg: opts.cfg,
50
+ accountId: opts.accountId ?? undefined,
51
+ client: opts.client,
52
+ timeoutMs: opts.timeoutMs,
53
+ });
54
+ return { eventId: eventId || null };
55
+ }
56
+
57
+ export async function deleteMatrixMessage(
58
+ roomId: string,
59
+ messageId: string,
60
+ opts: MatrixActionClientOpts & { reason?: string } = {},
61
+ ) {
62
+ await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
63
+ await client.redactEvent(resolvedRoom, messageId, opts.reason);
64
+ });
65
+ }
66
+
67
+ export async function readMatrixMessages(
68
+ roomId: string,
69
+ opts: MatrixActionClientOpts & {
70
+ limit?: number;
71
+ before?: string;
72
+ after?: string;
73
+ } = {},
74
+ ): Promise<{
75
+ messages: MatrixMessageSummary[];
76
+ nextBatch?: string | null;
77
+ prevBatch?: string | null;
78
+ }> {
79
+ return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
80
+ const limit = resolveMatrixActionLimit(opts.limit, 20);
81
+ const token = normalizeOptionalString(opts.before) ?? normalizeOptionalString(opts.after);
82
+ const dir = opts.after ? "f" : "b";
83
+ // Room history is queried via the low-level endpoint for compatibility.
84
+ const res = (await client.doRequest(
85
+ "GET",
86
+ `/_lobi/client/v3/rooms/${encodeURIComponent(resolvedRoom)}/messages`,
87
+ {
88
+ dir,
89
+ limit,
90
+ from: token,
91
+ },
92
+ )) as { chunk: MatrixRawEvent[]; start?: string; end?: string };
93
+ const hydratedChunk = await client.hydrateEvents(resolvedRoom, res.chunk);
94
+ const seenPollRoots = new Set<string>();
95
+ const messages: MatrixMessageSummary[] = [];
96
+ for (const event of hydratedChunk) {
97
+ if (event.unsigned?.redacted_because) {
98
+ continue;
99
+ }
100
+ if (event.type === EventType.RoomMessage) {
101
+ messages.push(summarizeMatrixRawEvent(event));
102
+ continue;
103
+ }
104
+ if (!isPollEventType(event.type)) {
105
+ continue;
106
+ }
107
+ const pollRootId = resolveMatrixPollRootEventId(event);
108
+ if (!pollRootId || seenPollRoots.has(pollRootId)) {
109
+ continue;
110
+ }
111
+ seenPollRoots.add(pollRootId);
112
+ const pollSummary = await fetchMatrixPollMessageSummary(client, resolvedRoom, event);
113
+ if (pollSummary) {
114
+ messages.push(pollSummary);
115
+ }
116
+ }
117
+ return {
118
+ messages,
119
+ nextBatch: res.end ?? null,
120
+ prevBatch: res.start ?? null,
121
+ };
122
+ });
123
+ }
@@ -0,0 +1,74 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+ import type { MatrixClient } from "../sdk.js";
3
+ import { listMatrixPins, pinMatrixMessage, unpinMatrixMessage } from "./pins.js";
4
+
5
+ function createPinsClient(seedPinned: string[], knownBodies: Record<string, string> = {}) {
6
+ let pinned = [...seedPinned];
7
+ const getRoomStateEvent = vi.fn(async () => ({ pinned: [...pinned] }));
8
+ const sendStateEvent = vi.fn(
9
+ async (_roomId: string, _type: string, _key: string, payload: unknown) => {
10
+ pinned = [...((payload as { pinned: string[] }).pinned ?? [])];
11
+ },
12
+ );
13
+ const getEvent = vi.fn(async (_roomId: string, eventId: string) => {
14
+ const body = knownBodies[eventId];
15
+ if (!body) {
16
+ throw new Error("missing");
17
+ }
18
+ return {
19
+ event_id: eventId,
20
+ sender: "@alice:example.org",
21
+ type: "m.room.message",
22
+ origin_server_ts: 123,
23
+ content: { msgtype: "m.text", body },
24
+ };
25
+ });
26
+
27
+ return {
28
+ client: {
29
+ getRoomStateEvent,
30
+ sendStateEvent,
31
+ getEvent,
32
+ stop: vi.fn(),
33
+ } as unknown as MatrixClient,
34
+ getPinned: () => pinned,
35
+ sendStateEvent,
36
+ };
37
+ }
38
+
39
+ describe("matrix pins actions", () => {
40
+ it("pins a message once even when asked twice", async () => {
41
+ const { client, getPinned, sendStateEvent } = createPinsClient(["$a"]);
42
+
43
+ const first = await pinMatrixMessage("!room:example.org", "$b", { client });
44
+ const second = await pinMatrixMessage("!room:example.org", "$b", { client });
45
+
46
+ expect(first.pinned).toEqual(["$a", "$b"]);
47
+ expect(second.pinned).toEqual(["$a", "$b"]);
48
+ expect(getPinned()).toEqual(["$a", "$b"]);
49
+ expect(sendStateEvent).toHaveBeenCalledTimes(2);
50
+ });
51
+
52
+ it("unpinds only the selected message id", async () => {
53
+ const { client, getPinned } = createPinsClient(["$a", "$b", "$c"]);
54
+
55
+ const result = await unpinMatrixMessage("!room:example.org", "$b", { client });
56
+
57
+ expect(result.pinned).toEqual(["$a", "$c"]);
58
+ expect(getPinned()).toEqual(["$a", "$c"]);
59
+ });
60
+
61
+ it("lists pinned ids and summarizes only resolvable events", async () => {
62
+ const { client } = createPinsClient(["$a", "$missing"], { $a: "hello" });
63
+
64
+ const result = await listMatrixPins("!room:example.org", { client });
65
+
66
+ expect(result.pinned).toEqual(["$a", "$missing"]);
67
+ expect(result.events).toEqual([
68
+ expect.objectContaining({
69
+ eventId: "$a",
70
+ body: "hello",
71
+ }),
72
+ ]);
73
+ });
74
+ });
@@ -0,0 +1,64 @@
1
+ import { withResolvedRoomAction } from "./client.js";
2
+ import { fetchEventSummary, readPinnedEvents } from "./summary.js";
3
+ import {
4
+ EventType,
5
+ type MatrixActionClientOpts,
6
+ type MatrixMessageSummary,
7
+ type RoomPinnedEventsEventContent,
8
+ } from "./types.js";
9
+
10
+ async function updateMatrixPins(
11
+ roomId: string,
12
+ messageId: string,
13
+ opts: MatrixActionClientOpts,
14
+ update: (current: string[]) => string[],
15
+ ): Promise<{ pinned: string[] }> {
16
+ return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
17
+ const current = await readPinnedEvents(client, resolvedRoom);
18
+ const next = update(current);
19
+ const payload: RoomPinnedEventsEventContent = { pinned: next };
20
+ await client.sendStateEvent(resolvedRoom, EventType.RoomPinnedEvents, "", payload);
21
+ return { pinned: next };
22
+ });
23
+ }
24
+
25
+ export async function pinMatrixMessage(
26
+ roomId: string,
27
+ messageId: string,
28
+ opts: MatrixActionClientOpts = {},
29
+ ): Promise<{ pinned: string[] }> {
30
+ return await updateMatrixPins(roomId, messageId, opts, (current) =>
31
+ current.includes(messageId) ? current : [...current, messageId],
32
+ );
33
+ }
34
+
35
+ export async function unpinMatrixMessage(
36
+ roomId: string,
37
+ messageId: string,
38
+ opts: MatrixActionClientOpts = {},
39
+ ): Promise<{ pinned: string[] }> {
40
+ return await updateMatrixPins(roomId, messageId, opts, (current) =>
41
+ current.filter((id) => id !== messageId),
42
+ );
43
+ }
44
+
45
+ export async function listMatrixPins(
46
+ roomId: string,
47
+ opts: MatrixActionClientOpts = {},
48
+ ): Promise<{ pinned: string[]; events: MatrixMessageSummary[] }> {
49
+ return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
50
+ const pinned = await readPinnedEvents(client, resolvedRoom);
51
+ const events = (
52
+ await Promise.all(
53
+ pinned.map(async (eventId) => {
54
+ try {
55
+ return await fetchEventSummary(client, resolvedRoom, eventId);
56
+ } catch {
57
+ return null;
58
+ }
59
+ }),
60
+ )
61
+ ).filter((event): event is MatrixMessageSummary => Boolean(event));
62
+ return { pinned, events };
63
+ });
64
+ }
@@ -0,0 +1,71 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+ import type { MatrixClient } from "../sdk.js";
3
+ import { voteMatrixPoll } from "./polls.js";
4
+
5
+ function createPollClient(pollContent?: Record<string, unknown>) {
6
+ const getEvent = vi.fn(async () => ({
7
+ type: "m.poll.start",
8
+ content: pollContent ?? {
9
+ "m.poll.start": {
10
+ question: { "m.text": "Favorite fruit?" },
11
+ max_selections: 1,
12
+ answers: [
13
+ { id: "apple", "m.text": "Apple" },
14
+ { id: "berry", "m.text": "Berry" },
15
+ ],
16
+ },
17
+ },
18
+ }));
19
+ const sendEvent = vi.fn(async () => "$vote1");
20
+
21
+ return {
22
+ client: {
23
+ getEvent,
24
+ sendEvent,
25
+ stop: vi.fn(),
26
+ } as unknown as MatrixClient,
27
+ getEvent,
28
+ sendEvent,
29
+ };
30
+ }
31
+
32
+ describe("matrix poll actions", () => {
33
+ it("votes by option index against the resolved room id", async () => {
34
+ const { client, getEvent, sendEvent } = createPollClient();
35
+
36
+ const result = await voteMatrixPoll("room:!room:example.org", "$poll", {
37
+ client,
38
+ optionIndex: 2,
39
+ });
40
+
41
+ expect(getEvent).toHaveBeenCalledWith("!room:example.org", "$poll");
42
+ expect(sendEvent).toHaveBeenCalledWith(
43
+ "!room:example.org",
44
+ "m.poll.response",
45
+ expect.objectContaining({
46
+ "m.poll.response": { answers: ["berry"] },
47
+ }),
48
+ );
49
+ expect(result).toEqual({
50
+ eventId: "$vote1",
51
+ roomId: "!room:example.org",
52
+ pollId: "$poll",
53
+ answerIds: ["berry"],
54
+ labels: ["Berry"],
55
+ maxSelections: 1,
56
+ });
57
+ });
58
+
59
+ it("rejects option indexes that are outside the poll range", async () => {
60
+ const { client, sendEvent } = createPollClient();
61
+
62
+ await expect(
63
+ voteMatrixPoll("room:!room:example.org", "$poll", {
64
+ client,
65
+ optionIndex: 3,
66
+ }),
67
+ ).rejects.toThrow("out of range");
68
+
69
+ expect(sendEvent).not.toHaveBeenCalled();
70
+ });
71
+ });