@itsliaaa/baileys 0.2.6 → 0.3.0-rc.10

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 (283) hide show
  1. package/README.md +214 -7
  2. package/WAProto/index.d.ts +392 -2465
  3. package/WAProto/index.js +13116 -3569
  4. package/lib/Defaults/index.d.ts +8 -9
  5. package/lib/Defaults/index.d.ts.map +1 -1
  6. package/lib/Defaults/index.js +13 -14
  7. package/lib/Defaults/index.js.map +1 -1
  8. package/lib/Signal/Group/ciphertext-message.d.ts.map +1 -1
  9. package/lib/Signal/Group/ciphertext-message.js.map +1 -1
  10. package/lib/Signal/Group/group-session-builder.d.ts.map +1 -1
  11. package/lib/Signal/Group/group-session-builder.js.map +1 -1
  12. package/lib/Signal/Group/group_cipher.d.ts.map +1 -1
  13. package/lib/Signal/Group/group_cipher.js.map +1 -1
  14. package/lib/Signal/Group/index.d.ts.map +1 -1
  15. package/lib/Signal/Group/index.js.map +1 -1
  16. package/lib/Signal/Group/keyhelper.d.ts.map +1 -1
  17. package/lib/Signal/Group/keyhelper.js.map +1 -1
  18. package/lib/Signal/Group/sender-chain-key.d.ts.map +1 -1
  19. package/lib/Signal/Group/sender-chain-key.js.map +1 -1
  20. package/lib/Signal/Group/sender-key-distribution-message.d.ts.map +1 -1
  21. package/lib/Signal/Group/sender-key-distribution-message.js.map +1 -1
  22. package/lib/Signal/Group/sender-key-message.d.ts.map +1 -1
  23. package/lib/Signal/Group/sender-key-message.js.map +1 -1
  24. package/lib/Signal/Group/sender-key-name.d.ts.map +1 -1
  25. package/lib/Signal/Group/sender-key-name.js.map +1 -1
  26. package/lib/Signal/Group/sender-key-record.d.ts.map +1 -1
  27. package/lib/Signal/Group/sender-key-record.js.map +1 -1
  28. package/lib/Signal/Group/sender-key-state.d.ts.map +1 -1
  29. package/lib/Signal/Group/sender-key-state.js.map +1 -1
  30. package/lib/Signal/Group/sender-message-key.d.ts.map +1 -1
  31. package/lib/Signal/Group/sender-message-key.js.map +1 -1
  32. package/lib/Signal/libsignal.d.ts +12 -0
  33. package/lib/Signal/libsignal.d.ts.map +1 -1
  34. package/lib/Signal/libsignal.js +41 -17
  35. package/lib/Signal/libsignal.js.map +1 -1
  36. package/lib/Signal/lid-mapping.d.ts +2 -2
  37. package/lib/Signal/lid-mapping.d.ts.map +1 -1
  38. package/lib/Signal/lid-mapping.js +2 -2
  39. package/lib/Signal/lid-mapping.js.map +1 -1
  40. package/lib/Socket/Client/index.d.ts.map +1 -1
  41. package/lib/Socket/Client/index.js.map +1 -1
  42. package/lib/Socket/Client/types.d.ts.map +1 -1
  43. package/lib/Socket/Client/types.js.map +1 -1
  44. package/lib/Socket/Client/websocket.d.ts.map +1 -1
  45. package/lib/Socket/Client/websocket.js.map +1 -1
  46. package/lib/Socket/business.d.ts +20 -41
  47. package/lib/Socket/business.d.ts.map +1 -1
  48. package/lib/Socket/business.js +1 -0
  49. package/lib/Socket/business.js.map +1 -1
  50. package/lib/Socket/chats.d.ts +12 -2
  51. package/lib/Socket/chats.d.ts.map +1 -1
  52. package/lib/Socket/chats.js +21 -37
  53. package/lib/Socket/chats.js.map +1 -1
  54. package/lib/Socket/communities.d.ts +22 -42
  55. package/lib/Socket/communities.d.ts.map +1 -1
  56. package/lib/Socket/communities.js +1 -0
  57. package/lib/Socket/communities.js.map +1 -1
  58. package/lib/Socket/groups.d.ts +14 -35
  59. package/lib/Socket/groups.d.ts.map +1 -1
  60. package/lib/Socket/groups.js +17 -4
  61. package/lib/Socket/groups.js.map +1 -1
  62. package/lib/Socket/index.d.ts +21 -41
  63. package/lib/Socket/index.d.ts.map +1 -1
  64. package/lib/Socket/index.js.map +1 -1
  65. package/lib/Socket/messages-recv.d.ts +19 -41
  66. package/lib/Socket/messages-recv.d.ts.map +1 -1
  67. package/lib/Socket/messages-recv.js +416 -297
  68. package/lib/Socket/messages-recv.js.map +1 -1
  69. package/lib/Socket/messages-send.d.ts +19 -35
  70. package/lib/Socket/messages-send.d.ts.map +1 -1
  71. package/lib/Socket/messages-send.js +68 -48
  72. package/lib/Socket/messages-send.js.map +1 -1
  73. package/lib/Socket/mex.d.ts.map +1 -1
  74. package/lib/Socket/mex.js.map +1 -1
  75. package/lib/Socket/newsletter.d.ts +14 -35
  76. package/lib/Socket/newsletter.d.ts.map +1 -1
  77. package/lib/Socket/newsletter.js +2 -3
  78. package/lib/Socket/newsletter.js.map +1 -1
  79. package/lib/Socket/socket.d.ts +7 -1
  80. package/lib/Socket/socket.d.ts.map +1 -1
  81. package/lib/Socket/socket.js +40 -23
  82. package/lib/Socket/socket.js.map +1 -1
  83. package/lib/Store/index.d.ts.map +1 -1
  84. package/lib/Store/index.js.map +1 -1
  85. package/lib/Store/make-in-memory-store.d.ts.map +1 -1
  86. package/lib/Store/make-in-memory-store.js.map +1 -1
  87. package/lib/Store/make-ordered-dictionary.d.ts.map +1 -1
  88. package/lib/Store/make-ordered-dictionary.js.map +1 -1
  89. package/lib/Store/object-repository.d.ts.map +1 -1
  90. package/lib/Store/object-repository.js.map +1 -1
  91. package/lib/Types/Auth.d.ts.map +1 -1
  92. package/lib/Types/Auth.js.map +1 -1
  93. package/lib/Types/Bussines.d.ts.map +1 -1
  94. package/lib/Types/Bussines.js.map +1 -1
  95. package/lib/Types/Call.d.ts.map +1 -1
  96. package/lib/Types/Call.js.map +1 -1
  97. package/lib/Types/Chat.d.ts.map +1 -1
  98. package/lib/Types/Chat.js.map +1 -1
  99. package/lib/Types/Contact.d.ts.map +1 -1
  100. package/lib/Types/Contact.js.map +1 -1
  101. package/lib/Types/Events.d.ts.map +1 -1
  102. package/lib/Types/Events.js.map +1 -1
  103. package/lib/Types/GroupMetadata.d.ts.map +1 -1
  104. package/lib/Types/GroupMetadata.js.map +1 -1
  105. package/lib/Types/Label.d.ts.map +1 -1
  106. package/lib/Types/Label.js.map +1 -1
  107. package/lib/Types/LabelAssociation.d.ts.map +1 -1
  108. package/lib/Types/LabelAssociation.js.map +1 -1
  109. package/lib/Types/Message.d.ts.map +1 -1
  110. package/lib/Types/Message.js.map +1 -1
  111. package/lib/Types/{Newsletter.d.ts → Mex.d.ts} +1 -1
  112. package/lib/Types/Mex.d.ts.map +1 -0
  113. package/lib/Types/{Newsletter.js → Mex.js} +9 -4
  114. package/lib/Types/Mex.js.map +1 -0
  115. package/lib/Types/Product.d.ts.map +1 -1
  116. package/lib/Types/Product.js.map +1 -1
  117. package/lib/Types/RichType.d.ts.map +1 -1
  118. package/lib/Types/RichType.js.map +1 -1
  119. package/lib/Types/Signal.d.ts.map +1 -1
  120. package/lib/Types/Signal.js.map +1 -1
  121. package/lib/Types/Socket.d.ts.map +1 -1
  122. package/lib/Types/Socket.js.map +1 -1
  123. package/lib/Types/State.d.ts +4 -0
  124. package/lib/Types/State.d.ts.map +1 -1
  125. package/lib/Types/State.js +43 -0
  126. package/lib/Types/State.js.map +1 -1
  127. package/lib/Types/USync.d.ts.map +1 -1
  128. package/lib/Types/USync.js.map +1 -1
  129. package/lib/Types/index.d.ts +1 -1
  130. package/lib/Types/index.d.ts.map +1 -1
  131. package/lib/Types/index.js +1 -1
  132. package/lib/Types/index.js.map +1 -1
  133. package/lib/Utils/auth-utils.d.ts +1 -0
  134. package/lib/Utils/auth-utils.d.ts.map +1 -1
  135. package/lib/Utils/auth-utils.js +12 -0
  136. package/lib/Utils/auth-utils.js.map +1 -1
  137. package/lib/Utils/browser-utils.d.ts +0 -1
  138. package/lib/Utils/browser-utils.d.ts.map +1 -1
  139. package/lib/Utils/browser-utils.js +1 -2
  140. package/lib/Utils/browser-utils.js.map +1 -1
  141. package/lib/Utils/business.d.ts.map +1 -1
  142. package/lib/Utils/business.js.map +1 -1
  143. package/lib/Utils/chat-utils.d.ts +5 -5
  144. package/lib/Utils/chat-utils.d.ts.map +1 -1
  145. package/lib/Utils/chat-utils.js +69 -36
  146. package/lib/Utils/chat-utils.js.map +1 -1
  147. package/lib/Utils/companion-reg-client-utils.d.ts +1 -12
  148. package/lib/Utils/companion-reg-client-utils.d.ts.map +1 -1
  149. package/lib/Utils/companion-reg-client-utils.js +20 -13
  150. package/lib/Utils/companion-reg-client-utils.js.map +1 -1
  151. package/lib/Utils/crypto.d.ts.map +1 -1
  152. package/lib/Utils/crypto.js.map +1 -1
  153. package/lib/Utils/decode-wa-message.d.ts +3 -1
  154. package/lib/Utils/decode-wa-message.d.ts.map +1 -1
  155. package/lib/Utils/decode-wa-message.js +17 -3
  156. package/lib/Utils/decode-wa-message.js.map +1 -1
  157. package/lib/Utils/event-buffer.d.ts.map +1 -1
  158. package/lib/Utils/event-buffer.js +30 -0
  159. package/lib/Utils/event-buffer.js.map +1 -1
  160. package/lib/Utils/generics.d.ts +1 -1
  161. package/lib/Utils/generics.d.ts.map +1 -1
  162. package/lib/Utils/generics.js +5 -5
  163. package/lib/Utils/generics.js.map +1 -1
  164. package/lib/Utils/history.d.ts +2 -0
  165. package/lib/Utils/history.d.ts.map +1 -1
  166. package/lib/Utils/history.js +1 -0
  167. package/lib/Utils/history.js.map +1 -1
  168. package/lib/Utils/identity-change-handler.d.ts.map +1 -1
  169. package/lib/Utils/identity-change-handler.js.map +1 -1
  170. package/lib/Utils/index.d.ts +1 -1
  171. package/lib/Utils/index.d.ts.map +1 -1
  172. package/lib/Utils/index.js +1 -1
  173. package/lib/Utils/index.js.map +1 -1
  174. package/lib/Utils/link-preview.d.ts.map +1 -1
  175. package/lib/Utils/link-preview.js +2 -2
  176. package/lib/Utils/link-preview.js.map +1 -1
  177. package/lib/Utils/logger.d.ts.map +1 -1
  178. package/lib/Utils/logger.js.map +1 -1
  179. package/lib/Utils/lt-hash.d.ts.map +1 -1
  180. package/lib/Utils/lt-hash.js.map +1 -1
  181. package/lib/Utils/make-mutex.d.ts.map +1 -1
  182. package/lib/Utils/make-mutex.js.map +1 -1
  183. package/lib/Utils/message-retry-manager.d.ts +4 -0
  184. package/lib/Utils/message-retry-manager.d.ts.map +1 -1
  185. package/lib/Utils/message-retry-manager.js +23 -0
  186. package/lib/Utils/message-retry-manager.js.map +1 -1
  187. package/lib/Utils/messages-media.d.ts +2 -1
  188. package/lib/Utils/messages-media.d.ts.map +1 -1
  189. package/lib/Utils/messages-media.js +19 -7
  190. package/lib/Utils/messages-media.js.map +1 -1
  191. package/lib/Utils/messages.d.ts +3 -12
  192. package/lib/Utils/messages.d.ts.map +1 -1
  193. package/lib/Utils/messages.js +210 -193
  194. package/lib/Utils/messages.js.map +1 -1
  195. package/lib/Utils/noise-handler.d.ts.map +1 -1
  196. package/lib/Utils/noise-handler.js.map +1 -1
  197. package/lib/Utils/offline-node-processor.d.ts.map +1 -1
  198. package/lib/Utils/offline-node-processor.js.map +1 -1
  199. package/lib/Utils/pre-key-manager.d.ts.map +1 -1
  200. package/lib/Utils/pre-key-manager.js.map +1 -1
  201. package/lib/Utils/process-message.d.ts.map +1 -1
  202. package/lib/Utils/process-message.js +18 -2
  203. package/lib/Utils/process-message.js.map +1 -1
  204. package/lib/Utils/reporting-utils.d.ts.map +1 -1
  205. package/lib/Utils/reporting-utils.js.map +1 -1
  206. package/lib/Utils/rich-message-utils.d.ts +8 -3
  207. package/lib/Utils/rich-message-utils.d.ts.map +1 -1
  208. package/lib/Utils/rich-message-utils.js +1 -1
  209. package/lib/Utils/rich-message-utils.js.map +1 -1
  210. package/lib/Utils/signal.d.ts +14 -1
  211. package/lib/Utils/signal.d.ts.map +1 -1
  212. package/lib/Utils/signal.js +42 -0
  213. package/lib/Utils/signal.js.map +1 -1
  214. package/lib/Utils/stanza-ack.d.ts.map +1 -1
  215. package/lib/Utils/stanza-ack.js.map +1 -1
  216. package/lib/Utils/sync-action-utils.d.ts.map +1 -1
  217. package/lib/Utils/sync-action-utils.js.map +1 -1
  218. package/lib/Utils/tc-token-utils.d.ts.map +1 -1
  219. package/lib/Utils/tc-token-utils.js +0 -1
  220. package/lib/Utils/tc-token-utils.js.map +1 -1
  221. package/lib/Utils/use-multi-file-auth-state.d.ts.map +1 -1
  222. package/lib/Utils/use-multi-file-auth-state.js.map +1 -1
  223. package/lib/Utils/use-single-file-auth-state.d.ts.map +1 -1
  224. package/lib/Utils/use-single-file-auth-state.js.map +1 -1
  225. package/lib/Utils/validate-connection.d.ts +1 -1
  226. package/lib/Utils/validate-connection.d.ts.map +1 -1
  227. package/lib/Utils/validate-connection.js +4 -8
  228. package/lib/Utils/validate-connection.js.map +1 -1
  229. package/lib/WABinary/constants.d.ts.map +1 -1
  230. package/lib/WABinary/constants.js.map +1 -1
  231. package/lib/WABinary/decode.d.ts.map +1 -1
  232. package/lib/WABinary/decode.js.map +1 -1
  233. package/lib/WABinary/encode.d.ts.map +1 -1
  234. package/lib/WABinary/encode.js.map +1 -1
  235. package/lib/WABinary/generic-utils.d.ts +1 -3
  236. package/lib/WABinary/generic-utils.d.ts.map +1 -1
  237. package/lib/WABinary/generic-utils.js +6 -7
  238. package/lib/WABinary/generic-utils.js.map +1 -1
  239. package/lib/WABinary/index.d.ts.map +1 -1
  240. package/lib/WABinary/index.js.map +1 -1
  241. package/lib/WABinary/jid-utils.d.ts.map +1 -1
  242. package/lib/WABinary/jid-utils.js.map +1 -1
  243. package/lib/WABinary/types.d.ts.map +1 -1
  244. package/lib/WABinary/types.js.map +1 -1
  245. package/lib/WAM/BinaryInfo.d.ts.map +1 -1
  246. package/lib/WAM/BinaryInfo.js.map +1 -1
  247. package/lib/WAM/constants.d.ts.map +1 -1
  248. package/lib/WAM/constants.js.map +1 -1
  249. package/lib/WAM/encode.d.ts.map +1 -1
  250. package/lib/WAM/encode.js.map +1 -1
  251. package/lib/WAM/index.d.ts.map +1 -1
  252. package/lib/WAM/index.js.map +1 -1
  253. package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts.map +1 -1
  254. package/lib/WAUSync/Protocols/USyncContactProtocol.js.map +1 -1
  255. package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts.map +1 -1
  256. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js.map +1 -1
  257. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts.map +1 -1
  258. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js.map +1 -1
  259. package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts.map +1 -1
  260. package/lib/WAUSync/Protocols/USyncStatusProtocol.js.map +1 -1
  261. package/lib/WAUSync/Protocols/USyncUsernameProtocol.d.ts.map +1 -1
  262. package/lib/WAUSync/Protocols/USyncUsernameProtocol.js +3 -1
  263. package/lib/WAUSync/Protocols/USyncUsernameProtocol.js.map +1 -1
  264. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts.map +1 -1
  265. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js.map +1 -1
  266. package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts.map +1 -1
  267. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js.map +1 -1
  268. package/lib/WAUSync/Protocols/index.d.ts.map +1 -1
  269. package/lib/WAUSync/Protocols/index.js.map +1 -1
  270. package/lib/WAUSync/USyncQuery.d.ts.map +1 -1
  271. package/lib/WAUSync/USyncQuery.js +1 -1
  272. package/lib/WAUSync/USyncQuery.js.map +1 -1
  273. package/lib/WAUSync/USyncUser.d.ts.map +1 -1
  274. package/lib/WAUSync/USyncUser.js.map +1 -1
  275. package/lib/WAUSync/index.d.ts.map +1 -1
  276. package/lib/WAUSync/index.js.map +1 -1
  277. package/lib/index.d.ts +1 -1
  278. package/lib/index.d.ts.map +1 -1
  279. package/lib/index.js +1 -1
  280. package/lib/index.js.map +1 -1
  281. package/package.json +34 -5
  282. package/lib/Types/Newsletter.d.ts.map +0 -1
  283. package/lib/Types/Newsletter.js.map +0 -1
