morok-bot-sdk 1.0.1
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/LICENSE +201 -0
- package/README.md +602 -0
- package/README.ru.md +602 -0
- package/dist/bot.d.ts +232 -0
- package/dist/bot.d.ts.map +1 -0
- package/dist/bot.js +558 -0
- package/dist/bot.js.map +1 -0
- package/dist/crypto/channel-cipher.d.ts +32 -0
- package/dist/crypto/channel-cipher.d.ts.map +1 -0
- package/dist/crypto/channel-cipher.js +77 -0
- package/dist/crypto/channel-cipher.js.map +1 -0
- package/dist/crypto/channel-key-store.d.ts +37 -0
- package/dist/crypto/channel-key-store.d.ts.map +1 -0
- package/dist/crypto/channel-key-store.js +149 -0
- package/dist/crypto/channel-key-store.js.map +1 -0
- package/dist/crypto/cross-signing.d.ts +57 -0
- package/dist/crypto/cross-signing.d.ts.map +1 -0
- package/dist/crypto/cross-signing.js +111 -0
- package/dist/crypto/cross-signing.js.map +1 -0
- package/dist/crypto/file-cipher.d.ts +36 -0
- package/dist/crypto/file-cipher.d.ts.map +1 -0
- package/dist/crypto/file-cipher.js +61 -0
- package/dist/crypto/file-cipher.js.map +1 -0
- package/dist/crypto/group-secret-cipher.d.ts +49 -0
- package/dist/crypto/group-secret-cipher.d.ts.map +1 -0
- package/dist/crypto/group-secret-cipher.js +69 -0
- package/dist/crypto/group-secret-cipher.js.map +1 -0
- package/dist/crypto/group-secret-store.d.ts +35 -0
- package/dist/crypto/group-secret-store.d.ts.map +1 -0
- package/dist/crypto/group-secret-store.js +149 -0
- package/dist/crypto/group-secret-store.js.map +1 -0
- package/dist/crypto/signal.d.ts +81 -0
- package/dist/crypto/signal.d.ts.map +1 -0
- package/dist/crypto/signal.js +125 -0
- package/dist/crypto/signal.js.map +1 -0
- package/dist/crypto/stores.d.ts +130 -0
- package/dist/crypto/stores.d.ts.map +1 -0
- package/dist/crypto/stores.js +314 -0
- package/dist/crypto/stores.js.map +1 -0
- package/dist/flow/attachments.d.ts +110 -0
- package/dist/flow/attachments.d.ts.map +1 -0
- package/dist/flow/attachments.js +409 -0
- package/dist/flow/attachments.js.map +1 -0
- package/dist/flow/conv-cache.d.ts +36 -0
- package/dist/flow/conv-cache.d.ts.map +1 -0
- package/dist/flow/conv-cache.js +84 -0
- package/dist/flow/conv-cache.js.map +1 -0
- package/dist/flow/direct.d.ts +109 -0
- package/dist/flow/direct.d.ts.map +1 -0
- package/dist/flow/direct.js +346 -0
- package/dist/flow/direct.js.map +1 -0
- package/dist/flow/groups.d.ts +146 -0
- package/dist/flow/groups.d.ts.map +1 -0
- package/dist/flow/groups.js +768 -0
- package/dist/flow/groups.js.map +1 -0
- package/dist/flow/prekeys.d.ts +45 -0
- package/dist/flow/prekeys.d.ts.map +1 -0
- package/dist/flow/prekeys.js +111 -0
- package/dist/flow/prekeys.js.map +1 -0
- package/dist/flow/receive.d.ts +125 -0
- package/dist/flow/receive.d.ts.map +1 -0
- package/dist/flow/receive.js +773 -0
- package/dist/flow/receive.js.map +1 -0
- package/dist/index.d.ts +15 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +6 -0
- package/dist/index.js.map +1 -0
- package/dist/morokbot-file.d.ts +14 -0
- package/dist/morokbot-file.d.ts.map +1 -0
- package/dist/morokbot-file.js +88 -0
- package/dist/morokbot-file.js.map +1 -0
- package/dist/ratelimit.d.ts +40 -0
- package/dist/ratelimit.d.ts.map +1 -0
- package/dist/ratelimit.js +76 -0
- package/dist/ratelimit.js.map +1 -0
- package/dist/sessions.d.ts +34 -0
- package/dist/sessions.d.ts.map +1 -0
- package/dist/sessions.js +69 -0
- package/dist/sessions.js.map +1 -0
- package/dist/state-lock.d.ts +17 -0
- package/dist/state-lock.d.ts.map +1 -0
- package/dist/state-lock.js +66 -0
- package/dist/state-lock.js.map +1 -0
- package/dist/transport/http.d.ts +48 -0
- package/dist/transport/http.d.ts.map +1 -0
- package/dist/transport/http.js +112 -0
- package/dist/transport/http.js.map +1 -0
- package/dist/transport/ws.d.ts +65 -0
- package/dist/transport/ws.d.ts.map +1 -0
- package/dist/transport/ws.js +219 -0
- package/dist/transport/ws.js.map +1 -0
- package/dist/types.d.ts +254 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +59 -0
package/dist/bot.js
ADDED
|
@@ -0,0 +1,558 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { readMorokbotFile } from './morokbot-file.js';
|
|
3
|
+
import { FileSignalStore } from './crypto/stores.js';
|
|
4
|
+
import { SignalEngine } from './crypto/signal.js';
|
|
5
|
+
import { signDeviceCert } from './crypto/cross-signing.js';
|
|
6
|
+
import { ChannelKeyStore } from './crypto/channel-key-store.js';
|
|
7
|
+
import { GroupSecretStore } from './crypto/group-secret-store.js';
|
|
8
|
+
import { StateLock } from './state-lock.js';
|
|
9
|
+
import { HttpClient } from './transport/http.js';
|
|
10
|
+
import { WsClient } from './transport/ws.js';
|
|
11
|
+
import { DirectFlow, uploadAndBuildPayload, uploadGalleryAndBuildPayload } from './flow/direct.js';
|
|
12
|
+
import { ReceiveFlow } from './flow/receive.js';
|
|
13
|
+
import { GroupsFlow } from './flow/groups.js';
|
|
14
|
+
import { ConvCache } from './flow/conv-cache.js';
|
|
15
|
+
import { PreKeyManager } from './flow/prekeys.js';
|
|
16
|
+
function deriveWsUrl(apiBaseUrl, wsUrl) {
|
|
17
|
+
if (wsUrl)
|
|
18
|
+
return wsUrl;
|
|
19
|
+
return apiBaseUrl
|
|
20
|
+
.replace(/^http:/i, 'ws:')
|
|
21
|
+
.replace(/^https:/i, 'wss:')
|
|
22
|
+
.replace(/\/+$/, '')
|
|
23
|
+
+ '/ws';
|
|
24
|
+
}
|
|
25
|
+
function isUsername(s) {
|
|
26
|
+
return typeof s === 'string' && s.length > 0;
|
|
27
|
+
}
|
|
28
|
+
export class MorokBot extends EventEmitter {
|
|
29
|
+
file;
|
|
30
|
+
stateDir;
|
|
31
|
+
apiBaseUrl;
|
|
32
|
+
wsUrlVal;
|
|
33
|
+
logger;
|
|
34
|
+
bgIntervalMs;
|
|
35
|
+
replenishThreshold;
|
|
36
|
+
replenishTarget;
|
|
37
|
+
autoBackfillOnJoin;
|
|
38
|
+
serveBackfillRequests;
|
|
39
|
+
signal;
|
|
40
|
+
http;
|
|
41
|
+
ws;
|
|
42
|
+
direct;
|
|
43
|
+
receive;
|
|
44
|
+
groups;
|
|
45
|
+
chanStore;
|
|
46
|
+
gsStore;
|
|
47
|
+
convCache;
|
|
48
|
+
stateLock;
|
|
49
|
+
prekey;
|
|
50
|
+
botUserId;
|
|
51
|
+
running = false;
|
|
52
|
+
starting = false;
|
|
53
|
+
stopRequested = false;
|
|
54
|
+
constructor(file, config) {
|
|
55
|
+
super();
|
|
56
|
+
this.file = file;
|
|
57
|
+
this.stateDir = config.stateDir ?? './bot-state';
|
|
58
|
+
this.apiBaseUrl = config.apiBaseUrl ?? 'https://app.morok.me';
|
|
59
|
+
this.wsUrlVal = deriveWsUrl(this.apiBaseUrl, config.wsUrl);
|
|
60
|
+
this.replenishThreshold = config.replenishThreshold ?? 100;
|
|
61
|
+
this.replenishTarget = config.replenishTarget ?? 200;
|
|
62
|
+
this.bgIntervalMs = config.backgroundIntervalMs ?? 5 * 60 * 1000;
|
|
63
|
+
this.autoBackfillOnJoin = config.autoBackfillOnJoin ?? false;
|
|
64
|
+
this.serveBackfillRequests = config.serveBackfillRequests ?? false;
|
|
65
|
+
if (config.logger !== undefined)
|
|
66
|
+
this.logger = config.logger;
|
|
67
|
+
this.on('error', (err) => {
|
|
68
|
+
this.logger?.warn({ err: err.message }, '[bot] unhandled error event');
|
|
69
|
+
});
|
|
70
|
+
}
|
|
71
|
+
static async fromFile(config) {
|
|
72
|
+
if (!config.tokenFile) {
|
|
73
|
+
throw new Error('MorokBot.fromFile: tokenFile is required');
|
|
74
|
+
}
|
|
75
|
+
const file = await readMorokbotFile(config.tokenFile, config.logger);
|
|
76
|
+
return new MorokBot(file, config);
|
|
77
|
+
}
|
|
78
|
+
async start() {
|
|
79
|
+
if (this.running || this.starting)
|
|
80
|
+
return;
|
|
81
|
+
this.starting = true;
|
|
82
|
+
this.stopRequested = false;
|
|
83
|
+
const checkpoint = (where) => {
|
|
84
|
+
if (this.stopRequested) {
|
|
85
|
+
throw new Error(`MorokBot.start: aborted at ${where} - stop() called during boot`);
|
|
86
|
+
}
|
|
87
|
+
};
|
|
88
|
+
try {
|
|
89
|
+
const stateLock = new StateLock(this.stateDir, this.logger);
|
|
90
|
+
await stateLock.acquire();
|
|
91
|
+
this.stateLock = stateLock;
|
|
92
|
+
checkpoint('stateLock.acquire');
|
|
93
|
+
const store = new FileSignalStore(this.stateDir);
|
|
94
|
+
await store.ensureLayout();
|
|
95
|
+
checkpoint('ensureLayout');
|
|
96
|
+
await store.importInitial({
|
|
97
|
+
botUserId: this.file.botUserId,
|
|
98
|
+
registrationId: this.file.registrationId,
|
|
99
|
+
deviceId: 1,
|
|
100
|
+
identityKeyPair: this.file.identityKey,
|
|
101
|
+
accountSigningKey: this.file.accountSigningKey,
|
|
102
|
+
signedPreKey: {
|
|
103
|
+
keyId: this.file.signedPreKey.keyId,
|
|
104
|
+
pub: this.file.signedPreKey.pub,
|
|
105
|
+
priv: this.file.signedPreKey.priv,
|
|
106
|
+
signature: this.file.signedPreKey.signature,
|
|
107
|
+
},
|
|
108
|
+
oneTimePreKeys: this.file.oneTimePreKeys.map(o => ({
|
|
109
|
+
keyId: o.keyId, pub: o.pub, priv: o.priv,
|
|
110
|
+
})),
|
|
111
|
+
});
|
|
112
|
+
checkpoint('importInitial');
|
|
113
|
+
const fsck = await store.fsck();
|
|
114
|
+
checkpoint('fsck');
|
|
115
|
+
if (fsck.quarantinedSessions.length + fsck.quarantinedPreKeys.length > 0) {
|
|
116
|
+
this.logger?.warn({ sessions: fsck.quarantinedSessions, preKeys: fsck.quarantinedPreKeys }, '[bot] fsck quarantined corrupted state files');
|
|
117
|
+
}
|
|
118
|
+
this.signal = new SignalEngine(store, this.logger);
|
|
119
|
+
this.http = new HttpClient({
|
|
120
|
+
apiBaseUrl: this.apiBaseUrl,
|
|
121
|
+
botToken: this.file.token,
|
|
122
|
+
...(this.logger !== undefined ? { logger: this.logger } : {}),
|
|
123
|
+
});
|
|
124
|
+
const session = await this.http.initialMint();
|
|
125
|
+
checkpoint('initialMint');
|
|
126
|
+
this.botUserId = session.userId;
|
|
127
|
+
this.assertResolvedUserId(session.userId);
|
|
128
|
+
const ws = new WsClient({
|
|
129
|
+
wsUrl: this.wsUrlVal,
|
|
130
|
+
deviceId: 1,
|
|
131
|
+
getJwt: () => this.http.getJwt(),
|
|
132
|
+
refreshJwt: async () => { await this.http.refresh(); },
|
|
133
|
+
...(this.logger !== undefined ? { logger: this.logger } : {}),
|
|
134
|
+
});
|
|
135
|
+
ws.on('close', (info) => {
|
|
136
|
+
let reason;
|
|
137
|
+
if (!info.willReconnect)
|
|
138
|
+
reason = 'shutdown';
|
|
139
|
+
else if (info.code === 4001)
|
|
140
|
+
reason = 'auth';
|
|
141
|
+
else
|
|
142
|
+
reason = 'transport';
|
|
143
|
+
this.emit('disconnect', {
|
|
144
|
+
reason,
|
|
145
|
+
code: info.code,
|
|
146
|
+
willReconnect: info.willReconnect,
|
|
147
|
+
});
|
|
148
|
+
});
|
|
149
|
+
ws.on('error', (err) => this.emit('error', err));
|
|
150
|
+
this.ws = ws;
|
|
151
|
+
await ws.start();
|
|
152
|
+
checkpoint('ws.start');
|
|
153
|
+
this.direct = new DirectFlow(this.http, ws, this.signal, this.logger);
|
|
154
|
+
this.chanStore = new ChannelKeyStore(this.stateDir, this.logger);
|
|
155
|
+
await this.chanStore.init();
|
|
156
|
+
checkpoint('chanStore.init');
|
|
157
|
+
this.gsStore = new GroupSecretStore(this.stateDir, this.logger);
|
|
158
|
+
await this.gsStore.init();
|
|
159
|
+
checkpoint('gsStore.init');
|
|
160
|
+
this.convCache = new ConvCache(session.userId, this.http, this.logger);
|
|
161
|
+
this.groups = new GroupsFlow(session.userId, this.http, ws, this.signal, this.chanStore, this.gsStore, this.convCache, this.logger);
|
|
162
|
+
this.receive = new ReceiveFlow(session.userId, this.http, ws, this.signal, this.convCache, { emit: (e) => {
|
|
163
|
+
switch (e.kind) {
|
|
164
|
+
case 'message':
|
|
165
|
+
this.emit('message', e.payload);
|
|
166
|
+
break;
|
|
167
|
+
case 'command':
|
|
168
|
+
this.emit('command', e.payload);
|
|
169
|
+
break;
|
|
170
|
+
case 'start':
|
|
171
|
+
this.emit('start', e.payload);
|
|
172
|
+
break;
|
|
173
|
+
case 'stop':
|
|
174
|
+
this.emit('stop', e.payload);
|
|
175
|
+
break;
|
|
176
|
+
case 'conversation_added':
|
|
177
|
+
this.emit('conversation_added', e.payload);
|
|
178
|
+
break;
|
|
179
|
+
case 'conversation_kicked':
|
|
180
|
+
this.emit('conversation_kicked', e.payload);
|
|
181
|
+
break;
|
|
182
|
+
case 'reaction':
|
|
183
|
+
this.emit('reaction', e.payload);
|
|
184
|
+
break;
|
|
185
|
+
case 'control':
|
|
186
|
+
this.emit('control', e.payload);
|
|
187
|
+
break;
|
|
188
|
+
case 'error':
|
|
189
|
+
this.emit('error', e.error);
|
|
190
|
+
break;
|
|
191
|
+
}
|
|
192
|
+
} }, this.groups, { autoBackfillOnJoin: this.autoBackfillOnJoin, serveBackfillRequests: this.serveBackfillRequests }, this.logger);
|
|
193
|
+
this.prekey = new PreKeyManager(this.http, this.signal, store, {
|
|
194
|
+
deviceId: 1,
|
|
195
|
+
replenishThreshold: this.replenishThreshold,
|
|
196
|
+
replenishTarget: this.replenishTarget,
|
|
197
|
+
backgroundIntervalMs: this.bgIntervalMs,
|
|
198
|
+
...(this.logger !== undefined ? { logger: this.logger } : {}),
|
|
199
|
+
});
|
|
200
|
+
await this.prekey.start();
|
|
201
|
+
checkpoint('prekey.start');
|
|
202
|
+
ws.on('frame', (frame) => {
|
|
203
|
+
if (frame.type === 'prekeys_low') {
|
|
204
|
+
this.prekey?.requestReplenish().catch(() => { });
|
|
205
|
+
}
|
|
206
|
+
const http = this.http;
|
|
207
|
+
if (http && http.needsProactiveRefresh()) {
|
|
208
|
+
void http.refresh().catch(() => { });
|
|
209
|
+
}
|
|
210
|
+
});
|
|
211
|
+
void this.ensureCrossSigning();
|
|
212
|
+
this.running = true;
|
|
213
|
+
this.logger?.info({ userId: session.userId, username: session.username }, '[bot] ready');
|
|
214
|
+
}
|
|
215
|
+
catch (err) {
|
|
216
|
+
try {
|
|
217
|
+
this.prekey?.stop();
|
|
218
|
+
}
|
|
219
|
+
catch { }
|
|
220
|
+
try {
|
|
221
|
+
this.direct?.shutdown();
|
|
222
|
+
}
|
|
223
|
+
catch { }
|
|
224
|
+
try {
|
|
225
|
+
this.groups?.shutdown();
|
|
226
|
+
}
|
|
227
|
+
catch { }
|
|
228
|
+
try {
|
|
229
|
+
this.ws?.stop();
|
|
230
|
+
}
|
|
231
|
+
catch { }
|
|
232
|
+
this.prekey = undefined;
|
|
233
|
+
this.direct = undefined;
|
|
234
|
+
this.receive = undefined;
|
|
235
|
+
this.groups = undefined;
|
|
236
|
+
this.chanStore = undefined;
|
|
237
|
+
this.gsStore = undefined;
|
|
238
|
+
this.convCache = undefined;
|
|
239
|
+
try {
|
|
240
|
+
await this.stateLock?.release();
|
|
241
|
+
}
|
|
242
|
+
catch { }
|
|
243
|
+
this.stateLock = undefined;
|
|
244
|
+
this.ws = undefined;
|
|
245
|
+
this.http = undefined;
|
|
246
|
+
this.signal = undefined;
|
|
247
|
+
this.botUserId = undefined;
|
|
248
|
+
throw err;
|
|
249
|
+
}
|
|
250
|
+
finally {
|
|
251
|
+
this.starting = false;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
assertResolvedUserId(resolvedUserId) {
|
|
255
|
+
if (resolvedUserId !== this.file.botUserId) {
|
|
256
|
+
throw new Error(`MorokBot.start: .morokbot says botUserId=${this.file.botUserId} but ` +
|
|
257
|
+
`/auth/bot-session resolved userId=${resolvedUserId} - file tampered or stateDir mismatched`);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
async ensureCrossSigning() {
|
|
261
|
+
const xsk = this.file.accountSigningKey;
|
|
262
|
+
const http = this.http;
|
|
263
|
+
const botUserId = this.botUserId;
|
|
264
|
+
if (!xsk) {
|
|
265
|
+
this.logger?.debug({}, '[bot] .morokbot has no accountSigningKey - skipping cross-signing (bot stays uncertified / TOFU)');
|
|
266
|
+
return;
|
|
267
|
+
}
|
|
268
|
+
if (!http || botUserId === undefined)
|
|
269
|
+
return;
|
|
270
|
+
try {
|
|
271
|
+
const cert = await signDeviceCert(xsk.priv, botUserId, 1, this.file.identityKey.pub);
|
|
272
|
+
if (this.stopRequested)
|
|
273
|
+
return;
|
|
274
|
+
await http.post('/crypto/cross-signing', {
|
|
275
|
+
accountSigningKey: xsk.pub,
|
|
276
|
+
deviceCertificate: cert,
|
|
277
|
+
});
|
|
278
|
+
this.logger?.info({ botUserId }, '[bot] device certificate published (cross-signing)');
|
|
279
|
+
}
|
|
280
|
+
catch (err) {
|
|
281
|
+
this.logger?.warn({ err: err.message }, '[bot] cross-signing publish failed; bot stays uncertified this run (peers use TOFU)');
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
async stop() {
|
|
285
|
+
if (!this.running && !this.starting)
|
|
286
|
+
return;
|
|
287
|
+
this.stopRequested = true;
|
|
288
|
+
this.running = false;
|
|
289
|
+
try {
|
|
290
|
+
this.prekey?.stop();
|
|
291
|
+
}
|
|
292
|
+
catch { }
|
|
293
|
+
try {
|
|
294
|
+
this.direct?.shutdown();
|
|
295
|
+
}
|
|
296
|
+
catch { }
|
|
297
|
+
try {
|
|
298
|
+
this.groups?.shutdown();
|
|
299
|
+
}
|
|
300
|
+
catch { }
|
|
301
|
+
try {
|
|
302
|
+
this.ws?.stop();
|
|
303
|
+
}
|
|
304
|
+
catch { }
|
|
305
|
+
try {
|
|
306
|
+
await this.stateLock?.release();
|
|
307
|
+
}
|
|
308
|
+
catch { }
|
|
309
|
+
this.stateLock = undefined;
|
|
310
|
+
}
|
|
311
|
+
get isConnected() {
|
|
312
|
+
return this.running && (this.ws?.isConnected ?? false);
|
|
313
|
+
}
|
|
314
|
+
get userId() {
|
|
315
|
+
if (this.botUserId === undefined) {
|
|
316
|
+
throw new Error('MorokBot.userId: call start() first');
|
|
317
|
+
}
|
|
318
|
+
return this.botUserId;
|
|
319
|
+
}
|
|
320
|
+
async send(args) {
|
|
321
|
+
if (!this.running || !this.direct || !this.receive || !this.http) {
|
|
322
|
+
throw new Error('MorokBot.send: not started');
|
|
323
|
+
}
|
|
324
|
+
const hasText = args.text !== undefined;
|
|
325
|
+
const hasAttachment = args.attachment !== undefined;
|
|
326
|
+
const hasGallery = Array.isArray(args.attachments) && args.attachments.length > 0;
|
|
327
|
+
if (hasAttachment && hasGallery) {
|
|
328
|
+
throw new Error('MorokBot.send: `attachment` and `attachments` are mutually exclusive');
|
|
329
|
+
}
|
|
330
|
+
if (!hasText && !hasAttachment && !hasGallery) {
|
|
331
|
+
throw new Error('MorokBot.send: must supply at least one of `text`, `attachment`, or `attachments`');
|
|
332
|
+
}
|
|
333
|
+
const hasPeer = args.peer !== undefined;
|
|
334
|
+
const hasGroup = args.conversation !== undefined;
|
|
335
|
+
if (hasPeer === hasGroup) {
|
|
336
|
+
throw new Error('MorokBot.send: must supply EXACTLY ONE of `peer` (for DM) or `conversation` (for group chat or channel)');
|
|
337
|
+
}
|
|
338
|
+
if (hasGroup) {
|
|
339
|
+
if (!this.groups)
|
|
340
|
+
throw new Error('MorokBot.send: groups flow not wired');
|
|
341
|
+
return this.sendGroup(args);
|
|
342
|
+
}
|
|
343
|
+
return this.sendDirect(args);
|
|
344
|
+
}
|
|
345
|
+
async sendDirect(args) {
|
|
346
|
+
const peerUserId = await this.resolvePeerUserId(args.peer);
|
|
347
|
+
const sendOpts = {};
|
|
348
|
+
if (args.replyToId !== undefined)
|
|
349
|
+
sendOpts.replyToId = args.replyToId;
|
|
350
|
+
if (args.replyToClientMsgId !== undefined)
|
|
351
|
+
sendOpts.replyToClientMsgId = args.replyToClientMsgId;
|
|
352
|
+
if (args.threadRootId !== undefined)
|
|
353
|
+
sendOpts.threadRootId = args.threadRootId;
|
|
354
|
+
const body = {};
|
|
355
|
+
if (args.text !== undefined)
|
|
356
|
+
body.text = args.text;
|
|
357
|
+
if (args.attachment !== undefined)
|
|
358
|
+
body.attachment = args.attachment;
|
|
359
|
+
if (args.attachments !== undefined)
|
|
360
|
+
body.attachments = args.attachments;
|
|
361
|
+
return this.direct.sendMessage(peerUserId, body, sendOpts);
|
|
362
|
+
}
|
|
363
|
+
async sendGroup(args) {
|
|
364
|
+
const conversationId = args.conversation;
|
|
365
|
+
if (!Number.isInteger(conversationId) || conversationId < 1) {
|
|
366
|
+
throw new Error(`MorokBot.send: conversationId must be a positive integer, got ${conversationId}`);
|
|
367
|
+
}
|
|
368
|
+
const hasText = typeof args.text === 'string' && args.text.length > 0;
|
|
369
|
+
const hasAttachment = args.attachment !== undefined;
|
|
370
|
+
const hasGallery = Array.isArray(args.attachments) && args.attachments.length > 0;
|
|
371
|
+
let plaintext;
|
|
372
|
+
let fileId;
|
|
373
|
+
let kind = 'text';
|
|
374
|
+
let additionalFileIds;
|
|
375
|
+
if (hasGallery) {
|
|
376
|
+
const built = await uploadGalleryAndBuildPayload(this.http, args.attachments, hasText ? args.text : undefined, this.logger);
|
|
377
|
+
plaintext = new TextEncoder().encode(built.payload);
|
|
378
|
+
fileId = built.headFileId;
|
|
379
|
+
kind = 'gallery';
|
|
380
|
+
additionalFileIds = built.additionalFileIds;
|
|
381
|
+
}
|
|
382
|
+
else if (hasAttachment) {
|
|
383
|
+
const built = await uploadAndBuildPayload(this.http, args.attachment, hasText ? args.text : undefined, this.logger);
|
|
384
|
+
plaintext = new TextEncoder().encode(built.payload);
|
|
385
|
+
fileId = built.ref.fileId;
|
|
386
|
+
kind = built.kind;
|
|
387
|
+
if (kind !== 'file' && hasText) {
|
|
388
|
+
this.logger?.warn({ kind, droppedTextLen: args.text.length }, '[bot] caption ignored: only kind="file" supports caption in groups');
|
|
389
|
+
}
|
|
390
|
+
}
|
|
391
|
+
else {
|
|
392
|
+
plaintext = new TextEncoder().encode(args.text);
|
|
393
|
+
}
|
|
394
|
+
const sendOpts = { kind };
|
|
395
|
+
if (args.replyToId !== undefined)
|
|
396
|
+
sendOpts.replyToId = args.replyToId;
|
|
397
|
+
if (args.replyToClientMsgId !== undefined)
|
|
398
|
+
sendOpts.replyToClientMsgId = args.replyToClientMsgId;
|
|
399
|
+
if (args.threadRootId !== undefined)
|
|
400
|
+
sendOpts.threadRootId = args.threadRootId;
|
|
401
|
+
if (fileId !== undefined)
|
|
402
|
+
sendOpts.fileId = fileId;
|
|
403
|
+
if (additionalFileIds !== undefined)
|
|
404
|
+
sendOpts.additionalFileIds = additionalFileIds;
|
|
405
|
+
if (args.expiresInSeconds !== undefined)
|
|
406
|
+
sendOpts.expiresInSeconds = args.expiresInSeconds;
|
|
407
|
+
const res = await this.groups.sendMessage(conversationId, plaintext, sendOpts);
|
|
408
|
+
return { messageId: res.messageId, clientMsgId: res.clientMsgId, conversationId: res.conversationId };
|
|
409
|
+
}
|
|
410
|
+
async rotateChannelKey(conversationId) {
|
|
411
|
+
if (!this.running || !this.groups) {
|
|
412
|
+
throw new Error('MorokBot.rotateChannelKey: not started');
|
|
413
|
+
}
|
|
414
|
+
return this.groups.rotateChannelKey(conversationId);
|
|
415
|
+
}
|
|
416
|
+
async backfillChannelKeys(conversationId, target, opts = {}) {
|
|
417
|
+
if (!this.running || !this.groups) {
|
|
418
|
+
throw new Error('MorokBot.backfillChannelKeys: not started');
|
|
419
|
+
}
|
|
420
|
+
return this.groups.backfillChannelKeys(conversationId, target, opts);
|
|
421
|
+
}
|
|
422
|
+
async rotateGroupSecret(conversationId) {
|
|
423
|
+
if (!this.running || !this.groups) {
|
|
424
|
+
throw new Error('MorokBot.rotateGroupSecret: not started');
|
|
425
|
+
}
|
|
426
|
+
return this.groups.rotateGroupSecret(conversationId);
|
|
427
|
+
}
|
|
428
|
+
async setMyCommands(commands) {
|
|
429
|
+
if (!this.running || !this.http) {
|
|
430
|
+
throw new Error('MorokBot.setMyCommands: not started');
|
|
431
|
+
}
|
|
432
|
+
const res = await this.http.post('/developer/bots/self/commands', { commands });
|
|
433
|
+
return { count: typeof res.data?.count === 'number' ? res.data.count : 0 };
|
|
434
|
+
}
|
|
435
|
+
async setMyControls(controls) {
|
|
436
|
+
if (!this.running || !this.http) {
|
|
437
|
+
throw new Error('MorokBot.setMyControls: not started');
|
|
438
|
+
}
|
|
439
|
+
const res = await this.http.post('/developer/bots/self/controls', { controls });
|
|
440
|
+
return { count: typeof res.data?.count === 'number' ? res.data.count : 0 };
|
|
441
|
+
}
|
|
442
|
+
async setControlsFor(peerUserId, controls) {
|
|
443
|
+
if (!this.running || !this.http) {
|
|
444
|
+
throw new Error('MorokBot.setControlsFor: not started');
|
|
445
|
+
}
|
|
446
|
+
const res = await this.http.post('/developer/bots/self/controls/conversation', { targetUserId: peerUserId, controls });
|
|
447
|
+
return { count: typeof res.data?.count === 'number' ? res.data.count : 0 };
|
|
448
|
+
}
|
|
449
|
+
async clearControlsFor(peerUserId) {
|
|
450
|
+
if (!this.running || !this.http) {
|
|
451
|
+
throw new Error('MorokBot.clearControlsFor: not started');
|
|
452
|
+
}
|
|
453
|
+
await this.http.post('/developer/bots/self/controls/conversation', { targetUserId: peerUserId, controls: null });
|
|
454
|
+
}
|
|
455
|
+
async reply(msg, args) {
|
|
456
|
+
if (args.text === undefined && args.attachment === undefined && args.attachments === undefined) {
|
|
457
|
+
throw new Error('MorokBot.reply: must supply at least one of `text`, `attachment`, or `attachments`');
|
|
458
|
+
}
|
|
459
|
+
const sendArgs = {
|
|
460
|
+
replyToId: msg.messageId,
|
|
461
|
+
};
|
|
462
|
+
if (msg.conversationType === 'DIRECT') {
|
|
463
|
+
sendArgs.peer = msg.sender.userId;
|
|
464
|
+
}
|
|
465
|
+
else {
|
|
466
|
+
sendArgs.conversation = msg.conversationId;
|
|
467
|
+
if (msg.threadRootId !== null)
|
|
468
|
+
sendArgs.threadRootId = msg.threadRootId;
|
|
469
|
+
}
|
|
470
|
+
if (args.text !== undefined)
|
|
471
|
+
sendArgs.text = args.text;
|
|
472
|
+
if (args.attachment !== undefined)
|
|
473
|
+
sendArgs.attachment = args.attachment;
|
|
474
|
+
if (args.attachments !== undefined)
|
|
475
|
+
sendArgs.attachments = args.attachments;
|
|
476
|
+
if (args.expiresInSeconds !== undefined)
|
|
477
|
+
sendArgs.expiresInSeconds = args.expiresInSeconds;
|
|
478
|
+
if (msg.clientMsgId !== null)
|
|
479
|
+
sendArgs.replyToClientMsgId = msg.clientMsgId;
|
|
480
|
+
return this.send(sendArgs);
|
|
481
|
+
}
|
|
482
|
+
async react(msg, unicode) {
|
|
483
|
+
if (!this.running || !this.ws || !this.direct || !this.groups) {
|
|
484
|
+
throw new Error('MorokBot.react: not started');
|
|
485
|
+
}
|
|
486
|
+
if (!unicode) {
|
|
487
|
+
throw new Error('MorokBot.react: unicode must be a non-empty string');
|
|
488
|
+
}
|
|
489
|
+
const { messageId, conversationId } = msg;
|
|
490
|
+
if (!Number.isInteger(messageId) || messageId < 1
|
|
491
|
+
|| !Number.isInteger(conversationId) || conversationId < 1) {
|
|
492
|
+
throw new Error('MorokBot.react: msg must carry a valid messageId and conversationId');
|
|
493
|
+
}
|
|
494
|
+
const payload = JSON.stringify({ emoji: unicode });
|
|
495
|
+
if (Buffer.byteLength(payload, 'utf8') > 256) {
|
|
496
|
+
throw new Error('MorokBot.react: unicode payload exceeds 256 bytes');
|
|
497
|
+
}
|
|
498
|
+
const plaintext = new TextEncoder().encode(payload);
|
|
499
|
+
let distributions;
|
|
500
|
+
if (msg.conversationType === 'DIRECT') {
|
|
501
|
+
const dists = await this.direct.encryptReaction(msg.sender.userId, plaintext);
|
|
502
|
+
if (dists.length === 0) {
|
|
503
|
+
throw new Error(`MorokBot.react: could not encrypt to any device of peer ${msg.sender.userId}`);
|
|
504
|
+
}
|
|
505
|
+
distributions = dists;
|
|
506
|
+
}
|
|
507
|
+
else {
|
|
508
|
+
const { ciphertext } = await this.groups.encryptForChannel(conversationId, plaintext);
|
|
509
|
+
distributions = [{ ciphertext, messageType: 8 }];
|
|
510
|
+
}
|
|
511
|
+
this.ws.send({ type: 'reaction_add', messageId, distributions });
|
|
512
|
+
}
|
|
513
|
+
async unreact(msg) {
|
|
514
|
+
if (!this.running || !this.ws) {
|
|
515
|
+
throw new Error('MorokBot.unreact: not started');
|
|
516
|
+
}
|
|
517
|
+
const { messageId } = msg;
|
|
518
|
+
if (!Number.isInteger(messageId) || messageId < 1) {
|
|
519
|
+
throw new Error('MorokBot.unreact: msg must carry a valid messageId');
|
|
520
|
+
}
|
|
521
|
+
this.ws.send({ type: 'reaction_remove', messageId });
|
|
522
|
+
}
|
|
523
|
+
async resolvePeerUserId(ref) {
|
|
524
|
+
if (typeof ref === 'number') {
|
|
525
|
+
if (!Number.isInteger(ref) || ref < 1) {
|
|
526
|
+
throw new Error(`peer userId must be a positive integer, got ${ref}`);
|
|
527
|
+
}
|
|
528
|
+
return ref;
|
|
529
|
+
}
|
|
530
|
+
if (!isUsername(ref)) {
|
|
531
|
+
throw new Error('peer must be a positive integer userId or a non-empty username string');
|
|
532
|
+
}
|
|
533
|
+
const username = ref.toLowerCase();
|
|
534
|
+
const res = await this.http.get(`/users/${encodeURIComponent(username)}`);
|
|
535
|
+
if (res.data.user.isDeleted) {
|
|
536
|
+
throw new Error(`peer ${username} is a deleted account`);
|
|
537
|
+
}
|
|
538
|
+
return res.data.user.id;
|
|
539
|
+
}
|
|
540
|
+
on(event, listener) {
|
|
541
|
+
return super.on(event, listener);
|
|
542
|
+
}
|
|
543
|
+
off(event, listener) {
|
|
544
|
+
return super.off(event, listener);
|
|
545
|
+
}
|
|
546
|
+
emit(event, ...args) {
|
|
547
|
+
try {
|
|
548
|
+
return super.emit(event, ...args);
|
|
549
|
+
}
|
|
550
|
+
catch (err) {
|
|
551
|
+
if (event === 'error')
|
|
552
|
+
throw err;
|
|
553
|
+
super.emit('error', err instanceof Error ? err : new Error(String(err)));
|
|
554
|
+
return false;
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
}
|
|
558
|
+
//# sourceMappingURL=bot.js.map
|
package/dist/bot.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bot.js","sourceRoot":"","sources":["../src/bot.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AES-256-GCM wire format for group/channel messages (Signal type 8), byte-identical to frontend/src/signal/channel.ts
|
|
3
|
+
* Any drift here breaks decryption end-to-end
|
|
4
|
+
*
|
|
5
|
+
* Wire: [ "MOK1" (4) || epoch (uint32 BE, 4) || iv (12) || ct+tag ]
|
|
6
|
+
*
|
|
7
|
+
* AAD is "morok-channel-${conversationId}", conv-scoped and not epoch-scoped
|
|
8
|
+
* The epoch sits in the wire header, the receiver reads it to pick the key, then runs AES-GCM with the conv-scoped AAD
|
|
9
|
+
*
|
|
10
|
+
* Legacy wires have no MOK1 prefix and decode as epoch=0
|
|
11
|
+
* The chance a random IV's first 4 bytes spell "MOK1" is 1 / 2^32 per message
|
|
12
|
+
*/
|
|
13
|
+
export declare const CHANNEL_KEY_BYTES = 32;
|
|
14
|
+
export declare const CHANNEL_IV_BYTES = 12;
|
|
15
|
+
export declare const CHANNEL_TAG_BYTES = 16;
|
|
16
|
+
export declare const CHANNEL_MAGIC: NodeJS.NonSharedUint8Array;
|
|
17
|
+
export declare const CHANNEL_MAGIC_BYTES: number;
|
|
18
|
+
export declare const CHANNEL_EPOCH_BYTES = 4;
|
|
19
|
+
export declare const CHANNEL_HEADER_BYTES: number;
|
|
20
|
+
export declare function channelAad(conversationId: number): Uint8Array;
|
|
21
|
+
export interface ParsedChannelWire {
|
|
22
|
+
epoch: number;
|
|
23
|
+
iv: Uint8Array;
|
|
24
|
+
ct: Uint8Array;
|
|
25
|
+
}
|
|
26
|
+
/** Splits a wire into (epoch, iv, ct). Handles the multi-epoch layout and the legacy [iv || ct+tag] as epoch=0 */
|
|
27
|
+
export declare function parseChannelWire(wire: Uint8Array): ParsedChannelWire;
|
|
28
|
+
/** Encrypt under the per-epoch 32-byte secret and return the full wire envelope. IV is random per call */
|
|
29
|
+
export declare function encryptChannelWire(secret: Uint8Array, conversationId: number, epoch: number, plaintext: Uint8Array): Promise<Uint8Array>;
|
|
30
|
+
/** Decrypt a parsed wire under the given secret. Caller picks the secret based on parsed.epoch */
|
|
31
|
+
export declare function decryptChannelWire(secret: Uint8Array, conversationId: number, parsed: ParsedChannelWire): Promise<Uint8Array>;
|
|
32
|
+
//# sourceMappingURL=channel-cipher.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"channel-cipher.d.ts","sourceRoot":"","sources":["../../src/crypto/channel-cipher.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAQH,eAAO,MAAM,iBAAiB,KAAO,CAAA;AACrC,eAAO,MAAM,gBAAgB,KAAQ,CAAA;AACrC,eAAO,MAAM,iBAAiB,KAAO,CAAA;AACrC,eAAO,MAAM,aAAa,4BAAyC,CAAA;AACnE,eAAO,MAAM,mBAAmB,QAAuB,CAAA;AACvD,eAAO,MAAM,mBAAmB,IAAI,CAAA;AACpC,eAAO,MAAM,oBAAoB,QAA+D,CAAA;AAWhG,wBAAgB,UAAU,CAAC,cAAc,EAAE,MAAM,GAAG,UAAU,CAK7D;AA6BD,MAAM,WAAW,iBAAiB;IAC9B,KAAK,EAAE,MAAM,CAAA;IACb,EAAE,EAAK,UAAU,CAAA;IACjB,EAAE,EAAK,UAAU,CAAA;CACpB;AAGD,kHAAkH;AAClH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,UAAU,GAAG,iBAAiB,CAepE;AAGD,0GAA0G;AAC1G,wBAAsB,kBAAkB,CACpC,MAAM,EAAU,UAAU,EAC1B,cAAc,EAAE,MAAM,EACtB,KAAK,EAAW,MAAM,EACtB,SAAS,EAAO,UAAU,GAC3B,OAAO,CAAC,UAAU,CAAC,CAmBrB;AAGD,kGAAkG;AAClG,wBAAsB,kBAAkB,CACpC,MAAM,EAAU,UAAU,EAC1B,cAAc,EAAE,MAAM,EACtB,MAAM,EAAU,iBAAiB,GAClC,OAAO,CAAC,UAAU,CAAC,CAQrB"}
|