@openclaw/crabline 0.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 (111) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +226 -0
  3. package/dist/src/bin/crabline.d.ts +2 -0
  4. package/dist/src/bin/crabline.js +7 -0
  5. package/dist/src/bin/crabline.js.map +1 -0
  6. package/dist/src/cli/program.d.ts +5 -0
  7. package/dist/src/cli/program.js +295 -0
  8. package/dist/src/cli/program.js.map +1 -0
  9. package/dist/src/config/load.d.ts +6 -0
  10. package/dist/src/config/load.js +41 -0
  11. package/dist/src/config/load.js.map +1 -0
  12. package/dist/src/config/schema.d.ts +1571 -0
  13. package/dist/src/config/schema.js +528 -0
  14. package/dist/src/config/schema.js.map +1 -0
  15. package/dist/src/core/errors.d.ts +12 -0
  16. package/dist/src/core/errors.js +28 -0
  17. package/dist/src/core/errors.js.map +1 -0
  18. package/dist/src/core/exit-codes.d.ts +13 -0
  19. package/dist/src/core/exit-codes.js +13 -0
  20. package/dist/src/core/exit-codes.js.map +1 -0
  21. package/dist/src/core/matcher.d.ts +2 -0
  22. package/dist/src/core/matcher.js +30 -0
  23. package/dist/src/core/matcher.js.map +1 -0
  24. package/dist/src/core/message-template.d.ts +2 -0
  25. package/dist/src/core/message-template.js +20 -0
  26. package/dist/src/core/message-template.js.map +1 -0
  27. package/dist/src/core/nonces.d.ts +2 -0
  28. package/dist/src/core/nonces.js +11 -0
  29. package/dist/src/core/nonces.js.map +1 -0
  30. package/dist/src/core/reporters.d.ts +3 -0
  31. package/dist/src/core/reporters.js +23 -0
  32. package/dist/src/core/reporters.js.map +1 -0
  33. package/dist/src/core/run.d.ts +30 -0
  34. package/dist/src/core/run.js +240 -0
  35. package/dist/src/core/run.js.map +1 -0
  36. package/dist/src/fake-servers/index.d.ts +10 -0
  37. package/dist/src/fake-servers/index.js +12 -0
  38. package/dist/src/fake-servers/index.js.map +1 -0
  39. package/dist/src/fake-servers/telegram.d.ts +29 -0
  40. package/dist/src/fake-servers/telegram.js +346 -0
  41. package/dist/src/fake-servers/telegram.js.map +1 -0
  42. package/dist/src/index.d.ts +15 -0
  43. package/dist/src/index.js +9 -0
  44. package/dist/src/index.js.map +1 -0
  45. package/dist/src/openclaw.d.ts +102 -0
  46. package/dist/src/openclaw.js +289 -0
  47. package/dist/src/openclaw.js.map +1 -0
  48. package/dist/src/providers/builtin/discord.d.ts +15 -0
  49. package/dist/src/providers/builtin/discord.js +92 -0
  50. package/dist/src/providers/builtin/discord.js.map +1 -0
  51. package/dist/src/providers/builtin/feishu.d.ts +10 -0
  52. package/dist/src/providers/builtin/feishu.js +95 -0
  53. package/dist/src/providers/builtin/feishu.js.map +1 -0
  54. package/dist/src/providers/builtin/googlechat.d.ts +11 -0
  55. package/dist/src/providers/builtin/googlechat.js +83 -0
  56. package/dist/src/providers/builtin/googlechat.js.map +1 -0
  57. package/dist/src/providers/builtin/imessage.d.ts +11 -0
  58. package/dist/src/providers/builtin/imessage.js +67 -0
  59. package/dist/src/providers/builtin/imessage.js.map +1 -0
  60. package/dist/src/providers/builtin/loopback.d.ts +63 -0
  61. package/dist/src/providers/builtin/loopback.js +174 -0
  62. package/dist/src/providers/builtin/loopback.js.map +1 -0
  63. package/dist/src/providers/builtin/matrix.d.ts +21 -0
  64. package/dist/src/providers/builtin/matrix.js +84 -0
  65. package/dist/src/providers/builtin/matrix.js.map +1 -0
  66. package/dist/src/providers/builtin/mattermost.d.ts +11 -0
  67. package/dist/src/providers/builtin/mattermost.js +67 -0
  68. package/dist/src/providers/builtin/mattermost.js.map +1 -0
  69. package/dist/src/providers/builtin/msteams.d.ts +13 -0
  70. package/dist/src/providers/builtin/msteams.js +72 -0
  71. package/dist/src/providers/builtin/msteams.js.map +1 -0
  72. package/dist/src/providers/builtin/native-local-mock.d.ts +48 -0
  73. package/dist/src/providers/builtin/native-local-mock.js +119 -0
  74. package/dist/src/providers/builtin/native-local-mock.js.map +1 -0
  75. package/dist/src/providers/builtin/script.d.ts +39 -0
  76. package/dist/src/providers/builtin/script.js +206 -0
  77. package/dist/src/providers/builtin/script.js.map +1 -0
  78. package/dist/src/providers/builtin/slack.d.ts +6 -0
  79. package/dist/src/providers/builtin/slack.js +106 -0
  80. package/dist/src/providers/builtin/slack.js.map +1 -0
  81. package/dist/src/providers/builtin/telegram.d.ts +21 -0
  82. package/dist/src/providers/builtin/telegram.js +123 -0
  83. package/dist/src/providers/builtin/telegram.js.map +1 -0
  84. package/dist/src/providers/builtin/whatsapp.d.ts +19 -0
  85. package/dist/src/providers/builtin/whatsapp.js +90 -0
  86. package/dist/src/providers/builtin/whatsapp.js.map +1 -0
  87. package/dist/src/providers/builtin/zalo.d.ts +17 -0
  88. package/dist/src/providers/builtin/zalo.js +70 -0
  89. package/dist/src/providers/builtin/zalo.js.map +1 -0
  90. package/dist/src/providers/catalog.d.ts +69 -0
  91. package/dist/src/providers/catalog.js +95 -0
  92. package/dist/src/providers/catalog.js.map +1 -0
  93. package/dist/src/providers/local-mock.d.ts +41 -0
  94. package/dist/src/providers/local-mock.js +243 -0
  95. package/dist/src/providers/local-mock.js.map +1 -0
  96. package/dist/src/providers/recorder.d.ts +19 -0
  97. package/dist/src/providers/recorder.js +160 -0
  98. package/dist/src/providers/recorder.js.map +1 -0
  99. package/dist/src/providers/registry.d.ts +8 -0
  100. package/dist/src/providers/registry.js +171 -0
  101. package/dist/src/providers/registry.js.map +1 -0
  102. package/dist/src/providers/types.d.ts +94 -0
  103. package/dist/src/providers/types.js +2 -0
  104. package/dist/src/providers/types.js.map +1 -0
  105. package/dist/src/providers/webhook-server.d.ts +12 -0
  106. package/dist/src/providers/webhook-server.js +118 -0
  107. package/dist/src/providers/webhook-server.js.map +1 -0
  108. package/docs/channel-setup.md +181 -0
  109. package/fixtures/examples/crabline.example.yaml +195 -0
  110. package/fixtures/examples/openclaw-bridge.yaml +125 -0
  111. package/package.json +64 -0