@@ -4,23 +4,26 @@ import { randomBytes } from 'crypto';
4
4
  import Long from 'long';
5
5
  import { proto } from '../../WAProto/index.js';
6
6
  import { DEFAULT_CACHE_TTLS, KEY_BUNDLE_TYPE, MIN_PREKEY_COUNT, PLACEHOLDER_MAX_AGE_SECONDS, STATUS_EXPIRY_SECONDS } from '../Defaults/index.js';
7
- import { WAMessageStatus, WAMessageStubType } from '../Types/index.js';
8
- import { aesDecryptCTR, aesEncryptGCM, cleanMessage, Curve, decodeMediaRetryNode, decodeMessageNode, decryptMessageNode, delay, derivePairingCodeKey, encodeBigEndian, encodeSignedDeviceIdentity, extractAddressingContext, getCallStatusFromNode, getHistoryMsg, getNextPreKeys, getStatusFromReceiptType, handleIdentityChange, hkdf, MISSING_KEYS_ERROR_TEXT, SERVER_ERROR_CODES, NACK_REASONS, NO_MESSAGE_FOUND_ERROR_TEXT, toNumber, unixTimestampSeconds, xmppPreKey, xmppSignedPreKey } from '../Utils/index.js';
7
+ import { ReachoutTimelockEnforcementType, WAMessageStatus, WAMessageStubType } from '../Types/index.js';
8
+ import { ACCOUNT_RESTRICTED_TEXT, aesDecryptCTR, aesEncryptGCM, cleanMessage, Curve, decodeMediaRetryNode, decodeMessageNode, decryptMessageNode, delay, derivePairingCodeKey, encodeBigEndian, encodeSignedDeviceIdentity, extractAddressingContext, extractE2ESessionFromRetryReceipt, getCallStatusFromNode, getHistoryMsg, getNextPreKeys, getStatusFromReceiptType, handleIdentityChange, hkdf, MISSING_KEYS_ERROR_TEXT, NACK_REASONS, NO_MESSAGE_FOUND_ERROR_TEXT, SERVER_ERROR_CODES, toNumber, unixTimestampSeconds, xmppPreKey, xmppSignedPreKey } from '../Utils/index.js';
9
9
  import { makeMutex } from '../Utils/make-mutex.js';
