@gakr-gakr/whatsapp 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.
Files changed (159) hide show
  1. package/action-runtime-api.ts +1 -0
  2. package/action-runtime.runtime.ts +1 -0
  3. package/api.ts +67 -0
  4. package/auth-presence.ts +80 -0
  5. package/autobot.plugin.json +23 -0
  6. package/channel-config-api.ts +1 -0
  7. package/channel-plugin-api.ts +3 -0
  8. package/config-api.ts +4 -0
  9. package/constants.ts +1 -0
  10. package/contract-api.ts +29 -0
  11. package/directory-contract-api.ts +4 -0
  12. package/doctor-contract-api.ts +8 -0
  13. package/index.ts +16 -0
  14. package/legacy-session-surface-api.ts +6 -0
  15. package/legacy-state-migrations-api.ts +1 -0
  16. package/light-runtime-api.ts +12 -0
  17. package/login-qr-api.ts +1 -0
  18. package/login-qr-runtime.ts +23 -0
  19. package/outbound-payload-test-api.ts +1 -0
  20. package/package.json +76 -0
  21. package/runtime-api.ts +84 -0
  22. package/secret-contract-api.ts +4 -0
  23. package/security-contract-api.ts +4 -0
  24. package/setup-entry.ts +21 -0
  25. package/setup-plugin-api.ts +3 -0
  26. package/src/account-config.ts +77 -0
  27. package/src/account-ids.ts +17 -0
  28. package/src/account-types.ts +5 -0
  29. package/src/accounts.ts +176 -0
  30. package/src/action-runtime-target-auth.ts +27 -0
  31. package/src/action-runtime.ts +76 -0
  32. package/src/active-listener.ts +17 -0
  33. package/src/agent-tools-login.ts +113 -0
  34. package/src/approval-auth.ts +27 -0
  35. package/src/auth-store.runtime.ts +1 -0
  36. package/src/auth-store.ts +494 -0
  37. package/src/auto-reply/config.runtime.ts +16 -0
  38. package/src/auto-reply/constants.ts +1 -0
  39. package/src/auto-reply/deliver-reply.ts +332 -0
  40. package/src/auto-reply/loggers.ts +6 -0
  41. package/src/auto-reply/mentions.ts +131 -0
  42. package/src/auto-reply/monitor/ack-reaction.ts +99 -0
  43. package/src/auto-reply/monitor/audio-preflight.runtime.ts +9 -0
  44. package/src/auto-reply/monitor/broadcast.ts +153 -0
  45. package/src/auto-reply/monitor/commands.ts +19 -0
  46. package/src/auto-reply/monitor/echo.ts +64 -0
  47. package/src/auto-reply/monitor/group-activation.runtime.ts +1 -0
  48. package/src/auto-reply/monitor/group-activation.ts +73 -0
  49. package/src/auto-reply/monitor/group-gating.runtime.ts +8 -0
  50. package/src/auto-reply/monitor/group-gating.ts +218 -0
  51. package/src/auto-reply/monitor/group-members.ts +65 -0
  52. package/src/auto-reply/monitor/inbound-context.ts +92 -0
  53. package/src/auto-reply/monitor/inbound-dispatch.runtime.ts +22 -0
  54. package/src/auto-reply/monitor/inbound-dispatch.ts +749 -0
  55. package/src/auto-reply/monitor/last-route.ts +61 -0
  56. package/src/auto-reply/monitor/listener-log.ts +28 -0
  57. package/src/auto-reply/monitor/message-line.runtime.ts +38 -0
  58. package/src/auto-reply/monitor/message-line.ts +54 -0
  59. package/src/auto-reply/monitor/on-message.ts +333 -0
  60. package/src/auto-reply/monitor/peer.ts +17 -0
  61. package/src/auto-reply/monitor/process-message.ts +584 -0
  62. package/src/auto-reply/monitor/runtime-api.ts +36 -0
  63. package/src/auto-reply/monitor/status-reaction.ts +108 -0
  64. package/src/auto-reply/monitor-state.ts +114 -0
  65. package/src/auto-reply/monitor.ts +720 -0
  66. package/src/auto-reply/reply-resolver.runtime.ts +1 -0
  67. package/src/auto-reply/types.ts +48 -0
  68. package/src/auto-reply/util.ts +62 -0
  69. package/src/auto-reply.impl.ts +6 -0
  70. package/src/auto-reply.ts +1 -0
  71. package/src/channel-actions.runtime.ts +7 -0
  72. package/src/channel-actions.ts +85 -0
  73. package/src/channel-outbound.ts +87 -0
  74. package/src/channel-react-action.runtime.ts +10 -0
  75. package/src/channel-react-action.ts +247 -0
  76. package/src/channel.runtime.ts +117 -0
  77. package/src/channel.setup.ts +32 -0
  78. package/src/channel.ts +356 -0
  79. package/src/command-policy.ts +7 -0
  80. package/src/config-accessors.ts +22 -0
  81. package/src/config-schema.ts +6 -0
  82. package/src/config-ui-hints.ts +24 -0
  83. package/src/connection-controller-registry.ts +49 -0
  84. package/src/connection-controller.ts +680 -0
  85. package/src/creds-files.ts +19 -0
  86. package/src/creds-persistence.ts +71 -0
  87. package/src/directory-config.ts +40 -0
  88. package/src/doctor-contract.ts +11 -0
  89. package/src/doctor.ts +56 -0
  90. package/src/document-filename.ts +17 -0
  91. package/src/group-intro.ts +15 -0
  92. package/src/group-policy.ts +40 -0
  93. package/src/group-session-contract.ts +20 -0
  94. package/src/group-session-key.ts +42 -0
  95. package/src/heartbeat.ts +34 -0
  96. package/src/identity.ts +164 -0
  97. package/src/inbound/access-control.ts +187 -0
  98. package/src/inbound/dedupe.ts +132 -0
  99. package/src/inbound/extract.ts +484 -0
  100. package/src/inbound/lifecycle.ts +39 -0
  101. package/src/inbound/media.ts +128 -0
  102. package/src/inbound/monitor.ts +1042 -0
  103. package/src/inbound/outbound-mentions.ts +260 -0
  104. package/src/inbound/runtime-api.ts +7 -0
  105. package/src/inbound/save-media.runtime.ts +1 -0
  106. package/src/inbound/send-api.ts +203 -0
  107. package/src/inbound/send-result.ts +109 -0
  108. package/src/inbound/types.ts +107 -0
  109. package/src/inbound-policy.ts +215 -0
  110. package/src/inbound.ts +9 -0
  111. package/src/login-qr.ts +542 -0
  112. package/src/login.ts +83 -0
  113. package/src/media.ts +10 -0
  114. package/src/monitor-inbox.allows-messages-from-senders-allowfrom-list.test-support.ts +417 -0
  115. package/src/monitor-inbox.append-upsert.test-support.ts +133 -0
  116. package/src/monitor-inbox.blocks-messages-from-unauthorized-senders-not-allowfrom.test-support.ts +418 -0
  117. package/src/monitor-inbox.captures-media-path-image-messages.test-support.ts +308 -0
  118. package/src/monitor-inbox.streams-inbound-messages.test-support.ts +824 -0
  119. package/src/normalize-target.ts +148 -0
  120. package/src/normalize.ts +8 -0
  121. package/src/outbound-adapter.ts +36 -0
  122. package/src/outbound-base.ts +256 -0
  123. package/src/outbound-media-contract.ts +307 -0
  124. package/src/outbound-media.runtime.ts +41 -0
  125. package/src/outbound-send-deps.ts +1 -0
  126. package/src/outbound-test-support.ts +16 -0
  127. package/src/qa-driver.runtime.ts +189 -0
  128. package/src/qr-image.ts +1 -0
  129. package/src/qr-terminal.ts +1 -0
  130. package/src/quoted-message.ts +184 -0
  131. package/src/reaction-level.ts +24 -0
  132. package/src/reconnect.ts +55 -0
  133. package/src/resolve-outbound-target.ts +58 -0
  134. package/src/runtime-api.ts +59 -0
  135. package/src/runtime-group-policy.ts +16 -0
  136. package/src/runtime.ts +9 -0
  137. package/src/security-contract.ts +47 -0
  138. package/src/security-fix.ts +71 -0
  139. package/src/send.ts +342 -0
  140. package/src/session-contract.ts +43 -0
  141. package/src/session-errors.ts +125 -0
  142. package/src/session-route.ts +32 -0
  143. package/src/session.runtime.ts +8 -0
  144. package/src/session.ts +327 -0
  145. package/src/setup-core.ts +52 -0
  146. package/src/setup-finalize.ts +450 -0
  147. package/src/setup-surface.ts +71 -0
  148. package/src/setup-test-helpers.ts +217 -0
  149. package/src/shared.ts +291 -0
  150. package/src/socket-timing.ts +38 -0
  151. package/src/state-migrations.ts +55 -0
  152. package/src/status-issues.ts +185 -0
  153. package/src/system-prompt.ts +31 -0
  154. package/src/targets-runtime.ts +221 -0
  155. package/src/text-runtime.ts +18 -0
  156. package/src/vcard.ts +84 -0
  157. package/targets.ts +5 -0
  158. package/test-api.ts +2 -0
  159. package/tsconfig.json +16 -0
