@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,757 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
2
+ import { getMatrixScopedEnvVarNames } from "../env-vars.js";
3
+ import type { CoreConfig } from "../types.js";
4
+ import {
5
+ listMatrixAccountIds,
6
+ resolveConfiguredMatrixBotUserIds,
7
+ resolveDefaultMatrixAccountId,
8
+ resolveMatrixAccount,
9
+ } from "./accounts.js";
10
+ import type { MatrixStoredCredentials } from "./credentials-read.js";
11
+
12
+ const loadMatrixCredentialsMock = vi.hoisted(() =>
13
+ vi.fn<(env?: NodeJS.ProcessEnv, accountId?: string | null) => MatrixStoredCredentials | null>(
14
+ () => null,
15
+ ),
16
+ );
17
+
18
+ vi.mock("./credentials-read.js", () => ({
19
+ loadMatrixCredentials: (env?: NodeJS.ProcessEnv, accountId?: string | null) =>
20
+ loadMatrixCredentialsMock(env, accountId),
21
+ credentialsMatchConfig: () => false,
22
+ }));
23
+
24
+ const envKeys = [
25
+ "LOBI_HOMESERVER",
26
+ "LOBI_USER_ID",
27
+ "LOBI_ACCESS_TOKEN",
28
+ "LOBI_PASSWORD",
29
+ "LOBI_DEVICE_NAME",
30
+ "MATRIX_DEFAULT_HOMESERVER",
31
+ "MATRIX_DEFAULT_ACCESS_TOKEN",
32
+ getMatrixScopedEnvVarNames("team-ops").homeserver,
33
+ getMatrixScopedEnvVarNames("team-ops").accessToken,
34
+ ];
35
+
36
+ type MatrixRoomScopeKey = "groups" | "rooms";
37
+
38
+ function createMatrixAccountConfig(accessToken: string) {
39
+ return {
40
+ homeserver: "https://matrix.example.org",
41
+ accessToken,
42
+ };
43
+ }
44
+
45
+ function createMatrixScopedEntriesConfig(scopeKey: MatrixRoomScopeKey): CoreConfig {
46
+ return {
47
+ channels: {
48
+ matrix: {
49
+ [scopeKey]: {
50
+ "!default-room:example.org": {
51
+ enabled: true,
52
+ account: "default",
53
+ },
54
+ "!axis-room:example.org": {
55
+ enabled: true,
56
+ account: "axis",
57
+ },
58
+ "!unassigned-room:example.org": {
59
+ enabled: true,
60
+ },
61
+ },
62
+ accounts: {
63
+ default: createMatrixAccountConfig("default-token"),
64
+ axis: createMatrixAccountConfig("axis-token"),
65
+ },
66
+ },
67
+ },
68
+ } as unknown as CoreConfig;
69
+ }
70
+
71
+ function createMatrixTopLevelDefaultScopedEntriesConfig(scopeKey: MatrixRoomScopeKey): CoreConfig {
72
+ return {
73
+ channels: {
74
+ matrix: {
75
+ ...createMatrixAccountConfig("default-token"),
76
+ [scopeKey]: {
77
+ "!default-room:example.org": {
78
+ enabled: true,
79
+ account: "default",
80
+ },
81
+ "!ops-room:example.org": {
82
+ enabled: true,
83
+ account: "ops",
84
+ },
85
+ "!shared-room:example.org": {
86
+ enabled: true,
87
+ },
88
+ },
89
+ accounts: {
90
+ ops: createMatrixAccountConfig("ops-token"),
91
+ },
92
+ },
93
+ },
94
+ } as unknown as CoreConfig;
95
+ }
96
+
97
+ function expectMatrixScopedEntries(
98
+ cfg: CoreConfig,
99
+ scopeKey: MatrixRoomScopeKey,
100
+ accountId: string,
101
+ expected: Record<string, { enabled: true; account?: string }>,
102
+ ): void {
103
+ expect(resolveMatrixAccount({ cfg, accountId }).config[scopeKey]).toEqual(expected);
104
+ }
105
+
106
+ function expectMultiAccountMatrixScopedEntries(
107
+ cfg: CoreConfig,
108
+ scopeKey: MatrixRoomScopeKey,
109
+ ): void {
110
+ expectMatrixScopedEntries(cfg, scopeKey, "default", {
111
+ "!default-room:example.org": {
112
+ enabled: true,
113
+ account: "default",
114
+ },
115
+ "!unassigned-room:example.org": {
116
+ enabled: true,
117
+ },
118
+ });
119
+ expectMatrixScopedEntries(cfg, scopeKey, "axis", {
120
+ "!axis-room:example.org": {
121
+ enabled: true,
122
+ account: "axis",
123
+ },
124
+ "!unassigned-room:example.org": {
125
+ enabled: true,
126
+ },
127
+ });
128
+ }
129
+
130
+ function expectTopLevelDefaultMatrixScopedEntries(
131
+ cfg: CoreConfig,
132
+ scopeKey: MatrixRoomScopeKey,
133
+ ): void {
134
+ expectMatrixScopedEntries(cfg, scopeKey, "default", {
135
+ "!default-room:example.org": {
136
+ enabled: true,
137
+ account: "default",
138
+ },
139
+ "!shared-room:example.org": {
140
+ enabled: true,
141
+ },
142
+ });
143
+ expectMatrixScopedEntries(cfg, scopeKey, "ops", {
144
+ "!ops-room:example.org": {
145
+ enabled: true,
146
+ account: "ops",
147
+ },
148
+ "!shared-room:example.org": {
149
+ enabled: true,
150
+ },
151
+ });
152
+ }
153
+
154
+ describe("resolveMatrixAccount", () => {
155
+ let prevEnv: Record<string, string | undefined> = {};
156
+
157
+ beforeEach(() => {
158
+ loadMatrixCredentialsMock.mockReset().mockReturnValue(null);
159
+ prevEnv = {};
160
+ for (const key of envKeys) {
161
+ prevEnv[key] = process.env[key];
162
+ delete process.env[key];
163
+ }
164
+ });
165
+
166
+ afterEach(() => {
167
+ for (const key of envKeys) {
168
+ const value = prevEnv[key];
169
+ if (value === undefined) {
170
+ delete process.env[key];
171
+ } else {
172
+ process.env[key] = value;
173
+ }
174
+ }
175
+ });
176
+
177
+ it("treats access-token-only config as configured", () => {
178
+ const cfg: CoreConfig = {
179
+ channels: {
180
+ matrix: {
181
+ homeserver: "https://matrix.example.org",
182
+ accessToken: "tok-access",
183
+ },
184
+ },
185
+ };
186
+
187
+ const account = resolveMatrixAccount({ cfg });
188
+ expect(account.configured).toBe(true);
189
+ });
190
+
191
+ it("treats SecretRef access-token config as configured", () => {
192
+ const cfg: CoreConfig = {
193
+ channels: {
194
+ matrix: {
195
+ homeserver: "https://matrix.example.org",
196
+ accessToken: { source: "file", provider: "matrix-file", id: "value" },
197
+ },
198
+ },
199
+ secrets: {
200
+ providers: {
201
+ "matrix-file": {
202
+ source: "file",
203
+ path: "/tmp/matrix-token",
204
+ },
205
+ },
206
+ },
207
+ };
208
+
209
+ const account = resolveMatrixAccount({ cfg });
210
+ expect(account.configured).toBe(true);
211
+ });
212
+
213
+ it("treats accounts.default SecretRef access-token config as configured", () => {
214
+ const cfg: CoreConfig = {
215
+ channels: {
216
+ matrix: {
217
+ accounts: {
218
+ default: {
219
+ homeserver: "https://matrix.example.org",
220
+ accessToken: { source: "file", provider: "matrix-file", id: "value" },
221
+ },
222
+ },
223
+ },
224
+ },
225
+ secrets: {
226
+ providers: {
227
+ "matrix-file": {
228
+ source: "file",
229
+ path: "/tmp/matrix-token",
230
+ },
231
+ },
232
+ },
233
+ };
234
+
235
+ const account = resolveMatrixAccount({ cfg });
236
+ expect(account.configured).toBe(true);
237
+ });
238
+
239
+ it("treats accounts.default SecretRef password config as configured", () => {
240
+ const cfg: CoreConfig = {
241
+ channels: {
242
+ matrix: {
243
+ accounts: {
244
+ default: {
245
+ homeserver: "https://matrix.example.org",
246
+ userId: "@bot:example.org",
247
+ password: { source: "file", provider: "matrix-file", id: "value" },
248
+ },
249
+ },
250
+ },
251
+ },
252
+ secrets: {
253
+ providers: {
254
+ "matrix-file": {
255
+ source: "file",
256
+ path: "/tmp/matrix-password",
257
+ },
258
+ },
259
+ },
260
+ };
261
+
262
+ const account = resolveMatrixAccount({ cfg });
263
+ expect(account.configured).toBe(true);
264
+ });
265
+
266
+ it("requires userId + password when no access token is set", () => {
267
+ const cfg: CoreConfig = {
268
+ channels: {
269
+ matrix: {
270
+ homeserver: "https://matrix.example.org",
271
+ userId: "@bot:example.org",
272
+ },
273
+ },
274
+ };
275
+
276
+ const account = resolveMatrixAccount({ cfg });
277
+ expect(account.configured).toBe(false);
278
+ });
279
+
280
+ it("marks password auth as configured when userId is present", () => {
281
+ const cfg: CoreConfig = {
282
+ channels: {
283
+ matrix: {
284
+ homeserver: "https://matrix.example.org",
285
+ userId: "@bot:example.org",
286
+ password: "secret",
287
+ },
288
+ },
289
+ };
290
+
291
+ const account = resolveMatrixAccount({ cfg });
292
+ expect(account.configured).toBe(true);
293
+ });
294
+
295
+ it("normalizes and de-duplicates configured account ids", () => {
296
+ const cfg: CoreConfig = {
297
+ channels: {
298
+ matrix: {
299
+ defaultAccount: "Main Bot",
300
+ accounts: {
301
+ "Main Bot": {
302
+ homeserver: "https://matrix.example.org",
303
+ accessToken: "main-token",
304
+ },
305
+ "main-bot": {
306
+ homeserver: "https://matrix.example.org",
307
+ accessToken: "duplicate-token",
308
+ },
309
+ OPS: {
310
+ homeserver: "https://matrix.example.org",
311
+ accessToken: "ops-token",
312
+ },
313
+ },
314
+ },
315
+ },
316
+ };
317
+
318
+ expect(listMatrixAccountIds(cfg)).toEqual(["main-bot", "ops"]);
319
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("main-bot");
320
+ });
321
+
322
+ it("returns the only named account when no explicit default is set", () => {
323
+ const cfg: CoreConfig = {
324
+ channels: {
325
+ matrix: {
326
+ accounts: {
327
+ ops: {
328
+ homeserver: "https://matrix.example.org",
329
+ accessToken: "ops-token",
330
+ },
331
+ },
332
+ },
333
+ },
334
+ };
335
+
336
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("ops");
337
+ });
338
+
339
+ it("uses configured defaultAccount when accountId is omitted", () => {
340
+ const cfg: CoreConfig = {
341
+ channels: {
342
+ matrix: {
343
+ defaultAccount: "ops",
344
+ homeserver: "https://matrix.example.org",
345
+ accessToken: "default-token",
346
+ accounts: {
347
+ ops: {
348
+ homeserver: "https://ops.example.org",
349
+ accessToken: "ops-token",
350
+ },
351
+ },
352
+ },
353
+ },
354
+ };
355
+
356
+ const account = resolveMatrixAccount({ cfg });
357
+ expect(account.accountId).toBe("ops");
358
+ expect(account.homeserver).toBe("https://ops.example.org");
359
+ expect(account.configured).toBe(true);
360
+ });
361
+
362
+ it("includes env-backed named accounts in plugin account enumeration", () => {
363
+ const keys = getMatrixScopedEnvVarNames("team-ops");
364
+ process.env[keys.homeserver] = "https://matrix.example.org";
365
+ process.env[keys.accessToken] = "ops-token";
366
+
367
+ const cfg: CoreConfig = {
368
+ channels: {
369
+ matrix: {},
370
+ },
371
+ };
372
+
373
+ expect(listMatrixAccountIds(cfg)).toEqual(["team-ops"]);
374
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("team-ops");
375
+ });
376
+
377
+ it("includes default accounts backed only by global env vars in plugin account enumeration", () => {
378
+ process.env.LOBI_HOMESERVER = "https://matrix.example.org";
379
+ process.env.LOBI_ACCESS_TOKEN = "default-token";
380
+
381
+ const cfg: CoreConfig = {};
382
+
383
+ expect(listMatrixAccountIds(cfg)).toEqual(["default"]);
384
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("default");
385
+ });
386
+
387
+ it("treats mixed default and named env-backed accounts as multi-account", () => {
388
+ const keys = getMatrixScopedEnvVarNames("team-ops");
389
+ process.env.LOBI_HOMESERVER = "https://matrix.example.org";
390
+ process.env.LOBI_ACCESS_TOKEN = "default-token";
391
+ process.env[keys.homeserver] = "https://matrix.example.org";
392
+ process.env[keys.accessToken] = "ops-token";
393
+
394
+ const cfg: CoreConfig = {
395
+ channels: {
396
+ matrix: {},
397
+ },
398
+ };
399
+
400
+ expect(listMatrixAccountIds(cfg)).toEqual(["default", "team-ops"]);
401
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("default");
402
+ });
403
+
404
+ it("includes a top-level configured default account alongside named accounts", () => {
405
+ const cfg: CoreConfig = {
406
+ channels: {
407
+ matrix: {
408
+ homeserver: "https://matrix.example.org",
409
+ accessToken: "default-token",
410
+ accounts: {
411
+ ops: {
412
+ homeserver: "https://matrix.example.org",
413
+ accessToken: "ops-token",
414
+ },
415
+ },
416
+ },
417
+ },
418
+ };
419
+
420
+ expect(listMatrixAccountIds(cfg)).toEqual(["default", "ops"]);
421
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("default");
422
+ });
423
+
424
+ it("does not materialize a default account from shared top-level defaults alone", () => {
425
+ const cfg: CoreConfig = {
426
+ channels: {
427
+ matrix: {
428
+ name: "Shared Defaults",
429
+ enabled: true,
430
+ accounts: {
431
+ ops: {
432
+ homeserver: "https://matrix.example.org",
433
+ accessToken: "ops-token",
434
+ },
435
+ },
436
+ },
437
+ },
438
+ };
439
+
440
+ expect(listMatrixAccountIds(cfg)).toEqual(["ops"]);
441
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("ops");
442
+ });
443
+
444
+ it('uses the synthetic "default" account when multiple named accounts need explicit selection', () => {
445
+ const cfg: CoreConfig = {
446
+ channels: {
447
+ matrix: {
448
+ accounts: {
449
+ alpha: {
450
+ homeserver: "https://matrix.example.org",
451
+ accessToken: "alpha-token",
452
+ },
453
+ beta: {
454
+ homeserver: "https://matrix.example.org",
455
+ accessToken: "beta-token",
456
+ },
457
+ },
458
+ },
459
+ },
460
+ };
461
+
462
+ expect(resolveDefaultMatrixAccountId(cfg)).toBe("default");
463
+ });
464
+
465
+ it("collects other configured Matrix account user ids for bot detection", () => {
466
+ const cfg: CoreConfig = {
467
+ channels: {
468
+ matrix: {
469
+ userId: "@main:example.org",
470
+ homeserver: "https://matrix.example.org",
471
+ accessToken: "main-token",
472
+ accounts: {
473
+ ops: {
474
+ homeserver: "https://matrix.example.org",
475
+ userId: "@ops:example.org",
476
+ accessToken: "ops-token",
477
+ },
478
+ alerts: {
479
+ homeserver: "https://matrix.example.org",
480
+ userId: "@alerts:example.org",
481
+ accessToken: "alerts-token",
482
+ },
483
+ },
484
+ },
485
+ },
486
+ };
487
+
488
+ expect(
489
+ Array.from(resolveConfiguredMatrixBotUserIds({ cfg, accountId: "ops" })).toSorted(),
490
+ ).toEqual(["@alerts:example.org", "@main:example.org"]);
491
+ });
492
+
493
+ it("honors injected env when detecting configured bot accounts", () => {
494
+ const env = {
495
+ LOBI_HOMESERVER: "https://matrix.example.org",
496
+ LOBI_USER_ID: "@main:example.org",
497
+ LOBI_ACCESS_TOKEN: "main-token",
498
+ MATRIX_ALERTS_HOMESERVER: "https://matrix.example.org",
499
+ MATRIX_ALERTS_USER_ID: "@alerts:example.org",
500
+ MATRIX_ALERTS_ACCESS_TOKEN: "alerts-token",
501
+ } as NodeJS.ProcessEnv;
502
+
503
+ const cfg: CoreConfig = {
504
+ channels: {
505
+ matrix: {},
506
+ },
507
+ };
508
+
509
+ expect(
510
+ Array.from(resolveConfiguredMatrixBotUserIds({ cfg, accountId: "ops", env })).toSorted(),
511
+ ).toEqual(["@alerts:example.org", "@main:example.org"]);
512
+ });
513
+
514
+ it("falls back to stored credentials when an access-token-only account omits userId", () => {
515
+ loadMatrixCredentialsMock.mockImplementation(
516
+ (env?: NodeJS.ProcessEnv, accountId?: string | null) =>
517
+ accountId === "ops"
518
+ ? {
519
+ homeserver: "https://matrix.example.org",
520
+ userId: "@ops:example.org",
521
+ accessToken: "ops-token",
522
+ createdAt: "2026-03-19T00:00:00.000Z",
523
+ }
524
+ : null,
525
+ );
526
+
527
+ const cfg: CoreConfig = {
528
+ channels: {
529
+ matrix: {
530
+ userId: "@main:example.org",
531
+ homeserver: "https://matrix.example.org",
532
+ accessToken: "main-token",
533
+ accounts: {
534
+ ops: {
535
+ homeserver: "https://matrix.example.org",
536
+ accessToken: "ops-token",
537
+ },
538
+ },
539
+ },
540
+ },
541
+ };
542
+
543
+ expect(Array.from(resolveConfiguredMatrixBotUserIds({ cfg, accountId: "default" }))).toEqual([
544
+ "@ops:example.org",
545
+ ]);
546
+ });
547
+
548
+ it("preserves shared nested dm and actions config when an account overrides one field", () => {
549
+ const account = resolveMatrixAccount({
550
+ cfg: {
551
+ channels: {
552
+ matrix: {
553
+ homeserver: "https://matrix.example.org",
554
+ accessToken: "main-token",
555
+ dm: {
556
+ enabled: true,
557
+ policy: "pairing",
558
+ },
559
+ actions: {
560
+ reactions: true,
561
+ messages: true,
562
+ },
563
+ accounts: {
564
+ ops: {
565
+ accessToken: "ops-token",
566
+ dm: {
567
+ allowFrom: ["@ops:example.org"],
568
+ },
569
+ actions: {
570
+ messages: false,
571
+ },
572
+ },
573
+ },
574
+ },
575
+ },
576
+ },
577
+ accountId: "ops",
578
+ });
579
+
580
+ expect(account.config.dm).toEqual({
581
+ enabled: true,
582
+ policy: "pairing",
583
+ allowFrom: ["@ops:example.org"],
584
+ });
585
+ expect(account.config.actions).toEqual({
586
+ reactions: true,
587
+ messages: false,
588
+ });
589
+ });
590
+
591
+ it("filters channel-level groups by room account in multi-account setups", () => {
592
+ expectMultiAccountMatrixScopedEntries(createMatrixScopedEntriesConfig("groups"), "groups");
593
+ });
594
+
595
+ it("filters channel-level groups when the default account is configured at the top level", () => {
596
+ expectTopLevelDefaultMatrixScopedEntries(
597
+ createMatrixTopLevelDefaultScopedEntriesConfig("groups"),
598
+ "groups",
599
+ );
600
+ });
601
+
602
+ it("filters legacy channel-level rooms by room account in multi-account setups", () => {
603
+ expectMultiAccountMatrixScopedEntries(createMatrixScopedEntriesConfig("rooms"), "rooms");
604
+ });
605
+
606
+ it("filters legacy channel-level rooms when the default account is configured at the top level", () => {
607
+ expectTopLevelDefaultMatrixScopedEntries(
608
+ createMatrixTopLevelDefaultScopedEntriesConfig("rooms"),
609
+ "rooms",
610
+ );
611
+ });
612
+
613
+ it("honors injected env when scoping room entries in multi-account setups", () => {
614
+ const env = {
615
+ LOBI_HOMESERVER: "https://matrix.example.org",
616
+ LOBI_ACCESS_TOKEN: "default-token",
617
+ LOBI_OPS_HOMESERVER: "https://matrix.example.org",
618
+ LOBI_OPS_ACCESS_TOKEN: "ops-token",
619
+ } as NodeJS.ProcessEnv;
620
+
621
+ const cfg = {
622
+ channels: {
623
+ matrix: {
624
+ groups: {
625
+ "!default-room:example.org": {
626
+ enabled: true,
627
+ account: "default",
628
+ },
629
+ "!ops-room:example.org": {
630
+ enabled: true,
631
+ account: "ops",
632
+ },
633
+ "!shared-room:example.org": {
634
+ enabled: true,
635
+ },
636
+ },
637
+ },
638
+ },
639
+ } as unknown as CoreConfig;
640
+
641
+ expect(resolveMatrixAccount({ cfg, accountId: "ops", env }).config.groups).toEqual({
642
+ "!ops-room:example.org": {
643
+ enabled: true,
644
+ account: "ops",
645
+ },
646
+ "!shared-room:example.org": {
647
+ enabled: true,
648
+ },
649
+ });
650
+ });
651
+
652
+ it("keeps scoped groups bound to their account even when only one account is active", () => {
653
+ const cfg = {
654
+ channels: {
655
+ matrix: {
656
+ groups: {
657
+ "!default-room:example.org": {
658
+ enabled: true,
659
+ account: "default",
660
+ },
661
+ "!shared-room:example.org": {
662
+ enabled: true,
663
+ },
664
+ },
665
+ accounts: {
666
+ ops: {
667
+ homeserver: "https://matrix.example.org",
668
+ accessToken: "ops-token",
669
+ },
670
+ },
671
+ },
672
+ },
673
+ } as unknown as CoreConfig;
674
+
675
+ expect(resolveMatrixAccount({ cfg, accountId: "ops" }).config.groups).toEqual({
676
+ "!shared-room:example.org": {
677
+ enabled: true,
678
+ },
679
+ });
680
+ });
681
+
682
+ it("keeps scoped legacy rooms bound to their account even when only one account is active", () => {
683
+ const cfg = {
684
+ channels: {
685
+ matrix: {
686
+ rooms: {
687
+ "!default-room:example.org": {
688
+ enabled: true,
689
+ account: "default",
690
+ },
691
+ "!shared-room:example.org": {
692
+ enabled: true,
693
+ },
694
+ },
695
+ accounts: {
696
+ ops: {
697
+ homeserver: "https://matrix.example.org",
698
+ accessToken: "ops-token",
699
+ },
700
+ },
701
+ },
702
+ },
703
+ } as unknown as CoreConfig;
704
+
705
+ expect(resolveMatrixAccount({ cfg, accountId: "ops" }).config.rooms).toEqual({
706
+ "!shared-room:example.org": {
707
+ enabled: true,
708
+ },
709
+ });
710
+ });
711
+
712
+ it("lets an account clear inherited groups with an explicit empty map", () => {
713
+ const cfg = {
714
+ channels: {
715
+ matrix: {
716
+ groups: {
717
+ "!shared-room:example.org": {
718
+ enabled: true,
719
+ },
720
+ },
721
+ accounts: {
722
+ ops: {
723
+ homeserver: "https://matrix.example.org",
724
+ accessToken: "ops-token",
725
+ groups: {},
726
+ },
727
+ },
728
+ },
729
+ },
730
+ } as unknown as CoreConfig;
731
+
732
+ expect(resolveMatrixAccount({ cfg, accountId: "ops" }).config.groups).toBeUndefined();
733
+ });
734
+
735
+ it("lets an account clear inherited legacy rooms with an explicit empty map", () => {
736
+ const cfg = {
737
+ channels: {
738
+ matrix: {
739
+ rooms: {
740
+ "!shared-room:example.org": {
741
+ enabled: true,
742
+ },
743
+ },
744
+ accounts: {
745
+ ops: {
746
+ homeserver: "https://matrix.example.org",
747
+ accessToken: "ops-token",
748
+ rooms: {},
749
+ },
750
+ },
751
+ },
752
+ },
753
+ } as unknown as CoreConfig;
754
+
755
+ expect(resolveMatrixAccount({ cfg, accountId: "ops" }).config.rooms).toBeUndefined();
756
+ });
757
+ });