10
10
  import { makeOfflineNodeProcessor } from '../Utils/offline-node-processor.js';
11
11
  import { buildAckStanza } from '../Utils/stanza-ack.js';
12
12
  import { buildMergedTcTokenIndexWrite, isTcTokenExpired, readTcTokenIndex, resolveIssuanceJid, resolveTcTokenJid, storeTcTokensFromIqResult, TC_TOKEN_INDEX_KEY } from '../Utils/tc-token-utils.js';
13
- import { areJidsSameUser, binaryNodeToString, getAllBinaryNodeChildren, getBinaryNodeChild, getBinaryNodeChildBuffer, getBinaryNodeChildren, getBinaryNodeChildString, isJidGroup, isJidNewsletter, isJidStatusBroadcast, isLidUser, isPnUser, jidDecode, jidNormalizedUser, S_WHATSAPP_NET } from '../WABinary/index.js';
13
+ import { areJidsSameUser, binaryNodeToString, getAllBinaryNodeChildren, getBinaryNodeChild, getBinaryNodeChildBuffer, getBinaryNodeChildren, getBinaryNodeChildString, getBinaryNodeChildUInt, isJidGroup, isJidNewsletter, isJidStatusBroadcast, isLidUser, isPnUser, jidDecode, jidNormalizedUser, S_WHATSAPP_NET } from '../WABinary/index.js';
14
14
  import { extractGroupMetadata } from './groups.js';
15
15
  import { makeMessagesSocket } from './messages-send.js';
16
+ const ENFORCEMENT_TYPE_VALUES = new Set(Object.values(ReachoutTimelockEnforcementType));
17
+ function isValidEnforcementType(value) {
18
+ return typeof value === 'string' && ENFORCEMENT_TYPE_VALUES.has(value);
19
+ }
16
20
  export const makeMessagesRecvSocket = (config) => {
17
21
  const { logger, retryRequestDelayMs, maxMsgRetryCount, getMessage, shouldIgnoreJid, enableAutoSessionRecreation } = config;
18
22
  const sock = makeMessagesSocket(config);
19
- const { ev, authState, ws, messageMutex, notificationMutex, receiptMutex, signalRepository, query, upsertMessage, resyncAppState, onUnexpectedError, assertSessions, sendNode, relayMessage, sendReceipt, uploadPreKeys, sendPeerDataOperationMessage, generateMessageTag, messageRetryManager, issuePrivacyTokens, registerSocketEndHandler } = sock;
23
+ const { userDevicesCache, devicesMutex, ev, authState, ws, messageMutex, notificationMutex, receiptMutex, signalRepository, query, upsertMessage, resyncAppState, onUnexpectedError, assertSessions, sendNode, relayMessage, sendReceipt, uploadPreKeys, sendPeerDataOperationMessage, messageRetryManager, registerSocketEndHandler, issuePrivacyTokens, fetchAccountReachoutTimelock, placeholderResendCache } = sock;
20
24
  const getLIDForPN = signalRepository.lidMapping.getLIDForPN.bind(signalRepository.lidMapping);
21
25
  /** this mutex ensures that each retryRequest will wait for the previous one to finish */
22
26
  const retryMutex = makeMutex();
23
- const devicesMutex = makeMutex();
24
27
  const msgRetryCache = config.msgRetryCounterCache ||
25
28
  new NodeCache({
26
29
  stdTTL: DEFAULT_CACHE_TTLS.MSG_RETRY, // 1 hour
@@ -31,16 +34,6 @@ export const makeMessagesRecvSocket = (config) => {
31
34
  stdTTL: DEFAULT_CACHE_TTLS.CALL_OFFER, // 5 mins
32
35
  useClones: false
33
36
  });
34
- const placeholderResendCache = config.placeholderResendCache ||
35
- new NodeCache({
36
- stdTTL: DEFAULT_CACHE_TTLS.MSG_RETRY, // 1 hour
37
- useClones: false
38
- });
39
- const userDevicesCache = config.userDevicesCache ??=
40
- new NodeCache({
41
- stdTTL: DEFAULT_CACHE_TTLS.USER_DEVICES, // 5 minutes
42
- useClones: false
43
- });
44
37
  // Debounce identity-change session refreshes per JID to avoid bursts
45
38
  const identityAssertDebounce = new NodeCache({ stdTTL: 5, useClones: false });
46
39
  let sendActiveReceipts = false;
@@ -94,25 +87,140 @@ export const makeMessagesRecvSocket = (config) => {
94
87
  }, 8000);
95
88
  return sendPeerDataOperationMessage(pdoMessage);
96
89
  };
