@sgintokic/baileys 0.0.2

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.

Potentially problematic release.


This version of @sgintokic/baileys might be problematic. Click here for more details.

Files changed (106) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +1097 -0
  3. package/WAProto/index.js +2 -0
  4. package/engine-requirements.js +1 -0
  5. package/lib/Defaults/index.js +155 -0
  6. package/lib/Signal/Group/ciphertext-message.js +11 -0
  7. package/lib/Signal/Group/group-session-builder.js +41 -0
  8. package/lib/Signal/Group/group_cipher.js +108 -0
  9. package/lib/Signal/Group/index.js +11 -0
  10. package/lib/Signal/Group/keyhelper.js +14 -0
  11. package/lib/Signal/Group/sender-chain-key.js +31 -0
  12. package/lib/Signal/Group/sender-key-distribution-message.js +66 -0
  13. package/lib/Signal/Group/sender-key-message.js +79 -0
  14. package/lib/Signal/Group/sender-key-name.js +49 -0
  15. package/lib/Signal/Group/sender-key-record.js +46 -0
  16. package/lib/Signal/Group/sender-key-state.js +104 -0
  17. package/lib/Signal/Group/sender-message-key.js +29 -0
  18. package/lib/Signal/libsignal.js +485 -0
  19. package/lib/Signal/lid-mapping.js +291 -0
  20. package/lib/Socket/Client/index.js +2 -0
  21. package/lib/Socket/Client/types.js +10 -0
  22. package/lib/Socket/Client/websocket.js +64 -0
  23. package/lib/Socket/business.js +293 -0
  24. package/lib/Socket/chats.js +1068 -0
  25. package/lib/Socket/communities.js +476 -0
  26. package/lib/Socket/groups.js +383 -0
  27. package/lib/Socket/index.js +8 -0
  28. package/lib/Socket/messages-recv.js +1830 -0
  29. package/lib/Socket/messages-send.js +1462 -0
  30. package/lib/Socket/mex.js +55 -0
  31. package/lib/Socket/newsletter.js +277 -0
  32. package/lib/Socket/socket.js +1087 -0
  33. package/lib/Store/index.js +3 -0
  34. package/lib/Store/make-in-memory-store.js +517 -0
  35. package/lib/Store/make-ordered-dictionary.js +75 -0
  36. package/lib/Store/object-repository.js +23 -0
  37. package/lib/Types/Auth.js +1 -0
  38. package/lib/Types/Bussines.js +1 -0
  39. package/lib/Types/Call.js +1 -0
  40. package/lib/Types/Chat.js +7 -0
  41. package/lib/Types/Contact.js +1 -0
  42. package/lib/Types/Events.js +1 -0
  43. package/lib/Types/GroupMetadata.js +1 -0
  44. package/lib/Types/Label.js +24 -0
  45. package/lib/Types/LabelAssociation.js +6 -0
  46. package/lib/Types/Message.js +18 -0
  47. package/lib/Types/Newsletter.js +33 -0
  48. package/lib/Types/Product.js +1 -0
  49. package/lib/Types/Signal.js +1 -0
  50. package/lib/Types/Socket.js +2 -0
  51. package/lib/Types/State.js +15 -0
  52. package/lib/Types/USync.js +1 -0
  53. package/lib/Types/index.js +31 -0
  54. package/lib/Utils/auth-utils.js +293 -0
  55. package/lib/Utils/browser-utils.js +32 -0
  56. package/lib/Utils/business.js +245 -0
  57. package/lib/Utils/chat-utils.js +959 -0
  58. package/lib/Utils/crypto.js +133 -0
  59. package/lib/Utils/decode-wa-message.js +376 -0
  60. package/lib/Utils/event-buffer.js +620 -0
  61. package/lib/Utils/generics.js +417 -0
  62. package/lib/Utils/history.js +150 -0
  63. package/lib/Utils/identity-change-handler.js +63 -0
  64. package/lib/Utils/index.js +21 -0
  65. package/lib/Utils/link-preview.js +91 -0
  66. package/lib/Utils/logger.js +2 -0
  67. package/lib/Utils/lt-hash.js +6 -0
  68. package/lib/Utils/make-mutex.js +31 -0
  69. package/lib/Utils/message-retry-manager.js +240 -0
  70. package/lib/Utils/messages-media.js +901 -0
  71. package/lib/Utils/messages.js +2052 -0
  72. package/lib/Utils/noise-handler.js +229 -0
  73. package/lib/Utils/offline-node-processor.js +50 -0
  74. package/lib/Utils/pre-key-manager.js +119 -0
  75. package/lib/Utils/process-message.js +641 -0
  76. package/lib/Utils/reporting-utils.js +346 -0
  77. package/lib/Utils/signal.js +188 -0
  78. package/lib/Utils/stanza-ack.js +33 -0
  79. package/lib/Utils/sync-action-utils.js +53 -0
  80. package/lib/Utils/tc-token-utils.js +15 -0
  81. package/lib/Utils/use-multi-file-auth-state.js +116 -0
  82. package/lib/Utils/use-single-file-auth-state.js +94 -0
  83. package/lib/Utils/validate-connection.js +235 -0
  84. package/lib/WABinary/constants.js +1300 -0
  85. package/lib/WABinary/decode.js +258 -0
  86. package/lib/WABinary/encode.js +219 -0
  87. package/lib/WABinary/generic-utils.js +203 -0
  88. package/lib/WABinary/index.js +5 -0
  89. package/lib/WABinary/jid-utils.js +93 -0
  90. package/lib/WABinary/types.js +1 -0
  91. package/lib/WAM/BinaryInfo.js +9 -0
  92. package/lib/WAM/constants.js +20669 -0
  93. package/lib/WAM/encode.js +151 -0
  94. package/lib/WAM/index.js +3 -0
  95. package/lib/WAUSync/Protocols/USyncContactProtocol.js +21 -0
  96. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +50 -0
  97. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +20 -0
  98. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +29 -0
  99. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +59 -0
  100. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +21 -0
  101. package/lib/WAUSync/Protocols/index.js +4 -0
  102. package/lib/WAUSync/USyncQuery.js +103 -0
  103. package/lib/WAUSync/USyncUser.js +22 -0
  104. package/lib/WAUSync/index.js +3 -0
  105. package/lib/index.js +11 -0
  106. package/package.json +58 -0
