@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.
- package/CHANGELOG.md +164 -0
- package/ENDOFFILE +0 -0
- package/EOF +0 -0
- package/LICENSE +21 -0
- package/SPEC-SUPPORT.md +116 -0
- package/YAMLEND +0 -0
- package/api.ts +18 -0
- package/archipelagolab-lobi-1.0.0.tgz +0 -0
- package/auth-presence.ts +56 -0
- package/channel-plugin-api.ts +3 -0
- package/cli-metadata.ts +11 -0
- package/contract-api.ts +17 -0
- package/docs/CHECKLIST.md +83 -0
- package/docs/FORK_SDK_GUIDE.md +279 -0
- package/helper-api.ts +3 -0
- package/index.test.ts +61 -0
- package/index.ts +65 -0
- package/openclaw.plugin.json +23 -0
- package/package.json +52 -0
- package/plugin-entry.handlers.runtime.ts +1 -0
- package/runtime-api.ts +54 -0
- package/runtime-heavy-api.ts +1 -0
- package/scripts/migrate-to-lobi.sh +72 -0
- package/secret-contract-api.ts +5 -0
- package/setup-entry.ts +13 -0
- package/src/account-selection.test.ts +124 -0
- package/src/account-selection.ts +226 -0
- package/src/actions.account-propagation.test.ts +251 -0
- package/src/actions.test.ts +251 -0
- package/src/actions.ts +336 -0
- package/src/approval-auth.test.ts +23 -0
- package/src/approval-auth.ts +25 -0
- package/src/approval-handler.runtime.test.ts +46 -0
- package/src/approval-handler.runtime.ts +400 -0
- package/src/approval-ids.ts +6 -0
- package/src/approval-native.test.ts +329 -0
- package/src/approval-native.ts +336 -0
- package/src/approval-reactions.test.ts +107 -0
- package/src/approval-reactions.ts +158 -0
- package/src/auth-precedence.ts +61 -0
- package/src/channel-account-paths.ts +92 -0
- package/src/channel.account-paths.test.ts +102 -0
- package/src/channel.directory.test.ts +601 -0
- package/src/channel.resolve.test.ts +38 -0
- package/src/channel.runtime.ts +16 -0
- package/src/channel.setup.test.ts +269 -0
- package/src/channel.ts +570 -0
- package/src/cli-metadata.ts +19 -0
- package/src/cli.test.ts +1015 -0
- package/src/cli.ts +1198 -0
- package/src/config-adapter.ts +41 -0
- package/src/config-schema.test.ts +90 -0
- package/src/config-schema.ts +114 -0
- package/src/directory-live.test.ts +200 -0
- package/src/directory-live.ts +238 -0
- package/src/doctor-contract.ts +287 -0
- package/src/doctor.test.ts +440 -0
- package/src/doctor.ts +262 -0
- package/src/env-vars.ts +92 -0
- package/src/exec-approval-resolver.test.ts +68 -0
- package/src/exec-approval-resolver.ts +23 -0
- package/src/exec-approvals.test.ts +483 -0
- package/src/exec-approvals.ts +290 -0
- package/src/group-mentions.ts +41 -0
- package/src/legacy-crypto-inspector-availability.test.ts +81 -0
- package/src/legacy-crypto-inspector-availability.ts +60 -0
- package/src/legacy-crypto.test.ts +234 -0
- package/src/legacy-crypto.ts +549 -0
- package/src/legacy-state.test.ts +86 -0
- package/src/legacy-state.ts +156 -0
- package/src/matrix/account-config.ts +150 -0
- package/src/matrix/accounts.readiness.test.ts +27 -0
- package/src/matrix/accounts.test.ts +757 -0
- package/src/matrix/accounts.ts +194 -0
- package/src/matrix/actions/client.test.ts +215 -0
- package/src/matrix/actions/client.ts +31 -0
- package/src/matrix/actions/devices.test.ts +114 -0
- package/src/matrix/actions/devices.ts +34 -0
- package/src/matrix/actions/limits.test.ts +15 -0
- package/src/matrix/actions/limits.ts +6 -0
- package/src/matrix/actions/messages.test.ts +289 -0
- package/src/matrix/actions/messages.ts +123 -0
- package/src/matrix/actions/pins.test.ts +74 -0
- package/src/matrix/actions/pins.ts +64 -0
- package/src/matrix/actions/polls.test.ts +71 -0
- package/src/matrix/actions/polls.ts +109 -0
- package/src/matrix/actions/profile.test.ts +109 -0
- package/src/matrix/actions/profile.ts +37 -0
- package/src/matrix/actions/reactions.test.ts +135 -0
- package/src/matrix/actions/reactions.ts +59 -0
- package/src/matrix/actions/room.test.ts +79 -0
- package/src/matrix/actions/room.ts +71 -0
- package/src/matrix/actions/summary.test.ts +87 -0
- package/src/matrix/actions/summary.ts +88 -0
- package/src/matrix/actions/types.ts +82 -0
- package/src/matrix/actions/verification.test.ts +105 -0
- package/src/matrix/actions/verification.ts +237 -0
- package/src/matrix/actions.ts +37 -0
- package/src/matrix/active-client.ts +26 -0
- package/src/matrix/async-lock.ts +18 -0
- package/src/matrix/backup-health.ts +115 -0
- package/src/matrix/client/config-runtime-api.ts +14 -0
- package/src/matrix/client/config-secret-input.runtime.ts +1 -0
- package/src/matrix/client/config.ts +982 -0
- package/src/matrix/client/create-client.test.ts +115 -0
- package/src/matrix/client/create-client.ts +101 -0
- package/src/matrix/client/env-auth.ts +6 -0
- package/src/matrix/client/file-sync-store.test.ts +265 -0
- package/src/matrix/client/file-sync-store.ts +289 -0
- package/src/matrix/client/logging.ts +123 -0
- package/src/matrix/client/migration-snapshot.runtime.ts +1 -0
- package/src/matrix/client/private-network-host.ts +56 -0
- package/src/matrix/client/runtime.ts +4 -0
- package/src/matrix/client/shared.test.ts +344 -0
- package/src/matrix/client/shared.ts +306 -0
- package/src/matrix/client/storage.test.ts +634 -0
- package/src/matrix/client/storage.ts +544 -0
- package/src/matrix/client/types.ts +50 -0
- package/src/matrix/client-bootstrap.test.ts +84 -0
- package/src/matrix/client-bootstrap.ts +164 -0
- package/src/matrix/client-resolver.test-helpers.ts +147 -0
- package/src/matrix/client.test.ts +1521 -0
- package/src/matrix/client.ts +23 -0
- package/src/matrix/config-paths.ts +31 -0
- package/src/matrix/config-update.test.ts +237 -0
- package/src/matrix/config-update.ts +291 -0
- package/src/matrix/credentials-read.ts +206 -0
- package/src/matrix/credentials-write.runtime.ts +26 -0
- package/src/matrix/credentials.test.ts +501 -0
- package/src/matrix/credentials.ts +95 -0
- package/src/matrix/deps.test.ts +74 -0
- package/src/matrix/deps.ts +225 -0
- package/src/matrix/device-health.test.ts +45 -0
- package/src/matrix/device-health.ts +31 -0
- package/src/matrix/direct-management.test.ts +350 -0
- package/src/matrix/direct-management.ts +347 -0
- package/src/matrix/direct-room.test.ts +61 -0
- package/src/matrix/direct-room.ts +128 -0
- package/src/matrix/draft-stream.test.ts +406 -0
- package/src/matrix/draft-stream.ts +216 -0
- package/src/matrix/encryption-guidance.ts +27 -0
- package/src/matrix/errors.ts +21 -0
- package/src/matrix/format.test.ts +340 -0
- package/src/matrix/format.ts +428 -0
- package/src/matrix/legacy-crypto-inspector.ts +95 -0
- package/src/matrix/media-errors.ts +20 -0
- package/src/matrix/media-text.ts +169 -0
- package/src/matrix/monitor/access-state.test.ts +45 -0
- package/src/matrix/monitor/access-state.ts +77 -0
- package/src/matrix/monitor/ack-config.test.ts +57 -0
- package/src/matrix/monitor/ack-config.ts +26 -0
- package/src/matrix/monitor/allowlist.test.ts +45 -0
- package/src/matrix/monitor/allowlist.ts +94 -0
- package/src/matrix/monitor/auto-join.test.ts +203 -0
- package/src/matrix/monitor/auto-join.ts +86 -0
- package/src/matrix/monitor/config.test.ts +197 -0
- package/src/matrix/monitor/config.ts +303 -0
- package/src/matrix/monitor/context-summary.ts +43 -0
- package/src/matrix/monitor/direct.test.ts +529 -0
- package/src/matrix/monitor/direct.ts +270 -0
- package/src/matrix/monitor/events.test.ts +1524 -0
- package/src/matrix/monitor/events.ts +213 -0
- package/src/matrix/monitor/handler.body-for-agent.test.ts +396 -0
- package/src/matrix/monitor/handler.group-history.test.ts +648 -0
- package/src/matrix/monitor/handler.media-failure.test.ts +267 -0
- package/src/matrix/monitor/handler.test-helpers.ts +308 -0
- package/src/matrix/monitor/handler.test.ts +2952 -0
- package/src/matrix/monitor/handler.thread-root-media.test.ts +82 -0
- package/src/matrix/monitor/handler.ts +1679 -0
- package/src/matrix/monitor/inbound-dedupe.test.ts +146 -0
- package/src/matrix/monitor/inbound-dedupe.ts +267 -0
- package/src/matrix/monitor/index.test.ts +920 -0
- package/src/matrix/monitor/index.ts +434 -0
- package/src/matrix/monitor/legacy-crypto-restore.test.ts +206 -0
- package/src/matrix/monitor/legacy-crypto-restore.ts +139 -0
- package/src/matrix/monitor/location.ts +100 -0
- package/src/matrix/monitor/media.test.ts +159 -0
- package/src/matrix/monitor/media.ts +119 -0
- package/src/matrix/monitor/mentions.test.ts +289 -0
- package/src/matrix/monitor/mentions.ts +177 -0
- package/src/matrix/monitor/reaction-events.test.ts +326 -0
- package/src/matrix/monitor/reaction-events.ts +187 -0
- package/src/matrix/monitor/recent-invite.test.ts +92 -0
- package/src/matrix/monitor/recent-invite.ts +30 -0
- package/src/matrix/monitor/replies.test.ts +265 -0
- package/src/matrix/monitor/replies.ts +136 -0
- package/src/matrix/monitor/reply-context.test.ts +276 -0
- package/src/matrix/monitor/reply-context.ts +92 -0
- package/src/matrix/monitor/room-history.test.ts +258 -0
- package/src/matrix/monitor/room-history.ts +301 -0
- package/src/matrix/monitor/room-info.test.ts +201 -0
- package/src/matrix/monitor/room-info.ts +126 -0
- package/src/matrix/monitor/rooms.test.ts +121 -0
- package/src/matrix/monitor/rooms.ts +52 -0
- package/src/matrix/monitor/route.test.ts +255 -0
- package/src/matrix/monitor/route.ts +178 -0
- package/src/matrix/monitor/runtime-api.ts +31 -0
- package/src/matrix/monitor/startup-verification.test.ts +294 -0
- package/src/matrix/monitor/startup-verification.ts +237 -0
- package/src/matrix/monitor/startup.test.ts +257 -0
- package/src/matrix/monitor/startup.ts +218 -0
- package/src/matrix/monitor/status.ts +111 -0
- package/src/matrix/monitor/sync-lifecycle.test.ts +224 -0
- package/src/matrix/monitor/sync-lifecycle.ts +91 -0
- package/src/matrix/monitor/task-runner.ts +38 -0
- package/src/matrix/monitor/thread-context.test.ts +149 -0
- package/src/matrix/monitor/thread-context.ts +108 -0
- package/src/matrix/monitor/threads.test.ts +68 -0
- package/src/matrix/monitor/threads.ts +85 -0
- package/src/matrix/monitor/types.ts +30 -0
- package/src/matrix/monitor/verification-events.ts +627 -0
- package/src/matrix/monitor/verification-utils.test.ts +47 -0
- package/src/matrix/monitor/verification-utils.ts +46 -0
- package/src/matrix/outbound-media-runtime.ts +1 -0
- package/src/matrix/poll-summary.ts +110 -0
- package/src/matrix/poll-types.test.ts +205 -0
- package/src/matrix/poll-types.ts +433 -0
- package/src/matrix/probe.runtime.ts +4 -0
- package/src/matrix/probe.test.ts +154 -0
- package/src/matrix/probe.ts +96 -0
- package/src/matrix/profile.test.ts +154 -0
- package/src/matrix/profile.ts +184 -0
- package/src/matrix/reaction-common.test.ts +96 -0
- package/src/matrix/reaction-common.ts +147 -0
- package/src/matrix/sdk/crypto-bootstrap.test.ts +505 -0
- package/src/matrix/sdk/crypto-bootstrap.ts +341 -0
- package/src/matrix/sdk/crypto-facade.test.ts +197 -0
- package/src/matrix/sdk/crypto-facade.ts +207 -0
- package/src/matrix/sdk/crypto-node.runtime.test.ts +27 -0
- package/src/matrix/sdk/crypto-node.runtime.ts +9 -0
- package/src/matrix/sdk/crypto-runtime.ts +11 -0
- package/src/matrix/sdk/decrypt-bridge.ts +356 -0
- package/src/matrix/sdk/event-helpers.test.ts +60 -0
- package/src/matrix/sdk/event-helpers.ts +71 -0
- package/src/matrix/sdk/http-client.test.ts +134 -0
- package/src/matrix/sdk/http-client.ts +87 -0
- package/src/matrix/sdk/idb-persistence-lock.ts +51 -0
- package/src/matrix/sdk/idb-persistence.lock-order.test.ts +108 -0
- package/src/matrix/sdk/idb-persistence.test-helpers.ts +88 -0
- package/src/matrix/sdk/idb-persistence.test.ts +149 -0
- package/src/matrix/sdk/idb-persistence.ts +283 -0
- package/src/matrix/sdk/logger.test.ts +25 -0
- package/src/matrix/sdk/logger.ts +108 -0
- package/src/matrix/sdk/read-response-with-limit.ts +19 -0
- package/src/matrix/sdk/recovery-key-store.test.ts +385 -0
- package/src/matrix/sdk/recovery-key-store.ts +430 -0
- package/src/matrix/sdk/transport.test.ts +161 -0
- package/src/matrix/sdk/transport.ts +344 -0
- package/src/matrix/sdk/types.ts +236 -0
- package/src/matrix/sdk/verification-manager.test.ts +509 -0
- package/src/matrix/sdk/verification-manager.ts +694 -0
- package/src/matrix/sdk/verification-status.ts +23 -0
- package/src/matrix/sdk.test.ts +2568 -0
- package/src/matrix/sdk.ts +1789 -0
- package/src/matrix/send/client.test.ts +174 -0
- package/src/matrix/send/client.ts +90 -0
- package/src/matrix/send/formatting.ts +189 -0
- package/src/matrix/send/media.ts +244 -0
- package/src/matrix/send/targets.test.ts +254 -0
- package/src/matrix/send/targets.ts +104 -0
- package/src/matrix/send/types.ts +134 -0
- package/src/matrix/send.test.ts +958 -0
- package/src/matrix/send.ts +609 -0
- package/src/matrix/session-store-metadata.ts +108 -0
- package/src/matrix/startup-abort.ts +44 -0
- package/src/matrix/sync-state.ts +27 -0
- package/src/matrix/target-ids.ts +102 -0
- package/src/matrix/thread-bindings-shared.ts +201 -0
- package/src/matrix/thread-bindings.test.ts +673 -0
- package/src/matrix/thread-bindings.ts +577 -0
- package/src/matrix-migration.runtime.ts +9 -0
- package/src/migration-config.test.ts +228 -0
- package/src/migration-config.ts +243 -0
- package/src/migration-snapshot-backup.ts +117 -0
- package/src/migration-snapshot.test.ts +184 -0
- package/src/migration-snapshot.ts +55 -0
- package/src/onboarding.resolve.test.ts +55 -0
- package/src/onboarding.test-harness.ts +158 -0
- package/src/onboarding.test.ts +665 -0
- package/src/onboarding.ts +773 -0
- package/src/outbound.test.ts +173 -0
- package/src/outbound.ts +78 -0
- package/src/plugin-entry.runtime.js +159 -0
- package/src/plugin-entry.runtime.test.ts +108 -0
- package/src/plugin-entry.runtime.ts +68 -0
- package/src/profile-update.ts +68 -0
- package/src/record-shared.ts +3 -0
- package/src/resolve-targets.test.ts +178 -0
- package/src/resolve-targets.ts +175 -0
- package/src/resolver.ts +21 -0
- package/src/runtime-api.ts +144 -0
- package/src/runtime.ts +7 -0
- package/src/secret-contract.ts +174 -0
- package/src/session-route.test.ts +315 -0
- package/src/session-route.ts +113 -0
- package/src/setup-bootstrap.ts +94 -0
- package/src/setup-config.ts +222 -0
- package/src/setup-contract.ts +89 -0
- package/src/setup-core.test.ts +326 -0
- package/src/setup-core.ts +50 -0
- package/src/setup-surface.ts +4 -0
- package/src/startup-maintenance.test.ts +227 -0
- package/src/startup-maintenance.ts +114 -0
- package/src/storage-paths.ts +92 -0
- package/src/test-helpers.ts +42 -0
- package/src/test-mocks.ts +55 -0
- package/src/test-runtime.ts +72 -0
- package/src/test-support/monitor-route-test-support.ts +8 -0
- package/src/tool-actions.runtime.ts +1 -0
- package/src/tool-actions.test.ts +422 -0
- package/src/tool-actions.ts +498 -0
- package/src/types.ts +230 -0
- package/test-api.ts +2 -0
- package/thread-bindings-runtime.ts +4 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,498 @@
|
|
|
1
|
+
import type { AgentToolResult } from "@mariozechner/pi-agent-core";
|
|
2
|
+
import { normalizeOptionalLowercaseString } from "openclaw/plugin-sdk/text-runtime";
|
|
3
|
+
import { resolveMatrixAccountConfig } from "./matrix/accounts.js";
|
|
4
|
+
import {
|
|
5
|
+
bootstrapMatrixVerification,
|
|
6
|
+
acceptMatrixVerification,
|
|
7
|
+
cancelMatrixVerification,
|
|
8
|
+
confirmMatrixVerificationReciprocateQr,
|
|
9
|
+
confirmMatrixVerificationSas,
|
|
10
|
+
deleteMatrixMessage,
|
|
11
|
+
editMatrixMessage,
|
|
12
|
+
generateMatrixVerificationQr,
|
|
13
|
+
getMatrixEncryptionStatus,
|
|
14
|
+
getMatrixRoomKeyBackupStatus,
|
|
15
|
+
getMatrixVerificationStatus,
|
|
16
|
+
getMatrixMemberInfo,
|
|
17
|
+
getMatrixRoomInfo,
|
|
18
|
+
getMatrixVerificationSas,
|
|
19
|
+
listMatrixPins,
|
|
20
|
+
listMatrixReactions,
|
|
21
|
+
listMatrixVerifications,
|
|
22
|
+
mismatchMatrixVerificationSas,
|
|
23
|
+
pinMatrixMessage,
|
|
24
|
+
readMatrixMessages,
|
|
25
|
+
requestMatrixVerification,
|
|
26
|
+
restoreMatrixRoomKeyBackup,
|
|
27
|
+
removeMatrixReactions,
|
|
28
|
+
scanMatrixVerificationQr,
|
|
29
|
+
sendMatrixMessage,
|
|
30
|
+
startMatrixVerification,
|
|
31
|
+
unpinMatrixMessage,
|
|
32
|
+
voteMatrixPoll,
|
|
33
|
+
verifyMatrixRecoveryKey,
|
|
34
|
+
} from "./matrix/actions.js";
|
|
35
|
+
import { reactMatrixMessage } from "./matrix/send.js";
|
|
36
|
+
import { applyMatrixProfileUpdate } from "./profile-update.js";
|
|
37
|
+
import {
|
|
38
|
+
createActionGate,
|
|
39
|
+
jsonResult,
|
|
40
|
+
readNumberParam,
|
|
41
|
+
readReactionParams,
|
|
42
|
+
readStringArrayParam,
|
|
43
|
+
readStringParam,
|
|
44
|
+
} from "./runtime-api.js";
|
|
45
|
+
import type { CoreConfig } from "./types.js";
|
|
46
|
+
|
|
47
|
+
const messageActions = new Set(["sendMessage", "editMessage", "deleteMessage", "readMessages"]);
|
|
48
|
+
const reactionActions = new Set(["react", "reactions"]);
|
|
49
|
+
const pinActions = new Set(["pinMessage", "unpinMessage", "listPins"]);
|
|
50
|
+
const pollActions = new Set(["pollVote"]);
|
|
51
|
+
const profileActions = new Set(["setProfile"]);
|
|
52
|
+
const verificationActions = new Set([
|
|
53
|
+
"encryptionStatus",
|
|
54
|
+
"verificationList",
|
|
55
|
+
"verificationRequest",
|
|
56
|
+
"verificationAccept",
|
|
57
|
+
"verificationCancel",
|
|
58
|
+
"verificationStart",
|
|
59
|
+
"verificationGenerateQr",
|
|
60
|
+
"verificationScanQr",
|
|
61
|
+
"verificationSas",
|
|
62
|
+
"verificationConfirm",
|
|
63
|
+
"verificationMismatch",
|
|
64
|
+
"verificationConfirmQr",
|
|
65
|
+
"verificationStatus",
|
|
66
|
+
"verificationBootstrap",
|
|
67
|
+
"verificationRecoveryKey",
|
|
68
|
+
"verificationBackupStatus",
|
|
69
|
+
"verificationBackupRestore",
|
|
70
|
+
]);
|
|
71
|
+
|
|
72
|
+
function readRoomId(params: Record<string, unknown>, required = true): string {
|
|
73
|
+
const direct = readStringParam(params, "roomId") ?? readStringParam(params, "channelId");
|
|
74
|
+
if (direct) {
|
|
75
|
+
return direct;
|
|
76
|
+
}
|
|
77
|
+
if (!required) {
|
|
78
|
+
return readStringParam(params, "to") ?? "";
|
|
79
|
+
}
|
|
80
|
+
return readStringParam(params, "to", { required: true });
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
function toSnakeCaseKey(key: string): string {
|
|
84
|
+
return normalizeOptionalLowercaseString(
|
|
85
|
+
key.replace(/([A-Z]+)([A-Z][a-z])/g, "$1_$2").replace(/([a-z0-9])([A-Z])/g, "$1_$2"),
|
|
86
|
+
)!;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
function readRawParam(params: Record<string, unknown>, key: string): unknown {
|
|
90
|
+
if (Object.hasOwn(params, key)) {
|
|
91
|
+
return params[key];
|
|
92
|
+
}
|
|
93
|
+
const snakeKey = toSnakeCaseKey(key);
|
|
94
|
+
if (snakeKey !== key && Object.hasOwn(params, snakeKey)) {
|
|
95
|
+
return params[snakeKey];
|
|
96
|
+
}
|
|
97
|
+
return undefined;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
function readStringAliasParam(
|
|
101
|
+
params: Record<string, unknown>,
|
|
102
|
+
keys: string[],
|
|
103
|
+
options: { required?: boolean } = {},
|
|
104
|
+
): string | undefined {
|
|
105
|
+
for (const key of keys) {
|
|
106
|
+
const raw = readRawParam(params, key);
|
|
107
|
+
if (typeof raw !== "string") {
|
|
108
|
+
continue;
|
|
109
|
+
}
|
|
110
|
+
const trimmed = raw.trim();
|
|
111
|
+
if (trimmed) {
|
|
112
|
+
return trimmed;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
if (options.required) {
|
|
116
|
+
throw new Error(`${keys[0]} required`);
|
|
117
|
+
}
|
|
118
|
+
return undefined;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function readNumericArrayParam(
|
|
122
|
+
params: Record<string, unknown>,
|
|
123
|
+
key: string,
|
|
124
|
+
options: { integer?: boolean } = {},
|
|
125
|
+
): number[] {
|
|
126
|
+
const { integer = false } = options;
|
|
127
|
+
const raw = readRawParam(params, key);
|
|
128
|
+
if (raw === undefined) {
|
|
129
|
+
return [];
|
|
130
|
+
}
|
|
131
|
+
return (Array.isArray(raw) ? raw : [raw])
|
|
132
|
+
.map((value) => {
|
|
133
|
+
if (typeof value === "number" && Number.isFinite(value)) {
|
|
134
|
+
return value;
|
|
135
|
+
}
|
|
136
|
+
if (typeof value === "string") {
|
|
137
|
+
const trimmed = value.trim();
|
|
138
|
+
if (!trimmed) {
|
|
139
|
+
return null;
|
|
140
|
+
}
|
|
141
|
+
const parsed = Number(trimmed);
|
|
142
|
+
return Number.isFinite(parsed) ? parsed : null;
|
|
143
|
+
}
|
|
144
|
+
return null;
|
|
145
|
+
})
|
|
146
|
+
.filter((value): value is number => value !== null)
|
|
147
|
+
.map((value) => (integer ? Math.trunc(value) : value));
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
export async function handleMatrixAction(
|
|
151
|
+
params: Record<string, unknown>,
|
|
152
|
+
cfg: CoreConfig,
|
|
153
|
+
opts: { mediaLocalRoots?: readonly string[] } = {},
|
|
154
|
+
): Promise<AgentToolResult<unknown>> {
|
|
155
|
+
const action = readStringParam(params, "action", { required: true });
|
|
156
|
+
const accountId = readStringParam(params, "accountId") ?? undefined;
|
|
157
|
+
const isActionEnabled = createActionGate(resolveMatrixAccountConfig({ cfg, accountId }).actions);
|
|
158
|
+
const clientOpts = {
|
|
159
|
+
cfg,
|
|
160
|
+
...(accountId ? { accountId } : {}),
|
|
161
|
+
};
|
|
162
|
+
|
|
163
|
+
if (reactionActions.has(action)) {
|
|
164
|
+
if (!isActionEnabled("reactions")) {
|
|
165
|
+
throw new Error("Matrix reactions are disabled.");
|
|
166
|
+
}
|
|
167
|
+
const roomId = readRoomId(params);
|
|
168
|
+
const messageId = readStringParam(params, "messageId", { required: true });
|
|
169
|
+
if (action === "react") {
|
|
170
|
+
const { emoji, remove, isEmpty } = readReactionParams(params, {
|
|
171
|
+
removeErrorMessage: "Emoji is required to remove a Matrix reaction.",
|
|
172
|
+
});
|
|
173
|
+
if (remove || isEmpty) {
|
|
174
|
+
const result = await removeMatrixReactions(roomId, messageId, {
|
|
175
|
+
...clientOpts,
|
|
176
|
+
emoji: remove ? emoji : undefined,
|
|
177
|
+
});
|
|
178
|
+
return jsonResult({ ok: true, removed: result.removed });
|
|
179
|
+
}
|
|
180
|
+
await reactMatrixMessage(roomId, messageId, emoji, clientOpts);
|
|
181
|
+
return jsonResult({ ok: true, added: emoji });
|
|
182
|
+
}
|
|
183
|
+
const limit = readNumberParam(params, "limit", { integer: true });
|
|
184
|
+
const reactions = await listMatrixReactions(roomId, messageId, {
|
|
185
|
+
...clientOpts,
|
|
186
|
+
limit: limit ?? undefined,
|
|
187
|
+
});
|
|
188
|
+
return jsonResult({ ok: true, reactions });
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
if (pollActions.has(action)) {
|
|
192
|
+
const roomId = readRoomId(params);
|
|
193
|
+
const pollId = readStringAliasParam(params, ["pollId", "messageId"], { required: true });
|
|
194
|
+
if (!pollId) {
|
|
195
|
+
throw new Error("pollId required");
|
|
196
|
+
}
|
|
197
|
+
const optionId = readStringParam(params, "pollOptionId");
|
|
198
|
+
const optionIndex = readNumberParam(params, "pollOptionIndex", { integer: true });
|
|
199
|
+
const optionIds = [
|
|
200
|
+
...(readStringArrayParam(params, "pollOptionIds") ?? []),
|
|
201
|
+
...(optionId ? [optionId] : []),
|
|
202
|
+
];
|
|
203
|
+
const optionIndexes = [
|
|
204
|
+
...readNumericArrayParam(params, "pollOptionIndexes", { integer: true }),
|
|
205
|
+
...(optionIndex !== undefined ? [optionIndex] : []),
|
|
206
|
+
];
|
|
207
|
+
const result = await voteMatrixPoll(roomId, pollId, {
|
|
208
|
+
...clientOpts,
|
|
209
|
+
optionIds,
|
|
210
|
+
optionIndexes,
|
|
211
|
+
});
|
|
212
|
+
return jsonResult({ ok: true, result });
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
if (messageActions.has(action)) {
|
|
216
|
+
if (!isActionEnabled("messages")) {
|
|
217
|
+
throw new Error("Matrix messages are disabled.");
|
|
218
|
+
}
|
|
219
|
+
switch (action) {
|
|
220
|
+
case "sendMessage": {
|
|
221
|
+
const to = readStringParam(params, "to", { required: true });
|
|
222
|
+
const mediaUrl =
|
|
223
|
+
readStringParam(params, "mediaUrl", { trim: false }) ??
|
|
224
|
+
readStringParam(params, "media", { trim: false }) ??
|
|
225
|
+
readStringParam(params, "filePath", { trim: false }) ??
|
|
226
|
+
readStringParam(params, "path", { trim: false });
|
|
227
|
+
const content = readStringParam(params, "content", {
|
|
228
|
+
required: !mediaUrl,
|
|
229
|
+
allowEmpty: true,
|
|
230
|
+
});
|
|
231
|
+
const replyToId =
|
|
232
|
+
readStringParam(params, "replyToId") ?? readStringParam(params, "replyTo");
|
|
233
|
+
const threadId = readStringParam(params, "threadId");
|
|
234
|
+
const audioAsVoice =
|
|
235
|
+
typeof readRawParam(params, "audioAsVoice") === "boolean"
|
|
236
|
+
? (readRawParam(params, "audioAsVoice") as boolean)
|
|
237
|
+
: typeof readRawParam(params, "asVoice") === "boolean"
|
|
238
|
+
? (readRawParam(params, "asVoice") as boolean)
|
|
239
|
+
: undefined;
|
|
240
|
+
const result = await sendMatrixMessage(to, content, {
|
|
241
|
+
mediaUrl: mediaUrl ?? undefined,
|
|
242
|
+
mediaLocalRoots: opts.mediaLocalRoots,
|
|
243
|
+
replyToId: replyToId ?? undefined,
|
|
244
|
+
threadId: threadId ?? undefined,
|
|
245
|
+
audioAsVoice,
|
|
246
|
+
...clientOpts,
|
|
247
|
+
});
|
|
248
|
+
return jsonResult({ ok: true, result });
|
|
249
|
+
}
|
|
250
|
+
case "editMessage": {
|
|
251
|
+
const roomId = readRoomId(params);
|
|
252
|
+
const messageId = readStringParam(params, "messageId", { required: true });
|
|
253
|
+
const content = readStringParam(params, "content", { required: true });
|
|
254
|
+
const result = await editMatrixMessage(roomId, messageId, content, clientOpts);
|
|
255
|
+
return jsonResult({ ok: true, result });
|
|
256
|
+
}
|
|
257
|
+
case "deleteMessage": {
|
|
258
|
+
const roomId = readRoomId(params);
|
|
259
|
+
const messageId = readStringParam(params, "messageId", { required: true });
|
|
260
|
+
const reason = readStringParam(params, "reason");
|
|
261
|
+
await deleteMatrixMessage(roomId, messageId, {
|
|
262
|
+
reason: reason ?? undefined,
|
|
263
|
+
...clientOpts,
|
|
264
|
+
});
|
|
265
|
+
return jsonResult({ ok: true, deleted: true });
|
|
266
|
+
}
|
|
267
|
+
case "readMessages": {
|
|
268
|
+
const roomId = readRoomId(params);
|
|
269
|
+
const limit = readNumberParam(params, "limit", { integer: true });
|
|
270
|
+
const before = readStringParam(params, "before");
|
|
271
|
+
const after = readStringParam(params, "after");
|
|
272
|
+
const result = await readMatrixMessages(roomId, {
|
|
273
|
+
limit: limit ?? undefined,
|
|
274
|
+
before: before ?? undefined,
|
|
275
|
+
after: after ?? undefined,
|
|
276
|
+
...clientOpts,
|
|
277
|
+
});
|
|
278
|
+
return jsonResult({ ok: true, ...result });
|
|
279
|
+
}
|
|
280
|
+
default:
|
|
281
|
+
break;
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
if (pinActions.has(action)) {
|
|
286
|
+
if (!isActionEnabled("pins")) {
|
|
287
|
+
throw new Error("Matrix pins are disabled.");
|
|
288
|
+
}
|
|
289
|
+
const roomId = readRoomId(params);
|
|
290
|
+
if (action === "pinMessage") {
|
|
291
|
+
const messageId = readStringParam(params, "messageId", { required: true });
|
|
292
|
+
const result = await pinMatrixMessage(roomId, messageId, clientOpts);
|
|
293
|
+
return jsonResult({ ok: true, pinned: result.pinned });
|
|
294
|
+
}
|
|
295
|
+
if (action === "unpinMessage") {
|
|
296
|
+
const messageId = readStringParam(params, "messageId", { required: true });
|
|
297
|
+
const result = await unpinMatrixMessage(roomId, messageId, clientOpts);
|
|
298
|
+
return jsonResult({ ok: true, pinned: result.pinned });
|
|
299
|
+
}
|
|
300
|
+
const result = await listMatrixPins(roomId, clientOpts);
|
|
301
|
+
return jsonResult({ ok: true, pinned: result.pinned, events: result.events });
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
if (profileActions.has(action)) {
|
|
305
|
+
if (!isActionEnabled("profile")) {
|
|
306
|
+
throw new Error("Matrix profile updates are disabled.");
|
|
307
|
+
}
|
|
308
|
+
const avatarPath =
|
|
309
|
+
readStringParam(params, "avatarPath") ??
|
|
310
|
+
readStringParam(params, "path") ??
|
|
311
|
+
readStringParam(params, "filePath");
|
|
312
|
+
const result = await applyMatrixProfileUpdate({
|
|
313
|
+
cfg,
|
|
314
|
+
account: accountId,
|
|
315
|
+
displayName: readStringParam(params, "displayName") ?? readStringParam(params, "name"),
|
|
316
|
+
avatarUrl: readStringParam(params, "avatarUrl"),
|
|
317
|
+
avatarPath,
|
|
318
|
+
mediaLocalRoots: opts.mediaLocalRoots,
|
|
319
|
+
});
|
|
320
|
+
return jsonResult({ ok: true, ...result });
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
if (action === "memberInfo") {
|
|
324
|
+
if (!isActionEnabled("memberInfo")) {
|
|
325
|
+
throw new Error("Matrix member info is disabled.");
|
|
326
|
+
}
|
|
327
|
+
const userId = readStringParam(params, "userId", { required: true });
|
|
328
|
+
const roomId = readStringParam(params, "roomId") ?? readStringParam(params, "channelId");
|
|
329
|
+
const result = await getMatrixMemberInfo(userId, {
|
|
330
|
+
roomId: roomId ?? undefined,
|
|
331
|
+
...clientOpts,
|
|
332
|
+
});
|
|
333
|
+
return jsonResult({ ok: true, member: result });
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
if (action === "channelInfo") {
|
|
337
|
+
if (!isActionEnabled("channelInfo")) {
|
|
338
|
+
throw new Error("Matrix room info is disabled.");
|
|
339
|
+
}
|
|
340
|
+
const roomId = readRoomId(params);
|
|
341
|
+
const result = await getMatrixRoomInfo(roomId, clientOpts);
|
|
342
|
+
return jsonResult({ ok: true, room: result });
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
if (verificationActions.has(action)) {
|
|
346
|
+
if (!isActionEnabled("verification")) {
|
|
347
|
+
throw new Error("Matrix verification actions are disabled.");
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
const requestId =
|
|
351
|
+
readStringParam(params, "requestId") ??
|
|
352
|
+
readStringParam(params, "verificationId") ??
|
|
353
|
+
readStringParam(params, "id");
|
|
354
|
+
|
|
355
|
+
if (action === "encryptionStatus") {
|
|
356
|
+
const includeRecoveryKey = params.includeRecoveryKey === true;
|
|
357
|
+
const status = await getMatrixEncryptionStatus({ includeRecoveryKey, ...clientOpts });
|
|
358
|
+
return jsonResult({ ok: true, status });
|
|
359
|
+
}
|
|
360
|
+
if (action === "verificationStatus") {
|
|
361
|
+
const includeRecoveryKey = params.includeRecoveryKey === true;
|
|
362
|
+
const status = await getMatrixVerificationStatus({ includeRecoveryKey, ...clientOpts });
|
|
363
|
+
return jsonResult({ ok: true, status });
|
|
364
|
+
}
|
|
365
|
+
if (action === "verificationBootstrap") {
|
|
366
|
+
const recoveryKey =
|
|
367
|
+
readStringParam(params, "recoveryKey", { trim: false }) ??
|
|
368
|
+
readStringParam(params, "key", { trim: false });
|
|
369
|
+
const result = await bootstrapMatrixVerification({
|
|
370
|
+
recoveryKey: recoveryKey ?? undefined,
|
|
371
|
+
forceResetCrossSigning: params.forceResetCrossSigning === true,
|
|
372
|
+
...clientOpts,
|
|
373
|
+
});
|
|
374
|
+
return jsonResult({ ok: result.success, result });
|
|
375
|
+
}
|
|
376
|
+
if (action === "verificationRecoveryKey") {
|
|
377
|
+
const recoveryKey =
|
|
378
|
+
readStringParam(params, "recoveryKey", { trim: false }) ??
|
|
379
|
+
readStringParam(params, "key", { trim: false });
|
|
380
|
+
const result = await verifyMatrixRecoveryKey(
|
|
381
|
+
readStringParam({ recoveryKey }, "recoveryKey", { required: true, trim: false }),
|
|
382
|
+
clientOpts,
|
|
383
|
+
);
|
|
384
|
+
return jsonResult({ ok: result.success, result });
|
|
385
|
+
}
|
|
386
|
+
if (action === "verificationBackupStatus") {
|
|
387
|
+
const status = await getMatrixRoomKeyBackupStatus(clientOpts);
|
|
388
|
+
return jsonResult({ ok: true, status });
|
|
389
|
+
}
|
|
390
|
+
if (action === "verificationBackupRestore") {
|
|
391
|
+
const recoveryKey =
|
|
392
|
+
readStringParam(params, "recoveryKey", { trim: false }) ??
|
|
393
|
+
readStringParam(params, "key", { trim: false });
|
|
394
|
+
const result = await restoreMatrixRoomKeyBackup({
|
|
395
|
+
recoveryKey: recoveryKey ?? undefined,
|
|
396
|
+
...clientOpts,
|
|
397
|
+
});
|
|
398
|
+
return jsonResult({ ok: result.success, result });
|
|
399
|
+
}
|
|
400
|
+
if (action === "verificationList") {
|
|
401
|
+
const verifications = await listMatrixVerifications(clientOpts);
|
|
402
|
+
return jsonResult({ ok: true, verifications });
|
|
403
|
+
}
|
|
404
|
+
if (action === "verificationRequest") {
|
|
405
|
+
const userId = readStringParam(params, "userId");
|
|
406
|
+
const deviceId = readStringParam(params, "deviceId");
|
|
407
|
+
const roomId = readStringParam(params, "roomId") ?? readStringParam(params, "channelId");
|
|
408
|
+
const ownUser = typeof params.ownUser === "boolean" ? params.ownUser : undefined;
|
|
409
|
+
const verification = await requestMatrixVerification({
|
|
410
|
+
ownUser,
|
|
411
|
+
userId: userId ?? undefined,
|
|
412
|
+
deviceId: deviceId ?? undefined,
|
|
413
|
+
roomId: roomId ?? undefined,
|
|
414
|
+
...clientOpts,
|
|
415
|
+
});
|
|
416
|
+
return jsonResult({ ok: true, verification });
|
|
417
|
+
}
|
|
418
|
+
if (action === "verificationAccept") {
|
|
419
|
+
const verification = await acceptMatrixVerification(
|
|
420
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
421
|
+
clientOpts,
|
|
422
|
+
);
|
|
423
|
+
return jsonResult({ ok: true, verification });
|
|
424
|
+
}
|
|
425
|
+
if (action === "verificationCancel") {
|
|
426
|
+
const reason = readStringParam(params, "reason");
|
|
427
|
+
const code = readStringParam(params, "code");
|
|
428
|
+
const verification = await cancelMatrixVerification(
|
|
429
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
430
|
+
{ reason: reason ?? undefined, code: code ?? undefined, ...clientOpts },
|
|
431
|
+
);
|
|
432
|
+
return jsonResult({ ok: true, verification });
|
|
433
|
+
}
|
|
434
|
+
if (action === "verificationStart") {
|
|
435
|
+
const methodRaw = readStringParam(params, "method");
|
|
436
|
+
const method = normalizeOptionalLowercaseString(methodRaw);
|
|
437
|
+
if (method && method !== "sas") {
|
|
438
|
+
throw new Error(
|
|
439
|
+
"Matrix verificationStart only supports method=sas; use verificationGenerateQr/verificationScanQr for QR flows.",
|
|
440
|
+
);
|
|
441
|
+
}
|
|
442
|
+
const verification = await startMatrixVerification(
|
|
443
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
444
|
+
{ method: "sas", ...clientOpts },
|
|
445
|
+
);
|
|
446
|
+
return jsonResult({ ok: true, verification });
|
|
447
|
+
}
|
|
448
|
+
if (action === "verificationGenerateQr") {
|
|
449
|
+
const qr = await generateMatrixVerificationQr(
|
|
450
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
451
|
+
clientOpts,
|
|
452
|
+
);
|
|
453
|
+
return jsonResult({ ok: true, ...qr });
|
|
454
|
+
}
|
|
455
|
+
if (action === "verificationScanQr") {
|
|
456
|
+
const qrDataBase64 =
|
|
457
|
+
readStringParam(params, "qrDataBase64") ??
|
|
458
|
+
readStringParam(params, "qrData") ??
|
|
459
|
+
readStringParam(params, "qr");
|
|
460
|
+
const verification = await scanMatrixVerificationQr(
|
|
461
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
462
|
+
readStringParam({ qrDataBase64 }, "qrDataBase64", { required: true }),
|
|
463
|
+
clientOpts,
|
|
464
|
+
);
|
|
465
|
+
return jsonResult({ ok: true, verification });
|
|
466
|
+
}
|
|
467
|
+
if (action === "verificationSas") {
|
|
468
|
+
const sas = await getMatrixVerificationSas(
|
|
469
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
470
|
+
clientOpts,
|
|
471
|
+
);
|
|
472
|
+
return jsonResult({ ok: true, sas });
|
|
473
|
+
}
|
|
474
|
+
if (action === "verificationConfirm") {
|
|
475
|
+
const verification = await confirmMatrixVerificationSas(
|
|
476
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
477
|
+
clientOpts,
|
|
478
|
+
);
|
|
479
|
+
return jsonResult({ ok: true, verification });
|
|
480
|
+
}
|
|
481
|
+
if (action === "verificationMismatch") {
|
|
482
|
+
const verification = await mismatchMatrixVerificationSas(
|
|
483
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
484
|
+
clientOpts,
|
|
485
|
+
);
|
|
486
|
+
return jsonResult({ ok: true, verification });
|
|
487
|
+
}
|
|
488
|
+
if (action === "verificationConfirmQr") {
|
|
489
|
+
const verification = await confirmMatrixVerificationReciprocateQr(
|
|
490
|
+
readStringParam({ requestId }, "requestId", { required: true }),
|
|
491
|
+
clientOpts,
|
|
492
|
+
);
|
|
493
|
+
return jsonResult({ ok: true, verification });
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
|
|
497
|
+
throw new Error(`Unsupported Matrix action: ${action}`);
|
|
498
|
+
}
|