97
- // Handles mex newsletter notifications
98
- const handleMexNewsletterNotification = async (node) => {
90
+ const handleMexNotification = async (node) => {
91
+ const updateNode = getBinaryNodeChild(node, 'update');
92
+ if (updateNode) {
93
+ const opName = updateNode.attrs?.op_name;
94
+ if (!opName) {
95
+ logger.warn({ node: binaryNodeToString(node) }, 'mex notification missing op_name, fallback to legacy');
96
+ await handleLegacyMexNewsletterNotification(node);
97
+ return;
98
+ }
99
+ let mexResponse;
100
+ try {
101
+ mexResponse = JSON.parse(updateNode.content.toString());
102
+ }
103
+ catch (error) {
104
+ logger.error({ err: error, opName }, 'failed to parse mex notification JSON');
105
+ return;
106
+ }
107
+ if (mexResponse.errors?.length) {
108
+ logger.warn({ errors: mexResponse.errors, opName }, 'mex notification has GQL errors');
109
+ return;
110
+ }
111
+ const data = mexResponse.data;
112
+ if (!data) {
113
+ logger.warn({ opName }, 'mex notification has null data');
114
+ return;
115
+ }
116
+ logger.debug({ opName }, 'processing mex notification');
117
+ switch (opName) {
118
+ case 'NotificationUserReachoutTimelockUpdate':
119
+ handleReachoutTimelockNotification(data);
120
+ break;
121
+ case 'MessageCappingInfoNotification':
122
+ handleMessageCappingNotification(data);
123
+ break;
124
+ // newsletter ops still use the legacy <mex> child structure
125
+ case 'NotificationNewsletterUpdate':
126
+ case 'NotificationLinkedProfilesUpdates':
127
+ case 'NotificationNewsletterAdminPromote':
128
+ case 'NotificationNewsletterAdminDemote':
129
+ case 'NotificationNewsletterUserSettingChange':
130
+ case 'NotificationNewsletterJoin':
131
+ case 'NotificationNewsletterLeave':
132
+ case 'NotificationNewsletterStateChange':
133
+ case 'NotificationNewsletterAdminMetadataUpdate':
134
+ case 'NotificationNewsletterOwnerUpdate':
135
+ case 'NotificationNewsletterAdminInviteRevoke':
136
+ case 'NotificationNewsletterWamoSubStatusChange':
137
+ case 'NotificationNewsletterBlockUser':
138
+ case 'NotificationNewsletterPaidPartnership':
139
+ case 'NotificationNewsletterMilestone':
140
+ case 'NewsletterResponseStateUpdate':
141
+ await handleLegacyMexNewsletterNotification(node);
142
+ break;
143
+ default:
144
+ logger.debug({ opName }, 'unhandled mex notification');
145
+ break;
146
+ }
147
+ return;
148
+ }
149
+ await handleLegacyMexNewsletterNotification(node);
150
+ };
151
+ const handleReachoutTimelockNotification = (data) => {
152
+ const payload = data.xwa2_notify_account_reachout_timelock;
153
+ if (!payload) {
154
+ logger.warn('reachout timelock notification missing payload');
155
+ return;
156
+ }
157
+ if (!payload.is_active) {
158
+ logger.info('reachout timelock restriction lifted');
159
+ ev.emit('connection.update', {
160
+ reachoutTimeLock: {
161
+ isActive: false,
162
+ enforcementType: ReachoutTimelockEnforcementType.DEFAULT
163
+ }
164
+ });
165
+ return;
166
+ }
167
+ // WA Web defaults to now+60s when the server omits the expiry
168
+ const timeEnforcementEnds = payload.time_enforcement_ends
169
+ ? new Date(parseInt(payload.time_enforcement_ends, 10) * 1000)
170
+ : new Date(Date.now() + 60000);
171
+ const enforcementType = isValidEnforcementType(payload.enforcement_type)
172
+ ? payload.enforcement_type
173
+ : ReachoutTimelockEnforcementType.DEFAULT;
174
+ logger.info({ enforcementType, timeEnforcementEnds }, 'reachout timelock restriction set');
175
+ ev.emit('connection.update', {
176
+ reachoutTimeLock: {
177
+ isActive: true,
178
+ timeEnforcementEnds,
179
+ enforcementType
180
+ }
181
+ });
182
+ };
183
+ const handleMessageCappingNotification = (data) => {
184
+ const payload = data.xwa2_notify_new_chat_messages_capping_info_update;
185
+ if (!payload) {
186
+ logger.warn('message capping notification missing payload');
187
+ return;
188
+ }
189
+ logger.info({ payload }, 'received message capping update');
190
+ ev.emit('message-capping.update', payload);
191
+ };
192
+ const handleLegacyMexNewsletterNotification = async (node) => {
99
193
  const mexNode = getBinaryNodeChild(node, 'mex');
100
- if (!mexNode?.content) {
101
- logger.warn({ node }, 'Invalid mex newsletter notification');
194
+ const updateNode = mexNode?.content ? null : getBinaryNodeChild(node, 'update') || getAllBinaryNodeChildren(node)[0];
195
+ const payloadNode = mexNode?.content ? mexNode : updateNode;
196
+ if (!payloadNode?.content) {
197
+ logger.warn({ node: binaryNodeToString(node) }, 'invalid mex newsletter notification');
102
198
  return;
103
199
  }
104
200
  let data;
105
201
  try {
106
- data = JSON.parse(mexNode.content.toString());
202
+ const payloadContent = payloadNode.content;
203
+ if (Array.isArray(payloadContent)) {
204
+ logger.warn({ payloadNode }, 'invalid mex newsletter notification payload format');
205
+ return;
206
+ }
207
+ const contentBuf = typeof payloadContent === 'string' ? Buffer.from(payloadContent, 'binary') : Buffer.from(payloadContent);
208
+ data = JSON.parse(contentBuf.toString());
107
209
  }
108
210
  catch (error) {
109
- logger.error({ err: error, node }, 'Failed to parse mex newsletter notification');
211
+ logger.error({ err: error, node: binaryNodeToString(node) }, 'failed to parse mex newsletter notification');
110
212
  return;
111
213
  }
112
- const operation = data?.operation;
113
- const updates = data?.updates;
214
+ const operation = data?.operation ?? payloadNode?.attrs?.op_name;
215
+ let updates = data?.updates;
216
+ if (!updates) {
217
+ const linkedProfiles = data?.data?.xwa2_notify_linked_profiles;
218
+ if (linkedProfiles) {
219
+ updates = [linkedProfiles];
220
+ }
221
+ }
114
222
  if (!updates || !operation) {
115
- logger.warn({ data }, 'Invalid mex newsletter notification content');
223
+ logger.warn({ data }, 'invalid mex newsletter notification content');
116
224
  return;
117
225
  }
118
226
  logger.info({ operation, updates }, 'got mex newsletter notification');
@@ -140,91 +248,114 @@ export const makeMessagesRecvSocket = (config) => {
140
248
  }
141
249
  }
142
250
  break;
251
+ case 'NotificationLinkedProfilesUpdates':
252
+ for (const update of updates) {
253
+ const lid = update?.jid;
254
+ const addedProfiles = Array.isArray(update?.added_profiles) ? update.added_profiles : [];
255
+ const mappings = [];
256
+ for (const profile of addedProfiles) {
257
+ const pn = typeof profile === 'string' ? profile : (profile?.pn ?? profile?.jid ?? null);
258
+ if (lid && pn) {
259
+ const mapping = { lid, pn };
260
+ ev.emit('lid-mapping.update', mapping);
261
+ mappings.push(mapping);
262
+ }
263
+ }
264
+ await signalRepository.lidMapping.storeLIDPNMappings(mappings);
265
+ }
266
+ break;
143
267
  default:
144
- logger.info({ operation, data }, 'Unhandled mex newsletter notification');
268
+ logger.info({ operation, data }, 'unhandled mex newsletter notification');
145
269
  break;
146
270
  }
147
271
  };
148
272
  // Handles newsletter notifications
