@skyzopedia/baileys-mod 5.0.8 → 6.0.1

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 (212) hide show
  1. package/WAProto/index.js +133384 -57814
  2. package/engine-requirements.js +10 -0
  3. package/lib/Defaults/baileys-version.json +3 -0
  4. package/lib/Defaults/index.d.ts +53 -0
  5. package/lib/Defaults/index.js +141 -117
  6. package/lib/Defaults/phonenumber-mcc.json +223 -0
  7. package/lib/Signal/Group/ciphertext-message.d.ts +9 -0
  8. package/lib/Signal/Group/ciphertext-message.js +14 -12
  9. package/lib/Signal/Group/group-session-builder.d.ts +14 -0
  10. package/lib/Signal/Group/group-session-builder.js +42 -10
  11. package/lib/Signal/Group/group_cipher.d.ts +17 -0
  12. package/lib/Signal/Group/group_cipher.js +87 -75
  13. package/lib/Signal/Group/index.d.ts +11 -0
  14. package/lib/Signal/Group/index.js +57 -13
  15. package/lib/Signal/Group/keyhelper.d.ts +10 -0
  16. package/lib/Signal/Group/keyhelper.js +52 -17
  17. package/lib/Signal/Group/queue-job.d.ts +1 -0
  18. package/lib/Signal/Group/queue-job.js +57 -0
  19. package/lib/Signal/Group/sender-chain-key.d.ts +13 -0
  20. package/lib/Signal/Group/sender-chain-key.js +33 -27
  21. package/lib/Signal/Group/sender-key-distribution-message.d.ts +16 -0
  22. package/lib/Signal/Group/sender-key-distribution-message.js +63 -62
  23. package/lib/Signal/Group/sender-key-message.d.ts +18 -0
  24. package/lib/Signal/Group/sender-key-message.js +66 -65
  25. package/lib/Signal/Group/sender-key-name.d.ts +17 -0
  26. package/lib/Signal/Group/sender-key-name.js +44 -45
  27. package/lib/Signal/Group/sender-key-record.d.ts +30 -0
  28. package/lib/Signal/Group/sender-key-record.js +49 -39
  29. package/lib/Signal/Group/sender-key-state.d.ts +38 -0
  30. package/lib/Signal/Group/sender-key-state.js +93 -80
  31. package/lib/Signal/Group/sender-message-key.d.ts +11 -0
  32. package/lib/Signal/Group/sender-message-key.js +28 -27
  33. package/lib/Signal/libsignal.d.ts +3 -0
  34. package/lib/Signal/libsignal.js +163 -313
  35. package/lib/Socket/Client/abstract-socket-client.d.ts +17 -0
  36. package/lib/Socket/Client/abstract-socket-client.js +13 -0
  37. package/lib/Socket/Client/index.d.ts +3 -0
  38. package/lib/Socket/Client/index.js +19 -4
  39. package/lib/Socket/Client/mobile-socket-client.d.ts +13 -0
  40. package/lib/Socket/Client/mobile-socket-client.js +65 -0
  41. package/lib/Socket/Client/web-socket-client.d.ts +12 -0
  42. package/lib/Socket/Client/web-socket-client.js +62 -0
  43. package/lib/Socket/business.d.ts +171 -0
  44. package/lib/Socket/business.js +242 -359
  45. package/lib/Socket/chats.d.ts +267 -0
  46. package/lib/Socket/chats.js +935 -846
  47. package/lib/Socket/dugong.d.ts +254 -0
  48. package/lib/Socket/dugong.js +484 -0
  49. package/lib/Socket/groups.d.ts +115 -0
  50. package/lib/Socket/groups.js +309 -304
  51. package/lib/Socket/index.d.ts +173 -0
  52. package/lib/Socket/index.js +10 -15
  53. package/lib/Socket/messages-recv.d.ts +161 -0
  54. package/lib/Socket/messages-recv.js +1054 -1107
  55. package/lib/Socket/messages-send.d.ts +149 -0
  56. package/lib/Socket/messages-send.js +447 -706
  57. package/lib/Socket/newsletter.d.ts +134 -0
  58. package/lib/Socket/newsletter.js +314 -199
  59. package/lib/Socket/registration.d.ts +267 -0
  60. package/lib/Socket/registration.js +166 -0
  61. package/lib/Socket/socket.d.ts +43 -0
  62. package/lib/Socket/socket.js +650 -777
  63. package/lib/Socket/usync.d.ts +36 -0
  64. package/lib/Socket/usync.js +70 -0
  65. package/lib/Store/index.d.ts +3 -0
  66. package/lib/Store/index.js +10 -6
  67. package/lib/Store/make-cache-manager-store.d.ts +13 -0
  68. package/lib/Store/make-cache-manager-store.js +81 -73
  69. package/lib/Store/make-in-memory-store.d.ts +118 -0
  70. package/lib/Store/make-in-memory-store.js +423 -286
  71. package/lib/Store/make-ordered-dictionary.d.ts +13 -0
  72. package/lib/Store/make-ordered-dictionary.js +79 -77
  73. package/lib/Store/object-repository.d.ts +10 -0
  74. package/lib/Store/object-repository.js +26 -24
  75. package/lib/Types/Auth.d.ts +110 -0
  76. package/lib/Types/Auth.js +2 -3
  77. package/lib/Types/Call.d.ts +13 -0
  78. package/lib/Types/Call.js +2 -3
  79. package/lib/Types/Chat.d.ts +102 -0
  80. package/lib/Types/Chat.js +4 -9
  81. package/lib/Types/Contact.d.ts +19 -0
  82. package/lib/Types/Contact.js +2 -3
  83. package/lib/Types/Events.d.ts +157 -0
  84. package/lib/Types/Events.js +2 -3
  85. package/lib/Types/GroupMetadata.d.ts +55 -0
  86. package/lib/Types/GroupMetadata.js +2 -3
  87. package/lib/Types/Label.d.ts +35 -0
  88. package/lib/Types/Label.js +26 -24
  89. package/lib/Types/LabelAssociation.d.ts +29 -0
  90. package/lib/Types/LabelAssociation.js +8 -6
  91. package/lib/Types/Message.d.ts +273 -0
  92. package/lib/Types/Message.js +9 -12
  93. package/lib/Types/Newsletter.d.ts +103 -0
  94. package/lib/Types/Newsletter.js +38 -33
  95. package/lib/Types/Product.d.ts +78 -0
  96. package/lib/Types/Product.js +2 -3
  97. package/lib/Types/Signal.d.ts +57 -0
  98. package/lib/Types/Signal.js +2 -3
  99. package/lib/Types/Socket.d.ts +111 -0
  100. package/lib/Types/Socket.js +2 -4
  101. package/lib/Types/State.d.ts +27 -0
  102. package/lib/Types/State.js +2 -11
  103. package/lib/Types/USync.d.ts +25 -0
  104. package/lib/Types/USync.js +2 -3
  105. package/lib/Types/index.d.ts +57 -0
  106. package/lib/Types/index.js +41 -27
  107. package/lib/Utils/auth-utils.d.ts +18 -0
  108. package/lib/Utils/auth-utils.js +198 -211
  109. package/lib/Utils/baileys-event-stream.d.ts +16 -0
  110. package/lib/Utils/baileys-event-stream.js +61 -42
  111. package/lib/Utils/business.d.ts +22 -0
  112. package/lib/Utils/business.js +214 -213
  113. package/lib/Utils/chat-utils.d.ts +71 -0
  114. package/lib/Utils/chat-utils.js +687 -710
  115. package/lib/Utils/crypto.d.ts +41 -0
  116. package/lib/Utils/crypto.js +133 -112
  117. package/lib/Utils/decode-wa-message.d.ts +19 -0
  118. package/lib/Utils/decode-wa-message.js +183 -252
  119. package/lib/Utils/event-buffer.d.ts +35 -0
  120. package/lib/Utils/event-buffer.js +496 -510
  121. package/lib/Utils/generics.d.ts +92 -0
  122. package/lib/Utils/generics.js +392 -319
  123. package/lib/Utils/generics.js.bak +433 -0
  124. package/lib/Utils/history.d.ts +15 -0
  125. package/lib/Utils/history.js +92 -83
  126. package/lib/Utils/index.d.ts +17 -0
  127. package/lib/Utils/index.js +33 -21
  128. package/lib/Utils/link-preview.d.ts +21 -0
  129. package/lib/Utils/link-preview.js +83 -71
  130. package/lib/Utils/logger.d.ts +4 -0
  131. package/lib/Utils/logger.js +7 -5
  132. package/lib/Utils/lt-hash.d.ts +12 -0
  133. package/lib/Utils/lt-hash.js +46 -40
  134. package/lib/Utils/make-mutex.d.ts +7 -0
  135. package/lib/Utils/make-mutex.js +41 -34
  136. package/lib/Utils/messages-media.d.ts +116 -0
  137. package/lib/Utils/messages-media.js +768 -550
  138. package/lib/Utils/messages.d.ts +77 -0
  139. package/lib/Utils/messages.js +263 -362
  140. package/lib/Utils/noise-handler.d.ts +21 -0
  141. package/lib/Utils/noise-handler.js +149 -138
  142. package/lib/Utils/process-message.d.ts +41 -0
  143. package/lib/Utils/process-message.js +303 -323
  144. package/lib/Utils/signal.d.ts +32 -0
  145. package/lib/Utils/signal.js +141 -149
  146. package/lib/Utils/use-multi-file-auth-state.d.ts +13 -0
  147. package/lib/Utils/use-multi-file-auth-state.js +103 -95
  148. package/lib/Utils/validate-connection.d.ts +11 -0
  149. package/lib/Utils/validate-connection.js +214 -183
  150. package/lib/Utils/validate-connection.js.bak +237 -0
  151. package/lib/WABinary/constants.d.ts +30 -0
  152. package/lib/WABinary/constants.js +35 -1298
  153. package/lib/WABinary/decode.d.ts +7 -0
  154. package/lib/WABinary/decode.js +249 -237
  155. package/lib/WABinary/encode.d.ts +3 -0
  156. package/lib/WABinary/encode.js +260 -213
  157. package/lib/WABinary/generic-utils.d.ts +17 -0
  158. package/lib/WABinary/generic-utils.js +65 -56
  159. package/lib/WABinary/index.d.ts +5 -0
  160. package/lib/WABinary/index.js +21 -7
  161. package/lib/WABinary/jid-utils.d.ts +31 -0
  162. package/lib/WABinary/jid-utils.js +58 -89
  163. package/lib/WABinary/types.d.ts +18 -0
  164. package/lib/WABinary/types.js +2 -3
  165. package/lib/WAM/BinaryInfo.d.ts +17 -0
  166. package/lib/WAM/BinaryInfo.js +12 -10
  167. package/lib/WAM/constants.d.ts +38 -0
  168. package/lib/WAM/constants.js +15348 -22851
  169. package/lib/WAM/encode.d.ts +3 -0
  170. package/lib/WAM/encode.js +136 -135
  171. package/lib/WAM/index.d.ts +3 -0
  172. package/lib/WAM/index.js +19 -5
  173. package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts +9 -0
  174. package/lib/WAUSync/Protocols/USyncContactProtocol.js +30 -28
  175. package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts +22 -0
  176. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -49
  177. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts +12 -0
  178. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +28 -27
  179. package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts +12 -0
  180. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +39 -36
  181. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts +25 -0
  182. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -50
  183. package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts +8 -0
  184. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +20 -26
  185. package/lib/WAUSync/Protocols/index.d.ts +4 -0
  186. package/lib/WAUSync/Protocols/index.js +20 -6
  187. package/lib/WAUSync/USyncQuery.d.ts +28 -0
  188. package/lib/WAUSync/USyncQuery.js +85 -86
  189. package/lib/WAUSync/USyncUser.d.ts +12 -0
  190. package/lib/WAUSync/USyncUser.js +25 -23
  191. package/lib/WAUSync/index.d.ts +3 -0
  192. package/lib/WAUSync/index.js +19 -5
  193. package/lib/index.d.ts +12 -0
  194. package/lib/index.js +36 -24
  195. package/package.json +106 -98
  196. package/LICENSE +0 -21
  197. package/WAProto/WAProto.proto +0 -5311
  198. package/lib/KeyDB/BinarySearch.js +0 -20
  199. package/lib/KeyDB/KeyedDB.js +0 -167
  200. package/lib/KeyDB/index.js +0 -4
  201. package/lib/Signal/lid-mapping.js +0 -155
  202. package/lib/Socket/Client/types.js +0 -13
  203. package/lib/Socket/Client/websocket.js +0 -52
  204. package/lib/Socket/Client/websocket.js.bak +0 -53
  205. package/lib/Socket/communities.js +0 -413
  206. package/lib/Socket/mex.js +0 -45
  207. package/lib/Types/Bussines.js +0 -3
  208. package/lib/Types/Newsletter.js.bak +0 -33
  209. package/lib/Utils/browser-utils.js +0 -25
  210. package/lib/Utils/message-retry-manager.js +0 -113
  211. package/lib/Utils/messages.js.bak +0 -907
  212. package/lib/Utils/pre-key-manager.js +0 -85