@@ -0,0 +1,67 @@
1
+ import path from "node:path";
2
+ import { CrablineError } from "../../core/errors.js";
3
+ import { LocalMockProviderAdapter } from "../local-mock.js";
4
+ import { authorFromBotFlag, createNativeTargetCodec, genericMockPayloadWithNativeThread, isRecord, optionalRecord, optionalString, requireNativeInboundId, } from "./native-local-mock.js";
5
+ const IMESSAGE_THREAD_RULE = {
6
+ example: "+15551234567, user@example.com, or iMessage;-;chat-guid",
7
+ name: "iMessage recipient or chat GUID",
8
+ pattern: /^(?:\+[1-9]\d{6,14}|[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Za-z]{2,}|(?:iMessage|SMS);[+-];.+)$/u,
9
+ };
10
+ export function resolveIMessageAdapterConfig(config, env = process.env) {
11
+ return {
12
+ apiKey: config.imessage?.apiKey ?? env.IMESSAGE_API_KEY ?? "local-mock-imessage-api-key",
13
+ local: config.imessage?.local ?? true,
14
+ serverUrl: config.imessage?.serverUrl ?? env.IMESSAGE_SERVER_URL,
15
+ };
16
+ }
17
+ export class IMessageProviderAdapter extends LocalMockProviderAdapter {
18
+ constructor(id, config, _userName, _runtime) {
19
+ super({
20
+ codec: createNativeTargetCodec({
21
+ channel: IMESSAGE_THREAD_RULE,
22
+ channelLabel: "iMessage recipient or chat GUID",
23
+ }),
24
+ config,
25
+ id,
26
+ options: {
27
+ defaultWebhook: { host: "127.0.0.1", path: "/imessage/webhook", port: 8796 },
28
+ endpointLabel: "webhook endpoint",
29
+ normalizeWebhookPayload: normalizeIMessageWebhookPayload,
30
+ platform: "imessage",
31
+ publicUrl: config.imessage?.webhook.publicUrl,
32
+ recorderPath: config.imessage?.recorder.path
33
+ ? path.resolve(config.imessage.recorder.path)
34
+ : undefined,
35
+ webhook: config.imessage?.webhook,
36
+ },
37
+ });
38
+ }
39
+ }
40
+ function normalizeIMessageWebhookPayload(payload) {
41
+ if (!isRecord(payload)) {
42
+ throw new CrablineError("iMessage webhook payload must be an object", { kind: "inbound" });
43
+ }
44
+ if (optionalRecord(payload, "message")) {
45
+ return genericMockPayloadWithNativeThread({
46
+ channelRule: IMESSAGE_THREAD_RULE,
47
+ payload,
48
+ threadRule: IMESSAGE_THREAD_RULE,
49
+ });
50
+ }
51
+ const data = optionalRecord(payload, "data") ?? payload;
52
+ const threadId = optionalString(data, "chatGuid") ?? optionalString(data, "chatIdentifier");
53
+ const text = optionalString(data, "text") ?? optionalString(data, "message");
54
+ if (!threadId || !text) {
55
+ throw new CrablineError("iMessage webhook payload requires chatGuid and text", {
56
+ kind: "inbound",
57
+ });
58
+ }
59
+ return {
60
+ author: authorFromBotFlag(data.isFromMe === true),
61
+ ...(optionalString(data, "guid") ? { id: optionalString(data, "guid") } : {}),
62
+ raw: payload,
63
+ text,
64
+ threadId: requireNativeInboundId(threadId, IMESSAGE_THREAD_RULE, "iMessage chatGuid"),
65
+ };
66
+ }
67
+ //# sourceMappingURL=imessage.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"imessage.js","sourceRoot":"","sources":["../../../../src/providers/builtin/imessage.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,63 @@
1
+ import type { ProviderConfig } from "../../config/schema.js";
2
+ import { LocalMockProviderAdapter } from "../local-mock.js";
3
+ import type { LoopbackMessage, LoopbackRawMessage, ProviderAdapter } from "../types.js";
4
+ type ThreadAddress = {
5
+ channelId?: string | undefined;
6
+ id: string;
7
+ threadId?: string | undefined;
8
+ };
9
+ type PostableMessage = string | {
10
+ card: unknown;
11
+ fallbackText?: string;
12
+ } | {
13
+ markdown: string;
14
+ } | {
15
+ raw: string;
16
+ };
17
+ export declare class LoopbackChatAdapter {
18
+ #private;
19
+ readonly name = "loopback";
20
+ readonly persistMessageHistory = true;
21
+ readonly userName: string;
22
+ constructor(userName: string);
23
+ addReaction(): Promise<void>;
24
+ channelIdFromThreadId(threadId: string): string;
25
+ decodeThreadId(threadId: string): ThreadAddress;
26
+ deleteMessage(threadId: string, messageId: string): Promise<void>;
27
+ editMessage(threadId: string, messageId: string, message: PostableMessage): Promise<{
28
+ id: string;
29
+ raw: LoopbackRawMessage;
30
+ threadId: string;
31
+ }>;
32
+ encodeThreadId(platformData: ThreadAddress): string;
33
+ fetchMessages(threadId: string, options?: {
34
+ cursor?: string;
35
+ limit?: number;
36
+ }): Promise<{
37
+ messages: LoopbackMessage[];
38
+ nextCursor?: string;
39
+ }>;
40
+ fetchThread(threadId: string): Promise<{
41
+ channelId: string;
42
+ id: string;
43
+ isDM: boolean;
44
+ metadata: {};
45
+ }>;
46
+ handleWebhook(_request?: Request): Promise<Response>;
47
+ isDM(): boolean;
48
+ parseMessage(raw: LoopbackRawMessage): LoopbackMessage;
49
+ postMessage(threadId: string, message: PostableMessage): Promise<{
50
+ id: string;
51
+ raw: LoopbackRawMessage;
52
+ threadId: string;
53
+ }>;
54
+ removeReaction(): Promise<void>;
55
+ renderFormatted(content: string): string;
56
+ startTyping(): Promise<void>;
57
+ ingestUserMessage(threadId: string, text: string): LoopbackMessage;
58
+ listSince(threadId: string, since: string): LoopbackMessage[];
59
+ }
60
+ export declare class LoopbackProviderAdapter extends LocalMockProviderAdapter implements ProviderAdapter {
61
+ constructor(id: string, config: ProviderConfig, _userName: string);
62
+ }
63
+ export {};
@@ -0,0 +1,174 @@
1
+ import { CrablineError } from "../../core/errors.js";
2
+ import { createGenericLocalMockTargetCodec, LocalMockProviderAdapter } from "../local-mock.js";
3
+ function createMessageId() {
4
+ return `loopback-mock-${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 8)}`;
5
+ }
6
+ function toPostableText(message) {
7
+ if (typeof message === "string") {
8
+ return message;
9
+ }
10
+ if ("raw" in message) {
11
+ return message.raw;
12
+ }
13
+ if ("markdown" in message) {
14
+ return message.markdown;
15
+ }
16
+ return message.fallbackText ?? "[card]";
17
+ }
18
+ export class LoopbackChatAdapter {
19
+ name = "loopback";
20
+ persistMessageHistory = true;
21
+ userName;
22
+ #messages = new Map();
23
+ constructor(userName) {
24
+ this.userName = userName;
25
+ }
26
+ addReaction() {
27
+ return Promise.resolve();
28
+ }
29
+ channelIdFromThreadId(threadId) {
30
+ const [channelId = threadId] = threadId.split("::");
31
+ return channelId;
32
+ }
33
+ decodeThreadId(threadId) {
34
+ const [address = threadId, rawThreadId] = threadId.split("::");
35
+ const [, channelId = address, id = address] = address.split(":");
36
+ const decoded = { id };
37
+ if (channelId) {
38
+ decoded.channelId = channelId;
39
+ }
40
+ if (rawThreadId) {
41
+ decoded.threadId = rawThreadId;
42
+ }
43
+ return decoded;
44
+ }
45
+ deleteMessage(threadId, messageId) {
46
+ const messages = this.#messages.get(threadId) ?? [];
47
+ this.#messages.set(threadId, messages.filter((entry) => entry.id !== messageId));
48
+ return Promise.resolve();
49
+ }
50
+ editMessage(threadId, messageId, message) {
51
+ const messages = this.#messages.get(threadId) ?? [];
52
+ const existing = messages.find((entry) => entry.id === messageId);
53
+ if (!existing) {
54
+ throw new CrablineError(`Loopback message not found: ${messageId}`, { kind: "inbound" });
55
+ }
56
+ const text = toPostableText(message);
57
+ existing.text = text;
58
+ existing.formatted = text;
59
+ existing.metadata.edited = true;
60
+ existing.metadata.editedAt = new Date();
61
+ existing.raw.text = text;
62
+ return Promise.resolve({ id: existing.id, raw: existing.raw, threadId });
63
+ }
64
+ encodeThreadId(platformData) {
65
+ const channelId = platformData.channelId ?? `loopback:${platformData.id}`;
66
+ return platformData.threadId ? `${channelId}::${platformData.threadId}` : channelId;
67
+ }
68
+ fetchMessages(threadId, options) {
69
+ const messages = [...(this.#messages.get(threadId) ?? [])];
70
+ const limit = options?.limit ?? messages.length;
71
+ if (!options?.cursor) {
72
+ return Promise.resolve({
73
+ messages: messages.slice(-limit),
74
+ });
75
+ }
76
+ const offset = Number(options.cursor);
77
+ const result = {
78
+ messages: messages.slice(Math.max(0, offset - limit), offset),
79
+ };
80
+ if (offset - limit > 0) {
81
+ result.nextCursor = String(offset - limit);
82
+ }
83
+ return Promise.resolve(result);
84
+ }
85
+ fetchThread(threadId) {
86
+ return Promise.resolve({
87
+ channelId: this.channelIdFromThreadId(threadId),
88
+ id: threadId,
89
+ isDM: true,
90
+ metadata: {},
91
+ });
92
+ }
93
+ handleWebhook(_request) {
94
+ return Promise.resolve(new Response("loopback adapter has no webhook surface", { status: 501 }));
95
+ }
96
+ isDM() {
97
+ return true;
98
+ }
99
+ parseMessage(raw) {
100
+ return {
101
+ author: {
102
+ isMe: raw.author === "assistant",
103
+ userName: raw.author === "assistant" ? this.userName : "loopback",
104
+ },
105
+ formatted: raw.text,
106
+ id: raw.id,
107
+ metadata: {
108
+ dateSent: new Date(raw.timestamp),
109
+ edited: false,
110
+ },
111
+ raw,
112
+ text: raw.text,
113
+ threadId: raw.threadId,
114
+ };
115
+ }
116
+ postMessage(threadId, message) {
117
+ const text = toPostableText(message);
118
+ const raw = {
119
+ author: "assistant",
120
+ id: createMessageId(),
121
+ text,
122
+ threadId,
123
+ timestamp: new Date().toISOString(),
124
+ };
125
+ const parsed = this.parseMessage(raw);
126
+ this.#append(threadId, parsed);
127
+ return Promise.resolve({ id: raw.id, raw, threadId });
128
+ }
129
+ removeReaction() {
130
+ return Promise.resolve();
131
+ }
132
+ renderFormatted(content) {
133
+ return content;
134
+ }
135
+ startTyping() {
136
+ return Promise.resolve();
137
+ }
138
+ ingestUserMessage(threadId, text) {
139
+ const raw = {
140
+ author: "user",
141
+ id: createMessageId(),
142
+ text,
143
+ threadId,
144
+ timestamp: new Date().toISOString(),
145
+ };
146
+ const parsed = this.parseMessage(raw);
147
+ this.#append(threadId, parsed);
148
+ return parsed;
149
+ }
150
+ listSince(threadId, since) {
151
+ const sinceTime = new Date(since).getTime();
152
+ return (this.#messages.get(threadId) ?? []).filter((entry) => entry.metadata.dateSent.getTime() >= sinceTime);
153
+ }
154
+ #append(threadId, message) {
155
+ const bucket = this.#messages.get(threadId) ?? [];
156
+ bucket.push(message);
157
+ this.#messages.set(threadId, bucket);
158
+ }
159
+ }
160
+ export class LoopbackProviderAdapter extends LocalMockProviderAdapter {
161
+ constructor(id, config, _userName) {
162
+ super({
163
+ codec: createGenericLocalMockTargetCodec("loopback"),
164
+ config,
165
+ id,
166
+ options: {
167
+ defaultWebhook: { host: "127.0.0.1", path: "/loopback/webhook", port: 8786 },
168
+ endpointLabel: "webhook endpoint",
169
+ platform: "loopback",
170
+ },
171
+ });
172
+ }
173
+ }
174
+ //# sourceMappingURL=loopback.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"loopback.js","sourceRoot":"","sources":["../../../../src/providers/builtin/loopback.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ import type { ProviderConfig } from "../../config/schema.js";
2
+ import { LocalMockProviderAdapter } from "../local-mock.js";
3
+ import type { ProviderAdapter } from "../types.js";
4
+ export declare function resolveMatrixAdapterConfig(config: ProviderConfig, userName: string, env?: NodeJS.ProcessEnv): {
5
+ auth: {
6
+ accessToken: string;
7
+ type: "accessToken";
8
+ userID?: string | undefined;
9
+ } | {
10
+ password: string;
11
+ type: "password";
12
+ userID?: string | undefined;
13
+ username: string;
14
+ };
15
+ baseURL: string;
16
+ commandPrefix: string | undefined;
17
+ recoveryKey: string | undefined;
18
+ };
19
+ export declare class MatrixProviderAdapter extends LocalMockProviderAdapter implements ProviderAdapter {
20
+ constructor(id: string, config: ProviderConfig, _userName: string, _runtime?: unknown);
21
+ }
@@ -0,0 +1,84 @@
1
+ import path from "node:path";
2
+ import { CrablineError } from "../../core/errors.js";
3
+ import { LocalMockProviderAdapter } from "../local-mock.js";
4
+ import { authorFromBotFlag, createNativeTargetCodec, genericMockPayloadWithNativeThread, isRecord, optionalRecord, optionalString, requireNativeInboundId, } from "./native-local-mock.js";
5
+ const MATRIX_ROOM_ID_RULE = {
6
+ example: "!abcdef:matrix.org",
7
+ name: "Matrix room id",
8
+ pattern: /^![^:\s]+:[^\s]+$/u,
9
+ };
10
+ const MATRIX_EVENT_ID_RULE = {
11
+ example: "$eventid:matrix.org",
12
+ name: "Matrix event id",
13
+ pattern: /^\$[^\s]+(?::[^\s]+)?$/u,
14
+ };
15
+ export function resolveMatrixAdapterConfig(config, userName, env = process.env) {
16
+ return {
17
+ auth: config.matrix?.auth ?? {
18
+ accessToken: env.MATRIX_ACCESS_TOKEN ?? "local-mock-matrix-token",
19
+ type: "accessToken",
20
+ userID: `@${userName}:matrix.local`,
21
+ },
22
+ baseURL: config.matrix?.baseURL ?? env.MATRIX_BASE_URL ?? "http://matrix.local",
23
+ commandPrefix: config.matrix?.commandPrefix,
24
+ recoveryKey: config.matrix?.recoveryKey ?? env.MATRIX_RECOVERY_KEY,
25
+ };
26
+ }
27
+ export class MatrixProviderAdapter extends LocalMockProviderAdapter {
28
+ constructor(id, config, _userName, _runtime) {
29
+ super({
30
+ codec: createNativeTargetCodec({
31
+ channel: MATRIX_ROOM_ID_RULE,
32
+ channelLabel: "Matrix room_id",
33
+ thread: MATRIX_EVENT_ID_RULE,
34
+ threadLabel: "Matrix event_id",
35
+ }),
36
+ config,
37
+ id,
38
+ options: {
39
+ defaultWebhook: { host: "127.0.0.1", path: "/matrix/webhook", port: 8797 },
40
+ endpointLabel: "webhook endpoint",
41
+ normalizeWebhookPayload: normalizeMatrixWebhookPayload,
42
+ platform: "matrix",
43
+ publicUrl: config.matrix?.webhook.publicUrl,
44
+ recorderPath: config.matrix?.recorder.path
45
+ ? path.resolve(config.matrix.recorder.path)
46
+ : undefined,
47
+ webhook: config.matrix?.webhook,
48
+ },
49
+ });
50
+ }
51
+ }
52
+ function normalizeMatrixWebhookPayload(payload) {
53
+ if (!isRecord(payload)) {
54
+ throw new CrablineError("Matrix webhook payload must be an object", { kind: "inbound" });
55
+ }
56
+ if (optionalRecord(payload, "message")) {
57
+ return genericMockPayloadWithNativeThread({
58
+ channelRule: MATRIX_ROOM_ID_RULE,
59
+ payload,
60
+ threadRule: MATRIX_EVENT_ID_RULE,
61
+ });
62
+ }
63
+ const content = optionalRecord(payload, "content");
64
+ const roomId = optionalString(payload, "room_id");
65
+ const eventId = optionalString(payload, "event_id");
66
+ const text = content ? optionalString(content, "body") : undefined;
67
+ if (!roomId || !text) {
68
+ throw new CrablineError("Matrix event payload requires room_id and content.body", {
69
+ kind: "inbound",
70
+ });
71
+ }
72
+ return {
73
+ author: authorFromBotFlag(false),
74
+ ...(eventId
75
+ ? { id: requireNativeInboundId(eventId, MATRIX_EVENT_ID_RULE, "Matrix event_id") }
76
+ : {}),
77
+ raw: payload,
78
+ text,
79
+ threadId: eventId
80
+ ? requireNativeInboundId(eventId, MATRIX_EVENT_ID_RULE, "Matrix event_id")
81
+ : requireNativeInboundId(roomId, MATRIX_ROOM_ID_RULE, "Matrix room_id"),
82
+ };
83
+ }
84
+ //# sourceMappingURL=matrix.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"matrix.js","sourceRoot":"","sources":["../../../../src/providers/builtin/matrix.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { ProviderConfig } from "../../config/schema.js";
2
+ import { LocalMockProviderAdapter } from "../local-mock.js";
3
+ import type { ProviderAdapter } from "../types.js";
4
+ export declare function resolveMattermostAdapterConfig(config: ProviderConfig, env?: NodeJS.ProcessEnv): {
5
+ baseUrl: string;
6
+ botToken: string;
7
+ userName: string | undefined;
8
+ };
9
+ export declare class MattermostProviderAdapter extends LocalMockProviderAdapter implements ProviderAdapter {
10
+ constructor(id: string, config: ProviderConfig, _userName: string, _runtime?: unknown);
11
+ }
@@ -0,0 +1,67 @@
1
+ import path from "node:path";
2
+ import { CrablineError } from "../../core/errors.js";
3
+ import { LocalMockProviderAdapter } from "../local-mock.js";
4
+ import { authorFromBotFlag, createNativeTargetCodec, genericMockPayloadWithNativeThread, isRecord, optionalRecord, optionalString, requireNativeInboundId, } from "./native-local-mock.js";
5
+ const MATTERMOST_ID_RULE = {
6
+ example: "abcdefghijklmnopqrstuvwx12",
7
+ name: "Mattermost id",
8
+ pattern: /^[a-z0-9]{26}$/u,
9
+ };
10
+ export function resolveMattermostAdapterConfig(config, env = process.env) {
11
+ return {
12
+ baseUrl: config.mattermost?.baseUrl ?? env.MATTERMOST_BASE_URL ?? "http://mattermost.local",
13
+ botToken: config.mattermost?.botToken ?? env.MATTERMOST_BOT_TOKEN ?? "local-mock-token",
14
+ userName: config.mattermost?.userName,
15
+ };
16
+ }
17
+ export class MattermostProviderAdapter extends LocalMockProviderAdapter {
18
+ constructor(id, config, _userName, _runtime) {
19
+ super({
20
+ codec: createNativeTargetCodec({
21
+ channel: MATTERMOST_ID_RULE,
22
+ channelLabel: "Mattermost channel_id",
23
+ }),
24
+ config,
25
+ id,
26
+ options: {
27
+ defaultWebhook: { host: "127.0.0.1", path: "/mattermost/webhook", port: 8793 },
28
+ endpointLabel: "webhook endpoint",
29
+ normalizeWebhookPayload: normalizeMattermostWebhookPayload,
30
+ platform: "mattermost",
31
+ publicUrl: config.mattermost?.webhook.publicUrl,
32
+ recorderPath: config.mattermost?.recorder.path
33
+ ? path.resolve(config.mattermost.recorder.path)
34
+ : undefined,
35
+ webhook: config.mattermost?.webhook,
36
+ },
37
+ });
38
+ }
39
+ }
40
+ function normalizeMattermostWebhookPayload(payload) {
41
+ if (!isRecord(payload)) {
42
+ throw new CrablineError("Mattermost webhook payload must be an object", { kind: "inbound" });
43
+ }
44
+ if (optionalRecord(payload, "message")) {
45
+ return genericMockPayloadWithNativeThread({
46
+ channelRule: MATTERMOST_ID_RULE,
47
+ payload,
48
+ threadRule: MATTERMOST_ID_RULE,
49
+ });
50
+ }
51
+ const channelId = optionalString(payload, "channel_id");
52
+ const threadId = optionalString(payload, "root_id") ?? channelId;
53
+ const text = optionalString(payload, "text");
54
+ if (!channelId || !threadId || !text) {
55
+ throw new CrablineError("Mattermost webhook payload requires channel_id and text", {
56
+ kind: "inbound",
57
+ });
58
+ }
59
+ return {
60
+ author: authorFromBotFlag(false),
61
+ ...(optionalString(payload, "post_id") ? { id: optionalString(payload, "post_id") } : {}),
62
+ raw: payload,
63
+ text,
64
+ threadId: requireNativeInboundId(threadId, MATTERMOST_ID_RULE, "Mattermost root_id"),
65
+ };
66
+ }
67
+ //# sourceMappingURL=mattermost.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mattermost.js","sourceRoot":"","sources":["../../../../src/providers/builtin/mattermost.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ import type { ProviderConfig } from "../../config/schema.js";
2
+ import { LocalMockProviderAdapter } from "../local-mock.js";
3
+ import type { ProviderAdapter } from "../types.js";
4
+ export declare function resolveMsTeamsAdapterConfig(config: ProviderConfig, env?: NodeJS.ProcessEnv): {
5
+ appId: string;
6
+ appPassword: string;
7
+ appTenantId: string | undefined;
8
+ appType: "MultiTenant" | "SingleTenant" | undefined;
9
+ userName: string | undefined;
10
+ };
11
+ export declare class MsTeamsProviderAdapter extends LocalMockProviderAdapter implements ProviderAdapter {
12
+ constructor(id: string, config: ProviderConfig, _userName: string, _runtime?: unknown);
13
+ }
@@ -0,0 +1,72 @@
1
+ import path from "node:path";
2
+ import { CrablineError } from "../../core/errors.js";
3
+ import { LocalMockProviderAdapter } from "../local-mock.js";
4
+ import { authorFromBotFlag, createNativeTargetCodec, genericMockPayloadWithNativeThread, isRecord, optionalRecord, optionalString, requireNativeInboundId, } from "./native-local-mock.js";
5
+ const MSTEAMS_CONVERSATION_ID_RULE = {
6
+ example: "19:meeting_abc@thread.v2",
7
+ name: "Microsoft Teams conversation id",
8
+ pattern: /^19:[^@]+@thread\.[A-Za-z0-9.]+$/u,
9
+ };
10
+ export function resolveMsTeamsAdapterConfig(config, env = process.env) {
11
+ return {
12
+ appId: config.msteams?.appId ?? env.TEAMS_APP_ID ?? "local-mock-teams-app",
13
+ appPassword: config.msteams?.appPassword ?? env.TEAMS_APP_PASSWORD ?? "local-mock-secret",
14
+ appTenantId: config.msteams?.appTenantId,
15
+ appType: config.msteams?.appType,
16
+ userName: config.msteams?.userName,
17
+ };
18
+ }
19
+ export class MsTeamsProviderAdapter extends LocalMockProviderAdapter {
20
+ constructor(id, config, _userName, _runtime) {
21
+ super({
22
+ codec: createNativeTargetCodec({
23
+ channel: MSTEAMS_CONVERSATION_ID_RULE,
24
+ channelLabel: "Microsoft Teams conversation.id",
25
+ }),
26
+ config,
27
+ id,
28
+ options: {
29
+ defaultWebhook: { host: "127.0.0.1", path: "/msteams/webhook", port: 8791 },
30
+ endpointLabel: "webhook endpoint",
31
+ normalizeWebhookPayload: normalizeMsTeamsWebhookPayload,
32
+ platform: "msteams",
33
+ publicUrl: config.msteams?.webhook.publicUrl,
34
+ recorderPath: config.msteams?.recorder.path
35
+ ? path.resolve(config.msteams.recorder.path)
36
+ : undefined,
37
+ webhook: config.msteams?.webhook,
38
+ },
39
+ });
40
+ }
41
+ }
42
+ function normalizeMsTeamsWebhookPayload(payload) {
43
+ if (!isRecord(payload)) {
44
+ throw new CrablineError("Microsoft Teams webhook payload must be an object", {
45
+ kind: "inbound",
46
+ });
47
+ }
48
+ if (optionalRecord(payload, "message")) {
49
+ return genericMockPayloadWithNativeThread({
50
+ channelRule: MSTEAMS_CONVERSATION_ID_RULE,
51
+ payload,
52
+ threadRule: MSTEAMS_CONVERSATION_ID_RULE,
53
+ });
54
+ }
55
+ const conversation = optionalRecord(payload, "conversation");
56
+ const from = optionalRecord(payload, "from");
57
+ const conversationId = conversation ? optionalString(conversation, "id") : undefined;
58
+ const text = optionalString(payload, "text");
59
+ if (!conversationId || !text) {
60
+ throw new CrablineError("Microsoft Teams activity payload requires conversation.id and text", {
61
+ kind: "inbound",
62
+ });
63
+ }
64
+ return {
65
+ author: authorFromBotFlag(optionalString(from ?? {}, "role") === "bot"),
66
+ ...(optionalString(payload, "id") ? { id: optionalString(payload, "id") } : {}),
67
+ raw: payload,
68
+ text,
69
+ threadId: requireNativeInboundId(conversationId, MSTEAMS_CONVERSATION_ID_RULE, "Microsoft Teams conversation.id"),
70
+ };
71
+ }
72
+ //# sourceMappingURL=msteams.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"msteams.js","sourceRoot":"","sources":["../../../../src/providers/builtin/msteams.ts"],"names":[],"mappings":"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"}