149
273
  const handleNewsletterNotification = async (node) => {
150
274
  const from = node.attrs.from;
151
- const child = getAllBinaryNodeChildren(node)[0];
275
+ const children = getAllBinaryNodeChildren(node);
152
276
  const author = node.attrs.participant;
153
- logger.info({ from, child }, 'got newsletter notification');
154
- switch (child.tag) {
155
- case 'reaction':
156
- const reactionUpdate = {
157
- id: from,
158
- server_id: child.attrs.message_id,
159
- reaction: {
160
- code: getBinaryNodeChildString(child, 'reaction'),
161
- count: 1
162
- }
163
- };
164
- ev.emit('newsletter.reaction', reactionUpdate);
165
- break;
166
- case 'view':
167
- const viewUpdate = {
168
- id: from,
169
- server_id: child.attrs.message_id,
170
- count: parseInt(child.content?.toString() || '0', 10)
171
- };
172
- ev.emit('newsletter.view', viewUpdate);
173
- break;
174
- case 'participant':
175
- const participantUpdate = {
176
- id: from,
177
- author,
178
- user: child.attrs.jid,
179
- action: child.attrs.action,
180
- new_role: child.attrs.role
181
- };
182
- ev.emit('newsletter-participants.update', participantUpdate);
183
- break;
184
- case 'update':
185
- const settingsNode = getBinaryNodeChild(child, 'settings');
186
- if (settingsNode) {
187
- const update = {};
188
- const nameNode = getBinaryNodeChild(settingsNode, 'name');
189
- if (nameNode?.content)
190
- update.name = nameNode.content.toString();
191
- const descriptionNode = getBinaryNodeChild(settingsNode, 'description');
192
- if (descriptionNode?.content)
193
- update.description = descriptionNode.content.toString();
194
- ev.emit('newsletter-settings.update', {
277
+ for (const child of children) {
278
+ logger.debug({ from, child }, 'got newsletter notification');
279
+ switch (child.tag) {
280
+ case 'reaction': {
281
+ const reactionUpdate = {
195
282
  id: from,
196
- update
197
- });
283
+ server_id: child.attrs.message_id,
284
+ reaction: {
285
+ code: getBinaryNodeChildString(child, 'reaction'),
286
+ count: 1
287
+ }
288
+ };
289
+ ev.emit('newsletter.reaction', reactionUpdate);
290
+ break;
198
291
  }
199
- break;
200
- case 'message':
201
- const plaintextNode = getBinaryNodeChild(child, 'plaintext');
202
- if (plaintextNode?.content) {
203
- try {
204
- const contentBuf = typeof plaintextNode.content === 'string'
205
- ? Buffer.from(plaintextNode.content, 'binary')
206
- : Buffer.from(plaintextNode.content);
207
- const messageProto = proto.Message.decode(contentBuf).toJSON();
208
- const fullMessage = proto.WebMessageInfo.fromObject({
209
- key: {
210
- remoteJid: from,
211
- id: child.attrs.message_id || child.attrs.server_id,
212
- fromMe: false // TODO: is this really true though
213
- },
214
- message: messageProto,
215
- messageTimestamp: +child.attrs.t
216
- }).toJSON();
217
- await upsertMessage(fullMessage, 'append');
218
- logger.info('Processed plaintext newsletter message');
292
+ case 'view': {
293
+ const viewUpdate = {
294
+ id: from,
295
+ server_id: child.attrs.message_id,
296
+ count: parseInt(child.content?.toString() || '0', 10)
297
+ };
298
+ ev.emit('newsletter.view', viewUpdate);
299
+ break;
300
+ }
301
+ case 'participant': {
302
+ const participantUpdate = {
303
+ id: from,
304
+ author,
305
+ user: child.attrs.jid,
306
+ action: child.attrs.action,
307
+ new_role: child.attrs.role
308
+ };
309
+ ev.emit('newsletter-participants.update', participantUpdate);
310
+ break;
311
+ }
312
+ case 'update': {
313
+ const settingsNode = getBinaryNodeChild(child, 'settings');
314
+ if (settingsNode) {
315
+ const update = {};
316
+ const nameNode = getBinaryNodeChild(settingsNode, 'name');
317
+ if (nameNode?.content)
318
+ update.name = nameNode.content.toString();
319
+ const descriptionNode = getBinaryNodeChild(settingsNode, 'description');
320
+ if (descriptionNode?.content)
321
+ update.description = descriptionNode.content.toString();
322
+ ev.emit('newsletter-settings.update', {
323
+ id: from,
324
+ update
325
+ });
219
326
  }
220
- catch (error) {
221
- logger.error({ error }, 'Failed to decode plaintext newsletter message');
327
+ break;
328
+ }
329
+ case 'message': {
330
+ const plaintextNode = getBinaryNodeChild(child, 'plaintext');
331
+ if (plaintextNode?.content) {
332
+ try {
333
+ const contentBuf = typeof plaintextNode.content === 'string'
334
+ ? Buffer.from(plaintextNode.content, 'binary')
335
+ : Buffer.from(plaintextNode.content);
336
+ const messageProto = proto.Message.decode(contentBuf).toJSON();
337
+ const fullMessage = proto.WebMessageInfo.fromObject({
338
+ key: {
339
+ remoteJid: from,
340
+ id: child.attrs.message_id || child.attrs.server_id,
341
+ fromMe: false // TODO: is this really true though
342
+ },
343
+ message: messageProto,
344
+ messageTimestamp: +child.attrs.t
345
+ }).toJSON();
346
+ await upsertMessage(fullMessage, 'append');
347
+ logger.debug('Processed plaintext newsletter message');
348
+ }
349
+ catch (error) {
350
+ logger.error({ error }, 'Failed to decode plaintext newsletter message');
351
+ }
222
352
  }
353
+ break;
223
354
  }
224
- break;
225
- default:
226
- logger.warn({ node }, 'Unknown newsletter notification');
227
- break;
355
+ default:
356
+ logger.warn({ node, child }, 'Unknown newsletter notification child');
357
+ break;
358
+ }
228
359
  }
229
360
  };
230
361
  const sendMessageAck = async (node, errorCode) => {
@@ -253,97 +384,6 @@ export const makeMessagesRecvSocket = (config) => {
253
384
  };
254
385
  await query(stanza);
255
386
  };
256
- // Lia@Note 01-03-26 --- Source: https://github.com/koptereli/Baileys/commit/575cd41e6f01a9b3e1d7e2708c2292fa93de91f2
257
- const initiateCall = async (jid, options = {}) => {
258
- const meId = authState.creds.me?.id;
259
- if (!meId) {
260
- throw new Boom('Not authenticated');
261
- }
262
- const callId = randomBytes(8).toString('hex');
263
- const isVideo = !!options.isVideo;
264
- const isGroup = isJidGroup(jid);
265
- const stanza = {
266
- tag: 'call',
267
- attrs: {
268
- id: generateMessageTag(),
269
- from: meId,
270
- to: jid,
271
- t: String(unixTimestampSeconds()),
272
- ...(authState.creds.me?.name ? { notify: authState.creds.me.name } : {})
273
- },
274
- content: [
275
- {
276
- tag: 'offer',
277
- attrs: {
278
- 'call-id': callId,
279
- 'call-creator': meId,
280
- count: '0'
281
- },
282
- content: [
283
- {
284
- tag: isVideo ? 'video' : 'audio',
285
- attrs: {}
286
- },
287
- {
288
- tag: 'net',
289
- attrs: {}
290
- },
291
- {
292
- tag: 'encopt',
293
- attrs: { key: randomBytes(2).toString('hex') }
294
- },
295
- {
296
- tag: 'relaylatency',
297
- attrs: {}
298
- },
299
- {
300
- tag: 'te',
301
- attrs: {}
302
- }
303
- ]
304
- }
305
- ]
306
- };
307
- await query(stanza);
308
- await callOfferCache.set(callId, {
309
- chatId: jid,
310
- from: meId,
311
- id: callId,
312
- date: new Date(),
313
- offline: false,
314
- status: 'offer',
315
- isVideo,
316
- isGroup,
317
- groupJid: isGroup ? jid : undefined
318
- });
319
- // TODO: implement ICE/DTLS-SRTP call media setup once full signaling requirements are mapped.
320
- return { callId, to: jid, isVideo };
321
- };
322
- const cancelCall = async (callId, callTo) => {
323
- const meId = authState.creds.me?.id;
324
- if (!meId) {
325
- throw new Boom('Not authenticated');
326
- }
327
- const stanza = {
328
- tag: 'call',
329
- attrs: {
330
- from: meId,
331
- to: callTo
332
- },
333
- content: [
334
- {
335
- tag: 'terminate',
336
- attrs: {
337
- 'call-id': callId,
338
- 'call-creator': meId,
339
- count: '0'
340
- }
341
- }
342
- ]
343
- };
344
- await query(stanza);
345
- await callOfferCache.del(callId);
346
- };
347
387
  const sendRetryRequest = async (node, forceIncludeKeys = false) => {
348
388
  const { fullMessage } = decodeMessageNode(node, authState.creds.me.id, authState.creds.me.lid || '');
349
389
  const { key: msgKey } = fullMessage;
@@ -475,6 +515,8 @@ export const makeMessagesRecvSocket = (config) => {
475
515
  logger.info({ msgAttrs: node.attrs, retryCount }, 'sent retry receipt');
476
516
  }, authState?.creds?.me?.id || 'sendRetryRequest');
477
517
  };
518
+ // Mirrors WAWeb/Handle/PreKeyLow.js: skip a re-issued notification with the same stanza id.
519
+ const inFlightPreKeyLow = new Set();
478
520
  /**
479
521
  * Fire-and-forget tctoken re-issuance after a peer's device identity changed.
480
522
  * Mirrors WAWebSendTcTokenWhenDeviceIdentityChange — runs in parallel with
@@ -507,12 +549,24 @@ export const makeMessagesRecvSocket = (config) => {
507
549
  const handleEncryptNotification = async (node) => {
508
550
  const from = node.attrs.from;
509
551
  if (from === S_WHATSAPP_NET) {
552
+ const stanzaId = node.attrs.id;
553
+ if (stanzaId && inFlightPreKeyLow.has(stanzaId)) {
554
+ return;
555
+ }
510
556
  const countChild = getBinaryNodeChild(node, 'count');
511
557
  const count = +countChild.attrs.value;
512
558
  const shouldUploadMorePreKeys = count < MIN_PREKEY_COUNT;
513
559
  logger.debug({ count, shouldUploadMorePreKeys }, 'recv pre-key count');
514
560
  if (shouldUploadMorePreKeys) {
515
- await uploadPreKeys();
561
+ if (stanzaId)
562
+ inFlightPreKeyLow.add(stanzaId);
563
+ try {
564
+ await uploadPreKeys();
565
+ }
566
+ finally {
567
+ if (stanzaId)
568
+ inFlightPreKeyLow.delete(stanzaId);
569
+ }
516
570
  }
517
571
  }
518
572
  else {
@@ -659,59 +713,83 @@ export const makeMessagesRecvSocket = (config) => {
659
713
  const handleDevicesNotification = async (node) => {
660
714
  const [child] = getAllBinaryNodeChildren(node);
661
715
  const from = jidNormalizedUser(node.attrs.from);
716
+ if (!child) {
717
+ logger.debug({ from }, 'devices notification missing child, skipping');
718
+ return;
719
+ }
720
+ const tag = child.tag;
721
+ const deviceHash = child.attrs.device_hash;
662
722
  const devices = getBinaryNodeChildren(child, 'device');
663
- if (areJidsSameUser(from, authState.creds.me.id) ||
664
- areJidsSameUser(from, authState.creds.me.lid)) {
723
+ if (areJidsSameUser(from, authState.creds.me.id) || areJidsSameUser(from, authState.creds.me.lid)) {
665
724
  const deviceJids = devices.map(d => d.attrs.jid);
666
725
  logger.info({ deviceJids }, 'got my own devices');
667
726
  }
668
- if (!devices || !devices.length || !devices[0]) {
669
- logger.debug({ from }, 'no devices in notification, skipping');
727
+ if (!devices.length) {
728
+ logger.debug({ from, tag }, 'no devices in notification, skipping');
670
729
  return;
671
730
  }
672
- const deviceJid = devices[0].attrs.jid;
673
- const decoded = jidDecode(deviceJid);
674
- if (!decoded)
675
- return;
676
- const { user, device } = decoded;
677
- const tag = child.tag;
678
- if (!deviceJid) {
679
- logger.debug({ tag }, 'no device jid in notification, skipping');
680
- return;
731
+ const decoded = [];
732
+ for (const d of devices) {
733
+ const jid = d.attrs.jid;
734
+ if (!jid)
735
+ continue;
736
+ const parts = jidDecode(jid);
737
+ if (!parts) {
738
+ logger.debug({ jid }, 'failed to decode device jid, skipping');
739
+ continue;
740
+ }
741
+ decoded.push({ jid, user: parts.user, server: parts.server, device: parts.device });
681
742
  }
743
+ if (!decoded.length)
744
+ return;
682
745
  await devicesMutex.mutex(async () => {
683
- if (tag === 'update') {
684
- logger.debug({ user }, `${user}'s device list updated, dropping cached devices`);
685
- if (userDevicesCache) {
686
- await userDevicesCache.del(user);
687
- }
688
- return;
689
- }
690
- const existingCache = (await (userDevicesCache?.get(user))) || [];
691
- if (!existingCache.length) {
692
- logger.debug({ user, tag }, 'device list not cached, skipping cache update');
693
- return;
746
+ const byUser = new Map();
747
+ for (const d of decoded) {
748
+ const list = byUser.get(d.user) || [];
749
+ list.push(d);
750
+ byUser.set(d.user, list);
694
751
  }
695
- const deviceHash = child.attrs.device_hash;
696
- let updatedDevices = [];
697
- switch (tag) {
698
- case 'add':
699
- logger.info({ deviceHash }, 'device added');
700
- updatedDevices = [
701
- ...existingCache.filter(d => d.device !== device),
702
- { user, device }
703
- ];
704
- break;
705
- case 'remove':
706
- logger.info({ deviceHash }, 'device removed');
707
- updatedDevices = existingCache.filter(d => d.device !== device);
708
- break;
709
- default:
710
- logger.debug({ tag }, 'Unknown device list change tag');
711
- return;
712
- }
713
- if (updatedDevices.length > 0 && userDevicesCache) {
714
- await userDevicesCache.set(user, updatedDevices);
752
+ for (const [user, entries] of byUser) {
753
+ if (tag === 'update') {
754
+ logger.debug({ user }, `${user}'s device list updated, dropping cached devices`);
755
+ await userDevicesCache?.del(user);
756
+ continue;
757
+ }
758
+ if (tag === 'remove') {
759
+ await signalRepository.deleteSession(entries.map(e => e.jid));
760
+ }
761
+ const existingCache = (await userDevicesCache?.get(user)) || [];
762
+ if (!existingCache.length) {
763
+ // No baseline yet; skip applying the delta so getUSyncDevices can
764
+ // later fetch the full device list. Caching just the notification
765
+ // entries would make a partial list look authoritative.
766
+ logger.debug({ user, tag }, 'device list not cached, deferring to USync refresh');
767
+ continue;
768
+ }
769
+ const affected = new Set(entries.map(e => e.device));
770
+ let updatedDevices;
771
+ switch (tag) {
772
+ case 'add':
773
+ logger.info({ deviceHash, count: entries.length }, 'devices added');
774
+ updatedDevices = [
775
+ ...existingCache.filter(d => !affected.has(d.device)),
776
+ ...entries.map(e => ({ user: e.user, server: e.server, device: e.device }))
777
+ ];
778
+ break;
779
+ case 'remove':
780
+ logger.info({ deviceHash, count: entries.length }, 'devices removed');
781
+ updatedDevices = existingCache.filter(d => !affected.has(d.device));
782
+ break;
783
+ default:
784
+ logger.debug({ tag }, 'Unknown device list change tag');
785
+ continue;
786
+ }
787
+ if (updatedDevices.length === 0) {
788
+ await userDevicesCache?.del(user);
789
+ }
790
+ else {
791
+ await userDevicesCache?.set(user, updatedDevices);
792
+ }
715
793
  }
716
794
  });
717
795
  };
@@ -725,7 +803,7 @@ export const makeMessagesRecvSocket = (config) => {
725
803
  await handleNewsletterNotification(node);
726
804
  break;
727
805
  case 'mex':
728
- await handleMexNewsletterNotification(node);
806
+ await handleMexNotification(node);
729
807
  break;
730
808
  case 'w:gp2':
731
809
  // TODO: HANDLE PARTICIPANT_PN
@@ -879,9 +957,8 @@ export const makeMessagesRecvSocket = (config) => {
879
957
  const tcTokenIndexLoaded = (async () => {
880
958
  try {
881
959
  const jids = await readTcTokenIndex(authState.keys);
882
- for (const jid of jids) {
960
+ for (const jid of jids)
883
961
  tcTokenKnownJids.add(jid);
884
- }
885
962
  logger.debug({ count: tcTokenKnownJids.size }, 'loaded tctoken index');
886
963
  }
887
964
  catch (err) {
@@ -899,7 +976,6 @@ export const makeMessagesRecvSocket = (config) => {
899
976
  const write = await buildMergedTcTokenIndexWrite(authState.keys, tcTokenKnownJids);
900
977
  return authState.keys.set({ tctoken: write });
901
978
  }
902
- ;
903
979
  function scheduleTcTokenIndexSave() {
904
980
  if (tcTokenIndexTimer) {
905
981
  clearTimeout(tcTokenIndexTimer);
@@ -911,7 +987,6 @@ export const makeMessagesRecvSocket = (config) => {
911
987
  });
912
988
  }, 5000);
913
989
  }
914
- ;
915
990
  function trackTcTokenJid(jid) {
916
991
  if (jid && jid !== TC_TOKEN_INDEX_KEY && !tcTokenKnownJids.has(jid)) {
917
992
  tcTokenKnownJids.add(jid);
@@ -962,10 +1037,11 @@ export const makeMessagesRecvSocket = (config) => {
962
1037
  const newValue = ((await msgRetryCache.get(key)) || 0) + 1;
963
1038
  await msgRetryCache.set(key, newValue);
964
1039
  };
965
- const sendMessagesAgain = async (key, ids, retryNode) => {
1040
+ const sendMessagesAgain = async (key, ids, retryNode, receiptNode) => {
966
1041
  const remoteJid = key.remoteJid;
967
1042
  const participant = key.participant || remoteJid;
968
1043
  const retryCount = +retryNode.attrs.count || 1;
1044
+ const msgId = ids[0];
969
1045
  // Try to get messages from cache first, then fallback to getMessage
970
1046
  const msgs = [];
971
1047
  for (const id of ids) {
@@ -997,12 +1073,49 @@ export const makeMessagesRecvSocket = (config) => {
997
1073
  // just re-send the message to everyone
998
1074
  // prevents the first message decryption failure
999
1075
  const sendToAll = !jidDecode(participant)?.device;
1000
- // Check if we should recreate session for this retry
1076
+ const sessionId = signalRepository.jidToSignalProtocolAddress(participant);
1077
+ let injectedFromBundle = false;
1078
+ const bundle = extractE2ESessionFromRetryReceipt(receiptNode);
1079
+ if (bundle) {
1080
+ try {
1081
+ await signalRepository.injectE2ESession({ jid: participant, session: bundle });
1082
+ injectedFromBundle = true;
1083
+ logger.debug({ participant, retryCount }, 'injected session from retry receipt key bundle');
1084
+ }
1085
+ catch (error) {
1086
+ logger.warn({ error, participant }, 'failed to inject session from retry receipt');
1087
+ }
1088
+ }
1089
+ if (!injectedFromBundle) {
1090
+ const receivedRegId = getBinaryNodeChildUInt(receiptNode, 'registration', 4);
1091
+ if (typeof receivedRegId === 'number' && Number.isInteger(receivedRegId)) {
1092
+ const info = await signalRepository.getSessionInfo(participant);
1093
+ if (info && info.registrationId !== 0 && info.registrationId !== receivedRegId) {
1094
+ logger.info({ participant, stored: info.registrationId, received: receivedRegId }, 'reg id mismatch on retry without bundle, deleting session');
1095
+ await authState.keys.set({ session: { [sessionId]: null } });
1096
+ }
1097
+ }
1098
+ }
1099
+ const BASE_KEY_CHECK_RETRY = 2;
1100
+ if (msgId && messageRetryManager) {
1101
+ const info = await signalRepository.getSessionInfo(participant);
1102
+ if (info) {
1103
+ if (retryCount === BASE_KEY_CHECK_RETRY) {
1104
+ messageRetryManager.saveBaseKey(sessionId, msgId, info.baseKey);
1105
+ }
1106
+ else if (retryCount > BASE_KEY_CHECK_RETRY) {
1107
+ if (messageRetryManager.hasSameBaseKey(sessionId, msgId, info.baseKey)) {
1108
+ logger.warn({ participant, retryCount }, 'base key collision on retry, forcing fresh session');
1109
+ await authState.keys.set({ session: { [sessionId]: null } });
1110
+ }
1111
+ messageRetryManager.deleteBaseKey(sessionId, msgId);
1112
+ }
1113
+ }
1114
+ }
1001
1115
  let shouldRecreateSession = false;
1002
1116
  let recreateReason = '';
1003
- if (enableAutoSessionRecreation && messageRetryManager && retryCount > 1) {
1117
+ if (enableAutoSessionRecreation && messageRetryManager && retryCount > 1 && !injectedFromBundle) {
1004
1118
  try {
1005
- const sessionId = signalRepository.jidToSignalProtocolAddress(participant);
1006
1119
  const hasSession = await signalRepository.validateSession(participant);
1007
1120
  const result = messageRetryManager.shouldRecreateSession(participant, hasSession.exists);
1008
1121
  shouldRecreateSession = result.recreate;
@@ -1016,11 +1129,13 @@ export const makeMessagesRecvSocket = (config) => {
1016
1129
  logger.warn({ error, participant }, 'failed to check session recreation for outgoing retry');
1017
1130
  }
1018
1131
  }
1019
- await assertSessions([participant], true);
1132
+ if (!injectedFromBundle) {
1133
+ await assertSessions([participant], true);
1134
+ }
1020
1135
  if (isJidGroup(remoteJid)) {
1021
1136
  await authState.keys.set({ 'sender-key-memory': { [remoteJid]: null } });
1022
1137
  }
1023
- logger.debug({ participant, sendToAll, shouldRecreateSession, recreateReason }, 'forced new session for retry recp');
1138
+ logger.debug({ participant, sendToAll, shouldRecreateSession, recreateReason, injectedFromBundle }, 'prepared session for retry resend');
1024
1139
  for (const [i, msg] of msgs.entries()) {
1025
1140
  if (!ids[i])
1026
1141
  continue;
@@ -1096,7 +1211,7 @@ export const makeMessagesRecvSocket = (config) => {
1096
1211
  try {
1097
1212
  await updateSendMessageAgainCount(ids[0], key.participant);
1098
1213
  logger.debug({ attrs, key }, 'recv retry request');
1099
- await sendMessagesAgain(key, ids, retryNode);
1214
+ await sendMessagesAgain(key, ids, retryNode, node);
1100
1215
  }
1101
1216
  catch (error) {
1102
1217
  logger.error({ key, ids, trace: error instanceof Error ? error.stack : 'Unknown error' }, 'error in sending message again');
@@ -1131,7 +1246,7 @@ export const makeMessagesRecvSocket = (config) => {
1131
1246
  fromMe,
1132
1247
  participant: node.attrs.participant,
1133
1248
  participantAlt,
1134
- username: attrs.participant_username || attrs.username || undefined,
1249
+ participantUsername: node.attrs.participant_username,
1135
1250
  addressingMode,
1136
1251
  id: node.attrs.id,
1137
1252
  ...(msg.key || {})
@@ -1252,29 +1367,14 @@ export const makeMessagesRecvSocket = (config) => {
1252
1367
  return sendMessageAck(node);
1253
1368
  }
1254
1369
  }
1255
- const errorMessage = msg?.messageStubParameters?.[0] || '';
1256
- const isPreKeyError = errorMessage.includes('PreKey');
1257
- logger.debug(`[handleMessage] Attempting retry request for failed decryption`);
1258
- // Handle both pre-key and normal retries in single mutex
1370
+ logger.debug('[handleMessage] Attempting retry request for failed decryption');
1371
+ // WAWeb only retry-receipts here; server emits PreKeyLow if prekeys run low.
1259
1372
  await retryMutex.mutex(async () => {
1260
1373
  try {
1261
1374
  if (!ws.isOpen) {
1262
1375
  logger.debug({ node }, 'Connection closed, skipping retry');
1263
1376
  return;
1264
1377
  }
1265
- // Handle pre-key errors with upload and delay
1266
- if (isPreKeyError) {
1267
- logger.info({ error: errorMessage }, 'PreKey error detected, uploading and retrying');
1268
- try {
1269
- logger.debug('Uploading pre-keys for error recovery');
1270
- await uploadPreKeys(5);
1271
- logger.debug('Waiting for server to process new pre-keys');
1272
- await delay(1000);
1273
- }
1274
- catch (uploadErr) {
1275
- logger.error({ uploadErr }, 'Pre-key upload failed, proceeding with retry anyway');
1276
- }
1277
- }
1278
1378
  const encNode = getBinaryNodeChild(node, 'enc');
1279
1379
  await sendRetryRequest(node, !encNode);
1280
1380
  if (retryRequestDelayMs) {
@@ -1282,15 +1382,7 @@ export const makeMessagesRecvSocket = (config) => {
1282
1382
  }
1283
1383
  }
1284
1384
  catch (err) {
1285
- logger.error({ err, isPreKeyError }, 'Failed to handle retry, attempting basic retry');
1286
- // Still attempt retry even if pre-key upload failed
1287
- try {
1288
- const encNode = getBinaryNodeChild(node, 'enc');
1289
- await sendRetryRequest(node, !encNode);
1290
- }
1291
- catch (retryErr) {
1292
- logger.error({ retryErr }, 'Failed to send retry after error handling');
1293
- }
1385
+ logger.error({ err }, 'Failed to send retry');
1294
1386
  }
1295
1387
  acked = true;
1296
1388
  await sendMessageAck(node, NACK_REASONS.UnhandledError);
@@ -1417,16 +1509,49 @@ export const makeMessagesRecvSocket = (config) => {
1417
1509
  // error in acknowledgement,
1418
1510
  // device could not display the message
1419
1511
  if (attrs.error) {
1420
- if (attrs.error === SERVER_ERROR_CODES.MissingTcToken) {
1421
- // 463 = account restricted + no tctoken for this contact.
1422
- // WA Web prevents this client-side (disables compose bar).
1423
- // No retry retrying worsens the restriction by counting
1424
- // as another "reach out" to an unknown contact.
1512
+ const isReachoutTimelocked = attrs.error === String(NACK_REASONS.SenderReachoutTimelocked);
1513
+ if (attrs.error === SERVER_ERROR_CODES.MessageAccountRestriction) {
1514
+ // 463 = 1:1 message missing privacy token (tctoken). Usually means the
1515
+ // account is restricted: WhatsApp blocks starting new chats but preserves
1516
+ // existing ones, since established chats already carry a tctoken.
1517
+ // WA Web prevents this client-side (disables the compose bar).
1518
+ // No retry — retrying counts as another "reach out" and worsens the restriction.
1425
1519
  logger.warn({ msgId: attrs.id, from: attrs.from }, 'error 463: account restricted or missing tctoken for contact');
1520
+ const ackFrom = attrs.from;
1521
+ if (ackFrom && !inFlight463Recoveries.has(ackFrom)) {
1522
+ inFlight463Recoveries.add(ackFrom);
1523
+ void (async () => {
1524
+ try {
1525
+ const getPNForLID = signalRepository.lidMapping.getPNForLID.bind(signalRepository.lidMapping);
1526
+ const tcStorageJid = await resolveTcTokenJid(ackFrom, getLIDForPN);
1527
+ const issueJid = await resolveIssuanceJid(ackFrom, sock.serverProps.lidTrustedTokenIssueToLid, getLIDForPN, getPNForLID);
1528
+ const result = await issuePrivacyTokens([issueJid], unixTimestampSeconds());
1529
+ await storeTcTokensFromIqResult({
1530
+ result,
1531
+ fallbackJid: tcStorageJid,
1532
+ keys: authState.keys,
1533
+ getLIDForPN,
1534
+ onNewJidStored: trackTcTokenJid
1535
+ });
1536
+ logger.debug({ from: ackFrom }, 'completed 463 token recovery issuance');
1537
+ }
1538
+ catch (err) {
1539
+ logger.debug({ from: ackFrom, err: err?.message }, 'failed 463 token recovery issuance');
1540
+ }
1541
+ finally {
1542
+ inFlight463Recoveries.delete(ackFrom);
1543
+ }
1544
+ })();
1545
+ }
1426
1546
  }
1427
1547
  else if (attrs.error === SERVER_ERROR_CODES.SmaxInvalid) {
1428
1548
  logger.warn({ msgId: attrs.id, from: attrs.from }, 'smax-invalid (479): stanza rejected by server — likely stale device session or malformed addressing');
1429
1549
  }
1550
+ else if (isReachoutTimelocked) {
1551
+ // user is temporarily restricted, fetch current restriction details
1552
+ await fetchAccountReachoutTimelock().catch(err => logger.warn({ err }, 'failed to fetch reachout timelock'));
1553
+ logger.warn({ attrs }, 'received error in ack');
1554
+ }
1430
1555
  else {
1431
1556
  logger.warn({ attrs }, 'received error in ack');
1432
1557
  }
@@ -1435,7 +1560,7 @@ export const makeMessagesRecvSocket = (config) => {
1435
1560
  key,
1436
1561
  update: {
1437
1562
  status: WAMessageStatus.ERROR,
1438
- messageStubParameters: [attrs.error]
1563
+ messageStubParameters: isReachoutTimelocked ? [attrs.error, ACCOUNT_RESTRICTED_TEXT] : [attrs.error]
1439
1564
  }
1440
1565
  }
1441
1566
  ]);
@@ -1532,6 +1657,8 @@ export const makeMessagesRecvSocket = (config) => {
1532
1657
  });
1533
1658
  /** timestamp of last tctoken prune run — throttles to once per 24h */
1534
1659
  let lastTcTokenPruneTs = 0;
1660
+ /** dedupe in-flight 463 recovery token issuance by target JID */
1661
+ const inFlight463Recoveries = new Set();
1535
1662
  ev.on('connection.update', ({ isOnline, connection }) => {
1536
1663
  if (typeof isOnline !== 'undefined') {
1537
1664
  sendActiveReceipts = isOnline;
@@ -1561,6 +1688,16 @@ export const makeMessagesRecvSocket = (config) => {
1561
1688
  }
1562
1689
  }
1563
1690
  });
1691
+ registerSocketEndHandler(() => {
1692
+ if (!config.msgRetryCounterCache && msgRetryCache.close) {
1693
+ msgRetryCache.close();
1694
+ }
1695
+ if (!config.callOfferCache && callOfferCache.close) {
1696
+ callOfferCache.close();
1697
+ }
1698
+ identityAssertDebounce.close();
1699
+ sendActiveReceipts = false;
1700
+ });
1564
1701
  async function pruneExpiredTcTokens() {
1565
1702
  try {
1566
1703
  await tcTokenIndexLoaded;
@@ -1568,9 +1705,8 @@ export const makeMessagesRecvSocket = (config) => {
1568
1705
  // (history sync) without needing inter-module wiring.
1569
1706
  const persisted = await readTcTokenIndex(authState.keys);
1570
1707
  const allJids = new Set(tcTokenKnownJids);
1571
- for (const jid of persisted) {
1708
+ for (const jid of persisted)
1572
1709
  allJids.add(jid);
1573
- }
1574
1710
  if (!allJids.size)
1575
1711
  return;
1576
1712
  const jids = [...allJids];
@@ -1615,36 +1751,19 @@ export const makeMessagesRecvSocket = (config) => {
1615
1751
  }
1616
1752
  });
1617
1753
  tcTokenKnownJids.clear();
1618
- for (const jid of survivors) {
1754
+ for (const jid of survivors)
1619
1755
  tcTokenKnownJids.add(jid);
1620
- }
1621
1756
  logger.debug({ mutated, remaining: survivors.size }, 'pruned expired tctokens');
1622
1757
  }
1623
1758
  catch (err) {
1624
1759
  logger.warn({ err: err?.message }, 'failed to prune expired tctokens');
1625
1760
  }
1626
1761
  }
1627
- ;
1628
- registerSocketEndHandler(() => {
1629
- if (!config.msgRetryCounterCache && msgRetryCache.close) {
1630
- msgRetryCache.close();
1631
- }
1632
- if (!config.callOfferCache && callOfferCache.close) {
1633
- callOfferCache.close();
1634
- }
1635
- if (!config.placeholderResendCache && placeholderResendCache.close) {
1636
- placeholderResendCache.close();
1637
- }
1638
- identityAssertDebounce.close();
1639
- sendActiveReceipts = false;
1640
- });
1641
1762
  return {
1642
1763
  ...sock,
1643
1764
  sendMessageAck,
1644
1765
  sendRetryRequest,
1645
1766
  rejectCall,
1646
- initiateCall,
1647
- cancelCall,
1648
1767
  fetchMessageHistory,
1649
1768
  requestPlaceholderResend,
1650
1769
  messageRetryManager