@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,680 @@
1
+ import { DisconnectReason, type WASocket } from "baileys";
2
+ import { info } from "autobot/plugin-sdk/runtime-env";
3
+ import type { RuntimeEnv } from "autobot/plugin-sdk/runtime-env";
4
+ import {
5
+ registerWhatsAppConnectionController,
6
+ unregisterWhatsAppConnectionController,
7
+ } from "./connection-controller-registry.js";
8
+ import type { ActiveWebListener, WebListenerCloseReason } from "./inbound/types.js";
9
+ import { computeBackoff, sleepWithAbort, type ReconnectPolicy } from "./reconnect.js";
10
+ import {
11
+ createWaSocket,
12
+ formatError,
13
+ getStatusCode,
14
+ logoutWeb,
15
+ waitForWaConnection,
16
+ } from "./session.js";
17
+ import type { WhatsAppSocketTimingOptions } from "./socket-timing.js";
18
+
19
+ const LOGGED_OUT_STATUS = DisconnectReason?.loggedOut ?? 401;
20
+ const WHATSAPP_LOGIN_RESTART_MESSAGE =
21
+ "WhatsApp asked for a restart after pairing (code 515); waiting for creds to save…";
22
+ const WHATSAPP_LOGGED_OUT_RELINK_MESSAGE =
23
+ "WhatsApp reported the session is logged out. Cleared cached web session; please rerun autobot channels login and scan the QR again.";
24
+ export const WHATSAPP_LOGGED_OUT_QR_MESSAGE =
25
+ "WhatsApp reported the session is logged out. Cleared cached web session; please scan a new QR.";
26
+ export const WHATSAPP_WATCHDOG_TIMEOUT_ERROR = "watchdog-timeout";
27
+
28
+ type TimerHandle = ReturnType<typeof setInterval>;
29
+ type WaSocket = Awaited<ReturnType<typeof createWaSocket>>;
30
+
31
+ export type ManagedWhatsAppListener = ActiveWebListener & {
32
+ close?: () => Promise<void>;
33
+ onClose?: Promise<WebListenerCloseReason>;
34
+ signalClose?: (reason?: WebListenerCloseReason) => void;
35
+ };
36
+
37
+ type WhatsAppLiveConnection = {
38
+ connectionId: string;
39
+ startedAt: number;
40
+ sock: WASocket;
41
+ listener: ManagedWhatsAppListener;
42
+ heartbeat: TimerHandle | null;
43
+ watchdogTimer: TimerHandle | null;
44
+ lastInboundAt: number | null;
45
+ lastTransportActivityAt: number;
46
+ handledMessages: number;
47
+ unregisterUnhandled: (() => void) | null;
48
+ unregisterTransportActivity: (() => void) | null;
49
+ openedAfterRecentInbound: boolean;
50
+ backgroundTasks: Set<Promise<unknown>>;
51
+ closePromise: Promise<WebListenerCloseReason>;
52
+ resolveClose: (reason: WebListenerCloseReason) => void;
53
+ };
54
+
55
+ type WhatsAppConnectionSnapshot = {
56
+ connectionId: string;
57
+ startedAt: number;
58
+ lastInboundAt: number | null;
59
+ lastTransportActivityAt: number;
60
+ handledMessages: number;
61
+ reconnectAttempts: number;
62
+ uptimeMs: number;
63
+ };
64
+
65
+ type NormalizedConnectionCloseReason = {
66
+ statusCode?: number;
67
+ statusLabel: number | "unknown";
68
+ isLoggedOut: boolean;
69
+ error?: unknown;
70
+ errorText: string;
71
+ };
72
+
73
+ type WhatsAppConnectionCloseDecision = {
74
+ action: "stop" | "retry";
75
+ delayMs?: number;
76
+ reconnectAttempts: number;
77
+ healthState: "logged-out" | "conflict" | "stopped" | "reconnecting";
78
+ normalized: NormalizedConnectionCloseReason;
79
+ };
80
+
81
+ type WhatsAppReconnectAttemptDecision = {
82
+ action: "stop" | "retry";
83
+ delayMs?: number;
84
+ reconnectAttempts: number;
85
+ healthState: "stopped" | "reconnecting";
86
+ };
87
+
88
+ function createNeverResolvePromise<T>(): Promise<T> {
89
+ return new Promise<T>(() => {});
90
+ }
91
+
92
+ type SocketActivityEmitter = {
93
+ on?: (event: string, listener: (...args: unknown[]) => void) => void;
94
+ off?: (event: string, listener: (...args: unknown[]) => void) => void;
95
+ removeListener?: (event: string, listener: (...args: unknown[]) => void) => void;
96
+ };
97
+
98
+ function createLiveConnection(params: {
99
+ connectionId: string;
100
+ sock: WASocket;
101
+ listener: ManagedWhatsAppListener;
102
+ openedAfterRecentInbound: boolean;
103
+ }): WhatsAppLiveConnection {
104
+ let closeResolved = false;
105
+ let resolveClosePromise = (_reason: WebListenerCloseReason) => {};
106
+ const closePromise = new Promise<WebListenerCloseReason>((resolve) => {
107
+ resolveClosePromise = (reason: WebListenerCloseReason) => {
108
+ if (closeResolved) {
109
+ return;
110
+ }
111
+ closeResolved = true;
112
+ resolve(reason);
113
+ };
114
+ });
115
+
116
+ return {
117
+ connectionId: params.connectionId,
118
+ startedAt: Date.now(),
119
+ sock: params.sock,
120
+ listener: params.listener,
121
+ heartbeat: null,
122
+ watchdogTimer: null,
123
+ lastInboundAt: null,
124
+ lastTransportActivityAt: Date.now(),
125
+ handledMessages: 0,
126
+ unregisterUnhandled: null,
127
+ unregisterTransportActivity: null,
128
+ openedAfterRecentInbound: params.openedAfterRecentInbound,
129
+ backgroundTasks: new Set<Promise<unknown>>(),
130
+ closePromise,
131
+ resolveClose: resolveClosePromise,
132
+ };
133
+ }
134
+
135
+ export function closeWaSocket(
136
+ sock:
137
+ | {
138
+ end?: (error: Error | undefined) => void;
139
+ ws?: { close?: () => void };
140
+ }
141
+ | null
142
+ | undefined,
143
+ ): void {
144
+ try {
145
+ if (typeof sock?.end === "function") {
146
+ sock.end(new Error("AutoBot WhatsApp socket close"));
147
+ return;
148
+ }
149
+ sock?.ws?.close?.();
150
+ } catch {
151
+ // ignore best-effort shutdown failures
152
+ }
153
+ }
154
+
155
+ export function closeWaSocketSoon(
156
+ sock:
157
+ | {
158
+ end?: (error: Error | undefined) => void;
159
+ ws?: { close?: () => void };
160
+ }
161
+ | null
162
+ | undefined,
163
+ delayMs = 500,
164
+ ): void {
165
+ setTimeout(() => {
166
+ closeWaSocket(sock);
167
+ }, delayMs);
168
+ }
169
+
170
+ type WhatsAppLoginWaitResult =
171
+ | {
172
+ outcome: "connected";
173
+ restarted: boolean;
174
+ sock: WaSocket;
175
+ }
176
+ | {
177
+ outcome: "logged-out";
178
+ message: string;
179
+ statusCode: number;
180
+ error: unknown;
181
+ }
182
+ | {
183
+ outcome: "failed";
184
+ message: string;
185
+ statusCode?: number;
186
+ error: unknown;
187
+ };
188
+
189
+ export async function waitForWhatsAppLoginResult(params: {
190
+ sock: WaSocket;
191
+ authDir: string;
192
+ isLegacyAuthDir: boolean;
193
+ verbose: boolean;
194
+ runtime: RuntimeEnv;
195
+ waitForConnection?: typeof waitForWaConnection;
196
+ createSocket?: typeof createWaSocket;
197
+ socketTiming?: WhatsAppSocketTimingOptions;
198
+ onQr?: (qr: string) => void;
199
+ onSocketReplaced?: (sock: WaSocket) => void;
200
+ }): Promise<WhatsAppLoginWaitResult> {
201
+ const wait = params.waitForConnection ?? waitForWaConnection;
202
+ const createSocket = params.createSocket ?? createWaSocket;
203
+ let currentSock = params.sock;
204
+ let restarted = false;
205
+
206
+ while (true) {
207
+ try {
208
+ await wait(currentSock);
209
+ return {
210
+ outcome: "connected",
211
+ restarted,
212
+ sock: currentSock,
213
+ };
214
+ } catch (err) {
215
+ const statusCode = getStatusCode(err);
216
+ if (statusCode === 515 && !restarted) {
217
+ restarted = true;
218
+ params.runtime.log(info(WHATSAPP_LOGIN_RESTART_MESSAGE));
219
+ closeWaSocket(currentSock);
220
+ try {
221
+ currentSock = await createSocket(false, params.verbose, {
222
+ authDir: params.authDir,
223
+ ...params.socketTiming,
224
+ onQr: params.onQr,
225
+ });
226
+ params.onSocketReplaced?.(currentSock);
227
+ continue;
228
+ } catch (createErr) {
229
+ return {
230
+ outcome: "failed",
231
+ message: formatError(createErr),
232
+ statusCode: getStatusCode(createErr),
233
+ error: createErr,
234
+ };
235
+ }
236
+ }
237
+
238
+ if (statusCode === LOGGED_OUT_STATUS) {
239
+ await logoutWeb({
240
+ authDir: params.authDir,
241
+ isLegacyAuthDir: params.isLegacyAuthDir,
242
+ runtime: params.runtime,
243
+ });
244
+ return {
245
+ outcome: "logged-out",
246
+ message: WHATSAPP_LOGGED_OUT_RELINK_MESSAGE,
247
+ statusCode: LOGGED_OUT_STATUS,
248
+ error: err,
249
+ };
250
+ }
251
+
252
+ return {
253
+ outcome: "failed",
254
+ message: formatError(err),
255
+ statusCode,
256
+ error: err,
257
+ };
258
+ }
259
+ }
260
+ }
261
+
262
+ export class WhatsAppConnectionController {
263
+ readonly accountId: string;
264
+ readonly authDir: string;
265
+ readonly socketRef: { current: WASocket | null };
266
+
267
+ private readonly reconnectPolicy: ReconnectPolicy;
268
+ private readonly heartbeatSeconds: number;
269
+ private readonly keepAlive: boolean;
270
+ private readonly transportTimeoutMs: number;
271
+ private readonly messageTimeoutMs: number;
272
+ private readonly appSilenceTimeoutMs: number;
273
+ private readonly watchdogCheckMs: number;
274
+ private readonly verbose: boolean;
275
+ private readonly abortSignal?: AbortSignal;
276
+ private readonly sleep: (ms: number, signal?: AbortSignal) => Promise<void>;
277
+ private readonly isNonRetryableStatus: (statusCode: unknown) => boolean;
278
+ private readonly socketTiming: WhatsAppSocketTimingOptions;
279
+ private readonly abortPromise?: Promise<"aborted">;
280
+ private readonly disconnectRetryController = new AbortController();
281
+
282
+ private current: WhatsAppLiveConnection | null = null;
283
+ private reconnectAttempts = 0;
284
+ private lastHandledInboundAt: number | null = null;
285
+
286
+ constructor(params: {
287
+ accountId: string;
288
+ authDir: string;
289
+ verbose: boolean;
290
+ keepAlive: boolean;
291
+ heartbeatSeconds: number;
292
+ transportTimeoutMs: number;
293
+ messageTimeoutMs: number;
294
+ watchdogCheckMs: number;
295
+ reconnectPolicy: ReconnectPolicy;
296
+ abortSignal?: AbortSignal;
297
+ sleep?: (ms: number, signal?: AbortSignal) => Promise<void>;
298
+ isNonRetryableStatus?: (statusCode: unknown) => boolean;
299
+ socketTiming?: WhatsAppSocketTimingOptions;
300
+ }) {
301
+ this.accountId = params.accountId;
302
+ this.authDir = params.authDir;
303
+ this.verbose = params.verbose;
304
+ this.keepAlive = params.keepAlive;
305
+ this.heartbeatSeconds = params.heartbeatSeconds;
306
+ this.transportTimeoutMs = params.transportTimeoutMs;
307
+ this.messageTimeoutMs = params.messageTimeoutMs;
308
+ this.appSilenceTimeoutMs = Math.max(params.messageTimeoutMs, params.messageTimeoutMs * 4);
309
+ this.watchdogCheckMs = params.watchdogCheckMs;
310
+ this.reconnectPolicy = params.reconnectPolicy;
311
+ this.abortSignal = params.abortSignal;
312
+ this.sleep = params.sleep ?? ((ms: number, signal?: AbortSignal) => sleepWithAbort(ms, signal));
313
+ this.isNonRetryableStatus = params.isNonRetryableStatus ?? (() => false);
314
+ this.socketTiming = params.socketTiming ?? {};
315
+ this.socketRef = { current: null };
316
+ this.abortPromise =
317
+ params.abortSignal &&
318
+ new Promise<"aborted">((resolve) => {
319
+ params.abortSignal?.addEventListener("abort", () => resolve("aborted"), { once: true });
320
+ });
321
+
322
+ if (params.abortSignal?.aborted) {
323
+ this.stopDisconnectRetries();
324
+ } else {
325
+ params.abortSignal?.addEventListener("abort", () => this.stopDisconnectRetries(), {
326
+ once: true,
327
+ });
328
+ }
329
+ }
330
+
331
+ getActiveListener(): ActiveWebListener | null {
332
+ return this.current?.listener ?? null;
333
+ }
334
+
335
+ getReconnectAttempts(): number {
336
+ return this.reconnectAttempts;
337
+ }
338
+
339
+ isStopRequested(): boolean {
340
+ return this.abortSignal?.aborted === true;
341
+ }
342
+
343
+ shouldRetryDisconnect(): boolean {
344
+ return (
345
+ this.keepAlive && !this.isStopRequested() && !this.disconnectRetryController.signal.aborted
346
+ );
347
+ }
348
+
349
+ getDisconnectRetryAbortSignal(): AbortSignal {
350
+ return this.disconnectRetryController.signal;
351
+ }
352
+
353
+ noteInbound(timestamp = Date.now()): void {
354
+ if (!this.current) {
355
+ return;
356
+ }
357
+ this.current.handledMessages += 1;
358
+ this.current.lastInboundAt = timestamp;
359
+ this.current.lastTransportActivityAt = timestamp;
360
+ this.current.openedAfterRecentInbound = false;
361
+ this.lastHandledInboundAt = timestamp;
362
+ }
363
+
364
+ noteTransportActivity(timestamp = Date.now()): void {
365
+ if (!this.current) {
366
+ return;
367
+ }
368
+ this.current.lastTransportActivityAt = timestamp;
369
+ }
370
+
371
+ getCurrentSnapshot(
372
+ connection: WhatsAppLiveConnection | null = this.current,
373
+ ): WhatsAppConnectionSnapshot | null {
374
+ if (!connection) {
375
+ return null;
376
+ }
377
+ return {
378
+ connectionId: connection.connectionId,
379
+ startedAt: connection.startedAt,
380
+ lastInboundAt: connection.lastInboundAt,
381
+ lastTransportActivityAt: connection.lastTransportActivityAt,
382
+ handledMessages: connection.handledMessages,
383
+ reconnectAttempts: this.reconnectAttempts,
384
+ uptimeMs: Date.now() - connection.startedAt,
385
+ };
386
+ }
387
+
388
+ setUnhandledRejectionCleanup(unregister: (() => void) | null): void {
389
+ if (!this.current) {
390
+ unregister?.();
391
+ return;
392
+ }
393
+ this.current.unregisterUnhandled?.();
394
+ this.current.unregisterUnhandled = unregister;
395
+ }
396
+
397
+ async openConnection(params: {
398
+ connectionId: string;
399
+ createListener: (context: {
400
+ sock: WASocket;
401
+ connection: WhatsAppLiveConnection;
402
+ }) => Promise<ManagedWhatsAppListener>;
403
+ onHeartbeat?: (snapshot: WhatsAppConnectionSnapshot) => void;
404
+ onWatchdogTimeout?: (snapshot: WhatsAppConnectionSnapshot) => void;
405
+ }): Promise<WhatsAppLiveConnection> {
406
+ if (this.current) {
407
+ await this.closeCurrentConnection();
408
+ }
409
+
410
+ let sock: WaSocket | null = null;
411
+ let connection: WhatsAppLiveConnection | null = null;
412
+ try {
413
+ sock = await createWaSocket(false, this.verbose, {
414
+ authDir: this.authDir,
415
+ ...this.socketTiming,
416
+ });
417
+ await waitForWaConnection(sock);
418
+
419
+ this.socketRef.current = sock;
420
+ const placeholderListener = {} as ManagedWhatsAppListener;
421
+ connection = createLiveConnection({
422
+ connectionId: params.connectionId,
423
+ sock,
424
+ listener: placeholderListener,
425
+ openedAfterRecentInbound: this.isOpeningAfterRecentInbound(),
426
+ });
427
+ const listener = await params.createListener({ sock, connection });
428
+ connection.listener = listener;
429
+ this.current = connection;
430
+ connection.unregisterTransportActivity = this.attachTransportActivityListener(sock);
431
+ registerWhatsAppConnectionController(this.accountId, this);
432
+ this.startTimers(connection, {
433
+ onHeartbeat: params.onHeartbeat,
434
+ onWatchdogTimeout: params.onWatchdogTimeout,
435
+ });
436
+ return connection;
437
+ } catch (err) {
438
+ if (this.socketRef.current === sock) {
439
+ this.socketRef.current = null;
440
+ }
441
+ closeWaSocket(sock);
442
+ if (connection?.unregisterUnhandled) {
443
+ connection.unregisterUnhandled();
444
+ }
445
+ connection?.unregisterTransportActivity?.();
446
+ throw err;
447
+ }
448
+ }
449
+
450
+ async waitForClose(): Promise<WebListenerCloseReason | "aborted"> {
451
+ const connection = this.current;
452
+ if (!connection) {
453
+ return "aborted";
454
+ }
455
+ const listenerClose =
456
+ connection.listener.onClose?.catch((err) => ({
457
+ status: 500,
458
+ isLoggedOut: false,
459
+ error: err,
460
+ })) ?? createNeverResolvePromise<WebListenerCloseReason>();
461
+
462
+ return await Promise.race([
463
+ connection.closePromise,
464
+ listenerClose,
465
+ this.abortPromise ?? createNeverResolvePromise<"aborted">(),
466
+ ]);
467
+ }
468
+
469
+ normalizeCloseReason(reason: WebListenerCloseReason): NormalizedConnectionCloseReason {
470
+ const statusCode =
471
+ (typeof reason === "object" && reason && "status" in reason
472
+ ? (reason as { status?: number }).status
473
+ : undefined) ?? undefined;
474
+ return {
475
+ statusCode,
476
+ statusLabel: typeof statusCode === "number" ? statusCode : "unknown",
477
+ isLoggedOut:
478
+ typeof reason === "object" &&
479
+ reason !== null &&
480
+ "isLoggedOut" in reason &&
481
+ (reason as { isLoggedOut?: boolean }).isLoggedOut === true,
482
+ error: reason?.error,
483
+ errorText: formatError(reason),
484
+ };
485
+ }
486
+
487
+ resolveCloseDecision(
488
+ reason: WebListenerCloseReason | "aborted",
489
+ ): WhatsAppConnectionCloseDecision | "aborted" {
490
+ if (reason === "aborted" || this.isStopRequested()) {
491
+ return "aborted";
492
+ }
493
+
494
+ const current = this.current;
495
+ if (current && Date.now() - current.startedAt > this.heartbeatSeconds * 1000) {
496
+ this.reconnectAttempts = 0;
497
+ }
498
+
499
+ const normalized = this.normalizeCloseReason(reason);
500
+ if (normalized.isLoggedOut) {
501
+ return {
502
+ action: "stop",
503
+ reconnectAttempts: this.reconnectAttempts,
504
+ healthState: "logged-out",
505
+ normalized,
506
+ };
507
+ }
508
+
509
+ if (this.isNonRetryableStatus(normalized.statusCode)) {
510
+ return {
511
+ action: "stop",
512
+ reconnectAttempts: this.reconnectAttempts,
513
+ healthState: "conflict",
514
+ normalized,
515
+ };
516
+ }
517
+
518
+ const retryDecision = this.consumeReconnectAttempt();
519
+ if (retryDecision.action === "stop") {
520
+ return {
521
+ action: "stop",
522
+ reconnectAttempts: retryDecision.reconnectAttempts,
523
+ healthState: retryDecision.healthState,
524
+ normalized,
525
+ };
526
+ }
527
+
528
+ return {
529
+ action: "retry",
530
+ delayMs: retryDecision.delayMs,
531
+ reconnectAttempts: retryDecision.reconnectAttempts,
532
+ healthState: retryDecision.healthState,
533
+ normalized,
534
+ };
535
+ }
536
+
537
+ consumeReconnectAttempt(): WhatsAppReconnectAttemptDecision {
538
+ this.reconnectAttempts += 1;
539
+ if (
540
+ this.reconnectPolicy.maxAttempts > 0 &&
541
+ this.reconnectAttempts >= this.reconnectPolicy.maxAttempts
542
+ ) {
543
+ return {
544
+ action: "stop",
545
+ reconnectAttempts: this.reconnectAttempts,
546
+ healthState: "stopped",
547
+ };
548
+ }
549
+
550
+ return {
551
+ action: "retry",
552
+ delayMs: computeBackoff(this.reconnectPolicy, this.reconnectAttempts),
553
+ reconnectAttempts: this.reconnectAttempts,
554
+ healthState: "reconnecting",
555
+ };
556
+ }
557
+
558
+ forceClose(reason: WebListenerCloseReason): void {
559
+ const connection = this.current;
560
+ if (!connection) {
561
+ return;
562
+ }
563
+ connection.resolveClose(reason);
564
+ connection.listener.signalClose?.(reason);
565
+ }
566
+
567
+ async closeCurrentConnection(): Promise<void> {
568
+ const connection = this.current;
569
+ if (!connection) {
570
+ return;
571
+ }
572
+ this.current = null;
573
+
574
+ if (this.socketRef.current === connection.sock) {
575
+ this.socketRef.current = null;
576
+ }
577
+ connection.unregisterUnhandled?.();
578
+ connection.unregisterTransportActivity?.();
579
+ if (connection.heartbeat) {
580
+ clearInterval(connection.heartbeat);
581
+ }
582
+ if (connection.watchdogTimer) {
583
+ clearInterval(connection.watchdogTimer);
584
+ }
585
+ if (connection.backgroundTasks.size > 0) {
586
+ await Promise.allSettled(connection.backgroundTasks);
587
+ connection.backgroundTasks.clear();
588
+ }
589
+ try {
590
+ await connection.listener.close?.();
591
+ } catch {
592
+ // best-effort close
593
+ }
594
+ closeWaSocket(connection.sock);
595
+ }
596
+
597
+ async waitBeforeRetry(delayMs: number): Promise<void> {
598
+ await this.sleep(delayMs, this.abortSignal);
599
+ }
600
+
601
+ async shutdown(): Promise<void> {
602
+ this.stopDisconnectRetries();
603
+ await this.closeCurrentConnection();
604
+ unregisterWhatsAppConnectionController(this.accountId, this);
605
+ }
606
+
607
+ private startTimers(
608
+ connection: WhatsAppLiveConnection,
609
+ hooks: {
610
+ onHeartbeat?: (snapshot: WhatsAppConnectionSnapshot) => void;
611
+ onWatchdogTimeout?: (snapshot: WhatsAppConnectionSnapshot) => void;
612
+ },
613
+ ): void {
614
+ if (!this.keepAlive) {
615
+ return;
616
+ }
617
+
618
+ connection.heartbeat = setInterval(() => {
619
+ const snapshot = this.getCurrentSnapshot(connection);
620
+ if (!snapshot) {
621
+ return;
622
+ }
623
+ hooks.onHeartbeat?.(snapshot);
624
+ }, this.heartbeatSeconds * 1000);
625
+
626
+ connection.watchdogTimer = setInterval(() => {
627
+ const now = Date.now();
628
+ const transportStaleForMs = now - connection.lastTransportActivityAt;
629
+ const appBaselineAt = connection.lastInboundAt ?? connection.startedAt;
630
+ const appSilentForMs = now - appBaselineAt;
631
+ const appSilenceTimeoutMs = connection.openedAfterRecentInbound
632
+ ? this.messageTimeoutMs
633
+ : this.appSilenceTimeoutMs;
634
+ if (transportStaleForMs <= this.transportTimeoutMs && appSilentForMs <= appSilenceTimeoutMs) {
635
+ return;
636
+ }
637
+ const snapshot = this.getCurrentSnapshot(connection);
638
+ if (!snapshot) {
639
+ return;
640
+ }
641
+ hooks.onWatchdogTimeout?.(snapshot);
642
+ this.forceClose({
643
+ status: 499,
644
+ isLoggedOut: false,
645
+ error: WHATSAPP_WATCHDOG_TIMEOUT_ERROR,
646
+ });
647
+ }, this.watchdogCheckMs);
648
+ }
649
+
650
+ private attachTransportActivityListener(sock: WASocket): (() => void) | null {
651
+ const ws = sock.ws as SocketActivityEmitter | undefined;
652
+ if (!ws || typeof ws.on !== "function") {
653
+ return null;
654
+ }
655
+
656
+ const noteActivity = () => this.noteTransportActivity();
657
+ ws.on("frame", noteActivity);
658
+
659
+ return () => {
660
+ if (typeof ws.off === "function") {
661
+ ws.off("frame", noteActivity);
662
+ return;
663
+ }
664
+ ws.removeListener?.("frame", noteActivity);
665
+ };
666
+ }
667
+
668
+ private isOpeningAfterRecentInbound(): boolean {
669
+ if (this.reconnectAttempts <= 0 || this.lastHandledInboundAt === null) {
670
+ return false;
671
+ }
672
+ return Date.now() - this.lastHandledInboundAt <= this.appSilenceTimeoutMs;
673
+ }
674
+
675
+ private stopDisconnectRetries(): void {
676
+ if (!this.disconnectRetryController.signal.aborted) {
677
+ this.disconnectRetryController.abort();
678
+ }
679
+ }
680
+ }
@@ -0,0 +1,19 @@
1
+ import fsSync from "node:fs";
2
+ import path from "node:path";
3
+
4
+ export function resolveWebCredsPath(authDir: string): string {
5
+ return path.join(authDir, "creds.json");
6
+ }
7
+
8
+ export function resolveWebCredsBackupPath(authDir: string): string {
9
+ return path.join(authDir, "creds.json.bak");
10
+ }
11
+
12
+ export function hasWebCredsSync(authDir: string): boolean {
13
+ try {
14
+ const stats = fsSync.statSync(resolveWebCredsPath(authDir));
15
+ return stats.isFile() && stats.size > 1;
16
+ } catch {
17
+ return false;
18
+ }
19
+ }