@@ -1,290 +1,427 @@
1
- //===================================//
2
- import { md5, toNumber, updateMessageWithReceipt, updateMessageWithReaction } from "../Utils/index.js"
3
- import { DEFAULT_CONNECTION_CONFIG } from "../Defaults/index.js"
4
- import { makeOrderedDictionary } from "./make-ordered-dictionary.js"
5
- import { LabelAssociationType } from "../Types/LabelAssociation.js"
6
- import { jidDecode, jidNormalizedUser } from "../WABinary/index.js"
7
- import { proto } from "../../WAProto/index.js";
8
- import { ObjectRepository } from "./object-repository.js"
9
- import KeyedDB from "../KeyDB/KeyedDB.js"
10
- //===================================//
11
- export const waChatKey = (pin) => ({
12
- key: (c) => (pin ? (c.pinned ? "1" : "0") : "") + (c.archived ? "0" : "1") + (c.conversationTimestamp ? c.conversationTimestamp.toString(16).padStart(8, "0") : "") + c.id,
13
- compare: (k1, k2) => k2.localeCompare(k1)
14
- })
15
- //===================================//
16
- export const waMessageID = (m) => m.key.id || ""
17
- //===================================//
18
- export const waLabelAssociationKey = {
19
- key: (la) => (la.type === LabelAssociationType.Chat ? la.chatId + la.labelId : la.chatId + la.messageId + la.labelId),
20
- compare: (k1, k2) => k2.localeCompare(k1)
21
- }
22
- //===================================//
23
- const makeMessagesDictionary = () => makeOrderedDictionary(waMessageID)
24
- //===================================//
25
- export const makeInMemoryStore = (config) => {
26
- const socket = config.socket
27
- const chatKey = config.chatKey || waChatKey(true)
28
- const labelAssociationKey = config.labelAssociationKey || waLabelAssociationKey
29
- const logger = config.logger || DEFAULT_CONNECTION_CONFIG.logger.child({ stream: "in-mem-store" })
30
- const chats = new KeyedDB(chatKey, c => c.id)
31
- const messages = {}
32
- const contacts = {}
33
- const groupMetadata = {}
34
- const presences = {}
35
- const state = { connection: "close" }
36
- const labels = new ObjectRepository()
37
- const labelAssociations = new KeyedDB(labelAssociationKey, labelAssociationKey.key)
38
- const assertMessageList = (jid) => {
39
- if (!messages[jid]) messages[jid] = makeMessagesDictionary()
40
- return messages[jid]
41
- }
42
- const contactsUpsert = (newContacts) => {
43
- const oldContacts = new Set(Object.keys(contacts))
44
- for (const contact of newContacts) {
45
- oldContacts.delete(contact.id)
46
- contacts[contact.id] = Object.assign(contacts[contact.id] || {}, contact)
47
- }
48
- return oldContacts
49
- }
50
- const labelsUpsert = (newLabels) => {
51
- for (const label of newLabels) labels.upsertById(label.id, label)
52
- }
53
- const bind = (ev) => {
54
- ev.on("connection.update", update => Object.assign(state, update))
55
- ev.on("messaging-history.set", ({ chats: newChats, contacts: newContacts, messages: newMessages, isLatest, syncType }) => {
56
- if (syncType === proto.HistorySync.HistorySyncType.ON_DEMAND) return
57
- if (isLatest) {
58
- chats.clear()
59
- for (const id in messages) delete messages[id]
60
- }
61
- const chatsAdded = chats.insertIfAbsent(...newChats).length
62
- logger.debug({ chatsAdded }, "synced chats")
63
- const oldContacts = contactsUpsert(newContacts)
64
- if (isLatest) for (const jid of oldContacts) delete contacts[jid]
65
- logger.debug({ deletedContacts: isLatest ? oldContacts.size : 0, newContacts }, "synced contacts")
66
- for (const msg of newMessages) {
67
- const jid = msg.key.remoteJid
68
- const list = assertMessageList(jid)
69
- list.upsert(msg, "prepend")
70
- }
71
- logger.debug({ messages: newMessages.length }, "synced messages")
72
- })
73
- ev.on("contacts.upsert", contacts => contactsUpsert(contacts))
74
- ev.on("contacts.update", async (updates) => {
75
- for (const update of updates) {
76
- let contact = contacts[update.id]
77
- if (!contact) {
78
- const contactHashes = await Promise.all(
79
- Object.keys(contacts).map(async (contactId) => {
80
- const { user } = jidDecode(contactId)
81
- return [contactId, (await md5(Buffer.from(user + "WA_ADD_NOTIF", "utf8"))).toString("base64").slice(0, 3)]
82
- })
83
- )
84
- contact = contacts[contactHashes.find(([, b]) => b === update.id?.[0]) || ""]
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.waLabelAssociationKey = exports.waMessageID = exports.waChatKey = void 0;
7
+ const WAProto_1 = require("../../WAProto");
8
+ const Defaults_1 = require("../Defaults");
9
+ const LabelAssociation_1 = require("../Types/LabelAssociation");
10
+ const Utils_1 = require("../Utils");
11
+ const WABinary_1 = require("../WABinary");
12
+ const make_ordered_dictionary_1 = __importDefault(require("./make-ordered-dictionary"));
13
+ const object_repository_1 = require("./object-repository");
14
+ const waChatKey = (pin) => ({
15
+ key: (c) => (pin ? (c.pinned ? '1' : '0') : '') + (c.archived ? '0' : '1') + (c.conversationTimestamp ? c.conversationTimestamp.toString(16).padStart(8, '0') : '') + c.id,
16
+ compare: (k1, k2) => k2.localeCompare(k1)
17
+ });
18
+ exports.waChatKey = waChatKey;
19
+ const waMessageID = (m) => m.key.id || '';
20
+ exports.waMessageID = waMessageID;
21
+ exports.waLabelAssociationKey = {
22
+ key: (la) => (la.type === LabelAssociation_1.LabelAssociationType.Chat ? la.chatId + la.labelId : la.chatId + la.messageId + la.labelId),
23
+ compare: (k1, k2) => k2.localeCompare(k1)
24
+ };
25
+ const makeMessagesDictionary = () => (0, make_ordered_dictionary_1.default)(exports.waMessageID);
26
+ exports.default = (config) => {
27
+ const socket = config.socket;
28
+ const chatKey = config.chatKey || (0, exports.waChatKey)(true);
29
+ const labelAssociationKey = config.labelAssociationKey || exports.waLabelAssociationKey;
30
+ const logger = config.logger || Defaults_1.DEFAULT_CONNECTION_CONFIG.logger.child({ stream: 'in-mem-store' });
31
+ const KeyedDB = require('@adiwajshing/keyed-db').default;
32
+ const chats = new KeyedDB(chatKey, c => c.id);
33
+ const messages = {};
34
+ const contacts = {};
35
+ const groupMetadata = {};
36
+ const presences = {};
37
+ const state = { connection: 'close' };
38
+ const labels = new object_repository_1.ObjectRepository();
39
+ const labelAssociations = new KeyedDB(labelAssociationKey, labelAssociationKey.key);
40
+ const assertMessageList = (jid) => {
41
+ if (!messages[jid]) {
42
+ messages[jid] = makeMessagesDictionary();
85
43
  }
86
- if (contact) {
87
- if (update.imgUrl === "changed") contact.imgUrl = socket ? await socket.profilePictureUrl(contact.id) : undefined
88
- else if (update.imgUrl === "removed") delete contact.imgUrl
89
- } else return logger.debug({ update }, "got update for non-existant contact")
90
-
91
- Object.assign(contacts[contact.id], contact)
92
- }
93
- })
94
- ev.on("chats.upsert", newChats => chats.upsert(...newChats))
95
- ev.on("chats.update", updates => {
96
- for (let update of updates) {
97
- const result = chats.update(update.id, chat => {
98
- if (update.unreadCount > 0) {
99
- update = { ...update }
100
- update.unreadCount = (chat.unreadCount || 0) + update.unreadCount
101
- }
102
- Object.assign(chat, update)
103
- })
104
- if (!result) logger.debug({ update }, "got update for non-existant chat")
105
- }
106
- })
107
- ev.on("labels.edit", (label) => {
108
- if (label.deleted) return labels.deleteById(label.id)
109
- if (labels.count() < 20) return labels.upsertById(label.id, label)
110
- logger.error("Labels count exceed")
111
- })
112
- ev.on("labels.association", ({ type, association }) => {
113
- switch (type) {
114
- case "add": labelAssociations.upsert(association); break
115
- case "remove": labelAssociations.delete(association); break
116
- default: console.error(`unknown operation type [${type}]`)
117
- }
118
- })
119
- ev.on("presence.update", ({ id, presences: update }) => {
120
- presences[id] = presences[id] || {}
121
- Object.assign(presences[id], update)
122
- })
123
- ev.on("chats.delete", deletions => {
124
- for (const item of deletions) if (chats.get(item)) chats.deleteById(item)
125
- })
126
- ev.on("messages.upsert", ({ messages: newMessages, type }) => {
127
- switch (type) {
128
- case "append":
129
- case "notify":
130
- for (const msg of newMessages) {
131
- const jid = jidNormalizedUser(msg.key.remoteJid)
132
- const list = assertMessageList(jid)
133
- list.upsert(msg, "append")
134
- if (type === "notify" && !chats.get(jid)) {
135
- ev.emit("chats.upsert", [{
136
- id: jid,
137
- conversationTimestamp: toNumber(msg.messageTimestamp),
138
- unreadCount: 1
139
- }])
44
+ return messages[jid];
45
+ };
46
+ const contactsUpsert = (newContacts) => {
47
+ const oldContacts = new Set(Object.keys(contacts));
48
+ for (const contact of newContacts) {
49
+ oldContacts.delete(contact.id);
50
+ contacts[contact.id] = Object.assign(contacts[contact.id] || {}, contact);
51
+ }
52
+ return oldContacts;
53
+ };
54
+ const labelsUpsert = (newLabels) => {
55
+ for (const label of newLabels) {
56
+ labels.upsertById(label.id, label);
57
+ }
58
+ };
59
+ const getValidContacts = () => {
60
+ for (const contact of Object.keys(contacts)) {
61
+ if (contact.indexOf('@') < 0) {
62
+ delete contacts[contact];
140
63
  }
141
- }
142
- break
143
- }
144
- })
145
- ev.on("messages.update", updates => {
146
- for (const { update, key } of updates) {
147
- const list = assertMessageList(jidNormalizedUser(key.remoteJid))
148
- if (update?.status) {
149
- const listStatus = list.get(key.id)?.status
150
- if (listStatus && update.status <= listStatus) {
151
- logger.debug({ update, storedStatus: listStatus }, "status stored newer then update")
152
- delete update.status
153
- logger.debug({ update }, "new update object")
154
- }
155
64
  }
156
- const result = list.updateAssign(key.id, update)
157
- if (!result) logger.debug({ update }, "got update for non-existent message")
158
- }
159
- })
160
- ev.on("messages.delete", item => {
161
- if ("all" in item) messages[item.jid]?.clear()
162
- else {
163
- const jid = item.keys[0].remoteJid
164
- const list = messages[jid]
165
- if (list) {
166
- const idSet = new Set(item.keys.map(k => k.id))
167
- list.filter(m => !idSet.has(m.key.id))
65
+ return Object.keys(contacts);
66
+ };
67
+ /**
68
+ * binds to a BaileysEventEmitter.
69
+ * It listens to all events and constructs a state that you can query accurate data from.
70
+ * Eg. can use the store to fetch chats, contacts, messages etc.
71
+ * @param ev typically the event emitter from the socket connection
72
+ */
73
+ const bind = (ev) => {
74
+ ev.on('connection.update', update => {
75
+ Object.assign(state, update);
76
+ });
77
+ ev.on('messaging-history.set', ({ chats: newChats, contacts: newContacts, messages: newMessages, isLatest }) => {
78
+ if (isLatest) {
79
+ chats.clear();
80
+ for (const id in messages) {
81
+ delete messages[id];
82
+ }
83
+ }
84
+ const chatsAdded = chats.insertIfAbsent(...newChats).length;
85
+ logger.debug({ chatsAdded }, 'synced chats');
86
+ const oldContacts = contactsUpsert(newContacts);
87
+ if (isLatest) {
88
+ for (const jid of oldContacts) {
89
+ delete contacts[jid];
90
+ }
91
+ }
92
+ logger.debug({ deletedContacts: isLatest ? oldContacts.size : 0, newContacts }, 'synced contacts');
93
+ for (const msg of newMessages) {
94
+ const jid = msg.key.remoteJid;
95
+ const list = assertMessageList(jid);
96
+ list.upsert(msg, 'prepend');
97
+ }
98
+ logger.debug({ messages: newMessages.length }, 'synced messages');
99
+ });
100
+ ev.on('contacts.upsert', contacts => {
101
+ contactsUpsert(contacts);
102
+ });
103
+ ev.on('contacts.update', async (updates) => {
104
+ var _a;
105
+ for (const update of updates) {
106
+ let contact;
107
+ if (contacts[update.id]) {
108
+ contact = contacts[update.id];
109
+ }
110
+ else {
111
+ const validContacts = getValidContacts();
112
+ const contactHashes = validContacts.map((contactId) => {
113
+ const { user } = (0, WABinary_1.jidDecode)(contactId);
114
+ return [contactId, ((0, Utils_1.md5)(Buffer.from(user + 'WA_ADD_NOTIF', 'utf8'))).toString('base64').slice(0, 3)];
115
+ });
116
+ contact = contacts[((_a = contactHashes.find(([, b]) => b === update.id)) === null || _a === void 0 ? void 0 : _a[0]) || '']; // find contact by attrs.hash, when user is not saved as a contact
117
+ }
118
+ if (contact) {
119
+ if (update.imgUrl === 'changed') {
120
+ contact.imgUrl = socket ? await (socket === null || socket === void 0 ? void 0 : socket.profilePictureUrl(contact.id)) : undefined;
121
+ }
122
+ else if (update.imgUrl === 'removed') {
123
+ delete contact.imgUrl;
124
+ }
125
+ Object.assign(contacts[contact.id], contact);
126
+ }
127
+ else {
128
+ logger.debug({ update }, 'got update for non-existant contact');
129
+ }
130
+ }
131
+ });
132
+ ev.on('chats.upsert', newChats => {
133
+ chats.upsert(...newChats);
134
+ });
135
+ ev.on('chats.update', updates => {
136
+ for (let update of updates) {
137
+ const result = chats.update(update.id, chat => {
138
+ if (update.unreadCount > 0) {
139
+ update = { ...update };
140
+ update.unreadCount = (chat.unreadCount || 0) + update.unreadCount;
141
+ }
142
+ Object.assign(chat, update);
143
+ });
144
+ if (!result) {
145
+ logger.debug({ update }, 'got update for non-existant chat');
146
+ }
147
+ }
148
+ });
149
+ ev.on('labels.edit', (label) => {
150
+ if (label.deleted) {
151
+ return labels.deleteById(label.id);
152
+ }
153
+ // WhatsApp can store only up to 20 labels
154
+ if (labels.count() < 20) {
155
+ return labels.upsertById(label.id, label);
156
+ }
157
+ logger.error('Labels count exceed');
158
+ });
159
+ ev.on('labels.association', ({ type, association }) => {
160
+ switch (type) {
161
+ case 'add':
162
+ labelAssociations.upsert(association);
163
+ break;
164
+ case 'remove':
165
+ labelAssociations.delete(association);
166
+ break;
167
+ default:
168
+ console.error(`unknown operation type [${type}]`);
169
+ }
170
+ });
171
+ ev.on('presence.update', ({ id, presences: update }) => {
172
+ presences[id] = presences[id] || {};
173
+ Object.assign(presences[id], update);
174
+ });
175
+ ev.on('chats.delete', deletions => {
176
+ for (const item of deletions) {
177
+ if (chats.get(item)) {
178
+ chats.deleteById(item);
179
+ }
180
+ }
181
+ });
182
+ ev.on('messages.upsert', ({ messages: newMessages, type }) => {
183
+ switch (type) {
184
+ case 'append':
185
+ case 'notify':
186
+ for (const msg of newMessages) {
187
+ const jid = (0, WABinary_1.jidNormalizedUser)(msg.key.remoteJid);
188
+ const list = assertMessageList(jid);
189
+ list.upsert(msg, 'append');
190
+ if (type === 'notify') {
191
+ if (!chats.get(jid)) {
192
+ ev.emit('chats.upsert', [
193
+ {
194
+ id: jid,
195
+ conversationTimestamp: (0, Utils_1.toNumber)(msg.messageTimestamp),
196
+ unreadCount: 1
197
+ }
198
+ ]);
199
+ }
200
+ }
201
+ }
202
+ break;
203
+ }
204
+ });
205
+ ev.on('messages.update', updates => {
206
+ var _a;
207
+ for (const { update, key } of updates) {
208
+ const list = assertMessageList((0, WABinary_1.jidNormalizedUser)(key.remoteJid));
209
+ if (update === null || update === void 0 ? void 0 : update.status) {
210
+ const listStatus = (_a = list.get(key.id)) === null || _a === void 0 ? void 0 : _a.status;
211
+ if (listStatus && (update === null || update === void 0 ? void 0 : update.status) <= listStatus) {
212
+ logger.debug({ update, storedStatus: listStatus }, 'status stored newer then update');
213
+ delete update.status;
214
+ logger.debug({ update }, 'new update object');
215
+ }
216
+ }
217
+ const result = list.updateAssign(key.id, update);
218
+ if (!result) {
219
+ logger.debug({ update }, 'got update for non-existent message');
220
+ }
221
+ }
222
+ });
223
+ ev.on('messages.delete', item => {
224
+ if ('all' in item) {
225
+ const list = messages[item.jid];
226
+ list === null || list === void 0 ? void 0 : list.clear();
227
+ }
228
+ else {
229
+ const jid = item.keys[0].remoteJid;
230
+ const list = messages[jid];
231
+ if (list) {
232
+ const idSet = new Set(item.keys.map(k => k.id));
233
+ list.filter(m => !idSet.has(m.key.id));
234
+ }
235
+ }
236
+ });
237
+ ev.on('groups.update', updates => {
238
+ for (const update of updates) {
239
+ const id = update.id;
240
+ if (groupMetadata[id]) {
241
+ Object.assign(groupMetadata[id], update);
242
+ }
243
+ else {
244
+ logger.debug({ update }, 'got update for non-existant group metadata');
245
+ }
246
+ }
247
+ });
248
+ ev.on('group-participants.update', ({ id, participants, action }) => {
249
+ const metadata = groupMetadata[id];
250
+ if (metadata) {
251
+ switch (action) {
252
+ case 'add':
253
+ metadata.participants.push(...participants.map(id => ({ id, isAdmin: false, isSuperAdmin: false })));
254
+ break;
255
+ case 'demote':
256
+ case 'promote':
257
+ for (const participant of metadata.participants) {
258
+ if (participants.includes(participant.id)) {
259
+ participant.isAdmin = action === 'promote';
260
+ }
261
+ }
262
+ break;
263
+ case 'remove':
264
+ metadata.participants = metadata.participants.filter(p => !participants.includes(p.id));
265
+ break;
266
+ }
267
+ }
268
+ });
269
+ ev.on('message-receipt.update', updates => {
270
+ for (const { key, receipt } of updates) {
271
+ const obj = messages[key.remoteJid];
272
+ const msg = obj === null || obj === void 0 ? void 0 : obj.get(key.id);
273
+ if (msg) {
274
+ (0, Utils_1.updateMessageWithReceipt)(msg, receipt);
275
+ }
276
+ }
277
+ });
278
+ ev.on('messages.reaction', (reactions) => {
279
+ for (const { key, reaction } of reactions) {
280
+ const obj = messages[key.remoteJid];
281
+ const msg = obj === null || obj === void 0 ? void 0 : obj.get(key.id);
282
+ if (msg) {
283
+ (0, Utils_1.updateMessageWithReaction)(msg, reaction);
284
+ }
285
+ }
286
+ });
287
+ };
288
+ const toJSON = () => ({
289
+ chats,
290
+ contacts,
291
+ messages,
292
+ labels,
293
+ labelAssociations
294
+ });
295
+ const fromJSON = (json) => {
296
+ chats.upsert(...json.chats);
297
+ labelAssociations.upsert(...json.labelAssociations || []);
298
+ contactsUpsert(Object.values(json.contacts));
299
+ labelsUpsert(Object.values(json.labels || {}));
300
+ for (const jid in json.messages) {
301
+ const list = assertMessageList(jid);
302
+ for (const msg of json.messages[jid]) {
303
+ list.upsert(WAProto_1.proto.WebMessageInfo.fromObject(msg), 'append');
304
+ }
305
+ }
306
+ };
307
+ return {
308
+ chats,
309
+ contacts,
310
+ messages,
311
+ groupMetadata,
312
+ state,
313
+ presences,
314
+ labels,
315
+ labelAssociations,
316
+ bind,
317
+ /** loads messages from the store, if not found -- uses the legacy connection */
318
+ loadMessages: async (jid, count, cursor) => {
319
+ const list = assertMessageList(jid);
320
+ const mode = !cursor || 'before' in cursor ? 'before' : 'after';
321
+ const cursorKey = !!cursor ? ('before' in cursor ? cursor.before : cursor.after) : undefined;
322
+ const cursorValue = cursorKey ? list.get(cursorKey.id) : undefined;
323
+ let messages;
324
+ if (list && mode === 'before' && (!cursorKey || cursorValue)) {
325
+ if (cursorValue) {
326
+ const msgIdx = list.array.findIndex(m => m.key.id === (cursorKey === null || cursorKey === void 0 ? void 0 : cursorKey.id));
327
+ messages = list.array.slice(0, msgIdx);
328
+ }
329
+ else {
330
+ messages = list.array;
331
+ }
332
+ const diff = count - messages.length;
333
+ if (diff < 0) {
334
+ messages = messages.slice(-count); // get the last X messages
335
+ }
336
+ }
337
+ else {
338
+ messages = [];
339
+ }
340
+ return messages;
341
+ },
342
+ /**
343
+ * Get all available labels for profile
344
+ *
345
+ * Keep in mind that the list is formed from predefined tags and tags
346
+ * that were "caught" during their editing.
347
+ */
348
+ getLabels: () => {
349
+ return labels;
350
+ },
351
+ /**
352
+ * Get labels for chat
353
+ *
354
+ * @returns Label IDs
355
+ **/
356
+ getChatLabels: (chatId) => {
357
+ return labelAssociations.filter((la) => la.chatId === chatId).all();
358
+ },
359
+ /**
360
+ * Get labels for message
361
+ *
362
+ * @returns Label IDs
363
+ **/
364
+ getMessageLabels: (messageId) => {
365
+ const associations = labelAssociations
366
+ .filter((la) => la.messageId === messageId)
367
+ .all();
368
+ return associations.map(({ labelId }) => labelId);
369
+ },
370
+ loadMessage: async (jid, id) => { var _a; return (_a = messages[jid]) === null || _a === void 0 ? void 0 : _a.get(id); },
371
+ mostRecentMessage: async (jid) => {
372
+ var _a;
373
+ const message = (_a = messages[jid]) === null || _a === void 0 ? void 0 : _a.array.slice(-1)[0];
374
+ return message;
375
+ },
376
+ fetchImageUrl: async (jid, sock) => {
377
+ const contact = contacts[jid];
378
+ if (!contact) {
379
+ return sock === null || sock === void 0 ? void 0 : sock.profilePictureUrl(jid);
380
+ }
381
+ if (typeof contact.imgUrl === 'undefined') {
382
+ contact.imgUrl = await (sock === null || sock === void 0 ? void 0 : sock.profilePictureUrl(jid));
383
+ }
384
+ return contact.imgUrl;
385
+ },
386
+ fetchGroupMetadata: async (jid, sock) => {
387
+ if (!groupMetadata[jid]) {
388
+ const metadata = await (sock === null || sock === void 0 ? void 0 : sock.groupMetadata(jid));
389
+ if (metadata) {
390
+ groupMetadata[jid] = metadata;
391
+ }
392
+ }
393
+ return groupMetadata[jid];
394
+ },
395
+ // fetchBroadcastListInfo: async(jid: string, sock: WASocket | undefined) => {
396
+ // if(!groupMetadata[jid]) {
397
+ // const metadata = await sock?.getBroadcastListInfo(jid)
398
+ // if(metadata) {
399
+ // groupMetadata[jid] = metadata
400
+ // }
401
+ // }
402
+ // return groupMetadata[jid]
403
+ // },
404
+ fetchMessageReceipts: async ({ remoteJid, id }) => {
405
+ const list = messages[remoteJid];
406
+ const msg = list === null || list === void 0 ? void 0 : list.get(id);
407
+ return msg === null || msg === void 0 ? void 0 : msg.userReceipt;
408
+ },
409
+ toJSON,
410
+ fromJSON,
411
+ writeToFile: (path) => {
412
+ // require fs here so that in case "fs" is not available -- the app does not crash
413
+ const { writeFileSync } = require('fs');
414
+ writeFileSync(path, JSON.stringify(toJSON()));
415
+ },
416
+ readFromFile: (path) => {
417
+ // require fs here so that in case "fs" is not available -- the app does not crash
418
+ const { readFileSync, existsSync } = require('fs');
419
+ if (existsSync(path)) {
420
+ logger.debug({ path }, 'reading from file');
421
+ const jsonStr = readFileSync(path, { encoding: 'utf-8' });
422
+ const json = JSON.parse(jsonStr);
423
+ fromJSON(json);
424
+ }
168
425
  }
169
- }
170
- })
171
- ev.on("groups.update", updates => {
172
- for (const update of updates) {
173
- const id = update.id
174
- if (groupMetadata[id]) Object.assign(groupMetadata[id], update)
175
- else logger.debug({ update }, "got update for non-existant group metadata")
176
- }
177
- })
178
- ev.on("group-participants.update", ({ id, participants, action }) => {
179
- const metadata = groupMetadata[id]
180
- if (!metadata) return
181
- switch (action) {
182
- case "add":
183
- metadata.participants.push(...participants.map(id => ({ id, isAdmin: false, isSuperAdmin: false })))
184
- break
185
- case "promote":
186
- case "demote":
187
- for (const participant of metadata.participants) {
188
- if (participants.includes(participant.id)) participant.isAdmin = action === "promote"
189
- }
190
- break
191
- case "remove":
192
- metadata.participants = metadata.participants.filter(p => !participants.includes(p.id))
193
- break
194
- }
195
- })
196
- ev.on("message-receipt.update", updates => {
197
- for (const { key, receipt } of updates) {
198
- const obj = messages[key.remoteJid]
199
- const msg = obj?.get(key.id)
200
- if (msg) updateMessageWithReceipt(msg, receipt)
201
- }
202
- })
203
- ev.on("messages.reaction", reactions => {
204
- for (const { key, reaction } of reactions) {
205
- const obj = messages[key.remoteJid]
206
- const msg = obj?.get(key.id)
207
- if (msg) updateMessageWithReaction(msg, reaction)
208
- }
209
- })
210
- }
211
- const toJSON = () => ({ chats, contacts, messages, labels, labelAssociations })
212
- const fromJSON = (json) => {
213
- chats.upsert(...json.chats)
214
- labelAssociations.upsert(...json.labelAssociations || [])
215
- contactsUpsert(Object.values(json.contacts))
216
- labelsUpsert(Object.values(json.labels || {}))
217
- for (const jid in json.messages) {
218
- const list = assertMessageList(jid)
219
- for (const msg of json.messages[jid]) list.upsert(proto.WebMessageInfo.fromObject(msg), "append")
220
- }
221
- }
222
- return {
223
- chats,
224
- contacts,
225
- messages,
226
- groupMetadata,
227
- state,
228
- presences,
229
- labels,
230
- labelAssociations,
231
- bind,
232
- loadMessages: async (jid, count, cursor) => {
233
- const list = assertMessageList(jid)
234
- const mode = !cursor || "before" in cursor ? "before" : "after"
235
- const cursorKey = cursor ? ("before" in cursor ? cursor.before : cursor.after) : undefined
236
- const cursorValue = cursorKey ? list.get(cursorKey.id) : undefined
237
- let msgs
238
- if (list && mode === "before" && (!cursorKey || cursorValue)) {
239
- if (cursorValue) {
240
- const idx = list.array.findIndex(m => m.key.id === cursorKey?.id)
241
- msgs = list.array.slice(0, idx)
242
- } else msgs = list.array
243
- const diff = count - msgs.length
244
- if (diff < 0) msgs = msgs.slice(-count)
245
- } else msgs = []
246
- return msgs
247
- },
248
- getLabels: () => labels,
249
- getChatLabels: (chatId) => labelAssociations.filter(la => la.chatId === chatId).all(),
250
- getMessageLabels: (messageId) => labelAssociations.filter(la => la.messageId === messageId).all().map(l => l.labelId),
251
- loadMessage: async (jid, id) => messages[jid]?.get(id),
252
- mostRecentMessage: async (jid) => messages[jid]?.array.slice(-1)[0],
253
- fetchImageUrl: async (jid, baron) => {
254
- const contact = contacts[jid]
255
- if (!contact) return baron?.profilePictureUrl?.(jid)
256
- if (typeof contact.imgUrl === "undefined") {
257
- contact.imgUrl = await baron?.profilePictureUrl?.(jid)
258
- }
259
- return contact.imgUrl
260
- },
261
-
262
- fetchGroupMetadata: async (jid, baron) => {
263
- if (!groupMetadata[jid]) {
264
- const metadata = await baron?.groupMetadata(jid)
265
- if (metadata) groupMetadata[jid] = metadata
266
- }
267
- return groupMetadata[jid]
268
- },
269
- fetchMessageReceipts: async ({ remoteJid, id }) => {
270
- const list = messages[remoteJid]
271
- const msg = list?.get(id)
272
- return msg?.userReceipt
273
- },
274
- toJSON,
275
- fromJSON,
276
- writeToFile: (path) => {
277
- import("fs").then(fs => fs.writeFileSync(path, JSON.stringify(toJSON())))
278
- },
279
- readFromFile: async (path) => {
280
- const fs = await import("fs")
281
- if (fs.existsSync(path)) {
282
- logger.debug({ path }, "reading from file")
283
- const jsonStr = fs.readFileSync(path, { encoding: "utf-8" })
284
- const json = JSON.parse(jsonStr)
285
- fromJSON(json)
286
- }
287
- }
288
- }
289
- }
290
- //===================================//
426
+ };
427
+ };