@@ -0,0 +1,620 @@
1
+ import EventEmitter from "events";
2
+ import { WAMessageStatus } from "../Types/index.js";
3
+ import { trimUndefined } from "./generics.js";
4
+ import {
5
+ updateMessageWithReaction,
6
+ updateMessageWithReceipt,
7
+ } from "./messages.js";
8
+ import { isRealMessage, shouldIncrementChatUnread } from "./process-message.js";
9
+ const BUFFERABLE_EVENT = [
10
+ "messaging-history.set",
11
+ "chats.upsert",
12
+ "chats.update",
13
+ "chats.delete",
14
+ "contacts.upsert",
15
+ "contacts.update",
16
+ "messages.upsert",
17
+ "messages.update",
18
+ "messages.delete",
19
+ "messages.reaction",
20
+ "message-receipt.update",
21
+ "groups.update",
22
+ ];
23
+ const BUFFERABLE_EVENT_SET = new Set(BUFFERABLE_EVENT);
24
+ /**
25
+ * The event buffer logically consolidates different events into a single event
26
+ * making the data processing more efficient.
27
+ */ export const makeEventBuffer = (logger) => {
28
+ const ev = new EventEmitter();
29
+ const historyCache = new Set();
30
+ let data = makeBufferData();
31
+ let isBuffering = false;
32
+ let bufferTimeout = null;
33
+ let flushPendingTimeout = null; // Add a specific timer for the debounced flush to prevent leak
34
+ let bufferCount = 0;
35
+ const MAX_HISTORY_CACHE_SIZE = 1e4; // Limit the history cache size to prevent memory bloat
36
+ const BUFFER_TIMEOUT_MS = 3e4; // 30 seconds
37
+ // take the generic event and fire it as a baileys event
38
+ ev.on("event", (map) => {
39
+ for (const event in map) {
40
+ ev.emit(event, map[event]);
41
+ }
42
+ });
43
+ function buffer() {
44
+ if (!isBuffering) {
45
+ logger.debug("Event buffer activated");
46
+ isBuffering = true;
47
+ bufferCount = 0;
48
+ if (bufferTimeout) {
49
+ clearTimeout(bufferTimeout);
50
+ }
51
+ bufferTimeout = setTimeout(() => {
52
+ if (isBuffering) {
53
+ logger.warn("Buffer timeout reached, auto-flushing");
54
+ flush();
55
+ }
56
+ }, BUFFER_TIMEOUT_MS);
57
+ }
58
+ // Always increment count when requested
59
+ bufferCount++;
60
+ }
61
+ function flush() {
62
+ if (!isBuffering) {
63
+ return false;
64
+ }
65
+ logger.debug({ bufferCount: bufferCount }, "Flushing event buffer");
66
+ isBuffering = false;
67
+ bufferCount = 0;
68
+ // Clear timeout
69
+ if (bufferTimeout) {
70
+ clearTimeout(bufferTimeout);
71
+ bufferTimeout = null;
72
+ }
73
+ if (flushPendingTimeout) {
74
+ clearTimeout(flushPendingTimeout);
75
+ flushPendingTimeout = null;
76
+ }
77
+ // Clear history cache if it exceeds the max size
78
+ if (historyCache.size > MAX_HISTORY_CACHE_SIZE) {
79
+ logger.debug({ cacheSize: historyCache.size }, "Clearing history cache");
80
+ historyCache.clear();
81
+ }
82
+ const newData = makeBufferData();
83
+ const chatUpdates = Object.values(data.chatUpdates);
84
+ let conditionalChatUpdatesLeft = 0;
85
+ for (const update of chatUpdates) {
86
+ if (update.conditional) {
87
+ conditionalChatUpdatesLeft += 1;
88
+ newData.chatUpdates[update.id] = update;
89
+ delete data.chatUpdates[update.id];
90
+ }
91
+ }
92
+ const consolidatedData = consolidateEvents(data);
93
+ if (Object.keys(consolidatedData).length) {
94
+ ev.emit("event", consolidatedData);
95
+ }
96
+ data = newData;
97
+ logger.trace(
98
+ { conditionalChatUpdatesLeft: conditionalChatUpdatesLeft },
99
+ "released buffered events",
100
+ );
101
+ return true;
102
+ }
103
+ return {
104
+ process(handler) {
105
+ const listener = async (map) => {
106
+ await handler(map);
107
+ };
108
+ ev.on("event", listener);
109
+ return () => {
110
+ ev.off("event", listener);
111
+ };
112
+ },
113
+ emit(event, evData) {
114
+ // Check if this is a messages.upsert with a different type than what's buffered
115
+ // If so, flush the buffered messages first to avoid type overshadowing
116
+ if (event === "messages.upsert") {
117
+ const { type } = evData;
118
+ const existingUpserts = Object.values(data.messageUpserts);
119
+ if (existingUpserts.length > 0) {
120
+ const bufferedType = existingUpserts[0].type;
121
+ if (bufferedType !== type) {
122
+ logger.debug(
123
+ { bufferedType: bufferedType, newType: type },
124
+ "messages.upsert type mismatch, emitting buffered messages",
125
+ );
126
+ // Emit the buffered messages with their correct type
127
+ ev.emit("event", {
128
+ "messages.upsert": {
129
+ messages: existingUpserts.map((m) => m.message),
130
+ type: bufferedType,
131
+ },
132
+ });
133
+ // Clear the message upserts from the buffer
134
+ data.messageUpserts = {};
135
+ }
136
+ }
137
+ }
138
+ if (isBuffering && BUFFERABLE_EVENT_SET.has(event)) {
139
+ append(data, historyCache, event, evData, logger);
140
+ return true;
141
+ }
142
+ return ev.emit("event", { [event]: evData });
143
+ },
144
+ isBuffering() {
145
+ return isBuffering;
146
+ },
147
+ buffer: buffer,
148
+ flush: flush,
149
+ createBufferedFunction(work) {
150
+ return async (...args) => {
151
+ buffer();
152
+ try {
153
+ const result = await work(...args);
154
+ // If this is the only buffer, flush after a small delay
155
+ if (bufferCount === 1) {
156
+ setTimeout(() => {
157
+ if (isBuffering && bufferCount === 1) {
158
+ flush();
159
+ }
160
+ }, 100); // Small delay to allow nested buffers
161
+ }
162
+ return result;
163
+ } catch (error) {
164
+ throw error;
165
+ } finally {
166
+ bufferCount = Math.max(0, bufferCount - 1);
167
+ if (bufferCount === 0) {
168
+ // Only schedule ONE timeout, not 10,000
169
+ if (!flushPendingTimeout) {
170
+ flushPendingTimeout = setTimeout(flush, 100);
171
+ }
172
+ }
173
+ }
174
+ };
175
+ },
176
+ on: (...args) => ev.on(...args),
177
+ off: (...args) => ev.off(...args),
178
+ removeAllListeners: (...args) => ev.removeAllListeners(...args),
179
+ };
180
+ };
181
+ const makeBufferData = () => {
182
+ return {
183
+ historySets: {
184
+ chats: {},
185
+ messages: {},
186
+ contacts: {},
187
+ isLatest: false,
188
+ empty: true,
189
+ },
190
+ chatUpserts: {},
191
+ chatUpdates: {},
192
+ chatDeletes: new Set(),
193
+ contactUpserts: {},
194
+ contactUpdates: {},
195
+ messageUpserts: {},
196
+ messageUpdates: {},
197
+ messageReactions: {},
198
+ messageDeletes: {},
199
+ messageReceipts: {},
200
+ groupUpdates: {},
201
+ };
202
+ };
203
+ function append(
204
+ data,
205
+ historyCache,
206
+ event,
207
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
208
+ eventData,
209
+ logger,
210
+ ) {
211
+ switch (event) {
212
+ case "messaging-history.set":
213
+ for (const chat of eventData.chats) {
214
+ const id = chat.id || "";
215
+ const existingChat = data.historySets.chats[id];
216
+ if (existingChat) {
217
+ existingChat.endOfHistoryTransferType = chat.endOfHistoryTransferType;
218
+ }
219
+ if (!existingChat && !historyCache.has(id)) {
220
+ data.historySets.chats[id] = chat;
221
+ historyCache.add(id);
222
+ absorbingChatUpdate(chat);
223
+ }
224
+ }
225
+ for (const contact of eventData.contacts) {
226
+ const existingContact = data.historySets.contacts[contact.id];
227
+ if (existingContact) {
228
+ Object.assign(existingContact, trimUndefined(contact));
229
+ } else {
230
+ const historyContactId = `c:${contact.id}`;
231
+ const hasAnyName =
232
+ contact.notify || contact.name || contact.verifiedName;
233
+ if (!historyCache.has(historyContactId) || hasAnyName) {
234
+ data.historySets.contacts[contact.id] = contact;
235
+ historyCache.add(historyContactId);
236
+ }
237
+ }
238
+ }
239
+ for (const message of eventData.messages) {
240
+ const key = stringifyMessageKey(message.key);
241
+ const existingMsg = data.historySets.messages[key];
242
+ if (!existingMsg && !historyCache.has(key)) {
243
+ data.historySets.messages[key] = message;
244
+ historyCache.add(key);
245
+ }
246
+ }
247
+ data.historySets.empty = false;
248
+ data.historySets.syncType = eventData.syncType;
249
+ data.historySets.progress = eventData.progress;
250
+ data.historySets.peerDataRequestSessionId =
251
+ eventData.peerDataRequestSessionId;
252
+ data.historySets.isLatest =
253
+ eventData.isLatest || data.historySets.isLatest;
254
+ break;
255
+ case "chats.upsert":
256
+ for (const chat of eventData) {
257
+ const id = chat.id || "";
258
+ let upsert = data.chatUpserts[id];
259
+ if (id && !upsert) {
260
+ upsert = data.historySets.chats[id];
261
+ if (upsert) {
262
+ logger.debug({ chatId: id }, "absorbed chat upsert in chat set");
263
+ }
264
+ }
265
+ if (upsert) {
266
+ upsert = concatChats(upsert, chat);
267
+ } else {
268
+ upsert = chat;
269
+ data.chatUpserts[id] = upsert;
270
+ }
271
+ absorbingChatUpdate(upsert);
272
+ if (data.chatDeletes.has(id)) {
273
+ data.chatDeletes.delete(id);
274
+ }
275
+ }
276
+ break;
277
+ case "chats.update":
278
+ for (const update of eventData) {
279
+ const chatId = update.id;
280
+ const conditionMatches = update.conditional
281
+ ? update.conditional(data)
282
+ : true;
283
+ if (conditionMatches) {
284
+ delete update.conditional;
285
+ // if there is an existing upsert, merge the update into it
286
+ const upsert =
287
+ data.historySets.chats[chatId] || data.chatUpserts[chatId];
288
+ if (upsert) {
289
+ concatChats(upsert, update);
290
+ } else {
291
+ // merge the update into the existing update
292
+ const chatUpdate = data.chatUpdates[chatId] || {};
293
+ data.chatUpdates[chatId] = concatChats(chatUpdate, update);
294
+ }
295
+ } else if (conditionMatches === undefined) {
296
+ // condition yet to be fulfilled
297
+ data.chatUpdates[chatId] = update;
298
+ }
299
+ // otherwise -- condition not met, update is invalid
300
+ // if the chat has been updated
301
+ // ignore any existing chat delete
302
+ if (data.chatDeletes.has(chatId)) {
303
+ data.chatDeletes.delete(chatId);
304
+ }
305
+ }
306
+ break;
307
+ case "chats.delete":
308
+ for (const chatId of eventData) {
309
+ if (!data.chatDeletes.has(chatId)) {
310
+ data.chatDeletes.add(chatId);
311
+ }
312
+ // remove any prior updates & upserts
313
+ if (data.chatUpdates[chatId]) {
314
+ delete data.chatUpdates[chatId];
315
+ }
316
+ if (data.chatUpserts[chatId]) {
317
+ delete data.chatUpserts[chatId];
318
+ }
319
+ if (data.historySets.chats[chatId]) {
320
+ delete data.historySets.chats[chatId];
321
+ }
322
+ }
323
+ break;
324
+ case "contacts.upsert":
325
+ for (const contact of eventData) {
326
+ let upsert = data.contactUpserts[contact.id];
327
+ if (!upsert) {
328
+ upsert = data.historySets.contacts[contact.id];
329
+ if (upsert) {
330
+ logger.debug(
331
+ { contactId: contact.id },
332
+ "absorbed contact upsert in contact set",
333
+ );
334
+ }
335
+ }
336
+ if (upsert) {
337
+ upsert = Object.assign(upsert, trimUndefined(contact));
338
+ } else {
339
+ upsert = contact;
340
+ data.contactUpserts[contact.id] = upsert;
341
+ }
342
+ if (data.contactUpdates[contact.id]) {
343
+ upsert = Object.assign(
344
+ data.contactUpdates[contact.id],
345
+ trimUndefined(contact),
346
+ );
347
+ delete data.contactUpdates[contact.id];
348
+ }
349
+ }
350
+ break;
351
+ case "contacts.update":
352
+ const contactUpdates = eventData;
353
+ for (const update of contactUpdates) {
354
+ const id = update.id;
355
+ // merge into prior upsert
356
+ const upsert = data.historySets.contacts[id] || data.contactUpserts[id];
357
+ if (upsert) {
358
+ Object.assign(upsert, update);
359
+ } else {
360
+ // merge into prior update
361
+ const contactUpdate = data.contactUpdates[id] || {};
362
+ data.contactUpdates[id] = Object.assign(contactUpdate, update);
363
+ }
364
+ }
365
+ break;
366
+ case "messages.upsert":
367
+ const { messages, type } = eventData;
368
+ for (const message of messages) {
369
+ const key = stringifyMessageKey(message.key);
370
+ let existing = data.messageUpserts[key]?.message;
371
+ if (!existing) {
372
+ existing = data.historySets.messages[key];
373
+ if (existing) {
374
+ logger.debug(
375
+ { messageId: key },
376
+ "absorbed message upsert in message set",
377
+ );
378
+ }
379
+ }
380
+ if (existing) {
381
+ message.messageTimestamp = existing.messageTimestamp;
382
+ }
383
+ if (data.messageUpdates[key]) {
384
+ logger.debug("absorbed prior message update in message upsert");
385
+ Object.assign(message, data.messageUpdates[key].update);
386
+ delete data.messageUpdates[key];
387
+ }
388
+ if (data.historySets.messages[key]) {
389
+ data.historySets.messages[key] = message;
390
+ } else {
391
+ data.messageUpserts[key] = {
392
+ message: message,
393
+ type:
394
+ type === "notify" || data.messageUpserts[key]?.type === "notify"
395
+ ? "notify"
396
+ : type,
397
+ };
398
+ }
399
+ }
400
+ break;
401
+ case "messages.update":
402
+ const msgUpdates = eventData;
403
+ for (const { key, update } of msgUpdates) {
404
+ const keyStr = stringifyMessageKey(key);
405
+ const existing =
406
+ data.historySets.messages[keyStr] ||
407
+ data.messageUpserts[keyStr]?.message;
408
+ if (existing) {
409
+ Object.assign(existing, update);
410
+ // if the message was received & read by us
411
+ // the chat counter must have been incremented
412
+ // so we need to decrement it
413
+ if (update.status === WAMessageStatus.READ && !key.fromMe) {
414
+ decrementChatReadCounterIfMsgDidUnread(existing);
415
+ }
416
+ } else {
417
+ const msgUpdate = data.messageUpdates[keyStr] || {
418
+ key: key,
419
+ update: {},
420
+ };
421
+ Object.assign(msgUpdate.update, update);
422
+ data.messageUpdates[keyStr] = msgUpdate;
423
+ }
424
+ }
425
+ break;
426
+ case "messages.delete":
427
+ const deleteData = eventData;
428
+ if ("keys" in deleteData) {
429
+ const { keys } = deleteData;
430
+ for (const key of keys) {
431
+ const keyStr = stringifyMessageKey(key);
432
+ if (!data.messageDeletes[keyStr]) {
433
+ data.messageDeletes[keyStr] = key;
434
+ }
435
+ if (data.messageUpserts[keyStr]) {
436
+ delete data.messageUpserts[keyStr];
437
+ }
438
+ if (data.messageUpdates[keyStr]) {
439
+ delete data.messageUpdates[keyStr];
440
+ }
441
+ }
442
+ } else {
443
+ // TODO: add support
444
+ }
445
+ break;
446
+ case "messages.reaction":
447
+ const reactions = eventData;
448
+ for (const { key, reaction } of reactions) {
449
+ const keyStr = stringifyMessageKey(key);
450
+ const existing = data.messageUpserts[keyStr];
451
+ if (existing) {
452
+ updateMessageWithReaction(existing.message, reaction);
453
+ } else {
454
+ data.messageReactions[keyStr] = data.messageReactions[keyStr] || {
455
+ key: key,
456
+ reactions: [],
457
+ };
458
+ updateMessageWithReaction(data.messageReactions[keyStr], reaction);
459
+ }
460
+ }
461
+ break;
462
+ case "message-receipt.update":
463
+ const receipts = eventData;
464
+ for (const { key, receipt } of receipts) {
465
+ const keyStr = stringifyMessageKey(key);
466
+ const existing = data.messageUpserts[keyStr];
467
+ if (existing) {
468
+ updateMessageWithReceipt(existing.message, receipt);
469
+ } else {
470
+ data.messageReceipts[keyStr] = data.messageReceipts[keyStr] || {
471
+ key: key,
472
+ userReceipt: [],
473
+ };
474
+ updateMessageWithReceipt(data.messageReceipts[keyStr], receipt);
475
+ }
476
+ }
477
+ break;
478
+ case "groups.update":
479
+ const groupUpdates = eventData;
480
+ for (const update of groupUpdates) {
481
+ const id = update.id;
482
+ const groupUpdate = data.groupUpdates[id] || {};
483
+ if (!data.groupUpdates[id]) {
484
+ data.groupUpdates[id] = Object.assign(groupUpdate, update);
485
+ }
486
+ }
487
+ break;
488
+ default:
489
+ throw new Error(`"${event}" cannot be buffered`);
490
+ }
491
+ function absorbingChatUpdate(existing) {
492
+ const chatId = existing.id || "";
493
+ const update = data.chatUpdates[chatId];
494
+ if (update) {
495
+ const conditionMatches = update.conditional
496
+ ? update.conditional(data)
497
+ : true;
498
+ if (conditionMatches) {
499
+ delete update.conditional;
500
+ logger.debug(
501
+ { chatId: chatId },
502
+ "absorbed chat update in existing chat",
503
+ );
504
+ Object.assign(existing, concatChats(update, existing));
505
+ delete data.chatUpdates[chatId];
506
+ } else if (conditionMatches === false) {
507
+ logger.debug(
508
+ { chatId: chatId },
509
+ "chat update condition fail, removing",
510
+ );
511
+ delete data.chatUpdates[chatId];
512
+ }
513
+ }
514
+ }
515
+ function decrementChatReadCounterIfMsgDidUnread(message) {
516
+ // decrement chat unread counter
517
+ // if the message has already been marked read by us
518
+ const chatId = message.key.remoteJid;
519
+ const chat = data.chatUpdates[chatId] || data.chatUpserts[chatId];
520
+ if (
521
+ isRealMessage(message) &&
522
+ shouldIncrementChatUnread(message) &&
523
+ typeof chat?.unreadCount === "number" &&
524
+ chat.unreadCount > 0
525
+ ) {
526
+ logger.debug({ chatId: chat.id }, "decrementing chat counter");
527
+ chat.unreadCount -= 1;
528
+ if (chat.unreadCount === 0) {
529
+ delete chat.unreadCount;
530
+ }
531
+ }
532
+ }
533
+ }
534
+ function consolidateEvents(data) {
535
+ const map = {};
536
+ if (!data.historySets.empty) {
537
+ map["messaging-history.set"] = {
538
+ chats: Object.values(data.historySets.chats),
539
+ messages: Object.values(data.historySets.messages),
540
+ contacts: Object.values(data.historySets.contacts),
541
+ syncType: data.historySets.syncType,
542
+ progress: data.historySets.progress,
543
+ isLatest: data.historySets.isLatest,
544
+ peerDataRequestSessionId: data.historySets.peerDataRequestSessionId,
545
+ };
546
+ }
547
+ const chatUpsertList = Object.values(data.chatUpserts);
548
+ if (chatUpsertList.length) {
549
+ map["chats.upsert"] = chatUpsertList;
550
+ }
551
+ const chatUpdateList = Object.values(data.chatUpdates);
552
+ if (chatUpdateList.length) {
553
+ map["chats.update"] = chatUpdateList;
554
+ }
555
+ const chatDeleteList = Array.from(data.chatDeletes);
556
+ if (chatDeleteList.length) {
557
+ map["chats.delete"] = chatDeleteList;
558
+ }
559
+ const messageUpsertList = Object.values(data.messageUpserts);
560
+ if (messageUpsertList.length) {
561
+ const type = messageUpsertList[0].type;
562
+ map["messages.upsert"] = {
563
+ messages: messageUpsertList.map((m) => m.message),
564
+ type: type,
565
+ };
566
+ }
567
+ const messageUpdateList = Object.values(data.messageUpdates);
568
+ if (messageUpdateList.length) {
569
+ map["messages.update"] = messageUpdateList;
570
+ }
571
+ const messageDeleteList = Object.values(data.messageDeletes);
572
+ if (messageDeleteList.length) {
573
+ map["messages.delete"] = { keys: messageDeleteList };
574
+ }
575
+ const messageReactionList = Object.values(data.messageReactions).flatMap(
576
+ ({ key, reactions }) =>
577
+ reactions.flatMap((reaction) => ({ key: key, reaction: reaction })),
578
+ );
579
+ if (messageReactionList.length) {
580
+ map["messages.reaction"] = messageReactionList;
581
+ }
582
+ const messageReceiptList = Object.values(data.messageReceipts).flatMap(
583
+ ({ key, userReceipt }) =>
584
+ userReceipt.flatMap((receipt) => ({ key: key, receipt: receipt })),
585
+ );
586
+ if (messageReceiptList.length) {
587
+ map["message-receipt.update"] = messageReceiptList;
588
+ }
589
+ const contactUpsertList = Object.values(data.contactUpserts);
590
+ if (contactUpsertList.length) {
591
+ map["contacts.upsert"] = contactUpsertList;
592
+ }
593
+ const contactUpdateList = Object.values(data.contactUpdates);
594
+ if (contactUpdateList.length) {
595
+ map["contacts.update"] = contactUpdateList;
596
+ }
597
+ const groupUpdateList = Object.values(data.groupUpdates);
598
+ if (groupUpdateList.length) {
599
+ map["groups.update"] = groupUpdateList;
600
+ }
601
+ return map;
602
+ }
603
+ function concatChats(a, b) {
604
+ if (
605
+ b.unreadCount === null && // neutralize unread counter
606
+ a.unreadCount < 0
607
+ ) {
608
+ a.unreadCount = undefined;
609
+ b.unreadCount = undefined;
610
+ }
611
+ if (typeof a.unreadCount === "number" && typeof b.unreadCount === "number") {
612
+ b = { ...b };
613
+ if (b.unreadCount >= 0) {
614
+ b.unreadCount = Math.max(b.unreadCount, 0) + Math.max(a.unreadCount, 0);
615
+ }
616
+ }
617
+ return Object.assign(a, b);
618
+ }
619
+ const stringifyMessageKey = (key) =>
620
+ `${key.remoteJid},${key.id},${key.fromMe ? "1" : "0"}`;