@bellachu/litebails 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (116) hide show
  1. package/LICENSE +30 -0
  2. package/README.md +30 -0
  3. package/WAProto/GenerateStatics.sh +3 -0
  4. package/WAProto/WAProto.proto +6902 -0
  5. package/WAProto/fix-imports.js +85 -0
  6. package/WAProto/index.d.ts +79257 -0
  7. package/WAProto/index.js +242946 -0
  8. package/engine-requirements.js +10 -0
  9. package/lib/Defaults/index.js +130 -0
  10. package/lib/Signal/Group/ciphertext-message.js +12 -0
  11. package/lib/Signal/Group/group-session-builder.js +30 -0
  12. package/lib/Signal/Group/group_cipher.js +82 -0
  13. package/lib/Signal/Group/index.js +12 -0
  14. package/lib/Signal/Group/keyhelper.js +18 -0
  15. package/lib/Signal/Group/sender-chain-key.js +26 -0
  16. package/lib/Signal/Group/sender-key-distribution-message.js +63 -0
  17. package/lib/Signal/Group/sender-key-message.js +66 -0
  18. package/lib/Signal/Group/sender-key-name.js +48 -0
  19. package/lib/Signal/Group/sender-key-record.js +41 -0
  20. package/lib/Signal/Group/sender-key-state.js +84 -0
  21. package/lib/Signal/Group/sender-message-key.js +26 -0
  22. package/lib/Signal/libsignal.js +431 -0
  23. package/lib/Signal/lid-mapping.js +277 -0
  24. package/lib/Socket/Client/index.js +3 -0
  25. package/lib/Socket/Client/types.js +11 -0
  26. package/lib/Socket/Client/websocket.js +54 -0
  27. package/lib/Socket/business.js +379 -0
  28. package/lib/Socket/chats.js +1193 -0
  29. package/lib/Socket/communities.js +431 -0
  30. package/lib/Socket/groups.js +374 -0
  31. package/lib/Socket/index.js +12 -0
  32. package/lib/Socket/luxu.js +387 -0
  33. package/lib/Socket/messages-recv.js +1916 -0
  34. package/lib/Socket/messages-send.js +1453 -0
  35. package/lib/Socket/mex.js +41 -0
  36. package/lib/Socket/newsletter.js +279 -0
  37. package/lib/Socket/socket.js +980 -0
  38. package/lib/Socket/username.js +234 -0
  39. package/lib/Store/index.js +10 -0
  40. package/lib/Store/keyed-db.js +108 -0
  41. package/lib/Store/make-cache-manager-store.js +85 -0
  42. package/lib/Store/make-in-memory-store.js +198 -0
  43. package/lib/Store/make-ordered-dictionary.js +75 -0
  44. package/lib/Store/object-repository.js +32 -0
  45. package/lib/Types/Auth.js +2 -0
  46. package/lib/Types/Bussines.js +2 -0
  47. package/lib/Types/Call.js +2 -0
  48. package/lib/Types/Chat.js +8 -0
  49. package/lib/Types/Contact.js +2 -0
  50. package/lib/Types/Events.js +2 -0
  51. package/lib/Types/GroupMetadata.js +2 -0
  52. package/lib/Types/Label.js +25 -0
  53. package/lib/Types/LabelAssociation.js +7 -0
  54. package/lib/Types/Message.js +11 -0
  55. package/lib/Types/Mex.js +37 -0
  56. package/lib/Types/Product.js +2 -0
  57. package/lib/Types/Signal.js +2 -0
  58. package/lib/Types/Socket.js +3 -0
  59. package/lib/Types/State.js +56 -0
  60. package/lib/Types/USync.js +2 -0
  61. package/lib/Types/index.js +26 -0
  62. package/lib/Utils/auth-utils.js +302 -0
  63. package/lib/Utils/browser-utils.js +49 -0
  64. package/lib/Utils/business.js +231 -0
  65. package/lib/Utils/chat-utils.js +872 -0
  66. package/lib/Utils/companion-reg-client-utils.js +35 -0
  67. package/lib/Utils/crypto.js +118 -0
  68. package/lib/Utils/decode-wa-message.js +350 -0
  69. package/lib/Utils/event-buffer.js +622 -0
  70. package/lib/Utils/generics.js +403 -0
  71. package/lib/Utils/history.js +134 -0
  72. package/lib/Utils/identity-change-handler.js +50 -0
  73. package/lib/Utils/index.js +23 -0
  74. package/lib/Utils/link-preview.js +85 -0
  75. package/lib/Utils/logger.js +3 -0
  76. package/lib/Utils/lt-hash.js +8 -0
  77. package/lib/Utils/make-mutex.js +33 -0
  78. package/lib/Utils/message-composer.js +273 -0
  79. package/lib/Utils/message-retry-manager.js +265 -0
  80. package/lib/Utils/messages-media.js +788 -0
  81. package/lib/Utils/messages.js +1260 -0
  82. package/lib/Utils/noise-handler.js +201 -0
  83. package/lib/Utils/offline-node-processor.js +40 -0
  84. package/lib/Utils/pre-key-manager.js +106 -0
  85. package/lib/Utils/process-message.js +630 -0
  86. package/lib/Utils/reporting-utils.js +258 -0
  87. package/lib/Utils/signal.js +201 -0
  88. package/lib/Utils/stanza-ack.js +38 -0
  89. package/lib/Utils/sync-action-utils.js +49 -0
  90. package/lib/Utils/tc-token-utils.js +163 -0
  91. package/lib/Utils/use-multi-file-auth-state.js +121 -0
  92. package/lib/Utils/validate-connection.js +203 -0
  93. package/lib/WABinary/constants.js +1301 -0
  94. package/lib/WABinary/decode.js +262 -0
  95. package/lib/WABinary/encode.js +220 -0
  96. package/lib/WABinary/generic-utils.js +204 -0
  97. package/lib/WABinary/index.js +6 -0
  98. package/lib/WABinary/jid-utils.js +98 -0
  99. package/lib/WABinary/types.js +2 -0
  100. package/lib/WAM/BinaryInfo.js +10 -0
  101. package/lib/WAM/constants.js +22853 -0
  102. package/lib/WAM/encode.js +150 -0
  103. package/lib/WAM/index.js +4 -0
  104. package/lib/WAUSync/Protocols/USyncContactProtocol.js +52 -0
  105. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +54 -0
  106. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +27 -0
  107. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +38 -0
  108. package/lib/WAUSync/Protocols/USyncUsernameProtocol.js +25 -0
  109. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +51 -0
  110. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +29 -0
  111. package/lib/WAUSync/Protocols/index.js +6 -0
  112. package/lib/WAUSync/USyncQuery.js +98 -0
  113. package/lib/WAUSync/USyncUser.js +31 -0
  114. package/lib/WAUSync/index.js +4 -0
  115. package/lib/index.js +21 -0
  116. package/package.json +148 -0
