@gakr-gakr/matrix 0.1.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 +285 -0
- package/SPEC-SUPPORT.md +116 -0
- package/api.ts +38 -0
- package/auth-presence.ts +56 -0
- package/autobot.plugin.json +28 -0
- package/channel-plugin-api.ts +3 -0
- package/cli-metadata.ts +11 -0
- package/contract-api.ts +17 -0
- package/doctor-contract-api.ts +1 -0
- package/helper-api.ts +3 -0
- package/index.ts +55 -0
- package/package.json +101 -0
- package/plugin-entry.handlers.runtime.ts +1 -0
- package/runtime-api.ts +72 -0
- package/runtime-heavy-api.ts +1 -0
- package/runtime-setter-api.ts +3 -0
- package/secret-contract-api.ts +5 -0
- package/setup-entry.ts +17 -0
- package/setup-plugin-api.ts +3 -0
- package/src/account-selection.ts +223 -0
- package/src/actions.ts +346 -0
- package/src/approval-auth.ts +25 -0
- package/src/approval-handler.runtime.ts +595 -0
- package/src/approval-ids.ts +6 -0
- package/src/approval-native.ts +348 -0
- package/src/approval-reaction-auth.ts +45 -0
- package/src/approval-reactions.ts +313 -0
- package/src/auth-precedence.ts +61 -0
- package/src/channel-account-paths.ts +97 -0
- package/src/channel.runtime.ts +17 -0
- package/src/channel.setup.ts +48 -0
- package/src/channel.ts +667 -0
- package/src/cli-metadata.ts +19 -0
- package/src/cli.ts +2298 -0
- package/src/config-adapter.ts +41 -0
- package/src/config-schema.ts +159 -0
- package/src/config-ui-hints.ts +56 -0
- package/src/directory-live.ts +238 -0
- package/src/doctor-contract.ts +287 -0
- package/src/doctor.ts +262 -0
- package/src/env-vars.ts +92 -0
- package/src/exec-approval-resolver.ts +23 -0
- package/src/exec-approvals.ts +293 -0
- package/src/group-mentions.ts +41 -0
- package/src/legacy-crypto-inspector-availability.ts +60 -0
- package/src/legacy-crypto.ts +531 -0
- package/src/legacy-state.ts +156 -0
- package/src/matrix/account-config.ts +175 -0
- package/src/matrix/accounts.ts +194 -0
- package/src/matrix/actions/client.ts +31 -0
- package/src/matrix/actions/devices.ts +34 -0
- package/src/matrix/actions/limits.ts +6 -0
- package/src/matrix/actions/messages.ts +129 -0
- package/src/matrix/actions/pins.ts +63 -0
- package/src/matrix/actions/polls.ts +109 -0
- package/src/matrix/actions/profile.ts +37 -0
- package/src/matrix/actions/reactions.ts +59 -0
- package/src/matrix/actions/room.ts +71 -0
- package/src/matrix/actions/summary.ts +88 -0
- package/src/matrix/actions/types.ts +63 -0
- package/src/matrix/actions/verification.ts +589 -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 +124 -0
- package/src/matrix/client/config-runtime-api.ts +9 -0
- package/src/matrix/client/config-secret-input.runtime.ts +1 -0
- package/src/matrix/client/config.ts +853 -0
- package/src/matrix/client/create-client.ts +105 -0
- package/src/matrix/client/env-auth.ts +95 -0
- package/src/matrix/client/file-sync-store.ts +289 -0
- package/src/matrix/client/logging.ts +140 -0
- package/src/matrix/client/migration-snapshot.runtime.ts +1 -0
- package/src/matrix/client/private-network-host.ts +1 -0
- package/src/matrix/client/runtime.ts +4 -0
- package/src/matrix/client/shared.ts +316 -0
- package/src/matrix/client/storage.ts +543 -0
- package/src/matrix/client/types.ts +50 -0
- package/src/matrix/client/url-validation.ts +76 -0
- package/src/matrix/client-bootstrap.ts +173 -0
- package/src/matrix/client.ts +23 -0
- package/src/matrix/config-paths.ts +31 -0
- package/src/matrix/config-update.ts +292 -0
- package/src/matrix/credentials-read.ts +207 -0
- package/src/matrix/credentials-write.runtime.ts +35 -0
- package/src/matrix/credentials.ts +95 -0
- package/src/matrix/deps.ts +309 -0
- package/src/matrix/device-health.ts +31 -0
- package/src/matrix/direct-management.ts +349 -0
- package/src/matrix/direct-room.ts +128 -0
- package/src/matrix/draft-stream.ts +225 -0
- package/src/matrix/encryption-guidance.ts +24 -0
- package/src/matrix/errors.ts +21 -0
- package/src/matrix/format.ts +426 -0
- package/src/matrix/legacy-crypto-inspector.ts +95 -0
- package/src/matrix/media-errors.ts +20 -0
- package/src/matrix/media-text.ts +162 -0
- package/src/matrix/monitor/access-state.ts +145 -0
- package/src/matrix/monitor/ack-config.ts +27 -0
- package/src/matrix/monitor/allowlist.ts +92 -0
- package/src/matrix/monitor/auto-join.ts +86 -0
- package/src/matrix/monitor/config.ts +569 -0
- package/src/matrix/monitor/context-summary.ts +43 -0
- package/src/matrix/monitor/direct.ts +296 -0
- package/src/matrix/monitor/events.ts +397 -0
- package/src/matrix/monitor/handler.ts +2271 -0
- package/src/matrix/monitor/inbound-dedupe.ts +267 -0
- package/src/matrix/monitor/index.ts +540 -0
- package/src/matrix/monitor/legacy-crypto-restore.ts +139 -0
- package/src/matrix/monitor/location.ts +108 -0
- package/src/matrix/monitor/media.ts +119 -0
- package/src/matrix/monitor/mentions.ts +256 -0
- package/src/matrix/monitor/reaction-events.ts +197 -0
- package/src/matrix/monitor/recent-invite.ts +30 -0
- package/src/matrix/monitor/replies.ts +136 -0
- package/src/matrix/monitor/reply-context.ts +92 -0
- package/src/matrix/monitor/room-history.ts +301 -0
- package/src/matrix/monitor/room-info.ts +126 -0
- package/src/matrix/monitor/rooms.ts +52 -0
- package/src/matrix/monitor/route.ts +179 -0
- package/src/matrix/monitor/runtime-api.ts +28 -0
- package/src/matrix/monitor/startup-verification.ts +237 -0
- package/src/matrix/monitor/startup.ts +218 -0
- package/src/matrix/monitor/status.ts +120 -0
- package/src/matrix/monitor/sync-lifecycle.ts +91 -0
- package/src/matrix/monitor/task-runner.ts +38 -0
- package/src/matrix/monitor/test-events.ts +21 -0
- package/src/matrix/monitor/thread-context.ts +108 -0
- package/src/matrix/monitor/threads.ts +85 -0
- package/src/matrix/monitor/types.ts +30 -0
- package/src/matrix/monitor/verification-events.ts +643 -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.ts +429 -0
- package/src/matrix/probe.runtime.ts +4 -0
- package/src/matrix/probe.ts +97 -0
- package/src/matrix/profile.ts +184 -0
- package/src/matrix/reaction-common.ts +147 -0
- package/src/matrix/sdk/crypto-bootstrap.ts +438 -0
- package/src/matrix/sdk/crypto-facade.ts +242 -0
- package/src/matrix/sdk/crypto-node.runtime.ts +17 -0
- package/src/matrix/sdk/crypto-runtime.ts +14 -0
- package/src/matrix/sdk/decrypt-bridge.ts +410 -0
- package/src/matrix/sdk/event-helpers.ts +83 -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.ts +286 -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.ts +453 -0
- package/src/matrix/sdk/timeout-abort-signal.ts +1 -0
- package/src/matrix/sdk/transport-runtime-api.ts +18 -0
- package/src/matrix/sdk/transport.ts +352 -0
- package/src/matrix/sdk/types.ts +245 -0
- package/src/matrix/sdk/verification-manager.ts +795 -0
- package/src/matrix/sdk/verification-status.ts +23 -0
- package/src/matrix/sdk.ts +2152 -0
- package/src/matrix/send/client.ts +93 -0
- package/src/matrix/send/formatting.ts +189 -0
- package/src/matrix/send/media.ts +244 -0
- package/src/matrix/send/targets.ts +104 -0
- package/src/matrix/send/types.ts +131 -0
- package/src/matrix/send.ts +660 -0
- package/src/matrix/session-store-metadata.ts +108 -0
- package/src/matrix/startup-abort.ts +44 -0
- package/src/matrix/subagent-hooks.ts +308 -0
- package/src/matrix/sync-state.ts +27 -0
- package/src/matrix/target-ids.ts +79 -0
- package/src/matrix/thread-bindings-shared.ts +206 -0
- package/src/matrix/thread-bindings.ts +580 -0
- package/src/matrix-migration.runtime.ts +9 -0
- package/src/migration-config.ts +243 -0
- package/src/migration-snapshot-backup.ts +116 -0
- package/src/migration-snapshot.ts +53 -0
- package/src/onboarding.ts +775 -0
- package/src/outbound.ts +248 -0
- package/src/plugin-entry.runtime.js +115 -0
- package/src/plugin-entry.runtime.ts +70 -0
- package/src/profile-update.ts +71 -0
- package/src/record-shared.ts +3 -0
- package/src/resolve-targets.ts +175 -0
- package/src/resolver.runtime.ts +5 -0
- package/src/resolver.ts +21 -0
- package/src/runtime-api.ts +106 -0
- package/src/runtime.ts +13 -0
- package/src/secret-contract.ts +174 -0
- package/src/session-route.ts +126 -0
- package/src/setup-bootstrap.ts +102 -0
- package/src/setup-config.ts +222 -0
- package/src/setup-contract.ts +90 -0
- package/src/setup-core.ts +146 -0
- package/src/setup-dm-policy.ts +15 -0
- package/src/setup-surface.ts +4 -0
- package/src/startup-maintenance.ts +114 -0
- package/src/storage-paths.ts +92 -0
- package/src/thread-binding-api.ts +23 -0
- package/src/tool-actions.runtime.ts +1 -0
- package/src/tool-actions.ts +498 -0
- package/src/types.ts +257 -0
- package/subagent-hooks-api.ts +31 -0
- package/test-api.ts +21 -0
- package/thread-binding-api.ts +4 -0
- package/thread-bindings-runtime.ts +4 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import {
|
|
2
|
+
buildPollResponseContent,
|
|
3
|
+
isPollStartType,
|
|
4
|
+
parsePollStart,
|
|
5
|
+
type PollStartContent,
|
|
6
|
+
} from "../poll-types.js";
|
|
7
|
+
import { withResolvedRoomAction } from "./client.js";
|
|
8
|
+
import type { MatrixActionClientOpts } from "./types.js";
|
|
9
|
+
|
|
10
|
+
function normalizeOptionIndexes(indexes: number[]): number[] {
|
|
11
|
+
const normalized = indexes
|
|
12
|
+
.map((index) => Math.trunc(index))
|
|
13
|
+
.filter((index) => Number.isFinite(index) && index > 0);
|
|
14
|
+
return Array.from(new Set(normalized));
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
function normalizeOptionIds(optionIds: string[]): string[] {
|
|
18
|
+
return Array.from(
|
|
19
|
+
new Set(optionIds.map((optionId) => optionId.trim()).filter((optionId) => optionId.length > 0)),
|
|
20
|
+
);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
function resolveSelectedAnswerIds(params: {
|
|
24
|
+
optionIds?: string[];
|
|
25
|
+
optionIndexes?: number[];
|
|
26
|
+
pollContent: PollStartContent;
|
|
27
|
+
}): { answerIds: string[]; labels: string[]; maxSelections: number } {
|
|
28
|
+
const parsed = parsePollStart(params.pollContent);
|
|
29
|
+
if (!parsed) {
|
|
30
|
+
throw new Error("Matrix poll vote requires a valid poll start event.");
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const selectedById = normalizeOptionIds(params.optionIds ?? []);
|
|
34
|
+
const selectedByIndex = normalizeOptionIndexes(params.optionIndexes ?? []).map((index) => {
|
|
35
|
+
const answer = parsed.answers[index - 1];
|
|
36
|
+
if (!answer) {
|
|
37
|
+
throw new Error(
|
|
38
|
+
`Matrix poll option index ${index} is out of range for a poll with ${parsed.answers.length} options.`,
|
|
39
|
+
);
|
|
40
|
+
}
|
|
41
|
+
return answer.id;
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
const answerIds = normalizeOptionIds([...selectedById, ...selectedByIndex]);
|
|
45
|
+
if (answerIds.length === 0) {
|
|
46
|
+
throw new Error("Matrix poll vote requires at least one poll option id or index.");
|
|
47
|
+
}
|
|
48
|
+
if (answerIds.length > parsed.maxSelections) {
|
|
49
|
+
throw new Error(
|
|
50
|
+
`Matrix poll allows at most ${parsed.maxSelections} selection${parsed.maxSelections === 1 ? "" : "s"}.`,
|
|
51
|
+
);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const answerMap = new Map(parsed.answers.map((answer) => [answer.id, answer.text] as const));
|
|
55
|
+
const labels = answerIds.map((answerId) => {
|
|
56
|
+
const label = answerMap.get(answerId);
|
|
57
|
+
if (!label) {
|
|
58
|
+
throw new Error(
|
|
59
|
+
`Matrix poll option id "${answerId}" is not valid for poll ${parsed.question}.`,
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
return label;
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
return {
|
|
66
|
+
answerIds,
|
|
67
|
+
labels,
|
|
68
|
+
maxSelections: parsed.maxSelections,
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
export async function voteMatrixPoll(
|
|
73
|
+
roomId: string,
|
|
74
|
+
pollId: string,
|
|
75
|
+
opts: MatrixActionClientOpts & {
|
|
76
|
+
optionId?: string;
|
|
77
|
+
optionIds?: string[];
|
|
78
|
+
optionIndex?: number;
|
|
79
|
+
optionIndexes?: number[];
|
|
80
|
+
} = {},
|
|
81
|
+
) {
|
|
82
|
+
return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
|
|
83
|
+
const pollEvent = await client.getEvent(resolvedRoom, pollId);
|
|
84
|
+
const eventType = typeof pollEvent.type === "string" ? pollEvent.type : "";
|
|
85
|
+
if (!isPollStartType(eventType)) {
|
|
86
|
+
throw new Error(`Event ${pollId} is not a Matrix poll start event.`);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
const { answerIds, labels, maxSelections } = resolveSelectedAnswerIds({
|
|
90
|
+
optionIds: [...(opts.optionIds ?? []), ...(opts.optionId ? [opts.optionId] : [])],
|
|
91
|
+
optionIndexes: [
|
|
92
|
+
...(opts.optionIndexes ?? []),
|
|
93
|
+
...(opts.optionIndex !== undefined ? [opts.optionIndex] : []),
|
|
94
|
+
],
|
|
95
|
+
pollContent: pollEvent.content as PollStartContent,
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
const content = buildPollResponseContent(pollId, answerIds);
|
|
99
|
+
const eventId = await client.sendEvent(resolvedRoom, "m.poll.response", content);
|
|
100
|
+
return {
|
|
101
|
+
eventId: eventId ?? null,
|
|
102
|
+
roomId: resolvedRoom,
|
|
103
|
+
pollId,
|
|
104
|
+
answerIds,
|
|
105
|
+
labels,
|
|
106
|
+
maxSelections,
|
|
107
|
+
};
|
|
108
|
+
});
|
|
109
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { getMatrixRuntime } from "../../runtime.js";
|
|
2
|
+
import { syncMatrixOwnProfile, type MatrixProfileSyncResult } from "../profile.js";
|
|
3
|
+
import { withResolvedActionClient } from "./client.js";
|
|
4
|
+
import type { MatrixActionClientOpts } from "./types.js";
|
|
5
|
+
|
|
6
|
+
export async function updateMatrixOwnProfile(
|
|
7
|
+
opts: MatrixActionClientOpts & {
|
|
8
|
+
displayName?: string;
|
|
9
|
+
avatarUrl?: string;
|
|
10
|
+
avatarPath?: string;
|
|
11
|
+
} = {},
|
|
12
|
+
): Promise<MatrixProfileSyncResult> {
|
|
13
|
+
const displayName = opts.displayName?.trim();
|
|
14
|
+
const avatarUrl = opts.avatarUrl?.trim();
|
|
15
|
+
const avatarPath = opts.avatarPath?.trim();
|
|
16
|
+
const runtime = getMatrixRuntime();
|
|
17
|
+
return await withResolvedActionClient(
|
|
18
|
+
opts,
|
|
19
|
+
async (client) => {
|
|
20
|
+
const userId = await client.getUserId();
|
|
21
|
+
return await syncMatrixOwnProfile({
|
|
22
|
+
client,
|
|
23
|
+
userId,
|
|
24
|
+
displayName: displayName || undefined,
|
|
25
|
+
avatarUrl: avatarUrl || undefined,
|
|
26
|
+
avatarPath: avatarPath || undefined,
|
|
27
|
+
loadAvatarFromUrl: async (url, maxBytes) => await runtime.media.loadWebMedia(url, maxBytes),
|
|
28
|
+
loadAvatarFromPath: async (path, maxBytes) =>
|
|
29
|
+
await runtime.media.loadWebMedia(path, {
|
|
30
|
+
maxBytes,
|
|
31
|
+
localRoots: opts.mediaLocalRoots,
|
|
32
|
+
}),
|
|
33
|
+
});
|
|
34
|
+
},
|
|
35
|
+
"persist",
|
|
36
|
+
);
|
|
37
|
+
}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import {
|
|
2
|
+
buildMatrixReactionRelationsPath,
|
|
3
|
+
selectOwnMatrixReactionEventIds,
|
|
4
|
+
summarizeMatrixReactionEvents,
|
|
5
|
+
} from "../reaction-common.js";
|
|
6
|
+
import { withResolvedRoomAction } from "./client.js";
|
|
7
|
+
import { resolveMatrixActionLimit } from "./limits.js";
|
|
8
|
+
import {
|
|
9
|
+
type MatrixActionClientOpts,
|
|
10
|
+
type MatrixRawEvent,
|
|
11
|
+
type MatrixReactionSummary,
|
|
12
|
+
} from "./types.js";
|
|
13
|
+
|
|
14
|
+
type ActionClient = NonNullable<MatrixActionClientOpts["client"]>;
|
|
15
|
+
|
|
16
|
+
async function listMatrixReactionEvents(
|
|
17
|
+
client: ActionClient,
|
|
18
|
+
roomId: string,
|
|
19
|
+
messageId: string,
|
|
20
|
+
limit: number,
|
|
21
|
+
): Promise<MatrixRawEvent[]> {
|
|
22
|
+
const res = (await client.doRequest("GET", buildMatrixReactionRelationsPath(roomId, messageId), {
|
|
23
|
+
dir: "b",
|
|
24
|
+
limit,
|
|
25
|
+
})) as { chunk?: MatrixRawEvent[] };
|
|
26
|
+
return Array.isArray(res.chunk) ? res.chunk : [];
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export async function listMatrixReactions(
|
|
30
|
+
roomId: string,
|
|
31
|
+
messageId: string,
|
|
32
|
+
opts: MatrixActionClientOpts & { limit?: number } = {},
|
|
33
|
+
): Promise<MatrixReactionSummary[]> {
|
|
34
|
+
return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
|
|
35
|
+
const limit = resolveMatrixActionLimit(opts.limit, 100);
|
|
36
|
+
const chunk = await listMatrixReactionEvents(client, resolvedRoom, messageId, limit);
|
|
37
|
+
return summarizeMatrixReactionEvents(chunk);
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export async function removeMatrixReactions(
|
|
42
|
+
roomId: string,
|
|
43
|
+
messageId: string,
|
|
44
|
+
opts: MatrixActionClientOpts & { emoji?: string } = {},
|
|
45
|
+
): Promise<{ removed: number }> {
|
|
46
|
+
return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
|
|
47
|
+
const chunk = await listMatrixReactionEvents(client, resolvedRoom, messageId, 200);
|
|
48
|
+
const userId = await client.getUserId();
|
|
49
|
+
if (!userId) {
|
|
50
|
+
return { removed: 0 };
|
|
51
|
+
}
|
|
52
|
+
const toRemove = selectOwnMatrixReactionEventIds(chunk, userId, opts.emoji);
|
|
53
|
+
if (toRemove.length === 0) {
|
|
54
|
+
return { removed: 0 };
|
|
55
|
+
}
|
|
56
|
+
await Promise.all(toRemove.map((id) => client.redactEvent(resolvedRoom, id)));
|
|
57
|
+
return { removed: toRemove.length };
|
|
58
|
+
});
|
|
59
|
+
}
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import { resolveMatrixRoomId } from "../send.js";
|
|
2
|
+
import { withResolvedActionClient, withResolvedRoomAction } from "./client.js";
|
|
3
|
+
import { EventType, type MatrixActionClientOpts } from "./types.js";
|
|
4
|
+
|
|
5
|
+
export async function getMatrixMemberInfo(
|
|
6
|
+
userId: string,
|
|
7
|
+
opts: MatrixActionClientOpts & { roomId?: string } = {},
|
|
8
|
+
) {
|
|
9
|
+
return await withResolvedActionClient(opts, async (client) => {
|
|
10
|
+
const roomId = opts.roomId ? await resolveMatrixRoomId(client, opts.roomId) : undefined;
|
|
11
|
+
const profile = await client.getUserProfile(userId);
|
|
12
|
+
// Membership and power levels are not included in profile calls; fetch state separately if needed.
|
|
13
|
+
return {
|
|
14
|
+
userId,
|
|
15
|
+
profile: {
|
|
16
|
+
displayName: profile?.displayname ?? null,
|
|
17
|
+
avatarUrl: profile?.avatar_url ?? null,
|
|
18
|
+
},
|
|
19
|
+
membership: null, // Would need separate room state query
|
|
20
|
+
powerLevel: null, // Would need separate power levels state query
|
|
21
|
+
displayName: profile?.displayname ?? null,
|
|
22
|
+
roomId: roomId ?? null,
|
|
23
|
+
};
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export async function getMatrixRoomInfo(roomId: string, opts: MatrixActionClientOpts = {}) {
|
|
28
|
+
return await withResolvedRoomAction(roomId, opts, async (client, resolvedRoom) => {
|
|
29
|
+
let name: string | null = null;
|
|
30
|
+
let topic: string | null = null;
|
|
31
|
+
let canonicalAlias: string | null = null;
|
|
32
|
+
let memberCount: number | null = null;
|
|
33
|
+
|
|
34
|
+
try {
|
|
35
|
+
const nameState = await client.getRoomStateEvent(resolvedRoom, "m.room.name", "");
|
|
36
|
+
name = typeof nameState?.name === "string" ? nameState.name : null;
|
|
37
|
+
} catch {
|
|
38
|
+
// ignore
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
try {
|
|
42
|
+
const topicState = await client.getRoomStateEvent(resolvedRoom, EventType.RoomTopic, "");
|
|
43
|
+
topic = typeof topicState?.topic === "string" ? topicState.topic : null;
|
|
44
|
+
} catch {
|
|
45
|
+
// ignore
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
try {
|
|
49
|
+
const aliasState = await client.getRoomStateEvent(resolvedRoom, "m.room.canonical_alias", "");
|
|
50
|
+
canonicalAlias = typeof aliasState?.alias === "string" ? aliasState.alias : null;
|
|
51
|
+
} catch {
|
|
52
|
+
// ignore
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
try {
|
|
56
|
+
const members = await client.getJoinedRoomMembers(resolvedRoom);
|
|
57
|
+
memberCount = members.length;
|
|
58
|
+
} catch {
|
|
59
|
+
// ignore
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
return {
|
|
63
|
+
roomId: resolvedRoom,
|
|
64
|
+
name,
|
|
65
|
+
topic,
|
|
66
|
+
canonicalAlias,
|
|
67
|
+
altAliases: [], // Would need separate query
|
|
68
|
+
memberCount,
|
|
69
|
+
};
|
|
70
|
+
});
|
|
71
|
+
}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import { isMatrixNotFoundError } from "../errors.js";
|
|
2
|
+
import { resolveMatrixMessageAttachment, resolveMatrixMessageBody } from "../media-text.js";
|
|
3
|
+
import { fetchMatrixPollMessageSummary } from "../poll-summary.js";
|
|
4
|
+
import type { MatrixClient } from "../sdk.js";
|
|
5
|
+
import {
|
|
6
|
+
EventType,
|
|
7
|
+
type MatrixMessageSummary,
|
|
8
|
+
type MatrixRawEvent,
|
|
9
|
+
type RoomMessageEventContent,
|
|
10
|
+
type RoomPinnedEventsEventContent,
|
|
11
|
+
} from "./types.js";
|
|
12
|
+
|
|
13
|
+
export function summarizeMatrixRawEvent(event: MatrixRawEvent): MatrixMessageSummary {
|
|
14
|
+
const content = event.content as RoomMessageEventContent;
|
|
15
|
+
const relates = content["m.relates_to"];
|
|
16
|
+
let relType: string | undefined;
|
|
17
|
+
let eventId: string | undefined;
|
|
18
|
+
if (relates) {
|
|
19
|
+
if ("rel_type" in relates) {
|
|
20
|
+
relType = relates.rel_type;
|
|
21
|
+
eventId = relates.event_id;
|
|
22
|
+
} else if ("m.in_reply_to" in relates) {
|
|
23
|
+
eventId = relates["m.in_reply_to"]?.event_id;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
const relatesTo =
|
|
27
|
+
relType || eventId
|
|
28
|
+
? {
|
|
29
|
+
relType,
|
|
30
|
+
eventId,
|
|
31
|
+
}
|
|
32
|
+
: undefined;
|
|
33
|
+
return {
|
|
34
|
+
eventId: event.event_id,
|
|
35
|
+
sender: event.sender,
|
|
36
|
+
body: resolveMatrixMessageBody({
|
|
37
|
+
body: content.body,
|
|
38
|
+
filename: content.filename,
|
|
39
|
+
msgtype: content.msgtype,
|
|
40
|
+
}),
|
|
41
|
+
msgtype: content.msgtype,
|
|
42
|
+
attachment: resolveMatrixMessageAttachment({
|
|
43
|
+
body: content.body,
|
|
44
|
+
filename: content.filename,
|
|
45
|
+
msgtype: content.msgtype,
|
|
46
|
+
}),
|
|
47
|
+
timestamp: event.origin_server_ts,
|
|
48
|
+
relatesTo,
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export async function readPinnedEvents(client: MatrixClient, roomId: string): Promise<string[]> {
|
|
53
|
+
try {
|
|
54
|
+
const content = (await client.getRoomStateEvent(
|
|
55
|
+
roomId,
|
|
56
|
+
EventType.RoomPinnedEvents,
|
|
57
|
+
"",
|
|
58
|
+
)) as RoomPinnedEventsEventContent;
|
|
59
|
+
const pinned = content.pinned;
|
|
60
|
+
return pinned.filter((id) => id.trim().length > 0);
|
|
61
|
+
} catch (err: unknown) {
|
|
62
|
+
if (isMatrixNotFoundError(err)) {
|
|
63
|
+
return [];
|
|
64
|
+
}
|
|
65
|
+
throw err;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
export async function fetchEventSummary(
|
|
70
|
+
client: MatrixClient,
|
|
71
|
+
roomId: string,
|
|
72
|
+
eventId: string,
|
|
73
|
+
): Promise<MatrixMessageSummary | null> {
|
|
74
|
+
try {
|
|
75
|
+
const raw = (await client.getEvent(roomId, eventId)) as unknown as MatrixRawEvent;
|
|
76
|
+
if (raw.unsigned?.redacted_because) {
|
|
77
|
+
return null;
|
|
78
|
+
}
|
|
79
|
+
const pollSummary = await fetchMatrixPollMessageSummary(client, roomId, raw);
|
|
80
|
+
if (pollSummary) {
|
|
81
|
+
return pollSummary;
|
|
82
|
+
}
|
|
83
|
+
return summarizeMatrixRawEvent(raw);
|
|
84
|
+
} catch {
|
|
85
|
+
// Event not found, redacted, or inaccessible - return null
|
|
86
|
+
return null;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import type { CoreConfig } from "../../types.js";
|
|
2
|
+
import { MATRIX_REACTION_EVENT_TYPE } from "../reaction-common.js";
|
|
3
|
+
import type { MatrixClient, MessageEventContent } from "../sdk.js";
|
|
4
|
+
export type { MatrixRawEvent } from "../sdk.js";
|
|
5
|
+
export type { MatrixReactionSummary } from "../reaction-common.js";
|
|
6
|
+
|
|
7
|
+
export const EventType = {
|
|
8
|
+
RoomMessage: "m.room.message",
|
|
9
|
+
RoomPinnedEvents: "m.room.pinned_events",
|
|
10
|
+
RoomTopic: "m.room.topic",
|
|
11
|
+
Reaction: MATRIX_REACTION_EVENT_TYPE,
|
|
12
|
+
} as const;
|
|
13
|
+
|
|
14
|
+
export type RoomMessageEventContent = MessageEventContent & {
|
|
15
|
+
msgtype: string;
|
|
16
|
+
body: string;
|
|
17
|
+
"m.new_content"?: RoomMessageEventContent;
|
|
18
|
+
"m.relates_to"?: {
|
|
19
|
+
rel_type?: string;
|
|
20
|
+
event_id?: string;
|
|
21
|
+
"m.in_reply_to"?: { event_id?: string };
|
|
22
|
+
};
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
export type RoomPinnedEventsEventContent = {
|
|
26
|
+
pinned: string[];
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
export type MatrixActionClientOpts = {
|
|
30
|
+
client?: MatrixClient;
|
|
31
|
+
cfg?: CoreConfig;
|
|
32
|
+
mediaLocalRoots?: readonly string[];
|
|
33
|
+
timeoutMs?: number;
|
|
34
|
+
accountId?: string | null;
|
|
35
|
+
readiness?: "none" | "prepared" | "started";
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
export type MatrixMessageSummary = {
|
|
39
|
+
eventId?: string;
|
|
40
|
+
sender?: string;
|
|
41
|
+
body?: string;
|
|
42
|
+
msgtype?: string;
|
|
43
|
+
attachment?: MatrixMessageAttachmentSummary;
|
|
44
|
+
timestamp?: number;
|
|
45
|
+
relatesTo?: {
|
|
46
|
+
relType?: string;
|
|
47
|
+
eventId?: string;
|
|
48
|
+
key?: string;
|
|
49
|
+
};
|
|
50
|
+
};
|
|
51
|
+
|
|
52
|
+
export type MatrixMessageAttachmentKind = "audio" | "file" | "image" | "sticker" | "video";
|
|
53
|
+
|
|
54
|
+
export type MatrixMessageAttachmentSummary = {
|
|
55
|
+
kind: MatrixMessageAttachmentKind;
|
|
56
|
+
caption?: string;
|
|
57
|
+
filename?: string;
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
export type MatrixActionClient = {
|
|
61
|
+
client: MatrixClient;
|
|
62
|
+
stopOnDone: boolean;
|
|
63
|
+
};
|