@@ -0,0 +1,542 @@
1
+ import { randomUUID } from "node:crypto";
2
+ import { logInfo } from "autobot/plugin-sdk/logging-core";
3
+ import { getRuntimeConfig } from "autobot/plugin-sdk/runtime-config-snapshot";
4
+ import { danger, info, success } from "autobot/plugin-sdk/runtime-env";
5
+ import { defaultRuntime, type RuntimeEnv } from "autobot/plugin-sdk/runtime-env";
6
+ import { resolveWhatsAppAccount } from "./accounts.js";
7
+ import {
8
+ closeWaSocket,
9
+ waitForWhatsAppLoginResult,
10
+ WHATSAPP_LOGGED_OUT_QR_MESSAGE,
11
+ } from "./connection-controller.js";
12
+ import { renderQrPngDataUrl } from "./qr-image.js";
13
+ import {
14
+ createWaSocket,
15
+ readWebAuthExistsForDecision,
16
+ readWebSelfId,
17
+ WHATSAPP_AUTH_UNSTABLE_CODE,
18
+ } from "./session.js";
19
+ import { resolveWhatsAppSocketTiming, type WhatsAppSocketTimingOptions } from "./socket-timing.js";
20
+
21
+ type WaSocket = Awaited<ReturnType<typeof createWaSocket>>;
22
+ export type StartWebLoginWithQrResult = {
23
+ qrDataUrl?: string;
24
+ message: string;
25
+ connected?: boolean;
26
+ code?: typeof WHATSAPP_AUTH_UNSTABLE_CODE;
27
+ };
28
+
29
+ type ActiveLogin = {
30
+ accountId: string;
31
+ authDir: string;
32
+ isLegacyAuthDir: boolean;
33
+ id: string;
34
+ sock: WaSocket;
35
+ startedAt: number;
36
+ qr?: string;
37
+ qrDataUrl?: string;
38
+ qrDataUrlVersion?: number;
39
+ qrVersion: number;
40
+ connected: boolean;
41
+ error?: string;
42
+ errorStatus?: number;
43
+ waitPromise: Promise<void>;
44
+ qrUpdatePromise: Promise<void>;
45
+ resolveQrUpdate: (() => void) | null;
46
+ qrRenderPromise: Promise<string> | null;
47
+ verbose: boolean;
48
+ runtime: RuntimeEnv;
49
+ socketTiming: WhatsAppSocketTimingOptions;
50
+ };
51
+
52
+ type LoginQrRaceResult =
53
+ | { outcome: "qr"; qr: string }
54
+ | { outcome: "connected" }
55
+ | { outcome: "failed"; message: string };
56
+
57
+ function waitForNextTask(): Promise<void> {
58
+ return new Promise((resolve) => setImmediate(resolve));
59
+ }
60
+
61
+ const ACTIVE_LOGIN_TTL_MS = 3 * 60_000;
62
+ const MAX_QR_RENDER_CHASES = 10;
63
+ const activeLogins = new Map<string, ActiveLogin>();
64
+
65
+ function closeSocket(sock: WaSocket) {
66
+ closeWaSocket(sock);
67
+ }
68
+
69
+ async function resetActiveLogin(accountId: string, reason?: string) {
70
+ const login = activeLogins.get(accountId);
71
+ if (login) {
72
+ closeSocket(login.sock);
73
+ activeLogins.delete(accountId);
74
+ }
75
+ if (reason) {
76
+ logInfo(reason);
77
+ }
78
+ }
79
+
80
+ function isLoginFresh(login: ActiveLogin) {
81
+ return Date.now() - login.startedAt < ACTIVE_LOGIN_TTL_MS;
82
+ }
83
+
84
+ function resetQrUpdateSignal(login: ActiveLogin) {
85
+ login.qrUpdatePromise = new Promise((resolve) => {
86
+ login.resolveQrUpdate = resolve;
87
+ });
88
+ }
89
+
90
+ function notifyQrUpdate(login: ActiveLogin) {
91
+ const resolve = login.resolveQrUpdate;
92
+ resetQrUpdateSignal(login);
93
+ resolve?.();
94
+ }
95
+
96
+ function updateLoginQrState(login: ActiveLogin, qr: string): number {
97
+ login.qr = qr;
98
+ login.qrVersion += 1;
99
+ return login.qrVersion;
100
+ }
101
+
102
+ async function ensureQrDataUrl(params: {
103
+ accountId: string;
104
+ loginId: string;
105
+ qr: string;
106
+ qrVersion: number;
107
+ }): Promise<string> {
108
+ const current = activeLogins.get(params.accountId);
109
+ if (
110
+ current?.id !== params.loginId ||
111
+ current.qrVersion !== params.qrVersion ||
112
+ current.qr !== params.qr
113
+ ) {
114
+ return await renderQrPngDataUrl(params.qr);
115
+ }
116
+
117
+ if (current.qrDataUrl && current.qrDataUrlVersion === params.qrVersion) {
118
+ return current.qrDataUrl;
119
+ }
120
+
121
+ if (current.qrRenderPromise) {
122
+ return await current.qrRenderPromise;
123
+ }
124
+
125
+ const renderPromise = (async () => {
126
+ for (let attempt = 0; attempt < MAX_QR_RENDER_CHASES; attempt += 1) {
127
+ const latest = activeLogins.get(params.accountId);
128
+ if (!latest || latest.id !== params.loginId || !latest.qr) {
129
+ throw new Error("WhatsApp QR is no longer active.");
130
+ }
131
+ if (latest.qrDataUrl && latest.qrDataUrlVersion === latest.qrVersion) {
132
+ return latest.qrDataUrl;
133
+ }
134
+
135
+ const qr = latest.qr;
136
+ const qrVersion = latest.qrVersion;
137
+ const dataUrl = await renderQrPngDataUrl(qr);
138
+ const refreshed = activeLogins.get(params.accountId);
139
+ if (!refreshed || refreshed.id !== params.loginId) {
140
+ return dataUrl;
141
+ }
142
+ if (refreshed.qrVersion === qrVersion && refreshed.qr === qr) {
143
+ refreshed.qrDataUrl = dataUrl;
144
+ refreshed.qrDataUrlVersion = qrVersion;
145
+ notifyQrUpdate(refreshed);
146
+ return dataUrl;
147
+ }
148
+ }
149
+
150
+ throw new Error("WhatsApp QR kept refreshing before the latest image could render.");
151
+ })();
152
+
153
+ current.qrRenderPromise = renderPromise;
154
+ try {
155
+ return await renderPromise;
156
+ } finally {
157
+ const latest = activeLogins.get(params.accountId);
158
+ if (latest?.id === params.loginId && latest.qrRenderPromise === renderPromise) {
159
+ latest.qrRenderPromise = null;
160
+ }
161
+ }
162
+ }
163
+
164
+ function renderLatestQrDataUrlInBackground(params: {
165
+ accountId: string;
166
+ loginId: string;
167
+ qr: string;
168
+ qrVersion: number;
169
+ }) {
170
+ void ensureQrDataUrl(params).catch(() => {
171
+ // Ignore background QR render failures; the caller can still retry or surface
172
+ // the login state without clobbering the active session.
173
+ });
174
+ }
175
+
176
+ function attachLoginWaiter(accountId: string, login: ActiveLogin) {
177
+ login.waitPromise = waitForWhatsAppLoginResult({
178
+ sock: login.sock,
179
+ authDir: login.authDir,
180
+ isLegacyAuthDir: login.isLegacyAuthDir,
181
+ verbose: login.verbose,
182
+ runtime: login.runtime,
183
+ socketTiming: login.socketTiming,
184
+ onQr: (qr) => {
185
+ const current = activeLogins.get(accountId);
186
+ if (!current || current.id !== login.id) {
187
+ return;
188
+ }
189
+ const qrVersion = updateLoginQrState(current, qr);
190
+ renderLatestQrDataUrlInBackground({
191
+ accountId,
192
+ loginId: login.id,
193
+ qr,
194
+ qrVersion,
195
+ });
196
+ },
197
+ onSocketReplaced: (sock) => {
198
+ const current = activeLogins.get(accountId);
199
+ if (current?.id === login.id) {
200
+ current.sock = sock;
201
+ current.connected = false;
202
+ current.error = undefined;
203
+ current.errorStatus = undefined;
204
+ }
205
+ },
206
+ })
207
+ .then((result) => {
208
+ const current = activeLogins.get(accountId);
209
+ if (current?.id !== login.id) {
210
+ return;
211
+ }
212
+ if (result.outcome === "connected") {
213
+ current.sock = result.sock;
214
+ current.connected = true;
215
+ return;
216
+ }
217
+ current.error = result.message;
218
+ current.errorStatus = result.statusCode;
219
+ })
220
+ .catch((err) => {
221
+ const current = activeLogins.get(accountId);
222
+ if (current?.id !== login.id) {
223
+ return;
224
+ }
225
+ current.error = err instanceof Error ? err.message : String(err);
226
+ current.errorStatus = undefined;
227
+ });
228
+ }
229
+
230
+ async function waitForQrOrRecoveredLogin(params: {
231
+ accountId: string;
232
+ login: ActiveLogin;
233
+ qrPromise: Promise<string>;
234
+ }): Promise<LoginQrRaceResult> {
235
+ const qrResult = params.qrPromise.then(
236
+ (qr) => ({ outcome: "qr", qr }) as const,
237
+ (err) =>
238
+ ({
239
+ outcome: "failed",
240
+ message: `Failed to get QR: ${String(err)}`,
241
+ }) as const,
242
+ );
243
+ const loginResult = params.login.waitPromise.then(async () => {
244
+ const current = activeLogins.get(params.accountId);
245
+ if (current?.id !== params.login.id) {
246
+ return {
247
+ outcome: "failed",
248
+ message: "WhatsApp login was replaced by a newer request.",
249
+ } as const;
250
+ }
251
+
252
+ // A QR may already be queued for the next task even if the login waiter won first.
253
+ await waitForNextTask();
254
+ const latest = activeLogins.get(params.accountId);
255
+ if (latest?.id !== params.login.id) {
256
+ return {
257
+ outcome: "failed",
258
+ message: "WhatsApp login was replaced by a newer request.",
259
+ } as const;
260
+ }
261
+ if (latest.qr) {
262
+ return { outcome: "qr", qr: latest.qr } as const;
263
+ }
264
+ if (latest.connected) {
265
+ return { outcome: "connected" } as const;
266
+ }
267
+ return {
268
+ outcome: "failed",
269
+ message: latest.error ? `WhatsApp login failed: ${latest.error}` : "WhatsApp login failed.",
270
+ } as const;
271
+ });
272
+
273
+ return await Promise.race([qrResult, loginResult]);
274
+ }
275
+
276
+ export async function startWebLoginWithQr(
277
+ opts: {
278
+ verbose?: boolean;
279
+ timeoutMs?: number;
280
+ force?: boolean;
281
+ accountId?: string;
282
+ runtime?: RuntimeEnv;
283
+ } = {},
284
+ ): Promise<StartWebLoginWithQrResult> {
285
+ const runtime = opts.runtime ?? defaultRuntime;
286
+ const cfg = getRuntimeConfig();
287
+ const account = resolveWhatsAppAccount({ cfg, accountId: opts.accountId });
288
+ const socketTiming = resolveWhatsAppSocketTiming(cfg);
289
+ const authState = await readWebAuthExistsForDecision(account.authDir);
290
+ if (authState.outcome === "unstable") {
291
+ return {
292
+ code: WHATSAPP_AUTH_UNSTABLE_CODE,
293
+ message: "WhatsApp auth state is still stabilizing. Retry login in a moment.",
294
+ };
295
+ }
296
+ if (authState.exists && !opts.force) {
297
+ const selfId = readWebSelfId(account.authDir);
298
+ const who = selfId.e164 ?? selfId.jid ?? "unknown";
299
+ return {
300
+ message: `WhatsApp is already linked (${who}). Say “relink” if you want a fresh QR.`,
301
+ };
302
+ }
303
+
304
+ const existing = activeLogins.get(account.accountId);
305
+ if (existing && isLoginFresh(existing) && existing.qrDataUrl) {
306
+ return {
307
+ qrDataUrl: existing.qrDataUrl,
308
+ message: "QR already active. Scan it in WhatsApp → Linked Devices.",
309
+ };
310
+ }
311
+
312
+ await resetActiveLogin(account.accountId);
313
+
314
+ let resolveQr: ((qr: string) => void) | null = null;
315
+ let rejectQr: ((err: Error) => void) | null = null;
316
+ const qrPromise = new Promise<string>((resolve, reject) => {
317
+ resolveQr = resolve;
318
+ rejectQr = reject;
319
+ });
320
+
321
+ const qrTimer = setTimeout(
322
+ () => {
323
+ rejectQr?.(new Error("Timed out waiting for WhatsApp QR"));
324
+ },
325
+ Math.max(opts.timeoutMs ?? 30_000, 5000),
326
+ );
327
+
328
+ let sock: WaSocket;
329
+ let pendingQr: string | null = null;
330
+ const loginId = randomUUID();
331
+ try {
332
+ sock = await createWaSocket(false, Boolean(opts.verbose), {
333
+ authDir: account.authDir,
334
+ ...socketTiming,
335
+ onQr: (qr: string) => {
336
+ pendingQr = qr;
337
+ const current = activeLogins.get(account.accountId);
338
+ if (current && current.id === loginId) {
339
+ const qrVersion = updateLoginQrState(current, qr);
340
+ renderLatestQrDataUrlInBackground({
341
+ accountId: account.accountId,
342
+ loginId,
343
+ qr,
344
+ qrVersion,
345
+ });
346
+ }
347
+ if (resolveQr) {
348
+ clearTimeout(qrTimer);
349
+ resolveQr(qr);
350
+ resolveQr = null;
351
+ rejectQr = null;
352
+ }
353
+ runtime.log(info("WhatsApp QR received."));
354
+ },
355
+ });
356
+ } catch (err) {
357
+ clearTimeout(qrTimer);
358
+ await resetActiveLogin(account.accountId);
359
+ return {
360
+ message: `Failed to start WhatsApp login: ${String(err)}`,
361
+ };
362
+ }
363
+ const login: ActiveLogin = {
364
+ accountId: account.accountId,
365
+ authDir: account.authDir,
366
+ isLegacyAuthDir: account.isLegacyAuthDir,
367
+ id: loginId,
368
+ sock,
369
+ startedAt: Date.now(),
370
+ connected: false,
371
+ waitPromise: Promise.resolve(),
372
+ qrVersion: 0,
373
+ qrUpdatePromise: Promise.resolve(),
374
+ resolveQrUpdate: null,
375
+ qrRenderPromise: null,
376
+ verbose: Boolean(opts.verbose),
377
+ runtime,
378
+ socketTiming,
379
+ };
380
+ resetQrUpdateSignal(login);
381
+ activeLogins.set(account.accountId, login);
382
+ if (pendingQr) {
383
+ const qrVersion = updateLoginQrState(login, pendingQr);
384
+ renderLatestQrDataUrlInBackground({
385
+ accountId: account.accountId,
386
+ loginId: login.id,
387
+ qr: pendingQr,
388
+ qrVersion,
389
+ });
390
+ }
391
+ attachLoginWaiter(account.accountId, login);
392
+
393
+ const loginStartResult = await waitForQrOrRecoveredLogin({
394
+ accountId: account.accountId,
395
+ login,
396
+ qrPromise,
397
+ });
398
+ clearTimeout(qrTimer);
399
+
400
+ if (loginStartResult.outcome === "connected") {
401
+ const selfId = readWebSelfId(account.authDir);
402
+ const who = selfId.e164 ?? selfId.jid ?? "unknown";
403
+ await resetActiveLogin(account.accountId);
404
+ return {
405
+ message: `WhatsApp recovered the existing linked session (${who}).`,
406
+ connected: true,
407
+ };
408
+ }
409
+
410
+ if (loginStartResult.outcome === "failed") {
411
+ await resetActiveLogin(account.accountId);
412
+ return {
413
+ message: loginStartResult.message,
414
+ };
415
+ }
416
+
417
+ const qr = login.qr ?? loginStartResult.qr;
418
+ const qrVersion = login.qrVersion;
419
+ if (qrVersion === 0) {
420
+ await resetActiveLogin(account.accountId);
421
+ return {
422
+ message: "Failed to capture the active WhatsApp QR. Ask me to generate a new one.",
423
+ };
424
+ }
425
+
426
+ let qrDataUrl: string;
427
+ try {
428
+ qrDataUrl = await ensureQrDataUrl({
429
+ accountId: account.accountId,
430
+ loginId: login.id,
431
+ qr,
432
+ qrVersion,
433
+ });
434
+ } catch (err) {
435
+ const message =
436
+ err instanceof Error ? `Failed to render the WhatsApp QR: ${err.message}` : String(err);
437
+ await resetActiveLogin(account.accountId, message);
438
+ return { message };
439
+ }
440
+ return {
441
+ qrDataUrl,
442
+ message: "Scan this QR in WhatsApp → Linked Devices.",
443
+ };
444
+ }
445
+
446
+ export async function waitForWebLogin(
447
+ opts: {
448
+ timeoutMs?: number;
449
+ runtime?: RuntimeEnv;
450
+ accountId?: string;
451
+ currentQrDataUrl?: string;
452
+ } = {},
453
+ ): Promise<{ connected: boolean; message: string; qrDataUrl?: string }> {
454
+ const runtime = opts.runtime ?? defaultRuntime;
455
+ const cfg = getRuntimeConfig();
456
+ const account = resolveWhatsAppAccount({ cfg, accountId: opts.accountId });
457
+ const activeLogin = activeLogins.get(account.accountId);
458
+ if (!activeLogin) {
459
+ return {
460
+ connected: false,
461
+ message: "No active WhatsApp login in progress.",
462
+ };
463
+ }
464
+
465
+ const login = activeLogin;
466
+ if (!isLoginFresh(login)) {
467
+ await resetActiveLogin(account.accountId);
468
+ return {
469
+ connected: false,
470
+ message: "The login QR expired. Ask me to generate a new one.",
471
+ };
472
+ }
473
+ const timeoutMs = Math.max(opts.timeoutMs ?? 120_000, 1000);
474
+ const deadline = Date.now() + timeoutMs;
475
+ const currentQrDataUrl = opts.currentQrDataUrl;
476
+
477
+ while (true) {
478
+ if (login.error) {
479
+ if (login.errorStatus === 401) {
480
+ const message = WHATSAPP_LOGGED_OUT_QR_MESSAGE;
481
+ await resetActiveLogin(account.accountId, message);
482
+ runtime.log(danger(message));
483
+ return { connected: false, message };
484
+ }
485
+ const message = `WhatsApp login failed: ${login.error}`;
486
+ await resetActiveLogin(account.accountId, message);
487
+ runtime.log(danger(message));
488
+ return { connected: false, message };
489
+ }
490
+
491
+ if (login.connected) {
492
+ const message = "✅ Linked! WhatsApp is ready.";
493
+ runtime.log(success(message));
494
+ await resetActiveLogin(account.accountId);
495
+ return { connected: true, message };
496
+ }
497
+
498
+ if (login.qrDataUrl && currentQrDataUrl && login.qrDataUrl !== currentQrDataUrl) {
499
+ return {
500
+ connected: false,
501
+ message: "QR refreshed. Scan the latest code in WhatsApp → Linked Devices.",
502
+ qrDataUrl: login.qrDataUrl,
503
+ };
504
+ }
505
+
506
+ const remaining = deadline - Date.now();
507
+ if (remaining <= 0) {
508
+ return {
509
+ connected: false,
510
+ message: "Still waiting for the QR scan. Let me know when you’ve scanned it.",
511
+ };
512
+ }
513
+ const timeout = new Promise<"timeout">((resolve) =>
514
+ setTimeout(() => resolve("timeout"), remaining),
515
+ );
516
+ const result = await Promise.race([
517
+ login.waitPromise.then(() => "done" as const),
518
+ login.qrUpdatePromise.then(() => "qr-update" as const),
519
+ timeout,
520
+ ]);
521
+
522
+ if (result === "timeout") {
523
+ return {
524
+ connected: false,
525
+ message: "Still waiting for the QR scan. Let me know when you’ve scanned it.",
526
+ };
527
+ }
528
+
529
+ if (result === "qr-update") {
530
+ continue;
531
+ }
532
+
533
+ if (result === "done") {
534
+ if (login.connected || login.error) {
535
+ continue;
536
+ }
537
+ return { connected: false, message: "Login ended without a connection." };
538
+ }
539
+
540
+ return { connected: false, message: "Login ended without a connection." };
541
+ }
542
+ }
package/src/login.ts ADDED
@@ -0,0 +1,83 @@
1
+ import { formatCliCommand } from "autobot/plugin-sdk/cli-runtime";
2
+ import { logInfo } from "autobot/plugin-sdk/logging-core";
3
+ import { getRuntimeConfig } from "autobot/plugin-sdk/runtime-config-snapshot";
4
+ import { danger, success } from "autobot/plugin-sdk/runtime-env";
5
+ import { defaultRuntime, type RuntimeEnv } from "autobot/plugin-sdk/runtime-env";
6
+ import { resolveWhatsAppAccount } from "./accounts.js";
7
+ import { restoreCredsFromBackupIfNeeded } from "./auth-store.js";
8
+ import { closeWaSocketSoon, waitForWhatsAppLoginResult } from "./connection-controller.js";
9
+ import { renderQrTerminal } from "./qr-terminal.js";
10
+ import { createWaSocket, waitForWaConnection } from "./session.js";
11
+ import { resolveWhatsAppSocketTiming } from "./socket-timing.js";
12
+
13
+ export async function loginWeb(
14
+ verbose: boolean,
15
+ waitForConnection?: typeof waitForWaConnection,
16
+ runtime: RuntimeEnv = defaultRuntime,
17
+ accountId?: string,
18
+ ) {
19
+ const cfg = getRuntimeConfig();
20
+ const account = resolveWhatsAppAccount({ cfg, accountId });
21
+ const socketTiming = resolveWhatsAppSocketTiming(cfg);
22
+ const restoredFromBackup = await restoreCredsFromBackupIfNeeded(account.authDir);
23
+ const onQr = (qr: string) => {
24
+ runtime.log("Open the WhatsApp app, go to Linked Devices, then scan this QR:");
25
+ void renderQrTerminal(qr)
26
+ .then((output) => {
27
+ runtime.log(output.endsWith("\n") ? output.slice(0, -1) : output);
28
+ })
29
+ .catch((err) => {
30
+ runtime.error(`failed rendering WhatsApp QR: ${String(err)}`);
31
+ });
32
+ };
33
+ let sock = await createWaSocket(false, verbose, {
34
+ authDir: account.authDir,
35
+ ...socketTiming,
36
+ onQr,
37
+ });
38
+ logInfo("Waiting for WhatsApp connection...", runtime);
39
+ try {
40
+ const result = await waitForWhatsAppLoginResult({
41
+ sock,
42
+ authDir: account.authDir,
43
+ isLegacyAuthDir: account.isLegacyAuthDir,
44
+ verbose,
45
+ runtime,
46
+ waitForConnection,
47
+ socketTiming,
48
+ onQr,
49
+ onSocketReplaced: (replacementSock) => {
50
+ sock = replacementSock;
51
+ },
52
+ });
53
+ if (result.outcome === "connected") {
54
+ runtime.log(
55
+ success(
56
+ result.restarted
57
+ ? "✅ Linked after restart; web session ready."
58
+ : restoredFromBackup
59
+ ? "✅ Recovered from creds.json.bak; web session ready."
60
+ : "✅ Linked! Credentials saved for future sends.",
61
+ ),
62
+ );
63
+ return;
64
+ }
65
+
66
+ if (result.outcome === "logged-out") {
67
+ runtime.error(
68
+ danger(
69
+ `WhatsApp reported the session is logged out. Cleared cached web session; please rerun ${formatCliCommand("autobot channels login")} and scan the QR again.`,
70
+ ),
71
+ );
72
+ throw new Error("Session logged out; cache cleared. Re-run login.", {
73
+ cause: result.error,
74
+ });
75
+ }
76
+
77
+ runtime.error(danger(`WhatsApp Web connection ended before fully opening. ${result.message}`));
78
+ throw new Error(result.message, { cause: result.error });
79
+ } finally {
80
+ // Let Baileys flush any final events before closing the socket.
81
+ closeWaSocketSoon(sock);
82
+ }
83
+ }
package/src/media.ts ADDED
@@ -0,0 +1,10 @@
1
+ export {
2
+ getDefaultLocalRoots,
3
+ LocalMediaAccessError,
4
+ loadWebMedia,
5
+ loadWebMediaRaw,
6
+ optimizeImageToJpeg,
7
+ optimizeImageToPng,
8
+ type WebMediaResult,
9
+ } from "autobot/plugin-sdk/web-media";
10
+ export type { LocalMediaAccessErrorCode } from "autobot/plugin-sdk/web-media";