@@ -0,0 +1,1453 @@
1
+ import NodeCache from '@cacheable/node-cache';
2
+ import { Boom } from '@hapi/boom';
3
+ import { proto } from '../../WAProto/index.js';
4
+ import { DEFAULT_CACHE_TTLS, WA_DEFAULT_EPHEMERAL } from '../Defaults/index.js';
5
+ import { aggregateMessageKeysNotFromMe, assertMediaContent, assertMeId, bindWaitForEvent, decryptMediaRetryData, DEF_MEDIA_HOST, encodeNewsletterMessage, encodeSignedDeviceIdentity, encodeWAMessage, encryptMediaRetryRequest, extractDeviceJids, generateMessageIDV2, generateIOSMessageID, generateParticipantHashV2, generateWAMessage, getStatusCodeForMediaRetry, getUrlFromDirectPath, getWAUploadToServer, MessageRetryManager, normalizeMessageContent, parseAndInjectE2ESessions, unixTimestampSeconds, setBotMessageSecret } from '../Utils/index.js';
6
+ import { getUrlInfo } from '../Utils/link-preview.js';
7
+ import { makeKeyedMutex, makeMutex } from '../Utils/make-mutex.js';
8
+ import { getMessageReportingToken, shouldIncludeReportingToken } from '../Utils/reporting-utils.js';
9
+ import { buildMergedTcTokenIndexWrite, isTcTokenExpired, resolveIssuanceJid, resolveTcTokenJid, shouldSendNewTcToken, storeTcTokensFromIqResult } from '../Utils/tc-token-utils.js';
10
+ import { areJidsSameUser, getBinaryNodeChild, getBinaryNodeChildren, isHostedLidUser, isHostedPnUser, isJidBot, isJidGroup, isJidMetaAI, isLidUser, isPnUser, jidDecode, jidEncode, jidNormalizedUser, PSA_WID, S_WHATSAPP_NET, getAdditionalNode, getBinaryNodeFilter, getBinaryFilteredBizBot, isInteropUser } from '../WABinary/index.js';
11
+ import { USyncQuery, USyncUser } from '../WAUSync/index.js';
12
+ import { makeUsernameSocket } from './username.js';
13
+ import imup from './luxu.js';
14
+ import * as Utils_1 from '../Utils/index.js';
15
+ import { randomBytes } from 'crypto';
16
+ export const makeMessagesSocket = (config) => {
17
+ const { logger, linkPreviewImageThumbnailWidth, generateHighQualityLinkPreview, options: httpRequestOptions, patchMessageBeforeSending, cachedGroupMetadata, enableRecentMessageCache, maxMsgRetryCount, aiLabel } = config;
18
+ const sock = makeUsernameSocket(config);
19
+ const { ev, authState, messageMutex, signalRepository, upsertMessage, query, fetchPrivacySettings, sendNode, groupMetadata, groupToggleEphemeral, registerSocketEndHandler } = sock;
20
+ const getLIDForPN = signalRepository.lidMapping.getLIDForPN.bind(signalRepository.lidMapping);
21
+ /**
22
+ * Set of tctoken storage JIDs with a fire-and-forget `issuePrivacyTokens` IQ in flight.
23
+ * Prevents duplicate IQs from rapid back-to-back sends before `senderTimestamp` persists.
24
+ * Entries are always removed in `.finally()`, so the set is bounded by concurrency.
25
+ */
26
+ const inFlightTcTokenIssuance = new Set();
27
+ const userDevicesCache = config.userDevicesCache ||
28
+ new NodeCache({
29
+ stdTTL: DEFAULT_CACHE_TTLS.USER_DEVICES, // 5 minutes
30
+ useClones: false
31
+ });
32
+ /** Serializes writes to userDevicesCache across USync refresh and device-notification handling. */
33
+ const devicesMutex = makeMutex();
34
+ // Initialize message retry manager if enabled
35
+ const messageRetryManager = enableRecentMessageCache ? new MessageRetryManager(logger, maxMsgRetryCount) : null;
36
+ // Prevent race conditions in Signal session encryption by user
37
+ const encryptionMutex = makeKeyedMutex();
38
+ let mediaConn;
39
+ /** Per-socket media host; updated whenever media_conn is fetched. Defaults to the public WhatsApp host. */
40
+ let mediaHost = DEF_MEDIA_HOST;
41
+ const refreshMediaConn = async (forceGet = false) => {
42
+ const media = await mediaConn;
43
+ if (!media || forceGet || new Date().getTime() - media.fetchDate.getTime() > media.ttl * 1000) {
44
+ mediaConn = (async () => {
45
+ const result = await query({
46
+ tag: 'iq',
47
+ attrs: {
48
+ type: 'set',
49
+ xmlns: 'w:m',
50
+ to: S_WHATSAPP_NET
51
+ },
52
+ content: [{ tag: 'media_conn', attrs: {} }]
53
+ });
54
+ const mediaConnNode = getBinaryNodeChild(result, 'media_conn');
55
+ // TODO: explore full length of data that whatsapp provides
56
+ const node = {
57
+ hosts: getBinaryNodeChildren(mediaConnNode, 'host').map(({ attrs }) => ({
58
+ hostname: attrs.hostname,
59
+ maxContentLengthBytes: +attrs.maxContentLengthBytes
60
+ })),
61
+ auth: mediaConnNode.attrs.auth,
62
+ ttl: +mediaConnNode.attrs.ttl,
63
+ fetchDate: new Date()
64
+ };
65
+ logger.debug('fetched media conn');
66
+ if (node.hosts[0]) {
67
+ mediaHost = node.hosts[0].hostname;
68
+ }
69
+ return node;
70
+ })();
71
+ }
72
+ return mediaConn;
73
+ };
74
+ /**
75
+ * generic send receipt function
76
+ * used for receipts of phone call, read, delivery etc.
77
+ * */
78
+ const sendReceipt = async (jid, participant, messageIds, type) => {
79
+ if (!messageIds || messageIds.length === 0) {
80
+ throw new Boom('missing ids in receipt');
81
+ }
82
+ const node = {
83
+ tag: 'receipt',
84
+ attrs: {
85
+ id: messageIds[0]
86
+ }
87
+ };
88
+ const isReadReceipt = type === 'read' || type === 'read-self';
89
+ if (isReadReceipt) {
90
+ node.attrs.t = unixTimestampSeconds().toString();
91
+ }
92
+ if (type === 'sender' && (isPnUser(jid) || isLidUser(jid))) {
93
+ node.attrs.recipient = jid;
94
+ node.attrs.to = participant;
95
+ }
96
+ else {
97
+ node.attrs.to = jid;
98
+ if (participant) {
99
+ node.attrs.participant = participant;
100
+ }
101
+ }
102
+ if (type) {
103
+ node.attrs.type = type;
104
+ }
105
+ const remainingMessageIds = messageIds.slice(1);
106
+ if (remainingMessageIds.length) {
107
+ node.content = [
108
+ {
109
+ tag: 'list',
110
+ attrs: {},
111
+ content: remainingMessageIds.map(id => ({
112
+ tag: 'item',
113
+ attrs: { id }
114
+ }))
115
+ }
116
+ ];
117
+ }
118
+ logger.debug({ attrs: node.attrs, messageIds }, 'sending receipt for messages');
119
+ await sendNode(node);
120
+ };
121
+ /** Correctly bulk send receipts to multiple chats, participants */
122
+ const sendReceipts = async (keys, type) => {
123
+ const recps = aggregateMessageKeysNotFromMe(keys);
124
+ for (const { jid, participant, messageIds } of recps) {
125
+ await sendReceipt(jid, participant, messageIds, type);
126
+ }
127
+ };
128
+ /** Bulk read messages. Keys can be from different chats & participants */
129
+ const readMessages = async (keys) => {
130
+ const privacySettings = await fetchPrivacySettings();
131
+ // based on privacy settings, we have to change the read type
132
+ const readType = privacySettings.readreceipts === 'all' ? 'read' : 'read-self';
133
+ await sendReceipts(keys, readType);
134
+ };
135
+ /** Fetch all the devices we've to send a message to */
136
+ const getUSyncDevices = async (jids, useCache, ignoreZeroDevices) => {
137
+ const deviceResults = [];
138
+ if (!useCache) {
139
+ logger.debug('not using cache for devices');
140
+ }
141
+ const toFetch = [];
142
+ const jidsWithUser = jids
143
+ .map(jid => {
144
+ const decoded = jidDecode(jid);
145
+ const user = decoded?.user;
146
+ const device = decoded?.device;
147
+ const isExplicitDevice = typeof device === 'number' && device >= 0;
148
+ if (isExplicitDevice && user) {
149
+ deviceResults.push({
150
+ user,
151
+ device,
152
+ jid
153
+ });
154
+ return null;
155
+ }
156
+ jid = jidNormalizedUser(jid);
157
+ return { jid, user };
158
+ })
159
+ .filter(jid => jid !== null);
160
+ let mgetDevices;
161
+ if (useCache && userDevicesCache.mget) {
162
+ const usersToFetch = jidsWithUser.map(j => j?.user).filter(Boolean);
163
+ mgetDevices = await userDevicesCache.mget(usersToFetch);
164
+ }
165
+ for (const { jid, user } of jidsWithUser) {
166
+ if (useCache) {
167
+ const devices = mgetDevices?.[user] ||
168
+ (userDevicesCache.mget ? undefined : (await userDevicesCache.get(user)));
169
+ if (devices) {
170
+ const devicesWithJid = devices.map(d => ({
171
+ ...d,
172
+ jid: jidEncode(d.user, d.server, d.device)
173
+ }));
174
+ deviceResults.push(...devicesWithJid);
175
+ logger.trace({ user }, 'using cache for devices');
176
+ }
177
+ else {
178
+ toFetch.push(jid);
179
+ }
180
+ }
181
+ else {
182
+ toFetch.push(jid);
183
+ }
184
+ }
185
+ if (!toFetch.length) {
186
+ return deviceResults;
187
+ }
188
+ const requestedLidUsers = new Set();
189
+ for (const jid of toFetch) {
190
+ if (isLidUser(jid) || isHostedLidUser(jid)) {
191
+ const user = jidDecode(jid)?.user;
192
+ if (user)
193
+ requestedLidUsers.add(user);
194
+ }
195
+ }
196
+ const query = new USyncQuery().withContext('message').withDeviceProtocol().withLIDProtocol();
197
+ for (const jid of toFetch) {
198
+ query.withUser(new USyncUser().withId(jid)); // todo: investigate - the idea here is that <user> should have an inline lid field with the lid being the pn equivalent
199
+ }
200
+ const result = await sock.executeUSyncQuery(query);
201
+ if (result) {
202
+ // TODO: LID MAP this stuff (lid protocol will now return lid with devices)
203
+ const lidResults = result.list.filter(a => !!a.lid);
204
+ if (lidResults.length > 0) {
205
+ logger.trace('Storing LID maps from device call');
206
+ await signalRepository.lidMapping.storeLIDPNMappings(lidResults.map(a => ({ lid: a.lid, pn: a.id })));
207
+ // Force-refresh sessions for newly mapped LIDs to align identity addressing
208
+ try {
209
+ const lids = lidResults.map(a => a.lid);
210
+ if (lids.length) {
211
+ await assertSessions(lids, true);
212
+ }
213
+ }
214
+ catch (e) {
215
+ logger.warn({ e, count: lidResults.length }, 'failed to assert sessions for newly mapped LIDs');
216
+ }
217
+ }
218
+ const extracted = extractDeviceJids(result?.list, authState.creds.me.id, authState.creds.me.lid, ignoreZeroDevices);
219
+ const deviceMap = {};
220
+ for (const item of extracted) {
221
+ deviceMap[item.user] = deviceMap[item.user] || [];
222
+ deviceMap[item.user]?.push(item);
223
+ }
224
+ // Process each user's devices as a group for bulk LID migration
225
+ for (const [user, userDevices] of Object.entries(deviceMap)) {
226
+ const isLidUser = requestedLidUsers.has(user);
227
+ // Process all devices for this user
228
+ for (const item of userDevices) {
229
+ const finalJid = isLidUser
230
+ ? jidEncode(user, item.server, item.device)
231
+ : jidEncode(item.user, item.server, item.device);
232
+ deviceResults.push({
233
+ ...item,
234
+ jid: finalJid
235
+ });
236
+ logger.debug({
237
+ user: item.user,
238
+ device: item.device,
239
+ finalJid,
240
+ usedLid: isLidUser
241
+ }, 'Processed device with LID priority');
242
+ }
243
+ }
244
+ await devicesMutex.mutex(async () => {
245
+ if (userDevicesCache.mset) {
246
+ // if the cache supports mset, we can set all devices in one go
247
+ await userDevicesCache.mset(Object.entries(deviceMap).map(([key, value]) => ({ key, value })));
248
+ }
249
+ else {
250
+ for (const key in deviceMap) {
251
+ if (deviceMap[key])
252
+ await userDevicesCache.set(key, deviceMap[key]);
253
+ }
254
+ }
255
+ });
256
+ const userDeviceUpdates = {};
257
+ for (const [userId, devices] of Object.entries(deviceMap)) {
258
+ if (devices && devices.length > 0) {
259
+ userDeviceUpdates[userId] = devices.map(d => d.device?.toString() || '0');
260
+ }
261
+ }
262
+ if (Object.keys(userDeviceUpdates).length > 0) {
263
+ try {
264
+ await authState.keys.set({ 'device-list': userDeviceUpdates });
265
+ logger.debug({ userCount: Object.keys(userDeviceUpdates).length }, 'stored user device lists for bulk migration');
266
+ }
267
+ catch (error) {
268
+ logger.warn({ error }, 'failed to store user device lists');
269
+ }
270
+ }
271
+ }
272
+ return deviceResults;
273
+ };
274
+ /**
275
+ * Update Member Label
276
+ */
277
+ const updateMemberLabel = (jid, memberLabel) => {
278
+ return relayMessage(jid, {
279
+ protocolMessage: {
280
+ type: proto.Message.ProtocolMessage.Type.GROUP_MEMBER_LABEL_CHANGE,
281
+ memberLabel: {
282
+ label: memberLabel?.slice(0, 30),
283
+ labelTimestamp: unixTimestampSeconds()
284
+ }
285
+ }
286
+ }, {
287
+ additionalNodes: [
288
+ {
289
+ tag: 'meta',
290
+ attrs: {
291
+ tag_reason: 'user_update',
292
+ appdata: 'member_tag'
293
+ },
294
+ content: undefined
295
+ }
296
+ ]
297
+ });
298
+ };
299
+ const assertSessions = async (jids, force) => {
300
+ let didFetchNewSession = false;
301
+ const uniqueJids = [...new Set(jids)];
302
+ const jidsRequiringFetch = [];
303
+ logger.debug({ jids }, 'assertSessions call with jids');
304
+ for (const jid of uniqueJids) {
305
+ if (!force) {
306
+ const sessionValidation = await signalRepository.validateSession(jid);
307
+ if (sessionValidation.exists) {
308
+ continue;
309
+ }
310
+ }
311
+ jidsRequiringFetch.push(jid);
312
+ }
313
+ if (jidsRequiringFetch.length) {
314
+ // LID if mapped, otherwise original
315
+ const wireJids = [
316
+ ...jidsRequiringFetch.filter(jid => !!isLidUser(jid) || !!isHostedLidUser(jid)),
317
+ ...((await signalRepository.lidMapping.getLIDsForPNs(jidsRequiringFetch.filter(jid => !!isPnUser(jid) || !!isHostedPnUser(jid)))) || []).map(a => a.lid)
318
+ ];
319
+ logger.debug({ jidsRequiringFetch, wireJids }, 'fetching sessions');
320
+ const result = await query({
321
+ tag: 'iq',
322
+ attrs: {
323
+ xmlns: 'encrypt',
324
+ type: 'get',
325
+ to: S_WHATSAPP_NET
326
+ },
327
+ content: [
328
+ {
329
+ tag: 'key',
330
+ attrs: {},
331
+ content: wireJids.map(jid => {
332
+ const attrs = { jid };
333
+ if (force)
334
+ attrs.reason = 'identity';
335
+ return { tag: 'user', attrs };
336
+ })
337
+ }
338
+ ]
339
+ });
340
+ await parseAndInjectE2ESessions(result, signalRepository);
341
+ didFetchNewSession = true;
342
+ }
343
+ return didFetchNewSession;
344
+ };
345
+ const sendPeerDataOperationMessage = async (pdoMessage) => {
346
+ //TODO: for later, abstract the logic to send a Peer Message instead of just PDO - useful for App State Key Resync with phone
347
+ if (!authState.creds.me?.id) {
348
+ throw new Boom('Not authenticated');
349
+ }
350
+ const protocolMessage = {
351
+ protocolMessage: {
352
+ peerDataOperationRequestMessage: pdoMessage,
353
+ type: proto.Message.ProtocolMessage.Type.PEER_DATA_OPERATION_REQUEST_MESSAGE
354
+ }
355
+ };
356
+ const meJid = jidNormalizedUser(authState.creds.me.id);
357
+ const msgId = await relayMessage(meJid, protocolMessage, {
358
+ additionalAttributes: {
359
+ category: 'peer',
360
+ push_priority: 'high_force'
361
+ },
362
+ additionalNodes: [
363
+ {
364
+ tag: 'meta',
365
+ attrs: { appdata: 'default' }
366
+ }
367
+ ]
368
+ });
369
+ return msgId;
370
+ };
371
+ const createParticipantNodes = async (recipientJids, message, extraAttrs, dsmMessage) => {
372
+ if (!recipientJids.length) {
373
+ return { nodes: [], shouldIncludeDeviceIdentity: false };
374
+ }
375
+ const patched = await patchMessageBeforeSending(message, recipientJids);
376
+ const patchedMessages = Array.isArray(patched)
377
+ ? patched
378
+ : recipientJids.map(jid => ({ recipientJid: jid, message: patched }));
379
+ let shouldIncludeDeviceIdentity = false;
380
+ const meId = authState.creds.me.id;
381
+ const meLid = authState.creds.me?.lid;
382
+ const meLidUser = meLid ? jidDecode(meLid)?.user : null;
383
+ const encryptionPromises = patchedMessages.map(async ({ recipientJid: jid, message: patchedMessage }) => {
384
+ try {
385
+ if (!jid)
386
+ return null;
387
+ let msgToEncrypt = patchedMessage;
388
+ if (dsmMessage) {
389
+ const { user: targetUser } = jidDecode(jid);
390
+ const { user: ownPnUser } = jidDecode(meId);
391
+ const ownLidUser = meLidUser;
392
+ const isOwnUser = targetUser === ownPnUser || (ownLidUser && targetUser === ownLidUser);
393
+ const isExactSenderDevice = jid === meId || (meLid && jid === meLid);
394
+ if (isOwnUser && !isExactSenderDevice) {
395
+ msgToEncrypt = dsmMessage;
396
+ logger.debug({ jid, targetUser }, 'Using DSM for own device');
397
+ }
398
+ }
399
+ const bytes = encodeWAMessage(msgToEncrypt);
400
+ const mutexKey = jid;
401
+ const node = await encryptionMutex.mutex(mutexKey, async () => {
402
+ const { type, ciphertext } = await signalRepository.encryptMessage({ jid, data: bytes });
403
+ if (type === 'pkmsg') {
404
+ shouldIncludeDeviceIdentity = true;
405
+ }
406
+ return {
407
+ tag: 'to',
408
+ attrs: { jid },
409
+ content: [
410
+ {
411
+ tag: 'enc',
412
+ attrs: { v: '2', type, ...(extraAttrs || {}) },
413
+ content: ciphertext
414
+ }
415
+ ]
416
+ };
417
+ });
418
+ return node;
419
+ }
420
+ catch (err) {
421
+ logger.error({ jid, err }, 'Failed to encrypt for recipient');
422
+ return null;
423
+ }
424
+ });
425
+ const nodes = (await Promise.all(encryptionPromises)).filter(node => node !== null);
426
+ if (recipientJids.length > 0 && nodes.length === 0) {
427
+ throw new Boom('All encryptions failed', { statusCode: 500 });
428
+ }
429
+ return { nodes, shouldIncludeDeviceIdentity };
430
+ };
431
+ const relayMessage = async (
432
+ jid,
433
+ message,
434
+ {
435
+ messageId: msgId,
436
+ participant = false,
437
+ notMe = false,
438
+ additionalAttributes,
439
+ additionalNodes,
440
+ useUserDevicesCache,
441
+ useCachedGroupMetadata,
442
+ statusJidList
443
+ }
444
+ ) => {
445
+ const meId = authState.creds.me.id
446
+ const meLid = authState.creds.me?.lid
447
+ const isRetryResend = Boolean(participant?.jid)
448
+ let shouldIncludeDeviceIdentity = isRetryResend
449
+ const statusJid = 'status@broadcast'
450
+ const { user, server } = jidDecode(jid)
451
+ const isGroup = server === 'g.us'
452
+ const isStatus = jid === statusJid
453
+ const isLid = server === 'lid'
454
+ const isNewsletter = server === 'newsletter'
455
+ const isInterop = isInteropUser(jid)
456
+ const isGroupOrStatus = isGroup || isStatus
457
+ const finalJid = jid
458
+ const iosBros = config.browser[0] === "iOS" || config.browser[1] === "Safari";
459
+ msgId = iosBros ? generateIOSMessageID() : msgId ?? generateMessageIDV2(meId)
460
+ useUserDevicesCache = useUserDevicesCache!== false
461
+ useCachedGroupMetadata = useCachedGroupMetadata!== false &&!isStatus
462
+ const participants = []
463
+ const destinationJid =!isStatus? finalJid : statusJid
464
+ const binaryNodeContent = []
465
+ const devices = []
466
+ let reportingMessage
467
+ const meMsg = {
468
+ deviceSentMessage: { destinationJid, message },
469
+ messageContextInfo: message.messageContextInfo
470
+ }
471
+ const extraAttrs = {}
472
+ const regexGroupOld = /^(\d{1,15})-(\d+)@g\.us$/
473
+ const messages = normalizeMessageContent(message)
474
+ const buttonType = getButtonType(messages)
475
+ const pollMessage =
476
+ messages.pollCreationMessage || messages.pollCreationMessageV2 || messages.pollCreationMessageV3
477
+ await authState.keys.transaction(async () => {
478
+ const mediaType = getMediaType(message)
479
+ if (mediaType) extraAttrs.mediatype = mediaType
480
+ if (isNewsletter) {
481
+ const patched = patchMessageBeforeSending? await patchMessageBeforeSending(message, []) : message
482
+ const bytes = encodeNewsletterMessage(patched)
483
+ binaryNodeContent.push({ tag: 'plaintext', attrs: {}, content: bytes })
484
+ const stanza = {
485
+ tag: 'message',
486
+ attrs: {
487
+ to: jid,
488
+ id: msgId,
489
+ type: getMessageType(message),
490
+ ...(additionalAttributes || {})
491
+ },
492
+ content: binaryNodeContent
493
+ }
494
+ logger.debug({ msgId }, `sending newsletter message to ${jid}`)
495
+ await sendNode(stanza)
496
+ return
497
+ }
498
+ if (normalizeMessageContent(message)?.pinInChatMessage || normalizeMessageContent(message)?.reactionMessage) {
499
+ extraAttrs['decrypt-fail'] = 'hide'
500
+ }
501
+ if (isGroupOrStatus &&!isRetryResend) {
502
+ const [groupData, senderKeyMap] = await Promise.all([
503
+ (async () => {
504
+ let groupData = useCachedGroupMetadata && cachedGroupMetadata? await cachedGroupMetadata(jid) : undefined
505
+ if (groupData && Array.isArray(groupData?.participants)) {
506
+ logger.trace({ jid, participants: groupData.participants.length }, 'using cached group metadata')
507
+ } else if (!isStatus) {
508
+ groupData = await groupMetadata(jid)
509
+ }
510
+ return groupData
511
+ })(),
512
+ (async () => {
513
+ if (!participant &&!isStatus) {
514
+ const result = await authState.keys.get('sender-key-memory', [jid])
515
+ return result[jid] || {}
516
+ }
517
+ return {}
518
+ })()
519
+ ])
520
+ const participantsList = groupData? groupData.participants.map(p => p.id) : []
521
+ if (groupData?.ephemeralDuration && groupData.ephemeralDuration > 0) {
522
+ additionalAttributes = {...additionalAttributes, expiration: groupData.ephemeralDuration.toString() }
523
+ }
524
+ if (isStatus && statusJidList) participantsList.push(...statusJidList)
525
+ const additionalDevices = await getUSyncDevices(participantsList,!!useUserDevicesCache, false)
526
+ devices.push(...additionalDevices)
527
+ if (isGroup) {
528
+ additionalAttributes = {
529
+ ...additionalAttributes,
530
+ addressing_mode: groupData?.addressingMode || 'lid'
531
+ }
532
+ }
533
+ if (message?.groupStatusMessageV2 &&!message?.messageContextInfo?.messageSecret) {
534
+ message = {
535
+ ...message,
536
+ messageContextInfo: {
537
+ ...(message.messageContextInfo || {}),
538
+ messageSecret: randomBytes(32)
539
+ },
540
+ groupStatusMessageV2: {
541
+ ...message.groupStatusMessageV2,
542
+ message: {
543
+ ...(message.groupStatusMessageV2.message || {}),
544
+ messageContextInfo: {
545
+ ...(message.groupStatusMessageV2.message?.messageContextInfo || {}),
546
+ messageSecret: message.messageContextInfo?.messageSecret || randomBytes(32)
547
+ }
548
+ }
549
+ }
550
+ }
551
+ }
552
+ // list/buttons/template -> interactiveMessage
553
+ if (message.listMessage) {
554
+ const list = message.listMessage
555
+ message = {
556
+ interactiveMessage: {
557
+ nativeFlowMessage: {
558
+ buttons: [
559
+ {
560
+ name: 'single_select',
561
+ buttonParamsJson: JSON.stringify({
562
+ title: list.buttonText || 'Select',
563
+ sections: (list.sections || []).map(section => ({
564
+ title: section.title || '',
565
+ highlight_label: '',
566
+ rows: (section.rows || []).map(row => ({
567
+ header: '',
568
+ title: row.title || '',
569
+ description: row.description || '',
570
+ id: row.rowId || row.id || ''
571
+ }))
572
+ }))
573
+ })
574
+ }
575
+ ],
576
+ messageParamsJson: '',
577
+ messageVersion: 1
578
+ },
579
+ body: { text: list.description || '' },
580
+ footer: list.footerText? { text: list.footerText } : undefined,
581
+ header: list.title? { title: list.title, hasMediaAttachment: false, subtitle: '' } : undefined,
582
+ contextInfo: list.contextInfo
583
+ }
584
+ }
585
+ } else if (message.buttonsMessage) {
586
+ const bMsg = message.buttonsMessage
587
+ const buttons = (bMsg.buttons || []).map(btn => ({
588
+ name: 'quick_reply',
589
+ buttonParamsJson: JSON.stringify({
590
+ display_text: btn.buttonText?.displayText || btn.buttonText || '',
591
+ id: btn.buttonId || btn.buttonText?.displayText || ''
592
+ })
593
+ }))
594
+ message = {
595
+ interactiveMessage: {
596
+ nativeFlowMessage: { buttons, messageParamsJson: '', messageVersion: 1 },
597
+ body: { text: bMsg.contentText || bMsg.text || '' },
598
+ footer: bMsg.footerText? { text: bMsg.footerText } : undefined,
599
+ header: bMsg.text
600
+ ? { title: bMsg.text, hasMediaAttachment: false, subtitle: '' }
601
+ : bMsg.imageMessage || bMsg.videoMessage || bMsg.documentMessage
602
+ ? { hasMediaAttachment: true,...(bMsg.imageMessage? { imageMessage: bMsg.imageMessage } : {}),...(bMsg.videoMessage? { videoMessage: bMsg.videoMessage } : {}) }
603
+ : undefined,
604
+ contextInfo: bMsg.contextInfo
605
+ }
606
+ }
607
+ } else if (message.templateMessage) {
608
+ const tmpl = message.templateMessage.hydratedTemplate || message.templateMessage.fourRowTemplate
609
+ if (tmpl) {
610
+ const buttons = (tmpl.hydratedButtons || [])
611
+ .map(hBtn => {
612
+ if (hBtn.quickReplyButton) {
613
+ return { name: 'quick_reply', buttonParamsJson: JSON.stringify({ display_text: hBtn.quickReplyButton.displayText || '', id: hBtn.quickReplyButton.id || hBtn.quickReplyButton.displayText || '' }) }
614
+ } else if (hBtn.urlButton) {
615
+ return { name: 'cta_url', buttonParamsJson: JSON.stringify({ display_text: hBtn.urlButton.displayText || '', url: hBtn.urlButton.url || '', merchant_url: hBtn.urlButton.url || '' }) }
616
+ } else if (hBtn.callButton) {
617
+ return { name: 'cta_call', buttonParamsJson: JSON.stringify({ display_text: hBtn.callButton.displayText || '', phone_number: hBtn.callButton.phoneNumber || '' }) }
618
+ }
619
+ return null
620
+ })
621
+ .filter(Boolean)
622
+ message = {
623
+ interactiveMessage: {
624
+ nativeFlowMessage: { buttons, messageParamsJson: '', messageVersion: 1 },
625
+ body: { text: tmpl.hydratedContentText || tmpl.contentText || '' },
626
+ footer: tmpl.hydratedFooterText? { text: tmpl.hydratedFooterText } : undefined,
627
+ header: tmpl.hydratedTitleText
628
+ ? { title: tmpl.hydratedTitleText, hasMediaAttachment: false, subtitle: '' }
629
+ : tmpl.imageMessage || tmpl.videoMessage || tmpl.documentMessage
630
+ ? { hasMediaAttachment: true,...(tmpl.imageMessage? { imageMessage: tmpl.imageMessage } : {}),...(tmpl.videoMessage? { videoMessage: tmpl.videoMessage } : {}) }
631
+ : undefined,
632
+ contextInfo: tmpl.contextInfo
633
+ }
634
+ }
635
+ }
636
+ }
637
+
638
+ const patched = await patchMessageBeforeSending(message)
639
+ if (Array.isArray(patched)) throw new Boom('Per-jid patching is not supported in groups')
640
+ const bytes = encodeWAMessage(patched)
641
+ reportingMessage = patched
642
+ const groupAddressingMode = additionalAttributes?.['addressing_mode'] || groupData?.addressingMode || 'lid'
643
+ const groupSenderIdentity = groupAddressingMode === 'lid' && meLid? meLid : meId
644
+ const { ciphertext, senderKeyDistributionMessage } = await signalRepository.encryptGroupMessage({
645
+ group: destinationJid,
646
+ data: bytes,
647
+ meId: groupSenderIdentity
648
+ })
649
+ const senderKeyRecipients = []
650
+ for (const device of devices) {
651
+ const deviceJid = device.jid
652
+ const hasKey =!!senderKeyMap[deviceJid]
653
+ if (!hasKey ||!!participant &&!isHostedLidUser(deviceJid) &&!isHostedPnUser(deviceJid) && device.device!== 99) {
654
+ senderKeyRecipients.push(deviceJid)
655
+ senderKeyMap[deviceJid] = true
656
+ }
657
+ }
658
+ if (senderKeyRecipients.length) {
659
+ logger.debug({ senderKeyJids: senderKeyRecipients }, 'sending new sender key')
660
+ const senderKeyMsg = {
661
+ senderKeyDistributionMessage: {
662
+ axolotlSenderKeyDistributionMessage: senderKeyDistributionMessage,
663
+ groupId: destinationJid
664
+ }
665
+ }
666
+ await assertSessions(senderKeyRecipients)
667
+ const result = await createParticipantNodes(senderKeyRecipients, senderKeyMsg, extraAttrs)
668
+ shouldIncludeDeviceIdentity = shouldIncludeDeviceIdentity || result.shouldIncludeDeviceIdentity
669
+ participants.push(...result.nodes)
670
+ }
671
+ binaryNodeContent.push({ tag: 'enc', attrs: { v: '2', type: 'skmsg',...extraAttrs }, content: ciphertext })
672
+ await authState.keys.set({ 'sender-key-memory': { [jid]: senderKeyMap } })
673
+ } else {
674
+ let ownId = meId
675
+ if (isLid && meLid) {
676
+ ownId = meLid
677
+ logger.debug({ to: jid, ownId }, 'Using LID identity for @lid conversation')
678
+ } else {
679
+ logger.debug({ to: jid, ownId }, 'Using PN identity for @s.whatsapp.net conversation')
680
+ }
681
+ const { user: ownUser } = jidDecode(ownId)
682
+ if (!participant) {
683
+ const patchedForReporting = await patchMessageBeforeSending(message, [jid])
684
+ reportingMessage = Array.isArray(patchedForReporting)
685
+ ? patchedForReporting.find(item => item.recipientJid === jid) || patchedForReporting[0]
686
+ : patchedForReporting
687
+ }
688
+ if (!isRetryResend) {
689
+ const targetUserServer = isLid? 'lid' : isInterop? 'interop' : 's.whatsapp.net'
690
+ devices.push({ user, device: 0, jid: jidEncode(user, targetUserServer, 0) })
691
+ if (user!== ownUser &&!isInterop) {
692
+ const ownUserServer = isLid? 'lid' : 's.whatsapp.net'
693
+ const ownUserForAddressing = isLid && meLid? jidDecode(meLid).user : jidDecode(meId).user
694
+ devices.push({ user: ownUserForAddressing, device: 0, jid: jidEncode(ownUserForAddressing, ownUserServer, 0) })
695
+ }
696
+ if (additionalAttributes?.['category']!== 'peer' &&!isInterop) {
697
+ devices.length = 0
698
+ const senderIdentity = isLid && meLid
699
+ ? jidEncode(jidDecode(meLid)?.user, 'lid', undefined)
700
+ : jidEncode(jidDecode(meId)?.user, 's.whatsapp.net', undefined)
701
+ const sessionDevices = await getUSyncDevices([senderIdentity, jid], true, false)
702
+ devices.push(...sessionDevices)
703
+ logger.debug({ deviceCount: devices.length, devices: devices.map(d => `${d.user}:${d.device}@${jidDecode(d.jid)?.server}`) }, 'Device enumeration complete with unified addressing')
704
+ }
705
+ }
706
+ const allRecipients = []
707
+ const meRecipients = []
708
+ const otherRecipients = []
709
+ const { user: mePnUser } = jidDecode(meId)
710
+ const { user: meLidUser } = meLid? jidDecode(meLid) : { user: null }
711
+ for (const { user, jid } of devices) {
712
+ /** notMe: opsi untuk skip sync pesan ke device lain milik akun sendiri (private chat) */
713
+ const isExactSenderDevice = jid === meId || (meLid && jid === meLid)
714
+ if (isExactSenderDevice) {
715
+ logger.debug({ jid, meId, meLid }, 'Skipping exact sender device (whatsmeow pattern)')
716
+ continue
717
+ }
718
+ const isMe = user === mePnUser || user === meLidUser
719
+ let ptcp = false
720
+ if (notMe) {
721
+ if (!isJidGroup(jid) && !isStatus) {
722
+ if (!(!isMe)) ptcp = true
723
+ } else {
724
+ ptcp = false
725
+ }
726
+ }
727
+ if (!ptcp) {
728
+ if (isMe) {
729
+ meRecipients.push(jid)
730
+ } else {
731
+ otherRecipients.push(jid)
732
+ }
733
+ allRecipients.push(jid)
734
+ }
735
+ }
736
+ // Fix: detect silent delivery failure — if the destination is genuinely
737
+ // someone else (not our own number) and device resolution produced zero
738
+ // recipient devices for them, don't proceed as if the send succeeded.
739
+ const isSelfChatDestination = user === mePnUser || user === meLidUser
740
+ if (!isRetryResend && !isSelfChatDestination && otherRecipients.length === 0) {
741
+ logger.warn({ jid, deviceCount: devices.length, notMe }, 'relayMessage: no recipient devices resolved for the other party; aborting to avoid a silent no-op send')
742
+ throw new Boom('No devices resolved for recipient — message was not delivered to the other party', { statusCode: 421, data: { jid } })
743
+ }
744
+ await assertSessions(allRecipients)
745
+ const [
746
+ { nodes: meNodes, shouldIncludeDeviceIdentity: s1 },
747
+ { nodes: otherNodes, shouldIncludeDeviceIdentity: s2 }
748
+ ] = await Promise.all([
749
+ createParticipantNodes(meRecipients, meMsg || message, extraAttrs),
750
+ createParticipantNodes(otherRecipients, message, extraAttrs, meMsg)
751
+ ])
752
+ participants.push(...meNodes,...otherNodes)
753
+ if (meRecipients.length > 0 || otherRecipients.length > 0) {
754
+ extraAttrs.phash = generateParticipantHashV2([...meRecipients,...otherRecipients])
755
+ }
756
+ shouldIncludeDeviceIdentity = shouldIncludeDeviceIdentity || s1 || s2
757
+ }
758
+ if (isRetryResend) {
759
+ const isParticipantLid = jidDecode(participant.jid).server === 'lid'
760
+ const isMe = areJidsSameUser(participant.jid, isParticipantLid? meLid : meId)
761
+ const encodedMessageToSend = isMe
762
+ ? encodeWAMessage({ deviceSentMessage: { destinationJid, message } })
763
+ : encodeWAMessage(message)
764
+ const { type, ciphertext: encryptedContent } = await signalRepository.encryptMessage({
765
+ data: encodedMessageToSend,
766
+ jid: participant.jid
767
+ })
768
+ binaryNodeContent.push({
769
+ tag: 'enc',
770
+ attrs: { v: '2', type, count: (participant.count?? 0).toString() },
771
+ content: encryptedContent
772
+ })
773
+ }
774
+ if (participants.length) {
775
+ if (additionalAttributes?.['category'] === 'peer') {
776
+ const peerNode = participants[0]?.content?.[0]
777
+ if (peerNode) binaryNodeContent.push(peerNode)
778
+ } else if (isInterop) {
779
+ const recipientNode = participants.find(p => isInteropUser(p?.attrs?.jid))
780
+ const encNode = (recipientNode?? participants[0])?.content?.[0]
781
+ if (encNode) binaryNodeContent.push(encNode)
782
+ } else {
783
+ binaryNodeContent.push({ tag: 'participants', attrs: {}, content: participants })
784
+ }
785
+ }
786
+ const stanza = {
787
+ tag: 'message',
788
+ attrs: { id: msgId, to: destinationJid, type: getMessageType(message),...(additionalAttributes || {}) },
789
+ content: binaryNodeContent
790
+ }
791
+ if (shouldIncludeDeviceIdentity) {
792
+ stanza.content.push({ tag: 'device-identity', attrs: {}, content: encodeSignedDeviceIdentity(authState.creds.account, true) })
793
+ logger.debug({ jid }, 'adding device identity')
794
+ }
795
+
796
+ if (isGroup && regexGroupOld.test(jid) &&!message.reactionMessage) {
797
+ stanza.content.push({ tag: 'multicast', attrs: {} })
798
+ }
799
+ if (pollMessage || messages.eventMessage) {
800
+ stanza.content.push({
801
+ tag: 'meta',
802
+ attrs: messages.eventMessage
803
+ ? { event_type: 'creation' }
804
+ : isNewsletter
805
+ ? { polltype: 'creation', contenttype: pollMessage?.pollContentType === 2? 'image' : 'text' }
806
+ : { polltype: 'creation' }
807
+ })
808
+ }
809
+ if (!isNewsletter &&!isRetryResend && reportingMessage?.messageContextInfo?.messageSecret && shouldIncludeReportingToken(reportingMessage)) {
810
+ try {
811
+ const encoded = encodeWAMessage(reportingMessage)
812
+ const reportingKey = { id: msgId, fromMe: true, remoteJid: destinationJid, participant: participant?.jid }
813
+ const reportingNode = await getMessageReportingToken(encoded, reportingMessage, reportingKey)
814
+ if (reportingNode) {
815
+ stanza.content.push(reportingNode)
816
+ logger.trace({ jid }, 'added reporting token to message')
817
+ }
818
+ } catch (error) {
819
+ logger.warn({ jid, trace: error?.stack }, 'failed to attach reporting token')
820
+ }
821
+ }
822
+ let didPushAdditional = false
823
+ if (!isNewsletter && buttonType) {
824
+ const buttonsNode = getButtonArgs(messages)
825
+ const filteredButtons = getBinaryNodeFilter(additionalNodes? additionalNodes : [])
826
+ if (filteredButtons) {
827
+ stanza.content.push(...additionalNodes)
828
+ didPushAdditional = true
829
+ } else {
830
+ stanza.content.push(buttonsNode)
831
+ }
832
+ }
833
+ if (!aiLabel && isPnUser(destinationJid)) {
834
+ const alreadyHasBizBot = getBinaryFilteredBizBot(additionalNodes || []) || getBinaryFilteredBizBot(stanza.content)
835
+ if (!alreadyHasBizBot) stanza.content.push({ tag: 'bot', attrs: { biz_bot: '1' } })
836
+ } else if (aiLabel &&!isGroup &&!isStatus &&!isNewsletter) {
837
+ const existingBizBot = getBinaryFilteredBizBot(additionalNodes || [])
838
+ if (!existingBizBot) stanza.content.push({ tag: 'bot', attrs: { biz_bot: '1' } })
839
+ }
840
+ const isPeerMessage = additionalAttributes?.['category'] === 'peer'
841
+ const is1on1Send =!isGroup &&!isRetryResend &&!isStatus &&!isNewsletter &&!isPeerMessage
842
+ const tcTokenJid = is1on1Send? await resolveTcTokenJid(destinationJid, getLIDForPN) : destinationJid
843
+ const contactTcTokenData = is1on1Send? await authState.keys.get('tctoken', [tcTokenJid]) : {}
844
+ const existingTokenEntry = contactTcTokenData[tcTokenJid]
845
+ let tcTokenBuffer = existingTokenEntry?.token
846
+ if (tcTokenBuffer?.length && isTcTokenExpired(existingTokenEntry?.timestamp)) {
847
+ logger.debug({ jid: destinationJid, timestamp: existingTokenEntry?.timestamp }, 'tctoken expired, clearing')
848
+ tcTokenBuffer = undefined
849
+ const cleared = existingTokenEntry?.senderTimestamp!== undefined? { token: Buffer.alloc(0), senderTimestamp: existingTokenEntry.senderTimestamp } : null
850
+ try {
851
+ await authState.keys.set({ tctoken: { [tcTokenJid]: cleared } })
852
+ } catch (err) {
853
+ logger.debug({ jid: destinationJid, err: err?.message }, 'failed to persist tctoken expiry cleanup')
854
+ }
855
+ }
856
+ if (tcTokenBuffer?.length && sock.serverProps.privacyTokenOn1to1) {
857
+ stanza.content.push({ tag: 'tctoken', attrs: {}, content: tcTokenBuffer })
858
+ }
859
+ if (additionalNodes && additionalNodes.length > 0 &&!didPushAdditional) {
860
+ stanza.content.push(...additionalNodes)
861
+ }
862
+ logger.debug({ msgId }, `sending message to ${participants.length} devices`)
863
+ await sendNode(stanza)
864
+ if (message.messageContextInfo?.messageSecret) {
865
+ setBotMessageSecret(msgId, message.messageContextInfo.messageSecret, destinationJid)
866
+ }
867
+ const isProtocolMsg =!!normalizeMessageContent(message)?.protocolMessage
868
+ const isBotOrPSA = destinationJid === PSA_WID || isJidBot(destinationJid) || isJidMetaAI(destinationJid)
869
+ if (is1on1Send &&!isProtocolMsg &&!isBotOrPSA && shouldSendNewTcToken(existingTokenEntry?.senderTimestamp) &&!inFlightTcTokenIssuance.has(tcTokenJid)) {
870
+ inFlightTcTokenIssuance.add(tcTokenJid)
871
+ const issueTimestamp = unixTimestampSeconds()
872
+ const getPNForLID = signalRepository.lidMapping.getPNForLID.bind(signalRepository.lidMapping)
873
+ resolveIssuanceJid(destinationJid, sock.serverProps.lidTrustedTokenIssueToLid, getLIDForPN, getPNForLID)
874
+ .then(issueJid => issuePrivacyTokens([issueJid], issueTimestamp))
875
+ .then(async result => {
876
+ await storeTcTokensFromIqResult({ result, fallbackJid: tcTokenJid, keys: authState.keys, getLIDForPN })
877
+ const currentData = await authState.keys.get('tctoken', [tcTokenJid])
878
+ const currentEntry = currentData[tcTokenJid]
879
+ const indexWrite = await buildMergedTcTokenIndexWrite(authState.keys, [tcTokenJid])
880
+ await authState.keys.set({
881
+ tctoken: {
882
+ [tcTokenJid]: { token: Buffer.alloc(0),...currentEntry, senderTimestamp: issueTimestamp },
883
+ ...indexWrite
884
+ }
885
+ })
886
+ })
887
+ .catch(err => logger.debug({ jid: destinationJid, err: err?.message }, 'fire-and-forget tctoken issuance failed'))
888
+ .finally(() => inFlightTcTokenIssuance.delete(tcTokenJid))
889
+ }
890
+ if (messageRetryManager &&!participant) {
891
+ messageRetryManager.addRecentMessage(destinationJid, msgId, message)
892
+ }
893
+ if (isInterop &&!isRetryResend) {
894
+ await trustInteropContact(destinationJid).catch(err => logger.debug({ err, jid: destinationJid }, 'failed to trust interop contact'))
895
+ }
896
+ }, meId)
897
+ return msgId
898
+ }
899
+ const getMessageType = (message) => {
900
+ const normalizedMessage = normalizeMessageContent(message);
901
+ if (!normalizedMessage)
902
+ return 'text';
903
+ if (normalizedMessage.reactionMessage || normalizedMessage.encReactionMessage) {
904
+ return 'reaction';
905
+ }
906
+ if (normalizedMessage.pollCreationMessage ||
907
+ normalizedMessage.pollCreationMessageV2 ||
908
+ normalizedMessage.pollCreationMessageV3 ||
909
+ normalizedMessage.pollCreationMessageV4 ||
910
+ normalizedMessage.pollCreationMessageV5 ||
911
+ normalizedMessage.pollUpdateMessage) {
912
+ return 'poll';
913
+ }
914
+ if (normalizedMessage.eventMessage) {
915
+ return 'event';
916
+ }
917
+ if (getMediaType(normalizedMessage) !== '') {
918
+ return 'media';
919
+ }
920
+ return 'text';
921
+ };
922
+ const getMediaType = (message) => {
923
+ if (message.imageMessage) {
924
+ return 'image';
925
+ }
926
+ else if (message.videoMessage) {
927
+ return message.videoMessage.gifPlayback ? 'gif' : 'video';
928
+ }
929
+ else if (message.audioMessage) {
930
+ return message.audioMessage.ptt ? 'ptt' : 'audio';
931
+ }
932
+ else if (message.contactMessage) {
933
+ return 'vcard';
934
+ }
935
+ else if (message.documentMessage) {
936
+ return 'document';
937
+ }
938
+ else if (message.contactsArrayMessage) {
939
+ return 'contact_array';
940
+ }
941
+ else if (message.liveLocationMessage) {
942
+ return 'livelocation';
943
+ }
944
+ else if (message.stickerMessage) {
945
+ return 'sticker';
946
+ }
947
+ else if (message.listMessage) {
948
+ return 'list';
949
+ }
950
+ else if (message.listResponseMessage) {
951
+ return 'list_response';
952
+ }
953
+ else if (message.buttonsResponseMessage) {
954
+ return 'buttons_response';
955
+ }
956
+ else if (message.orderMessage) {
957
+ return 'order';
958
+ }
959
+ else if (message.productMessage) {
960
+ return 'product';
961
+ }
962
+ else if (message.interactiveResponseMessage) {
963
+ return 'native_flow_response';
964
+ }
965
+ else if (message.groupInviteMessage) {
966
+ return 'url';
967
+ }
968
+ return '';
969
+ };
970
+ const getButtonType = (message) => {
971
+ if (message.listMessage) {
972
+ return 'list'
973
+ }
974
+ else if (message.buttonsMessage) {
975
+ return 'buttons'
976
+ }
977
+ else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'review_and_pay') {
978
+ return 'review_and_pay'
979
+ }
980
+ else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'review_order') {
981
+ return 'review_order'
982
+ }
983
+ else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'payment_info') {
984
+ return 'payment_info'
985
+ } else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'payment_key_info') {
986
+ return 'payment_key_info'
987
+ } else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'payment_status') {
988
+ return 'payment_status'
989
+ }
990
+ else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'payment_method') {
991
+ return 'payment_method'
992
+ }
993
+ else if (message.interactiveMessage?.nativeFlowMessage?.buttons?.[0]?.name === 'catalog_message') {
994
+ return 'catalog_message'
995
+ }
996
+ else if (message.interactiveMessage && message.interactiveMessage?.nativeFlowMessage) {
997
+ return 'interactive'
998
+ }
999
+ else if (message.interactiveMessage?.nativeFlowMessage) {
1000
+ return 'native_flow'
1001
+ }
1002
+ };
1003
+ const getButtonArgs = (message) => {
1004
+ const nativeFlow = message.interactiveMessage?.nativeFlowMessage
1005
+ const firstButtonName = nativeFlow?.buttons?.[0]?.name
1006
+ const nativeFlowSpecials = [
1007
+ 'mpm',
1008
+ 'cta_catalog',
1009
+ 'send_location',
1010
+ 'call_permission_request',
1011
+ 'wa_payment_transaction_details',
1012
+ 'automated_greeting_message_view_catalog'
1013
+ ]
1014
+
1015
+ if (nativeFlow && (firstButtonName === 'review_and_pay' || firstButtonName === 'payment_info')) {
1016
+ return {
1017
+ tag: 'biz',
1018
+ attrs: {
1019
+ native_flow_name: firstButtonName === 'review_and_pay' ? 'order_details' : firstButtonName
1020
+ }
1021
+ }
1022
+ } else if (nativeFlow && nativeFlowSpecials.includes(firstButtonName)) {
1023
+ // Only works for WhatsApp Original, not WhatsApp Business
1024
+ return {
1025
+ tag: 'biz',
1026
+ attrs: {
1027
+ actual_actors: '2',
1028
+ host_storage: '2',
1029
+ privacy_mode_ts: Utils_1.unixTimestampSeconds().toString()
1030
+ },
1031
+ content: [
1032
+ {
1033
+ tag: 'interactive',
1034
+ attrs: {
1035
+ type: 'native_flow',
1036
+ v: '1'
1037
+ },
1038
+ content: [
1039
+ {
1040
+ tag: 'native_flow',
1041
+ attrs: {
1042
+ v: '2',
1043
+ name: firstButtonName
1044
+ }
1045
+ }
1046
+ ]
1047
+ },
1048
+ {
1049
+ tag: 'quality_control',
1050
+ attrs: {
1051
+ source_type: 'third_party'
1052
+ }
1053
+ }
1054
+ ]
1055
+ }
1056
+ } else if (nativeFlow || message.buttonsMessage) {
1057
+ // It works for whatsapp original and whatsapp business
1058
+ return {
1059
+ tag: 'biz',
1060
+ attrs: {
1061
+ actual_actors: '2',
1062
+ host_storage: '2',
1063
+ privacy_mode_ts: Utils_1.unixTimestampSeconds().toString()
1064
+ },
1065
+ content: [
1066
+ {
1067
+ tag: 'interactive',
1068
+ attrs: {
1069
+ type: 'native_flow',
1070
+ v: '1'
1071
+ },
1072
+ content: [
1073
+ {
1074
+ tag: 'native_flow',
1075
+ attrs: {
1076
+ v: '9',
1077
+ name: 'mixed'
1078
+ }
1079
+ }
1080
+ ]
1081
+ },
1082
+ {
1083
+ tag: 'quality_control',
1084
+ attrs: {
1085
+ source_type: 'third_party'
1086
+ }
1087
+ }
1088
+ ]
1089
+ }
1090
+ } else if (message.listMessage) {
1091
+ return {
1092
+ tag: 'biz',
1093
+ attrs: {
1094
+ actual_actors: '2',
1095
+ host_storage: '2',
1096
+ privacy_mode_ts: Utils_1.unixTimestampSeconds().toString()
1097
+ },
1098
+ content: [
1099
+ {
1100
+ tag: 'list',
1101
+ attrs: {
1102
+ v: '2',
1103
+ type: 'product_list'
1104
+ }
1105
+ },
1106
+ {
1107
+ tag: 'quality_control',
1108
+ attrs: {
1109
+ source_type: 'third_party'
1110
+ }
1111
+ }
1112
+ ]
1113
+ }
1114
+ } else {
1115
+ return {
1116
+ tag: 'biz',
1117
+ attrs: {
1118
+ actual_actors: '2',
1119
+ host_storage: '2',
1120
+ privacy_mode_ts: Utils_1.unixTimestampSeconds().toString()
1121
+ }
1122
+ }
1123
+ }
1124
+ }
1125
+ const issuePrivacyTokens = async (jids, timestamp) => {
1126
+ const t = (timestamp ?? unixTimestampSeconds()).toString();
1127
+ const result = await query({
1128
+ tag: 'iq',
1129
+ attrs: {
1130
+ to: S_WHATSAPP_NET,
1131
+ type: 'set',
1132
+ xmlns: 'privacy'
1133
+ },
1134
+ content: [
1135
+ {
1136
+ tag: 'tokens',
1137
+ attrs: {},
1138
+ content: jids.map(jid => ({
1139
+ tag: 'token',
1140
+ attrs: {
1141
+ jid: jidNormalizedUser(jid),
1142
+ t,
1143
+ type: 'trusted_contact'
1144
+ }
1145
+ }))
1146
+ }
1147
+ ]
1148
+ });
1149
+ return result;
1150
+ };
1151
+ const waUploadToServer = getWAUploadToServer(config, refreshMediaConn);
1152
+ const waitForMsgMediaUpdate = bindWaitForEvent(ev, 'messages.media-update');
1153
+ registerSocketEndHandler(() => {
1154
+ if (!config.userDevicesCache && userDevicesCache.close) {
1155
+ userDevicesCache.close();
1156
+ }
1157
+ mediaConn = undefined;
1158
+ if (messageRetryManager) {
1159
+ messageRetryManager.clear();
1160
+ }
1161
+ });
1162
+ return {
1163
+ ...sock,
1164
+ userDevicesCache,
1165
+ devicesMutex,
1166
+ issuePrivacyTokens,
1167
+ assertSessions,
1168
+ relayMessage,
1169
+ sendReceipt,
1170
+ sendReceipts,
1171
+ readMessages,
1172
+ refreshMediaConn,
1173
+ // Function (not getter) so the spread in chats.ts preserves the live closure binding.
1174
+ getMediaHost: () => mediaHost,
1175
+ waUploadToServer,
1176
+ fetchPrivacySettings,
1177
+ sendPeerDataOperationMessage,
1178
+ createParticipantNodes,
1179
+ getUSyncDevices,
1180
+ messageRetryManager,
1181
+ updateMemberLabel,
1182
+ updateMediaMessage: async (message) => {
1183
+ const content = assertMediaContent(message.message);
1184
+ const mediaKey = content.mediaKey;
1185
+ const meId = authState.creds.me.id;
1186
+ const node = encryptMediaRetryRequest(message.key, mediaKey, meId);
1187
+ let error = undefined;
1188
+ await Promise.all([
1189
+ sendNode(node),
1190
+ waitForMsgMediaUpdate(async (update) => {
1191
+ const result = update.find(c => c.key.id === message.key.id);
1192
+ if (result) {
1193
+ if (result.error) {
1194
+ error = result.error;
1195
+ }
1196
+ else {
1197
+ try {
1198
+ const media = decryptMediaRetryData(result.media, mediaKey, result.key.id);
1199
+ if (media.result !== proto.MediaRetryNotification.ResultType.SUCCESS) {
1200
+ const resultStr = proto.MediaRetryNotification.ResultType[media.result];
1201
+ throw new Boom(`Media re-upload failed by device (${resultStr})`, {
1202
+ data: media,
1203
+ statusCode: getStatusCodeForMediaRetry(media.result) || 404
1204
+ });
1205
+ }
1206
+ content.directPath = media.directPath;
1207
+ content.url = getUrlFromDirectPath(content.directPath, mediaHost);
1208
+ logger.debug({ directPath: media.directPath, key: result.key }, 'media update successful');
1209
+ }
1210
+ catch (err) {
1211
+ error = err;
1212
+ }
1213
+ }
1214
+ return true;
1215
+ }
1216
+ })
1217
+ ]);
1218
+ if (error) {
1219
+ throw error;
1220
+ }
1221
+ ev.emit('messages.update', [{ key: message.key, update: { message: message.message } }]);
1222
+ return message;
1223
+ },
1224
+ sendTable: async (jid, title, headers, rows, quoted, options = {}) => {
1225
+ const { message, messageId } = Utils_1.generateTableContent(title, headers, rows, quoted, options)
1226
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1227
+ return { message, messageId }
1228
+ },
1229
+ sendList: async (jid, title, items, quoted, options = {}) => {
1230
+ const { message, messageId } = Utils_1.generateListContent(title, items, quoted, options)
1231
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1232
+ return { message, messageId }
1233
+ },
1234
+ sendCodeBlock: async (jid, code, quoted, options = {}) => {
1235
+ const { message, messageId } = Utils_1.generateCodeBlockContent(code, quoted, options)
1236
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1237
+ return { message, messageId }
1238
+ },
1239
+ sendLatex: async (jid, quoted, options) => {
1240
+ const { message, messageId } = Utils_1.generateLatexContent(quoted, options)
1241
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1242
+ return { message, messageId }
1243
+ },
1244
+ sendLatexImage: async (jid, quoted, options, renderLatexToPng, uploadFn) => {
1245
+ const { message, messageId } = await Utils_1.generateLatexImageContent(
1246
+ quoted,
1247
+ options,
1248
+ uploadFn,
1249
+ renderLatexToPng
1250
+ )
1251
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1252
+ return { message, messageId }
1253
+ },
1254
+ sendLatexInlineImage: async (jid, quoted, options, renderLatexToPng, uploadFn) => {
1255
+ const { message, messageId } = await Utils_1.generateLatexInlineImageContent(
1256
+ quoted,
1257
+ options,
1258
+ uploadFn,
1259
+ renderLatexToPng
1260
+ )
1261
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1262
+ return { message, messageId }
1263
+ },
1264
+ captureUnifiedResponse: Utils_1.captureUnifiedResponse,
1265
+ sendUnifiedResponse: async (jid, quoted, captured) => {
1266
+ const { message, messageId } = Utils_1.generateUnifiedResponseContent(quoted, captured)
1267
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1268
+ return { message, messageId }
1269
+ },
1270
+ sendRichMessage: async (jid, submessages, quoted, options = {}) => {
1271
+ const { message, messageId } = Utils_1.generateRichMessageContent(submessages, quoted, options)
1272
+ await relayMessage(jid, message, { messageId, notMe: options.notMe })
1273
+ return { message, messageId }
1274
+ },
1275
+ sendMessage: async (jid, content, options = {}) => {
1276
+ const userJid = authState.creds.me.id;
1277
+ const luki = new imup(Utils_1, waUploadToServer, relayMessage)
1278
+ const { quoted, participant = false } = options;
1279
+ const messageType = luki.detectType(content);
1280
+ if (typeof content === 'object' &&
1281
+ 'disappearingMessagesInChat' in content &&
1282
+ typeof content['disappearingMessagesInChat'] !== 'undefined' &&
1283
+ isJidGroup(jid)) {
1284
+ const { disappearingMessagesInChat } = content;
1285
+ const value = typeof disappearingMessagesInChat === 'boolean'
1286
+ ? disappearingMessagesInChat
1287
+ ? WA_DEFAULT_EPHEMERAL
1288
+ : 0
1289
+ : disappearingMessagesInChat;
1290
+ await groupToggleEphemeral(jid, value);
1291
+ }
1292
+ else {
1293
+ if (messageType) {
1294
+ switch(messageType) {
1295
+ case 'PAYMENT':
1296
+ const paymentContent = await luki.handlePayment(content, quoted);
1297
+ return await relayMessage(jid, paymentContent, {
1298
+ messageId: Utils_1.generateMessageID(),
1299
+ notMe: options.notMe
1300
+ });
1301
+ case 'PRODUCT':
1302
+ const productContent = await luki.handleProduct(content, jid, quoted);
1303
+ const productMsg = await Utils_1.generateWAMessageFromContent(jid, productContent, { quoted });
1304
+ return await relayMessage(jid, productMsg.message, {
1305
+ messageId: productMsg.key.id,
1306
+ notMe: options.notMe
1307
+ });
1308
+
1309
+ case 'ALBUM':
1310
+ return await luki.handleAlbum(content, jid, quoted)
1311
+ case 'EVENT':
1312
+ return await luki.handleEvent(content, jid, quoted)
1313
+ case 'POLL_RESULT':
1314
+ return await luki.handlePollResult(content, jid, quoted)
1315
+ case 'ORDER':
1316
+ return await luki.handleOrderMessage(content, jid, quoted)
1317
+ case 'GROUP_STATUS':
1318
+ return await luki.handleGroupStory(content, jid, quoted)
1319
+ case 'GROUP_LABEL':
1320
+ return await luki.handleGbLabel(content, jid)
1321
+ }
1322
+ }
1323
+ const fullMsg = await generateWAMessage(jid, content, {
1324
+ logger,
1325
+ userJid,
1326
+ getUrlInfo: text => getUrlInfo(text, {
1327
+ thumbnailWidth: linkPreviewImageThumbnailWidth,
1328
+ fetchOpts: {
1329
+ timeout: 3000,
1330
+ ...(httpRequestOptions || {})
1331
+ },
1332
+ logger,
1333
+ uploadImage: generateHighQualityLinkPreview ? waUploadToServer : undefined
1334
+ }),
1335
+ //TODO: CACHE
1336
+ getProfilePicUrl: sock.profilePictureUrl,
1337
+ getCallLink: sock.createCallLink,
1338
+ upload: waUploadToServer,
1339
+ mediaCache: config.mediaCache,
1340
+ options: config.options,
1341
+ messageId: generateMessageIDV2(sock.user?.id),
1342
+ ...options
1343
+ });
1344
+ const isEventMsg = 'event' in content && !!content.event;
1345
+ const isDeleteMsg = 'delete' in content && !!content.delete;
1346
+ const isEditMsg = 'edit' in content && !!content.edit;
1347
+ const isPinMsg = 'pin' in content && !!content.pin;
1348
+ const isPollMessage = 'poll' in content && !!content.poll;
1349
+ const additionalAttributes = {};
1350
+ const additionalNodes = [];
1351
+ // required for delete
1352
+ if (isDeleteMsg) {
1353
+ // if the chat is a group, and I am not the author, then delete the message as an admin
1354
+ if (isJidGroup(content.delete?.remoteJid) && !content.delete?.fromMe) {
1355
+ additionalAttributes.edit = '8';
1356
+ }
1357
+ else {
1358
+ additionalAttributes.edit = '7';
1359
+ }
1360
+ }
1361
+ else if (isEditMsg) {
1362
+ additionalAttributes.edit = '1';
1363
+ }
1364
+ else if (isPinMsg) {
1365
+ additionalAttributes.edit = '2';
1366
+ }
1367
+ else if (isPollMessage) {
1368
+ additionalNodes.push({
1369
+ tag: 'meta',
1370
+ attrs: {
1371
+ polltype: 'creation'
1372
+ }
1373
+ });
1374
+ }
1375
+ else if (isEventMsg) {
1376
+ additionalNodes.push({
1377
+ tag: 'meta',
1378
+ attrs: {
1379
+ event_type: 'creation'
1380
+ }
1381
+ });
1382
+ }
1383
+ await relayMessage(jid, fullMsg.message, {
1384
+ messageId: fullMsg.key.id,
1385
+ useCachedGroupMetadata: options.useCachedGroupMetadata,
1386
+ additionalAttributes,
1387
+ statusJidList: options.statusJidList,
1388
+ additionalNodes: aiLabel ? additionalNodes : options.additionalNodes,
1389
+ participant,
1390
+ notMe: options.notMe
1391
+ });
1392
+ if (config.emitOwnEvents) {
1393
+ process.nextTick(async () => {
1394
+ await messageMutex.mutex(() => upsertMessage(fullMsg, 'append'));
1395
+ });
1396
+ }
1397
+ return fullMsg;
1398
+ }
1399
+ },
1400
+ sendMessageMembers: async (jid, message, options = {}) => {
1401
+ const {
1402
+ messageId: idm,
1403
+ quoted,
1404
+ delayMs = 1500,
1405
+ useUserDevicesCache = true,
1406
+ cachedGroupMetadata,
1407
+ onlyMember = true
1408
+ } = options;
1409
+ const { server } = jidDecode(jid);
1410
+ if (server !== "g.us") throw new Error("@g.us server required");
1411
+ const meId = authState.creds.me.id;
1412
+ const messages = Utils_1.normalizeMessageContent(message);
1413
+ const groupData = cachedGroupMetadata? await cachedGroupMetadata(jid) : await groupMetadata(jid);
1414
+ const isLid = groupData.addressingMode === "lid";
1415
+ const adminJids = groupData.participants.filter((x) => x.admin !== null).map((y) => y.id)
1416
+ let participantJids = groupData.participants.map(z => z.id);
1417
+ if (onlyMember) {
1418
+ // Fix: array selalu truthy di JS meski kosong ([] ? true).
1419
+ // Cek .length supaya fallback ke semua member beneran jalan
1420
+ // kalau data admin kosong, bukan diam-diam kirim ke 0 orang.
1421
+ participantJids = adminJids.length ? adminJids : participantJids;
1422
+ }
1423
+ logger.info(`Sending message to ${participantJids.length} members from ${jid}`);
1424
+ for (let i = 0; i < participantJids.length; i++) {
1425
+ const jid = participantJids[i];
1426
+ if (areJidsSameUser(jid, meId)) continue;
1427
+ try {
1428
+ const msgId = `${idm || Utils_1.generateMessageID()}_${i}`;
1429
+ const fullMsg = await Utils_1.generateWAMessageFromContent(jid, message, {
1430
+ messageId: msgId,
1431
+ quoted
1432
+ })
1433
+ await relayMessage(jid, fullMsg.message, {
1434
+ messageId: fullMsg.key.id,
1435
+ useUserDevicesCache,
1436
+ notMe: options.notMe
1437
+ });
1438
+ logger.debug(`Message successfully sent to ${jid}`);
1439
+ if (delayMs && i < participantJids.length - 1) {
1440
+ await new Promise(z => setTimeout(z, delayMs));
1441
+ }
1442
+ } catch (e) {
1443
+ logger.error({ jid, e }, "Error sending message to");
1444
+ }
1445
+ }
1446
+ return JSON.stringify({
1447
+ members_total: participantJids.length,
1448
+ message
1449
+ }, null, 4);
1450
+ }
1451
+ };
1452
+ };
1453
+ //# sourceMappingURL=messages-send.js.map