@auxiora/channels 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (89) hide show
  1. package/LICENSE +191 -0
  2. package/dist/adapters/bluebubbles.d.ts +63 -0
  3. package/dist/adapters/bluebubbles.d.ts.map +1 -0
  4. package/dist/adapters/bluebubbles.js +197 -0
  5. package/dist/adapters/bluebubbles.js.map +1 -0
  6. package/dist/adapters/discord.d.ts +27 -0
  7. package/dist/adapters/discord.d.ts.map +1 -0
  8. package/dist/adapters/discord.js +202 -0
  9. package/dist/adapters/discord.js.map +1 -0
  10. package/dist/adapters/email.d.ts +39 -0
  11. package/dist/adapters/email.d.ts.map +1 -0
  12. package/dist/adapters/email.js +359 -0
  13. package/dist/adapters/email.js.map +1 -0
  14. package/dist/adapters/googlechat.d.ts +77 -0
  15. package/dist/adapters/googlechat.d.ts.map +1 -0
  16. package/dist/adapters/googlechat.js +232 -0
  17. package/dist/adapters/googlechat.js.map +1 -0
  18. package/dist/adapters/matrix.d.ts +37 -0
  19. package/dist/adapters/matrix.d.ts.map +1 -0
  20. package/dist/adapters/matrix.js +262 -0
  21. package/dist/adapters/matrix.js.map +1 -0
  22. package/dist/adapters/signal.d.ts +32 -0
  23. package/dist/adapters/signal.d.ts.map +1 -0
  24. package/dist/adapters/signal.js +216 -0
  25. package/dist/adapters/signal.js.map +1 -0
  26. package/dist/adapters/slack.d.ts +29 -0
  27. package/dist/adapters/slack.d.ts.map +1 -0
  28. package/dist/adapters/slack.js +202 -0
  29. package/dist/adapters/slack.js.map +1 -0
  30. package/dist/adapters/teams.d.ts +66 -0
  31. package/dist/adapters/teams.d.ts.map +1 -0
  32. package/dist/adapters/teams.js +227 -0
  33. package/dist/adapters/teams.js.map +1 -0
  34. package/dist/adapters/telegram.d.ts +28 -0
  35. package/dist/adapters/telegram.d.ts.map +1 -0
  36. package/dist/adapters/telegram.js +170 -0
  37. package/dist/adapters/telegram.js.map +1 -0
  38. package/dist/adapters/twilio.d.ts +63 -0
  39. package/dist/adapters/twilio.d.ts.map +1 -0
  40. package/dist/adapters/twilio.js +193 -0
  41. package/dist/adapters/twilio.js.map +1 -0
  42. package/dist/adapters/whatsapp.d.ts +99 -0
  43. package/dist/adapters/whatsapp.d.ts.map +1 -0
  44. package/dist/adapters/whatsapp.js +218 -0
  45. package/dist/adapters/whatsapp.js.map +1 -0
  46. package/dist/adapters/zalo.d.ts +64 -0
  47. package/dist/adapters/zalo.d.ts.map +1 -0
  48. package/dist/adapters/zalo.js +216 -0
  49. package/dist/adapters/zalo.js.map +1 -0
  50. package/dist/index.d.ts +15 -0
  51. package/dist/index.d.ts.map +1 -0
  52. package/dist/index.js +16 -0
  53. package/dist/index.js.map +1 -0
  54. package/dist/manager.d.ts +35 -0
  55. package/dist/manager.d.ts.map +1 -0
  56. package/dist/manager.js +127 -0
  57. package/dist/manager.js.map +1 -0
  58. package/dist/types.d.ts +71 -0
  59. package/dist/types.d.ts.map +1 -0
  60. package/dist/types.js +2 -0
  61. package/dist/types.js.map +1 -0
  62. package/package.json +32 -0
  63. package/src/adapters/bluebubbles.ts +294 -0
  64. package/src/adapters/discord.ts +253 -0
  65. package/src/adapters/email.ts +457 -0
  66. package/src/adapters/googlechat.ts +364 -0
  67. package/src/adapters/matrix.ts +376 -0
  68. package/src/adapters/signal.ts +313 -0
  69. package/src/adapters/slack.ts +252 -0
  70. package/src/adapters/teams.ts +320 -0
  71. package/src/adapters/telegram.ts +208 -0
  72. package/src/adapters/twilio.ts +256 -0
  73. package/src/adapters/whatsapp.ts +342 -0
  74. package/src/adapters/zalo.ts +319 -0
  75. package/src/index.ts +78 -0
  76. package/src/manager.ts +180 -0
  77. package/src/types.ts +84 -0
  78. package/tests/bluebubbles.test.ts +438 -0
  79. package/tests/email.test.ts +136 -0
  80. package/tests/googlechat.test.ts +439 -0
  81. package/tests/matrix.test.ts +564 -0
  82. package/tests/signal.test.ts +404 -0
  83. package/tests/slack.test.ts +343 -0
  84. package/tests/teams.test.ts +429 -0
  85. package/tests/twilio.test.ts +269 -0
  86. package/tests/whatsapp.test.ts +530 -0
  87. package/tests/zalo.test.ts +499 -0
  88. package/tsconfig.json +8 -0
  89. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,77 @@
