@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,720 @@
1
+ import { resolveAccountEntry } from "autobot/plugin-sdk/account-core";
2
+ import { resolveInboundDebounceMs } from "autobot/plugin-sdk/channel-inbound-debounce";
3
+ import { formatCliCommand } from "autobot/plugin-sdk/cli-runtime";
4
+ import { hasControlCommand } from "autobot/plugin-sdk/command-detection";
5
+ import { drainPendingDeliveries } from "autobot/plugin-sdk/delivery-queue-runtime";
6
+ import { DEFAULT_GROUP_HISTORY_LIMIT } from "autobot/plugin-sdk/reply-history";
7
+ import { resolveAgentRoute } from "autobot/plugin-sdk/routing";
8
+ import { logVerbose } from "autobot/plugin-sdk/runtime-env";
9
+ import { registerUnhandledRejectionHandler } from "autobot/plugin-sdk/runtime-env";
10
+ import { getChildLogger } from "autobot/plugin-sdk/runtime-env";
11
+ import {
12
+ defaultRuntime,
13
+ formatDurationPrecise,
14
+ warn,
15
+ type RuntimeEnv,
16
+ } from "autobot/plugin-sdk/runtime-env";
17
+ import { enqueueSystemEvent } from "autobot/plugin-sdk/system-event-runtime";
18
+ import { resolveWhatsAppAccount, resolveWhatsAppMediaMaxBytes } from "../accounts.js";
19
+ import { WHATSAPP_AUTH_UNSTABLE_CODE, WhatsAppAuthUnstableError } from "../auth-store.js";
20
+ import {
21
+ WhatsAppConnectionController,
22
+ WHATSAPP_WATCHDOG_TIMEOUT_ERROR,
23
+ type ManagedWhatsAppListener,
24
+ } from "../connection-controller.js";
25
+ import { resolveWhatsAppInboundPolicy } from "../inbound-policy.js";
26
+ import { attachWebInboxToSocket, type WhatsAppGroupMetadataCache } from "../inbound/monitor.js";
27
+ import {
28
+ newConnectionId,
29
+ resolveHeartbeatSeconds,
30
+ resolveReconnectPolicy,
31
+ sleepWithAbort,
32
+ } from "../reconnect.js";
33
+ import {
34
+ formatError,
35
+ getStatusCode,
36
+ getWebAuthAgeMs,
37
+ logoutWeb,
38
+ readWebSelfId,
39
+ } from "../session.js";
40
+ import { resolveWhatsAppSocketTiming } from "../socket-timing.js";
41
+ import { getRuntimeConfig, getRuntimeConfigSourceSnapshot } from "./config.runtime.js";
42
+ import { whatsappHeartbeatLog, whatsappLog } from "./loggers.js";
43
+ import { buildMentionConfig } from "./mentions.js";
44
+ import { createWebChannelStatusController } from "./monitor-state.js";
45
+ import { createEchoTracker } from "./monitor/echo.js";
46
+ import { formatWhatsAppInboundListeningLog } from "./monitor/listener-log.js";
47
+ import { createWebOnMessageHandler } from "./monitor/on-message.js";
48
+ import type { WebInboundMsg, WebMonitorTuning } from "./types.js";
49
+ import { isLikelyWhatsAppCryptoError } from "./util.js";
50
+
51
+ function isNonRetryableWebCloseStatus(statusCode: unknown): boolean {
52
+ // WhatsApp 440 = session conflict ("Unknown Stream Errored (conflict)").
53
+ // This is persistent until the operator resolves the conflicting session.
54
+ // Baileys 428 = DisconnectReason.connectionClosed, a generic WebSocket close
55
+ // that is often transient and must stay on the reconnect path.
56
+ return statusCode === 440;
57
+ }
58
+
59
+ type ReplyResolver = typeof import("./reply-resolver.runtime.js").getReplyFromConfig;
60
+ type WhatsAppRuntimeConfig = ReturnType<typeof getRuntimeConfig>;
61
+
62
+ let replyResolverRuntimePromise: Promise<typeof import("./reply-resolver.runtime.js")> | null =
63
+ null;
64
+
65
+ function loadReplyResolverRuntime() {
66
+ replyResolverRuntimePromise ??= import("./reply-resolver.runtime.js");
67
+ return replyResolverRuntimePromise;
68
+ }
69
+
70
+ function resolveWebMonitorConfigSnapshot(params: {
71
+ cfg: WhatsAppRuntimeConfig;
72
+ accountId?: string | null;
73
+ }): {
74
+ cfg: WhatsAppRuntimeConfig;
75
+ account: ReturnType<typeof resolveWhatsAppAccount>;
76
+ } {
77
+ const account = resolveWhatsAppAccount({
78
+ cfg: params.cfg,
79
+ accountId: params.accountId,
80
+ });
81
+ const cfg = {
82
+ ...params.cfg,
83
+ channels: {
84
+ ...params.cfg.channels,
85
+ whatsapp: {
86
+ ...params.cfg.channels?.whatsapp,
87
+ ackReaction: account.ackReaction,
88
+ messagePrefix: account.messagePrefix,
89
+ allowFrom: account.allowFrom,
90
+ groupAllowFrom: account.groupAllowFrom,
91
+ groupPolicy: account.groupPolicy,
92
+ textChunkLimit: account.textChunkLimit,
93
+ chunkMode: account.chunkMode,
94
+ mediaMaxMb: account.mediaMaxMb,
95
+ blockStreaming: account.blockStreaming,
96
+ groups: account.groups,
97
+ },
98
+ },
99
+ } satisfies WhatsAppRuntimeConfig;
100
+ return { cfg, account };
101
+ }
102
+
103
+ function normalizeReconnectAccountId(accountId?: string | null): string {
104
+ return (accountId ?? "").trim() || "default";
105
+ }
106
+
107
+ function isNoListenerReconnectError(lastError?: string): boolean {
108
+ return typeof lastError === "string" && /No active WhatsApp Web listener/i.test(lastError);
109
+ }
110
+
111
+ function resolveExplicitWhatsAppDebounceOverride(params: {
112
+ cfg: ReturnType<typeof getRuntimeConfig>;
113
+ sourceCfg?: ReturnType<typeof getRuntimeConfig> | null;
114
+ accountId: string;
115
+ }): number | undefined {
116
+ const channel = params.sourceCfg?.channels?.whatsapp;
117
+ if (!channel) {
118
+ return undefined;
119
+ }
120
+
121
+ const accountId = normalizeReconnectAccountId(params.accountId);
122
+ const accountDebounce = resolveAccountEntry(channel.accounts, accountId)?.debounceMs;
123
+ if (accountDebounce !== undefined) {
124
+ return accountDebounce;
125
+ }
126
+ if (accountId !== "default") {
127
+ const defaultAccountDebounce = resolveAccountEntry(channel.accounts, "default")?.debounceMs;
128
+ if (defaultAccountDebounce !== undefined) {
129
+ return defaultAccountDebounce;
130
+ }
131
+ }
132
+
133
+ return channel.debounceMs;
134
+ }
135
+
136
+ function isRetryableAuthUnstableError(error: unknown): error is WhatsAppAuthUnstableError {
137
+ return (
138
+ error instanceof WhatsAppAuthUnstableError ||
139
+ (typeof error === "object" &&
140
+ error !== null &&
141
+ "code" in error &&
142
+ (error as { code?: unknown }).code === WHATSAPP_AUTH_UNSTABLE_CODE)
143
+ );
144
+ }
145
+
146
+ async function clearTerminalWebAuthState(params: {
147
+ account: ReturnType<typeof resolveWhatsAppAccount>;
148
+ runtime: RuntimeEnv;
149
+ statusLabel: number | "unknown";
150
+ healthState: "logged-out" | "conflict";
151
+ log: ReturnType<typeof getChildLogger>;
152
+ }) {
153
+ try {
154
+ const cleared = await logoutWeb({
155
+ authDir: params.account.authDir,
156
+ isLegacyAuthDir: params.account.isLegacyAuthDir,
157
+ runtime: params.runtime,
158
+ });
159
+ params.log.warn(
160
+ {
161
+ accountId: params.account.accountId,
162
+ cleared,
163
+ healthState: params.healthState,
164
+ status: params.statusLabel,
165
+ },
166
+ "web reconnect: cleared cached auth after terminal close",
167
+ );
168
+ } catch (error) {
169
+ params.log.warn(
170
+ {
171
+ accountId: params.account.accountId,
172
+ error: formatError(error),
173
+ healthState: params.healthState,
174
+ status: params.statusLabel,
175
+ },
176
+ "web reconnect: failed clearing cached auth after terminal close",
177
+ );
178
+ params.runtime.error(
179
+ `WhatsApp Web cleanup failed after terminal close (status ${params.statusLabel}). Run \`${formatCliCommand("autobot channels logout --channel whatsapp")}\`, then relink with \`${formatCliCommand("autobot channels login --channel whatsapp")}\`.`,
180
+ );
181
+ }
182
+ }
183
+ const DEFAULT_TRANSPORT_TIMEOUT_MS = 5 * 60 * 1000;
184
+
185
+ export async function monitorWebChannel(
186
+ verbose: boolean,
187
+ listenerFactory: typeof attachWebInboxToSocket | undefined = attachWebInboxToSocket,
188
+ keepAlive = true,
189
+ replyResolver?: ReplyResolver,
190
+ runtime: RuntimeEnv = defaultRuntime,
191
+ abortSignal?: AbortSignal,
192
+ tuning: WebMonitorTuning = {},
193
+ ) {
194
+ const activeReplyResolver =
195
+ replyResolver ?? (await loadReplyResolverRuntime()).getReplyFromConfig;
196
+ const runId = newConnectionId();
197
+ const replyLogger = getChildLogger({ module: "web-auto-reply", runId });
198
+ const heartbeatLogger = getChildLogger({ module: "web-heartbeat", runId });
199
+ const reconnectLogger = getChildLogger({ module: "web-reconnect", runId });
200
+ const statusController = createWebChannelStatusController(tuning.statusSink);
201
+ statusController.emit();
202
+
203
+ const baseCfg = getRuntimeConfig();
204
+ const sourceCfg = getRuntimeConfigSourceSnapshot();
205
+ const { cfg, account } = resolveWebMonitorConfigSnapshot({
206
+ cfg: baseCfg,
207
+ accountId: tuning.accountId,
208
+ });
209
+ const loadCurrentMonitorConfig = () =>
210
+ resolveWebMonitorConfigSnapshot({
211
+ cfg: getRuntimeConfig(),
212
+ accountId: account.accountId,
213
+ }).cfg;
214
+
215
+ const maxMediaBytes = resolveWhatsAppMediaMaxBytes(account);
216
+ const heartbeatSeconds = resolveHeartbeatSeconds(cfg, tuning.heartbeatSeconds);
217
+ const reconnectPolicy = resolveReconnectPolicy(cfg, tuning.reconnect);
218
+ const socketTiming = resolveWhatsAppSocketTiming(cfg, tuning.socketTiming);
219
+ const baseMentionConfig = buildMentionConfig(cfg);
220
+ const groupHistoryLimit =
221
+ account.historyLimit ??
222
+ cfg.channels?.whatsapp?.historyLimit ??
223
+ cfg.messages?.groupChat?.historyLimit ??
224
+ DEFAULT_GROUP_HISTORY_LIMIT;
225
+ const groupHistories = new Map<
226
+ string,
227
+ Array<{
228
+ sender: string;
229
+ body: string;
230
+ timestamp?: number;
231
+ id?: string;
232
+ senderJid?: string;
233
+ }>
234
+ >();
235
+ const groupMemberNames = new Map<string, Map<string, string>>();
236
+ const groupMetadataCache: WhatsAppGroupMetadataCache = new Map();
237
+ const echoTracker = createEchoTracker({ maxItems: 100, logVerbose });
238
+
239
+ const sleep =
240
+ tuning.sleep ??
241
+ ((ms: number, signal?: AbortSignal) => sleepWithAbort(ms, signal ?? abortSignal));
242
+ const stopRequested = () => abortSignal?.aborted === true;
243
+
244
+ // Avoid noisy MaxListenersExceeded warnings in test environments where
245
+ // multiple gateway instances may be constructed.
246
+ const currentMaxListeners = process.getMaxListeners?.() ?? 10;
247
+ if (process.setMaxListeners && currentMaxListeners < 50) {
248
+ process.setMaxListeners(50);
249
+ }
250
+
251
+ let sigintStop = false;
252
+ const handleSigint = () => {
253
+ sigintStop = true;
254
+ };
255
+ process.once("SIGINT", handleSigint);
256
+
257
+ const transportTimeoutMs = tuning.transportTimeoutMs ?? DEFAULT_TRANSPORT_TIMEOUT_MS;
258
+ const messageTimeoutMs = tuning.messageTimeoutMs ?? 30 * 60 * 1000;
259
+ const watchdogCheckMs = tuning.watchdogCheckMs ?? 60 * 1000;
260
+ const controller = new WhatsAppConnectionController({
261
+ accountId: account.accountId,
262
+ authDir: account.authDir,
263
+ verbose,
264
+ keepAlive,
265
+ heartbeatSeconds,
266
+ transportTimeoutMs,
267
+ messageTimeoutMs,
268
+ watchdogCheckMs,
269
+ reconnectPolicy,
270
+ socketTiming,
271
+ abortSignal,
272
+ sleep,
273
+ isNonRetryableStatus: isNonRetryableWebCloseStatus,
274
+ });
275
+
276
+ try {
277
+ while (true) {
278
+ if (stopRequested()) {
279
+ break;
280
+ }
281
+
282
+ const connectionId = newConnectionId();
283
+ const inboundDebounceMs = resolveInboundDebounceMs({
284
+ cfg,
285
+ channel: "whatsapp",
286
+ overrideMs: resolveExplicitWhatsAppDebounceOverride({
287
+ cfg,
288
+ sourceCfg,
289
+ accountId: account.accountId,
290
+ }),
291
+ });
292
+ const shouldDebounce = (msg: WebInboundMsg) => {
293
+ if (msg.mediaPath || msg.mediaType) {
294
+ return false;
295
+ }
296
+ if (msg.location) {
297
+ return false;
298
+ }
299
+ if (msg.replyToId || msg.replyToBody) {
300
+ return false;
301
+ }
302
+ return !hasControlCommand(msg.body, cfg);
303
+ };
304
+
305
+ let connection;
306
+ try {
307
+ connection = await controller.openConnection({
308
+ connectionId,
309
+ createListener: async ({ sock, connection }) => {
310
+ const onMessage = createWebOnMessageHandler({
311
+ cfg,
312
+ loadConfig: loadCurrentMonitorConfig,
313
+ verbose,
314
+ connectionId,
315
+ maxMediaBytes,
316
+ groupHistoryLimit,
317
+ groupHistories,
318
+ groupMemberNames,
319
+ echoTracker,
320
+ backgroundTasks: connection.backgroundTasks,
321
+ replyResolver: activeReplyResolver,
322
+ replyLogger,
323
+ baseMentionConfig,
324
+ account,
325
+ });
326
+
327
+ return (await (listenerFactory ?? attachWebInboxToSocket)({
328
+ cfg,
329
+ loadConfig: loadCurrentMonitorConfig,
330
+ verbose,
331
+ accountId: account.accountId,
332
+ authDir: account.authDir,
333
+ mediaMaxMb: account.mediaMaxMb,
334
+ selfChatMode: account.selfChatMode,
335
+ sendReadReceipts: account.sendReadReceipts,
336
+ debounceMs: inboundDebounceMs,
337
+ shouldDebounce,
338
+ socketRef: controller.socketRef,
339
+ shouldRetryDisconnect: () => !sigintStop && controller.shouldRetryDisconnect(),
340
+ disconnectRetryPolicy: reconnectPolicy,
341
+ disconnectRetryAbortSignal: controller.getDisconnectRetryAbortSignal(),
342
+ groupMetadataCache,
343
+ onMessage: async (msg: WebInboundMsg) => {
344
+ const inboundAt = Date.now();
345
+ controller.noteInbound(inboundAt);
346
+ statusController.noteInbound(inboundAt);
347
+ await onMessage(msg);
348
+ },
349
+ sock,
350
+ })) as ManagedWhatsAppListener;
351
+ },
352
+ onHeartbeat: (snapshot) => {
353
+ const authAgeMs = getWebAuthAgeMs(account.authDir);
354
+ const minutesSinceLastMessage = snapshot.lastInboundAt
355
+ ? Math.floor((Date.now() - snapshot.lastInboundAt) / 60000)
356
+ : null;
357
+
358
+ const logData = {
359
+ connectionId: snapshot.connectionId,
360
+ reconnectAttempts: snapshot.reconnectAttempts,
361
+ messagesHandled: snapshot.handledMessages,
362
+ lastInboundAt: snapshot.lastInboundAt,
363
+ lastTransportActivityAt: snapshot.lastTransportActivityAt,
364
+ authAgeMs,
365
+ uptimeMs: snapshot.uptimeMs,
366
+ ...(minutesSinceLastMessage !== null && minutesSinceLastMessage > 30
367
+ ? { minutesSinceLastMessage }
368
+ : {}),
369
+ };
370
+ statusController.noteTransportActivity(snapshot.lastTransportActivityAt);
371
+
372
+ if (minutesSinceLastMessage && minutesSinceLastMessage > 30) {
373
+ heartbeatLogger.warn(
374
+ logData,
375
+ "⚠️ web gateway heartbeat - no messages in 30+ minutes",
376
+ );
377
+ } else {
378
+ heartbeatLogger.info(logData, "web gateway heartbeat");
379
+ }
380
+ },
381
+ onWatchdogTimeout: (snapshot) => {
382
+ const now = Date.now();
383
+ const transportSilentMs = now - snapshot.lastTransportActivityAt;
384
+ const appBaselineAt = snapshot.lastInboundAt ?? snapshot.startedAt;
385
+ const minutesSinceTransportActivity = Math.floor(transportSilentMs / 60000);
386
+ const minutesSinceAppActivity = Math.floor((now - appBaselineAt) / 60000);
387
+ const watchdogReason =
388
+ transportSilentMs > transportTimeoutMs ? "transport-inactive" : "app-silent";
389
+ statusController.noteWatchdogStale();
390
+ heartbeatLogger.warn(
391
+ {
392
+ connectionId: snapshot.connectionId,
393
+ watchdogReason,
394
+ minutesSinceTransportActivity,
395
+ minutesSinceAppActivity,
396
+ lastInboundAt: snapshot.lastInboundAt ? new Date(snapshot.lastInboundAt) : null,
397
+ lastTransportActivityAt: new Date(snapshot.lastTransportActivityAt),
398
+ messagesHandled: snapshot.handledMessages,
399
+ },
400
+ "WhatsApp watchdog timeout detected - forcing reconnect",
401
+ );
402
+ whatsappHeartbeatLog.warn(
403
+ `WhatsApp watchdog timeout (${watchdogReason}) - restarting connection`,
404
+ );
405
+ },
406
+ });
407
+ } catch (error) {
408
+ if (getStatusCode(error) === 428) {
409
+ const retryDecision = controller.consumeReconnectAttempt();
410
+ statusController.noteReconnectAttempts(retryDecision.reconnectAttempts);
411
+ statusController.noteClose({
412
+ statusCode: 428,
413
+ error: formatError(error),
414
+ reconnectAttempts: retryDecision.reconnectAttempts,
415
+ healthState: retryDecision.healthState,
416
+ });
417
+ if (retryDecision.action === "stop") {
418
+ reconnectLogger.warn(
419
+ {
420
+ connectionId,
421
+ status: 428,
422
+ reconnectAttempts: retryDecision.reconnectAttempts,
423
+ maxAttempts: reconnectPolicy.maxAttempts,
424
+ },
425
+ "web reconnect: 428 during opening; max attempts reached",
426
+ );
427
+ runtime.error(
428
+ `WhatsApp Web connection closed during setup (status 428) after ${retryDecision.reconnectAttempts}/${reconnectPolicy.maxAttempts} attempts. Relink with \`${formatCliCommand("autobot channels login --channel whatsapp")}\` if the issue persists.`,
429
+ );
430
+ await controller.shutdown();
431
+ break;
432
+ }
433
+ reconnectLogger.info(
434
+ {
435
+ connectionId,
436
+ status: 428,
437
+ reconnectAttempts: retryDecision.reconnectAttempts,
438
+ delayMs: retryDecision.delayMs,
439
+ },
440
+ "web reconnect: 428 during opening; retrying",
441
+ );
442
+ runtime.error(
443
+ `WhatsApp Web connection closed during setup (status 428). Retry ${retryDecision.reconnectAttempts}/${reconnectPolicy.maxAttempts || "∞"} in ${formatDurationPrecise(retryDecision.delayMs ?? 0)}.`,
444
+ );
445
+ try {
446
+ await controller.waitBeforeRetry(retryDecision.delayMs ?? 0);
447
+ } catch {
448
+ break;
449
+ }
450
+ continue;
451
+ }
452
+ if (!isRetryableAuthUnstableError(error)) {
453
+ throw error;
454
+ }
455
+ const retryDecision = controller.consumeReconnectAttempt();
456
+ statusController.noteReconnectAttempts(retryDecision.reconnectAttempts);
457
+ statusController.noteClose({
458
+ error: error.message,
459
+ reconnectAttempts: retryDecision.reconnectAttempts,
460
+ healthState: retryDecision.healthState,
461
+ });
462
+ if (retryDecision.action === "stop") {
463
+ reconnectLogger.warn(
464
+ {
465
+ connectionId,
466
+ reconnectAttempts: retryDecision.reconnectAttempts,
467
+ maxAttempts: reconnectPolicy.maxAttempts,
468
+ },
469
+ "web reconnect: auth state stayed unstable; max attempts reached",
470
+ );
471
+ runtime.error(
472
+ `WhatsApp auth state is still stabilizing after ${retryDecision.reconnectAttempts}/${reconnectPolicy.maxAttempts} attempts. Stopping web monitoring.`,
473
+ );
474
+ await controller.shutdown();
475
+ break;
476
+ }
477
+ reconnectLogger.info(
478
+ {
479
+ connectionId,
480
+ reconnectAttempts: retryDecision.reconnectAttempts,
481
+ delayMs: retryDecision.delayMs,
482
+ },
483
+ "web reconnect: auth state still stabilizing during inbox attach; retrying",
484
+ );
485
+ runtime.error(
486
+ `WhatsApp auth state is still stabilizing. Retry ${retryDecision.reconnectAttempts}/${reconnectPolicy.maxAttempts || "∞"} for inbox attach in ${formatDurationPrecise(retryDecision.delayMs ?? 0)}.`,
487
+ );
488
+ try {
489
+ await controller.waitBeforeRetry(retryDecision.delayMs ?? 0);
490
+ } catch {
491
+ break;
492
+ }
493
+ continue;
494
+ }
495
+
496
+ statusController.noteConnected();
497
+ controller.setUnhandledRejectionCleanup(
498
+ registerUnhandledRejectionHandler((reason) => {
499
+ if (!isLikelyWhatsAppCryptoError(reason)) {
500
+ return false;
501
+ }
502
+ const errorStr = formatError(reason);
503
+ reconnectLogger.warn(
504
+ { connectionId: connection.connectionId, error: errorStr },
505
+ "web reconnect: unhandled rejection from WhatsApp socket; forcing reconnect",
506
+ );
507
+ controller.forceClose({
508
+ status: 499,
509
+ isLoggedOut: false,
510
+ error: reason,
511
+ });
512
+ return true;
513
+ }),
514
+ );
515
+
516
+ const { e164: selfE164 } = readWebSelfId(account.authDir);
517
+ const connectRoute = resolveAgentRoute({
518
+ cfg,
519
+ channel: "whatsapp",
520
+ accountId: account.accountId,
521
+ });
522
+ enqueueSystemEvent(`WhatsApp gateway connected${selfE164 ? ` as ${selfE164}` : ""}.`, {
523
+ sessionKey: connectRoute.sessionKey,
524
+ trusted: true,
525
+ });
526
+
527
+ const normalizedAccountId = normalizeReconnectAccountId(account.accountId);
528
+ void drainPendingDeliveries({
529
+ drainKey: `whatsapp:${normalizedAccountId}`,
530
+ logLabel: "WhatsApp reconnect drain",
531
+ cfg,
532
+ log: reconnectLogger,
533
+ selectEntry: (entry) => ({
534
+ match:
535
+ entry.channel === "whatsapp" &&
536
+ normalizeReconnectAccountId(entry.accountId) === normalizedAccountId,
537
+ bypassBackoff: isNoListenerReconnectError(entry.lastError),
538
+ }),
539
+ }).catch((err) => {
540
+ reconnectLogger.warn(
541
+ { connectionId: connection.connectionId, error: String(err) },
542
+ "reconnect drain failed",
543
+ );
544
+ });
545
+
546
+ const periodicDrainInterval = setInterval(() => {
547
+ void drainPendingDeliveries({
548
+ drainKey: `whatsapp:${normalizedAccountId}`,
549
+ logLabel: "WhatsApp periodic drain",
550
+ cfg,
551
+ log: reconnectLogger,
552
+ selectEntry: (entry) => ({
553
+ match:
554
+ entry.channel === "whatsapp" &&
555
+ normalizeReconnectAccountId(entry.accountId) === normalizedAccountId,
556
+ bypassBackoff: false,
557
+ }),
558
+ }).catch((err) => {
559
+ reconnectLogger.warn(
560
+ { connectionId: connection.connectionId, error: String(err) },
561
+ "periodic drain failed",
562
+ );
563
+ });
564
+ }, 30_000);
565
+
566
+ const inboundPolicy = resolveWhatsAppInboundPolicy({
567
+ cfg,
568
+ accountId: account.accountId,
569
+ selfE164: selfE164 ?? null,
570
+ });
571
+ whatsappLog.info(
572
+ formatWhatsAppInboundListeningLog({
573
+ groups: inboundPolicy.account.groups,
574
+ groupPolicy: inboundPolicy.groupPolicy,
575
+ hasGroupAllowFrom: inboundPolicy.groupAllowFrom.length > 0,
576
+ }),
577
+ );
578
+ if (process.stdout.isTTY || process.stderr.isTTY) {
579
+ whatsappLog.raw("Ctrl+C to stop.");
580
+ }
581
+
582
+ if (!keepAlive) {
583
+ clearInterval(periodicDrainInterval);
584
+ await controller.shutdown();
585
+ return;
586
+ }
587
+
588
+ const reason = await controller
589
+ .waitForClose()
590
+ .finally(() => clearInterval(periodicDrainInterval));
591
+ if (stopRequested() || sigintStop || reason === "aborted") {
592
+ await controller.shutdown();
593
+ break;
594
+ }
595
+
596
+ const decision = controller.resolveCloseDecision(reason);
597
+ if (decision === "aborted") {
598
+ await controller.shutdown();
599
+ break;
600
+ }
601
+ statusController.noteReconnectAttempts(controller.getReconnectAttempts());
602
+
603
+ reconnectLogger.info(
604
+ {
605
+ connectionId: connection.connectionId,
606
+ status: decision.normalized.statusLabel,
607
+ loggedOut: decision.normalized.isLoggedOut,
608
+ reconnectAttempts: decision.reconnectAttempts,
609
+ error: decision.normalized.errorText,
610
+ },
611
+ "web reconnect: connection closed",
612
+ );
613
+
614
+ enqueueSystemEvent(
615
+ `WhatsApp gateway disconnected (status ${decision.normalized.statusLabel})`,
616
+ {
617
+ sessionKey: connectRoute.sessionKey,
618
+ trusted: true,
619
+ },
620
+ );
621
+
622
+ if (decision.action === "stop") {
623
+ await controller.closeCurrentConnection();
624
+ statusController.noteClose({
625
+ statusCode: decision.normalized.statusCode,
626
+ loggedOut: decision.normalized.isLoggedOut,
627
+ error: decision.normalized.errorText,
628
+ reconnectAttempts: decision.reconnectAttempts,
629
+ healthState: decision.healthState,
630
+ });
631
+
632
+ if (decision.healthState === "logged-out") {
633
+ await clearTerminalWebAuthState({
634
+ account,
635
+ runtime,
636
+ statusLabel: decision.normalized.statusLabel,
637
+ healthState: decision.healthState,
638
+ log: reconnectLogger,
639
+ });
640
+ runtime.error(
641
+ `WhatsApp session logged out. Run \`${formatCliCommand("autobot channels login --channel whatsapp")}\` to relink.`,
642
+ );
643
+ } else if (decision.healthState === "conflict") {
644
+ await clearTerminalWebAuthState({
645
+ account,
646
+ runtime,
647
+ statusLabel: decision.normalized.statusLabel,
648
+ healthState: decision.healthState,
649
+ log: reconnectLogger,
650
+ });
651
+ reconnectLogger.warn(
652
+ {
653
+ connectionId: connection.connectionId,
654
+ status: decision.normalized.statusLabel,
655
+ error: decision.normalized.errorText,
656
+ },
657
+ "web reconnect: non-retryable close status; stopping monitor",
658
+ );
659
+ runtime.error(
660
+ `WhatsApp Web connection closed (status ${decision.normalized.statusLabel}: session conflict). Resolve conflicting WhatsApp Web sessions, then relink with \`${formatCliCommand("autobot channels login --channel whatsapp")}\`. Stopping web monitoring.`,
661
+ );
662
+ } else {
663
+ reconnectLogger.warn(
664
+ {
665
+ connectionId: connection.connectionId,
666
+ status: decision.normalized.statusLabel,
667
+ reconnectAttempts: decision.reconnectAttempts,
668
+ maxAttempts: reconnectPolicy.maxAttempts,
669
+ },
670
+ "web reconnect: max attempts reached; continuing in degraded mode",
671
+ );
672
+ runtime.error(
673
+ `WhatsApp Web reconnect: max attempts reached (${decision.reconnectAttempts}/${reconnectPolicy.maxAttempts}). Stopping web monitoring.`,
674
+ );
675
+ }
676
+
677
+ await controller.shutdown();
678
+ break;
679
+ }
680
+
681
+ const isWatchdogRecoveryReconnect =
682
+ decision.normalized.error === WHATSAPP_WATCHDOG_TIMEOUT_ERROR;
683
+ statusController.noteClose({
684
+ statusCode: decision.normalized.statusCode,
685
+ error: decision.normalized.errorText,
686
+ reconnectAttempts: decision.reconnectAttempts,
687
+ healthState: decision.healthState,
688
+ watchdogRecovery: isWatchdogRecoveryReconnect,
689
+ });
690
+ reconnectLogger.info(
691
+ {
692
+ connectionId: connection.connectionId,
693
+ status: decision.normalized.statusLabel,
694
+ reconnectAttempts: decision.reconnectAttempts,
695
+ maxAttempts: reconnectPolicy.maxAttempts || "unlimited",
696
+ delayMs: decision.delayMs,
697
+ },
698
+ "web reconnect: scheduling retry",
699
+ );
700
+ const reconnectMessage = isWatchdogRecoveryReconnect
701
+ ? `WhatsApp Web watchdog is recovering a stale connection (status ${decision.normalized.statusLabel}). Retry ${decision.reconnectAttempts}/${reconnectPolicy.maxAttempts || "∞"} in ${formatDurationPrecise(decision.delayMs ?? 0)}.`
702
+ : `WhatsApp Web connection closed (status ${decision.normalized.statusLabel}). Retry ${decision.reconnectAttempts}/${reconnectPolicy.maxAttempts || "∞"} in ${formatDurationPrecise(decision.delayMs ?? 0)}… (${decision.normalized.errorText})`;
703
+ if (isWatchdogRecoveryReconnect) {
704
+ runtime.log(warn(reconnectMessage));
705
+ } else {
706
+ runtime.error(reconnectMessage);
707
+ }
708
+ await controller.closeCurrentConnection();
709
+ try {
710
+ await controller.waitBeforeRetry(decision.delayMs ?? 0);
711
+ } catch {
712
+ break;
713
+ }
714
+ }
715
+ } finally {
716
+ statusController.markStopped();
717
+ process.removeListener("SIGINT", handleSigint);
718
+ await controller.shutdown();
719
+ }
720
+ }