1
+ import type { ChannelAdapter, InboundMessage, OutboundMessage, SendResult } from '../types.js';
2
+ export interface GoogleChatAdapterConfig {
3
+ serviceAccountKey: string;
4
+ allowedSpaces?: string[];
5
+ }
6
+ interface GoogleChatEvent {
7
+ type: string;
8
+ eventTime: string;
9
+ message?: {
10
+ name: string;
11
+ sender: {
12
+ name: string;
13
+ displayName: string;
14
+ type: string;
15
+ };
16
+ createTime: string;
17
+ text: string;
18
+ thread?: {
19
+ name: string;
20
+ };
21
+ space: {
22
+ name: string;
23
+ type: string;
24
+ displayName?: string;
25
+ };
26
+ argumentText?: string;
27
+ attachment?: Array<{
28
+ name: string;
29
+ contentName: string;
30
+ contentType: string;
31
+ downloadUri?: string;
32
+ thumbnailUri?: string;
33
+ source: string;
34
+ }>;
35
+ };
36
+ space?: {
37
+ name: string;
38
+ type: string;
39
+ displayName?: string;
40
+ };
41
+ user?: {
42
+ name: string;
43
+ displayName: string;
44
+ type: string;
45
+ };
46
+ configCompleteRedirectUrl?: string;
47
+ }
48
+ export declare class GoogleChatAdapter implements ChannelAdapter {
49
+ readonly type: "googlechat";
50
+ readonly name = "Google Chat";
51
+ private config;
52
+ private messageHandler?;
53
+ private errorHandler?;
54
+ private connected;
55
+ private accessToken?;
56
+ private tokenExpiry;
57
+ private serviceAccount;
58
+ constructor(config: GoogleChatAdapterConfig);
59
+ connect(): Promise<void>;
60
+ disconnect(): Promise<void>;
61
+ isConnected(): boolean;
62
+ private getAccessToken;
63
+ private importPrivateKey;
64
+ /**
65
+ * Handle incoming event from Google Chat.
66
+ * Call this from your HTTP server when receiving POST to the bot endpoint.
67
+ */
68
+ handleWebhook(event: GoogleChatEvent): Promise<void>;
69
+ private toInboundMessage;
70
+ send(channelId: string, message: OutboundMessage): Promise<SendResult>;
71
+ private chunkMessage;
72
+ startTyping(_channelId: string): Promise<() => void>;
73
+ onMessage(handler: (message: InboundMessage) => Promise<void>): void;
74
+ onError(handler: (error: Error) => void): void;
75
+ }
76
+ export {};
77
+ //# sourceMappingURL=googlechat.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"googlechat.d.ts","sourceRoot":"","sources":["../../src/adapters/googlechat.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACV,cAAc,EACd,cAAc,EACd,eAAe,EACf,UAAU,EACX,MAAM,aAAa,CAAC;AAErB,MAAM,WAAW,uBAAuB;IACtC,iBAAiB,EAAE,MAAM,CAAC;IAC1B,aAAa,CAAC,EAAE,MAAM,EAAE,CAAC;CAC1B;AAED,UAAU,eAAe;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE;QACR,IAAI,EAAE,MAAM,CAAC;QACb,MAAM,EAAE;YACN,IAAI,EAAE,MAAM,CAAC;YACb,WAAW,EAAE,MAAM,CAAC;YACpB,IAAI,EAAE,MAAM,CAAC;SACd,CAAC;QACF,UAAU,EAAE,MAAM,CAAC;QACnB,IAAI,EAAE,MAAM,CAAC;QACb,MAAM,CAAC,EAAE;YACP,IAAI,EAAE,MAAM,CAAC;SACd,CAAC;QACF,KAAK,EAAE;YACL,IAAI,EAAE,MAAM,CAAC;YACb,IAAI,EAAE,MAAM,CAAC;YACb,WAAW,CAAC,EAAE,MAAM,CAAC;SACtB,CAAC;QACF,YAAY,CAAC,EAAE,MAAM,CAAC;QACtB,UAAU,CAAC,EAAE,KAAK,CAAC;YACjB,IAAI,EAAE,MAAM,CAAC;YACb,WAAW,EAAE,MAAM,CAAC;YACpB,WAAW,EAAE,MAAM,CAAC;YACpB,WAAW,CAAC,EAAE,MAAM,CAAC;YACrB,YAAY,CAAC,EAAE,MAAM,CAAC;YACtB,MAAM,EAAE,MAAM,CAAC;SAChB,CAAC,CAAC;KACJ,CAAC;IACF,KAAK,CAAC,EAAE;QACN,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,MAAM,CAAC;QACb,WAAW,CAAC,EAAE,MAAM,CAAC;KACtB,CAAC;IACF,IAAI,CAAC,EAAE;QACL,IAAI,EAAE,MAAM,CAAC;QACb,WAAW,EAAE,MAAM,CAAC;QACpB,IAAI,EAAE,MAAM,CAAC;KACd,CAAC;IACF,yBAAyB,CAAC,EAAE,MAAM,CAAC;CACpC;AA8BD,qBAAa,iBAAkB,YAAW,cAAc;IACtD,QAAQ,CAAC,IAAI,EAAG,YAAY,CAAU;IACtC,QAAQ,CAAC,IAAI,iBAAiB;IAE9B,OAAO,CAAC,MAAM,CAA0B;IACxC,OAAO,CAAC,cAAc,CAAC,CAA6C;IACpE,OAAO,CAAC,YAAY,CAAC,CAAyB;IAC9C,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,WAAW,CAAC,CAAS;IAC7B,OAAO,CAAC,WAAW,CAAK;IACxB,OAAO,CAAC,cAAc,CAAoB;gBAE9B,MAAM,EAAE,uBAAuB;IASrC,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAWxB,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAOjC,WAAW,IAAI,OAAO;YAIR,cAAc;YAyDd,gBAAgB;IAgB9B;;;OAGG;IACG,aAAa,CAAC,KAAK,EAAE,eAAe,GAAG,OAAO,CAAC,IAAI,CAAC;IAoC1D,OAAO,CAAC,gBAAgB;IA8BlB,IAAI,CAAC,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,UAAU,CAAC;IAoD5E,OAAO,CAAC,YAAY;IA6Bd,WAAW,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,CAAC;IAK1D,SAAS,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,cAAc,KAAK,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI;IAIpE,OAAO,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,GAAG,IAAI;CAG/C"}
@@ -0,0 +1,232 @@
1
+ import { audit } from '@auxiora/audit';
2
+ const CHAT_API_BASE = 'https://chat.googleapis.com/v1';
3
+ const MAX_MESSAGE_LENGTH = 4096;
4
+ export class GoogleChatAdapter {
5
+ type = 'googlechat';
6
+ name = 'Google Chat';
7
+ config;
8
+ messageHandler;
9
+ errorHandler;
10
+ connected = false;
11
+ accessToken;
12
+ tokenExpiry = 0;
13
+ serviceAccount;
14
+ constructor(config) {
15
+ this.config = config;
16
+ try {
17
+ this.serviceAccount = JSON.parse(config.serviceAccountKey);
18
+ }
19
+ catch {
20
+ throw new Error('Invalid Google Chat service account key: must be valid JSON');
21
+ }
22
+ }
23
+ async connect() {
24
+ // Verify credentials by obtaining a token
25
+ await this.getAccessToken();
26
+ this.connected = true;
27
+ audit('channel.connected', {
28
+ channelType: 'googlechat',
29
+ serviceAccount: this.serviceAccount.client_email,
30
+ });
31
+ }
32
+ async disconnect() {
33
+ this.connected = false;
34
+ this.accessToken = undefined;
35
+ this.tokenExpiry = 0;
36
+ audit('channel.disconnected', { channelType: 'googlechat' });
37
+ }
38
+ isConnected() {
39
+ return this.connected;
40
+ }
41
+ async getAccessToken() {
42
+ if (this.accessToken && Date.now() < this.tokenExpiry) {
43
+ return this.accessToken;
44
+ }
45
+ const now = Math.floor(Date.now() / 1000);
46
+ const header = btoa(JSON.stringify({ alg: 'RS256', typ: 'JWT' }));
47
+ const payload = btoa(JSON.stringify({
48
+ iss: this.serviceAccount.client_email,
49
+ scope: 'https://www.googleapis.com/auth/chat.bot',
50
+ aud: this.serviceAccount.token_uri,
51
+ iat: now,
52
+ exp: now + 3600,
53
+ }));
54
+ // Sign the JWT using the Web Crypto API
55
+ const signingInput = `${header}.${payload}`;
56
+ const key = await this.importPrivateKey(this.serviceAccount.private_key);
57
+ const signature = await crypto.subtle.sign('RSASSA-PKCS1-v1_5', key, new TextEncoder().encode(signingInput));
58
+ const sig = btoa(String.fromCharCode(...new Uint8Array(signature)))
59
+ .replace(/\+/g, '-')
60
+ .replace(/\//g, '_')
61
+ .replace(/=+$/, '');
62
+ const jwt = `${header}.${payload}.${sig}`;
63
+ const params = new URLSearchParams({
64
+ grant_type: 'urn:ietf:params:oauth:grant-type:jwt-bearer',
65
+ assertion: jwt,
66
+ });
67
+ const response = await fetch(this.serviceAccount.token_uri, {
68
+ method: 'POST',
69
+ headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
70
+ body: params.toString(),
71
+ });
72
+ if (!response.ok) {
73
+ throw new Error(`Failed to obtain Google Chat token: ${response.status} ${response.statusText}`);
74
+ }
75
+ const data = (await response.json());
76
+ this.accessToken = data.access_token;
77
+ // Expire 5 minutes early
78
+ this.tokenExpiry = Date.now() + (data.expires_in - 300) * 1000;
79
+ return this.accessToken;
80
+ }
81
+ async importPrivateKey(pem) {
82
+ const pemBody = pem
83
+ .replace(/-----BEGIN PRIVATE KEY-----/, '')
84
+ .replace(/-----END PRIVATE KEY-----/, '')
85
+ .replace(/\s/g, '');
86
+ const binaryDer = Uint8Array.from(atob(pemBody), (c) => c.charCodeAt(0));
87
+ return crypto.subtle.importKey('pkcs8', binaryDer.buffer, { name: 'RSASSA-PKCS1-v1_5', hash: 'SHA-256' }, false, ['sign']);
88
+ }
89
+ /**
90
+ * Handle incoming event from Google Chat.
91
+ * Call this from your HTTP server when receiving POST to the bot endpoint.
92
+ */
93
+ async handleWebhook(event) {
94
+ if (event.type !== 'MESSAGE')
95
+ return;
96
+ if (!event.message)
97
+ return;
98
+ // Check allowed spaces
99
+ const spaceName = event.message.space.name;
100
+ if (this.config.allowedSpaces?.length &&
101
+ !this.config.allowedSpaces.includes(spaceName)) {
102
+ audit('message.filtered', {
103
+ channelType: 'googlechat',
104
+ senderId: event.message.sender.name,
105
+ channelId: spaceName,
106
+ reason: 'space_not_allowed',
107
+ });
108
+ return;
109
+ }
110
+ const inbound = this.toInboundMessage(event);
111
+ audit('message.received', {
112
+ channelType: 'googlechat',
113
+ senderId: inbound.senderId,
114
+ channelId: inbound.channelId,
115
+ });
116
+ if (this.messageHandler) {
117
+ try {
118
+ await this.messageHandler(inbound);
119
+ }
120
+ catch (error) {
121
+ this.errorHandler?.(error instanceof Error ? error : new Error(String(error)));
122
+ }
123
+ }
124
+ }
125
+ toInboundMessage(event) {
126
+ const message = event.message;
127
+ // Use argumentText if available (strips @mention), fall back to text
128
+ const content = message.argumentText?.trim() || message.text || '';
129
+ return {
130
+ id: message.name,
131
+ channelType: 'googlechat',
132
+ channelId: message.space.name,
133
+ senderId: message.sender.name,
134
+ senderName: message.sender.displayName,
135
+ content,
136
+ timestamp: new Date(message.createTime).getTime(),
137
+ replyToId: message.thread?.name,
138
+ attachments: message.attachment?.map((a) => ({
139
+ type: a.contentType.startsWith('image/')
140
+ ? 'image'
141
+ : a.contentType.startsWith('audio/')
142
+ ? 'audio'
143
+ : a.contentType.startsWith('video/')
144
+ ? 'video'
145
+ : 'file',
146
+ url: a.downloadUri,
147
+ mimeType: a.contentType,
148
+ filename: a.contentName,
149
+ })),
150
+ raw: event,
151
+ };
152
+ }
153
+ async send(channelId, message) {
154
+ try {
155
+ const token = await this.getAccessToken();
156
+ const chunks = this.chunkMessage(message.content);
157
+ let lastMessageName;
158
+ for (const chunk of chunks) {
159
+ const body = {
160
+ text: chunk,
161
+ };
162
+ if (message.replyToId) {
163
+ body.thread = { name: message.replyToId };
164
+ }
165
+ const url = `${CHAT_API_BASE}/${channelId}/messages`;
166
+ const response = await fetch(url, {
167
+ method: 'POST',
168
+ headers: {
169
+ Authorization: `Bearer ${token}`,
170
+ 'Content-Type': 'application/json',
171
+ },
172
+ body: JSON.stringify(body),
173
+ });
174
+ if (!response.ok) {
175
+ const errorText = await response.text().catch(() => response.statusText);
176
+ throw new Error(`Google Chat API error ${response.status}: ${errorText}`);
177
+ }
178
+ const result = (await response.json());
179
+ lastMessageName = result.name;
180
+ }
181
+ audit('message.sent', {
182
+ channelType: 'googlechat',
183
+ channelId,
184
+ messageId: lastMessageName,
185
+ });
186
+ return { success: true, messageId: lastMessageName };
187
+ }
188
+ catch (error) {
189
+ const errorMessage = error instanceof Error ? error.message : String(error);
190
+ audit('channel.error', {
191
+ channelType: 'googlechat',
192
+ action: 'send',
193
+ error: errorMessage,
194
+ });
195
+ return { success: false, error: errorMessage };
196
+ }
197
+ }
198
+ chunkMessage(content) {
199
+ if (content.length <= MAX_MESSAGE_LENGTH) {
200
+ return [content];
201
+ }
202
+ const chunks = [];
203
+ let remaining = content;
204
+ while (remaining.length > 0) {
205
+ if (remaining.length <= MAX_MESSAGE_LENGTH) {
206
+ chunks.push(remaining);
207
+ break;
208
+ }
209
+ let breakPoint = remaining.lastIndexOf('\n', MAX_MESSAGE_LENGTH);
210
+ if (breakPoint === -1 || breakPoint < MAX_MESSAGE_LENGTH / 2) {
211
+ breakPoint = remaining.lastIndexOf(' ', MAX_MESSAGE_LENGTH);
212
+ }
213
+ if (breakPoint === -1 || breakPoint < MAX_MESSAGE_LENGTH / 2) {
214
+ breakPoint = MAX_MESSAGE_LENGTH;
215
+ }
216
+ chunks.push(remaining.slice(0, breakPoint));
217
+ remaining = remaining.slice(breakPoint).trimStart();
218
+ }
219
+ return chunks;
220
+ }
221
+ async startTyping(_channelId) {
222
+ // Google Chat does not support typing indicators via the API
223
+ return () => { };
224
+ }
225
+ onMessage(handler) {
226
+ this.messageHandler = handler;
227
+ }
228
+ onError(handler) {
229
+ this.errorHandler = handler;
230
+ }
231
+ }
232
+ //# sourceMappingURL=googlechat.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"googlechat.js","sourceRoot":"","sources":["../../src/adapters/googlechat.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,37 @@
1
+ import type { ChannelAdapter, InboundMessage, OutboundMessage, SendResult } from '../types.js';
2
+ export interface MatrixAdapterConfig {
3
+ homeserverUrl: string;
4
+ userId: string;
5
+ accessToken: string;
6
+ autoJoinRooms?: boolean;
7
+ allowedUsers?: string[];
8
+ allowedRooms?: string[];
9
+ }
10
+ export declare class MatrixAdapter implements ChannelAdapter {
11
+ readonly type: "matrix";
12
+ readonly name = "Matrix";
13
+ private config;
14
+ private messageHandler?;
15
+ private errorHandler?;
16
+ private connected;
17
+ private syncToken?;
18
+ private syncAbort?;
19
+ private syncLoopRunning;
20
+ constructor(config: MatrixAdapterConfig);
21
+ private get baseUrl();
22
+ private get headers();
23
+ private matrixFetch;
24
+ connect(): Promise<void>;
25
+ disconnect(): Promise<void>;
26
+ isConnected(): boolean;
27
+ private startSyncLoop;
28
+ private syncLoop;
29
+ private handleEvent;
30
+ private toInboundMessage;
31
+ send(channelId: string, message: OutboundMessage): Promise<SendResult>;
32
+ private chunkMessage;
33
+ startTyping(channelId: string): Promise<() => void>;
34
+ onMessage(handler: (message: InboundMessage) => Promise<void>): void;
35
+ onError(handler: (error: Error) => void): void;
36
+ }
37
+ //# sourceMappingURL=matrix.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"matrix.d.ts","sourceRoot":"","sources":["../../src/adapters/matrix.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACV,cAAc,EACd,cAAc,EACd,eAAe,EACf,UAAU,EACX,MAAM,aAAa,CAAC;AAErB,MAAM,WAAW,mBAAmB;IAClC,aAAa,EAAE,MAAM,CAAC;IACtB,MAAM,EAAE,MAAM,CAAC;IACf,WAAW,EAAE,MAAM,CAAC;IACpB,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;IACxB,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;CACzB;AAmDD,qBAAa,aAAc,YAAW,cAAc;IAClD,QAAQ,CAAC,IAAI,EAAG,QAAQ,CAAU;IAClC,QAAQ,CAAC,IAAI,YAAY;IAEzB,OAAO,CAAC,MAAM,CAAsB;IACpC,OAAO,CAAC,cAAc,CAAC,CAA6C;IACpE,OAAO,CAAC,YAAY,CAAC,CAAyB;IAC9C,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,SAAS,CAAC,CAAS;IAC3B,OAAO,CAAC,SAAS,CAAC,CAAkB;IACpC,OAAO,CAAC,eAAe,CAAS;gBAEpB,MAAM,EAAE,mBAAmB;IAIvC,OAAO,KAAK,OAAO,GAGlB;IAED,OAAO,KAAK,OAAO,GAKlB;YAEa,WAAW;IAkBnB,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAcxB,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAOjC,WAAW,IAAI,OAAO;IAItB,OAAO,CAAC,aAAa;YAKP,QAAQ;YAqDR,WAAW;IAwCzB,OAAO,CAAC,gBAAgB;IA0BlB,IAAI,CAAC,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,eAAe,GAAG,OAAO,CAAC,UAAU,CAAC;IA+C5E,OAAO,CAAC,YAAY;IA6Bd,WAAW,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,CAAC;IA+BzD,SAAS,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,cAAc,KAAK,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI;IAIpE,OAAO,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,GAAG,IAAI;CAG/C"}
@@ -0,0 +1,262 @@
1
+ import { audit } from '@auxiora/audit';
2
+ const MAX_MESSAGE_LENGTH = 65536;
3
+ export class MatrixAdapter {
4
+ type = 'matrix';
5
+ name = 'Matrix';
6
+ config;
7
+ messageHandler;
8
+ errorHandler;
9
+ connected = false;
10
+ syncToken;
11
+ syncAbort;
12
+ syncLoopRunning = false;
13
+ constructor(config) {
14
+ this.config = config;
15
+ }
16
+ get baseUrl() {
17
+ const url = this.config.homeserverUrl.replace(/\/+$/, '');
18
+ return `${url}/_matrix/client/v3`;
19
+ }
20
+ get headers() {
21
+ return {
22
+ 'Authorization': `Bearer ${this.config.accessToken}`,
23
+ 'Content-Type': 'application/json',
24
+ };
25
+ }
26
+ async matrixFetch(path, options = {}) {
27
+ const url = `${this.baseUrl}${path}`;
28
+ const response = await fetch(url, {
29
+ ...options,
30
+ headers: { ...this.headers, ...(options.headers || {}) },
31
+ });
32
+ if (!response.ok) {
33
+ const errorText = await response.text().catch(() => response.statusText);
34
+ throw new Error(`Matrix API error ${response.status}: ${errorText}`);
35
+ }
36
+ return response.json();
37
+ }
38
+ async connect() {
39
+ // Verify credentials by calling whoami
40
+ await this.matrixFetch('/account/whoami');
41
+ this.connected = true;
42
+ audit('channel.connected', {
43
+ channelType: 'matrix',
44
+ userId: this.config.userId,
45
+ });
46
+ // Start sync loop
47
+ this.startSyncLoop();
48
+ }
49
+ async disconnect() {
50
+ this.syncLoopRunning = false;
51
+ this.syncAbort?.abort();
52
+ this.connected = false;
53
+ audit('channel.disconnected', { channelType: 'matrix' });
54
+ }
55
+ isConnected() {
56
+ return this.connected;
57
+ }
58
+ startSyncLoop() {
59
+ this.syncLoopRunning = true;
60
+ void this.syncLoop();
61
+ }
62
+ async syncLoop() {
63
+ while (this.syncLoopRunning && this.connected) {
64
+ try {
65
+ this.syncAbort = new AbortController();
66
+ const params = new URLSearchParams({
67
+ timeout: '30000',
68
+ });
69
+ if (this.syncToken) {
70
+ params.set('since', this.syncToken);
71
+ }
72
+ const response = await this.matrixFetch(`/sync?${params.toString()}`, { signal: this.syncAbort.signal });
73
+ // Handle invites (auto-join)
74
+ if (this.config.autoJoinRooms && response.rooms?.invite) {
75
+ for (const roomId of Object.keys(response.rooms.invite)) {
76
+ try {
77
+ await this.matrixFetch(`/rooms/${encodeURIComponent(roomId)}/join`, {
78
+ method: 'POST',
79
+ body: '{}',
80
+ });
81
+ }
82
+ catch {
83
+ // Ignore join failures
84
+ }
85
+ }
86
+ }
87
+ // Process messages from joined rooms
88
+ if (response.rooms?.join) {
89
+ for (const [roomId, room] of Object.entries(response.rooms.join)) {
90
+ const events = room.timeline?.events || [];
91
+ for (const event of events) {
92
+ await this.handleEvent(roomId, event);
93
+ }
94
+ }
95
+ }
96
+ this.syncToken = response.next_batch;
97
+ }
98
+ catch (error) {
99
+ if (error instanceof Error && error.name === 'AbortError') {
100
+ break;
101
+ }
102
+ this.errorHandler?.(error instanceof Error ? error : new Error(String(error)));
103
+ // Brief delay before retrying on error
104
+ await new Promise((resolve) => setTimeout(resolve, 5000));
105
+ }
106
+ }
107
+ }
108
+ async handleEvent(roomId, event) {
109
+ // Only process m.room.message events
110
+ if (event.type !== 'm.room.message')
111
+ return;
112
+ // Ignore own messages
113
+ if (event.sender === this.config.userId)
114
+ return;
115
+ // Only process text and notice messages
116
+ const msgtype = event.content.msgtype;
117
+ if (msgtype !== 'm.text' && msgtype !== 'm.notice')
118
+ return;
119
+ // Check allowed rooms
120
+ if (this.config.allowedRooms?.length && !this.config.allowedRooms.includes(roomId)) {
121
+ audit('message.filtered', { channelType: 'matrix', senderId: event.sender, roomId, reason: 'room_not_allowed' });
122
+ return;
123
+ }
124
+ // Check allowed users
125
+ if (this.config.allowedUsers?.length && !this.config.allowedUsers.includes(event.sender)) {
126
+ audit('message.filtered', { channelType: 'matrix', senderId: event.sender, roomId, reason: 'user_not_allowed' });
127
+ return;
128
+ }
129
+ const inbound = this.toInboundMessage(roomId, event);
130
+ audit('message.received', {
131
+ channelType: 'matrix',
132
+ senderId: inbound.senderId,
133
+ channelId: inbound.channelId,
134
+ });
135
+ if (this.messageHandler) {
136
+ try {
137
+ await this.messageHandler(inbound);
138
+ }
139
+ catch (error) {
140
+ this.errorHandler?.(error instanceof Error ? error : new Error(String(error)));
141
+ }
142
+ }
143
+ }
144
+ toInboundMessage(roomId, event) {
145
+ const replyTo = event.content['m.relates_to']?.['m.in_reply_to']?.event_id;
146
+ // Strip reply fallback from content if present
147
+ let content = event.content.body || '';
148
+ if (replyTo && content.startsWith('> ')) {
149
+ const lines = content.split('\n');
150
+ const nonQuoteIdx = lines.findIndex((l) => !l.startsWith('> ') && l !== '');
151
+ if (nonQuoteIdx > 0) {
152
+ content = lines.slice(nonQuoteIdx).join('\n').trim();
153
+ }
154
+ }
155
+ return {
156
+ id: event.event_id,
157
+ channelType: 'matrix',
158
+ channelId: roomId,
159
+ senderId: event.sender,
160
+ senderName: event.sender.split(':')[0].replace('@', ''),
161
+ content,
162
+ timestamp: event.origin_server_ts,
163
+ replyToId: replyTo,
164
+ raw: event,
165
+ };
166
+ }
167
+ async send(channelId, message) {
168
+ try {
169
+ const chunks = this.chunkMessage(message.content);
170
+ let lastEventId;
171
+ for (const chunk of chunks) {
172
+ const txnId = `m${Date.now()}.${Math.random().toString(36).slice(2)}`;
173
+ const body = {
174
+ msgtype: 'm.text',
175
+ body: chunk,
176
+ };
177
+ // Add reply relation
178
+ if (message.replyToId) {
179
+ body['m.relates_to'] = {
180
+ 'm.in_reply_to': {
181
+ event_id: message.replyToId,
182
+ },
183
+ };
184
+ }
185
+ const result = await this.matrixFetch(`/rooms/${encodeURIComponent(channelId)}/send/m.room.message/${encodeURIComponent(txnId)}`, { method: 'PUT', body: JSON.stringify(body) });
186
+ lastEventId = result.event_id;
187
+ }
188
+ audit('message.sent', {
189
+ channelType: 'matrix',
190
+ channelId,
191
+ messageId: lastEventId,
192
+ });
193
+ return { success: true, messageId: lastEventId };
194
+ }
195
+ catch (error) {
196
+ const errorMessage = error instanceof Error ? error.message : String(error);
197
+ audit('channel.error', {
198
+ channelType: 'matrix',
199
+ action: 'send',
200
+ error: errorMessage,
201
+ });
202
+ return { success: false, error: errorMessage };
203
+ }
204
+ }
205
+ chunkMessage(content) {
206
+ if (content.length <= MAX_MESSAGE_LENGTH) {
207
+ return [content];
208
+ }
209
+ const chunks = [];
210
+ let remaining = content;
211
+ while (remaining.length > 0) {
212
+ if (remaining.length <= MAX_MESSAGE_LENGTH) {
213
+ chunks.push(remaining);
214
+ break;
215
+ }
216
+ let breakPoint = remaining.lastIndexOf('\n', MAX_MESSAGE_LENGTH);
217
+ if (breakPoint === -1 || breakPoint < MAX_MESSAGE_LENGTH / 2) {
218
+ breakPoint = remaining.lastIndexOf(' ', MAX_MESSAGE_LENGTH);
219
+ }
220
+ if (breakPoint === -1 || breakPoint < MAX_MESSAGE_LENGTH / 2) {
221
+ breakPoint = MAX_MESSAGE_LENGTH;
222
+ }
223
+ chunks.push(remaining.slice(0, breakPoint));
224
+ remaining = remaining.slice(breakPoint).trimStart();
225
+ }
226
+ return chunks;
227
+ }
228
+ async startTyping(channelId) {
229
+ const userId = this.config.userId;
230
+ const typingPath = `/rooms/${encodeURIComponent(channelId)}/typing/${encodeURIComponent(userId)}`;
231
+ // Send typing with 30s timeout, repeat every 25s
232
+ let stopped = false;
233
+ const sendTyping = () => this.matrixFetch(typingPath, {
234
+ method: 'PUT',
235
+ body: JSON.stringify({ typing: true, timeout: 30000 }),
236
+ }).catch((e) => {
237
+ audit('channel.error', { channelType: 'matrix', action: 'typing', error: e.message });
238
+ });
239
+ sendTyping();
240
+ const interval = setInterval(() => {
241
+ if (stopped)
242
+ return;
243
+ sendTyping();
244
+ }, 25000);
245
+ return () => {
246
+ stopped = true;
247
+ clearInterval(interval);
248
+ // Send stop-typing signal
249
+ this.matrixFetch(typingPath, {
250
+ method: 'PUT',
251
+ body: JSON.stringify({ typing: false }),
252
+ }).catch(() => { });
253
+ };
254
+ }
255
+ onMessage(handler) {
256
+ this.messageHandler = handler;
257
+ }
258
+ onError(handler) {
259
+ this.errorHandler = handler;
260
+ }
261
+ }
262
+ //# sourceMappingURL=matrix.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"matrix.js","sourceRoot":"","sources":["../../src/adapters/matrix.ts"],"names":[],"mappings":"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"}