@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.
- package/README.md +214 -7
- package/WAProto/index.d.ts +392 -2465
- package/WAProto/index.js +13116 -3569
- package/lib/Defaults/index.d.ts +8 -9
- package/lib/Defaults/index.d.ts.map +1 -1
- package/lib/Defaults/index.js +13 -14
- package/lib/Defaults/index.js.map +1 -1
- package/lib/Signal/Group/ciphertext-message.d.ts.map +1 -1
- package/lib/Signal/Group/ciphertext-message.js.map +1 -1
- package/lib/Signal/Group/group-session-builder.d.ts.map +1 -1
- package/lib/Signal/Group/group-session-builder.js.map +1 -1
- package/lib/Signal/Group/group_cipher.d.ts.map +1 -1
- package/lib/Signal/Group/group_cipher.js.map +1 -1
- package/lib/Signal/Group/index.d.ts.map +1 -1
- package/lib/Signal/Group/index.js.map +1 -1
- package/lib/Signal/Group/keyhelper.d.ts.map +1 -1
- package/lib/Signal/Group/keyhelper.js.map +1 -1
- package/lib/Signal/Group/sender-chain-key.d.ts.map +1 -1
- package/lib/Signal/Group/sender-chain-key.js.map +1 -1
- package/lib/Signal/Group/sender-key-distribution-message.d.ts.map +1 -1
- package/lib/Signal/Group/sender-key-distribution-message.js.map +1 -1
- package/lib/Signal/Group/sender-key-message.d.ts.map +1 -1
- package/lib/Signal/Group/sender-key-message.js.map +1 -1
- package/lib/Signal/Group/sender-key-name.d.ts.map +1 -1
- package/lib/Signal/Group/sender-key-name.js.map +1 -1
- package/lib/Signal/Group/sender-key-record.d.ts.map +1 -1
- package/lib/Signal/Group/sender-key-record.js.map +1 -1
- package/lib/Signal/Group/sender-key-state.d.ts.map +1 -1
- package/lib/Signal/Group/sender-key-state.js.map +1 -1
- package/lib/Signal/Group/sender-message-key.d.ts.map +1 -1
- package/lib/Signal/Group/sender-message-key.js.map +1 -1
- package/lib/Signal/libsignal.d.ts +12 -0
- package/lib/Signal/libsignal.d.ts.map +1 -1
- package/lib/Signal/libsignal.js +41 -17
- package/lib/Signal/libsignal.js.map +1 -1
- package/lib/Signal/lid-mapping.d.ts +2 -2
- package/lib/Signal/lid-mapping.d.ts.map +1 -1
- package/lib/Signal/lid-mapping.js +2 -2
- package/lib/Signal/lid-mapping.js.map +1 -1
- package/lib/Socket/Client/index.d.ts.map +1 -1
- package/lib/Socket/Client/index.js.map +1 -1
- package/lib/Socket/Client/types.d.ts.map +1 -1
- package/lib/Socket/Client/types.js.map +1 -1
- package/lib/Socket/Client/websocket.d.ts.map +1 -1
- package/lib/Socket/Client/websocket.js.map +1 -1
- package/lib/Socket/business.d.ts +20 -41
- package/lib/Socket/business.d.ts.map +1 -1
- package/lib/Socket/business.js +1 -0
- package/lib/Socket/business.js.map +1 -1
- package/lib/Socket/chats.d.ts +12 -2
- package/lib/Socket/chats.d.ts.map +1 -1
- package/lib/Socket/chats.js +21 -37
- package/lib/Socket/chats.js.map +1 -1
- package/lib/Socket/communities.d.ts +22 -42
- package/lib/Socket/communities.d.ts.map +1 -1
- package/lib/Socket/communities.js +1 -0
- package/lib/Socket/communities.js.map +1 -1
- package/lib/Socket/groups.d.ts +14 -35
- package/lib/Socket/groups.d.ts.map +1 -1
- package/lib/Socket/groups.js +17 -4
- package/lib/Socket/groups.js.map +1 -1
- package/lib/Socket/index.d.ts +21 -41
- package/lib/Socket/index.d.ts.map +1 -1
- package/lib/Socket/index.js.map +1 -1
- package/lib/Socket/messages-recv.d.ts +19 -41
- package/lib/Socket/messages-recv.d.ts.map +1 -1
- package/lib/Socket/messages-recv.js +416 -297
- package/lib/Socket/messages-recv.js.map +1 -1
- package/lib/Socket/messages-send.d.ts +19 -35
- package/lib/Socket/messages-send.d.ts.map +1 -1
- package/lib/Socket/messages-send.js +68 -48
- package/lib/Socket/messages-send.js.map +1 -1
- package/lib/Socket/mex.d.ts.map +1 -1
- package/lib/Socket/mex.js.map +1 -1
- package/lib/Socket/newsletter.d.ts +14 -35
- package/lib/Socket/newsletter.d.ts.map +1 -1
- package/lib/Socket/newsletter.js +2 -3
- package/lib/Socket/newsletter.js.map +1 -1
- package/lib/Socket/socket.d.ts +7 -1
- package/lib/Socket/socket.d.ts.map +1 -1
- package/lib/Socket/socket.js +40 -23
- package/lib/Socket/socket.js.map +1 -1
- package/lib/Store/index.d.ts.map +1 -1
- package/lib/Store/index.js.map +1 -1
- package/lib/Store/make-in-memory-store.d.ts.map +1 -1
- package/lib/Store/make-in-memory-store.js.map +1 -1
- package/lib/Store/make-ordered-dictionary.d.ts.map +1 -1
- package/lib/Store/make-ordered-dictionary.js.map +1 -1
- package/lib/Store/object-repository.d.ts.map +1 -1
- package/lib/Store/object-repository.js.map +1 -1
- package/lib/Types/Auth.d.ts.map +1 -1
- package/lib/Types/Auth.js.map +1 -1
- package/lib/Types/Bussines.d.ts.map +1 -1
- package/lib/Types/Bussines.js.map +1 -1
- package/lib/Types/Call.d.ts.map +1 -1
- package/lib/Types/Call.js.map +1 -1
- package/lib/Types/Chat.d.ts.map +1 -1
- package/lib/Types/Chat.js.map +1 -1
- package/lib/Types/Contact.d.ts.map +1 -1
- package/lib/Types/Contact.js.map +1 -1
- package/lib/Types/Events.d.ts.map +1 -1
- package/lib/Types/Events.js.map +1 -1
- package/lib/Types/GroupMetadata.d.ts.map +1 -1
- package/lib/Types/GroupMetadata.js.map +1 -1
- package/lib/Types/Label.d.ts.map +1 -1
- package/lib/Types/Label.js.map +1 -1
- package/lib/Types/LabelAssociation.d.ts.map +1 -1
- package/lib/Types/LabelAssociation.js.map +1 -1
- package/lib/Types/Message.d.ts.map +1 -1
- package/lib/Types/Message.js.map +1 -1
- package/lib/Types/{Newsletter.d.ts → Mex.d.ts} +1 -1
- package/lib/Types/Mex.d.ts.map +1 -0
- package/lib/Types/{Newsletter.js → Mex.js} +9 -4
- package/lib/Types/Mex.js.map +1 -0
- package/lib/Types/Product.d.ts.map +1 -1
- package/lib/Types/Product.js.map +1 -1
- package/lib/Types/RichType.d.ts.map +1 -1
- package/lib/Types/RichType.js.map +1 -1
- package/lib/Types/Signal.d.ts.map +1 -1
- package/lib/Types/Signal.js.map +1 -1
- package/lib/Types/Socket.d.ts.map +1 -1
- package/lib/Types/Socket.js.map +1 -1
- package/lib/Types/State.d.ts +4 -0
- package/lib/Types/State.d.ts.map +1 -1
- package/lib/Types/State.js +43 -0
- package/lib/Types/State.js.map +1 -1
- package/lib/Types/USync.d.ts.map +1 -1
- package/lib/Types/USync.js.map +1 -1
- package/lib/Types/index.d.ts +1 -1
- package/lib/Types/index.d.ts.map +1 -1
- package/lib/Types/index.js +1 -1
- package/lib/Types/index.js.map +1 -1
- package/lib/Utils/auth-utils.d.ts +1 -0
- package/lib/Utils/auth-utils.d.ts.map +1 -1
- package/lib/Utils/auth-utils.js +12 -0
- package/lib/Utils/auth-utils.js.map +1 -1
- package/lib/Utils/browser-utils.d.ts +0 -1
- package/lib/Utils/browser-utils.d.ts.map +1 -1
- package/lib/Utils/browser-utils.js +1 -2
- package/lib/Utils/browser-utils.js.map +1 -1
- package/lib/Utils/business.d.ts.map +1 -1
- package/lib/Utils/business.js.map +1 -1
- package/lib/Utils/chat-utils.d.ts +5 -5
- package/lib/Utils/chat-utils.d.ts.map +1 -1
- package/lib/Utils/chat-utils.js +69 -36
- package/lib/Utils/chat-utils.js.map +1 -1
- package/lib/Utils/companion-reg-client-utils.d.ts +1 -12
- package/lib/Utils/companion-reg-client-utils.d.ts.map +1 -1
- package/lib/Utils/companion-reg-client-utils.js +20 -13
- package/lib/Utils/companion-reg-client-utils.js.map +1 -1
- package/lib/Utils/crypto.d.ts.map +1 -1
- package/lib/Utils/crypto.js.map +1 -1
- package/lib/Utils/decode-wa-message.d.ts +3 -1
- package/lib/Utils/decode-wa-message.d.ts.map +1 -1
- package/lib/Utils/decode-wa-message.js +17 -3
- package/lib/Utils/decode-wa-message.js.map +1 -1
- package/lib/Utils/event-buffer.d.ts.map +1 -1
- package/lib/Utils/event-buffer.js +30 -0
- package/lib/Utils/event-buffer.js.map +1 -1
- package/lib/Utils/generics.d.ts +1 -1
- package/lib/Utils/generics.d.ts.map +1 -1
- package/lib/Utils/generics.js +5 -5
- package/lib/Utils/generics.js.map +1 -1
- package/lib/Utils/history.d.ts +2 -0
- package/lib/Utils/history.d.ts.map +1 -1
- package/lib/Utils/history.js +1 -0
- package/lib/Utils/history.js.map +1 -1
- package/lib/Utils/identity-change-handler.d.ts.map +1 -1
- package/lib/Utils/identity-change-handler.js.map +1 -1
- package/lib/Utils/index.d.ts +1 -1
- package/lib/Utils/index.d.ts.map +1 -1
- package/lib/Utils/index.js +1 -1
- package/lib/Utils/index.js.map +1 -1
- package/lib/Utils/link-preview.d.ts.map +1 -1
- package/lib/Utils/link-preview.js +2 -2
- package/lib/Utils/link-preview.js.map +1 -1
- package/lib/Utils/logger.d.ts.map +1 -1
- package/lib/Utils/logger.js.map +1 -1
- package/lib/Utils/lt-hash.d.ts.map +1 -1
- package/lib/Utils/lt-hash.js.map +1 -1
- package/lib/Utils/make-mutex.d.ts.map +1 -1
- package/lib/Utils/make-mutex.js.map +1 -1
- package/lib/Utils/message-retry-manager.d.ts +4 -0
- package/lib/Utils/message-retry-manager.d.ts.map +1 -1
- package/lib/Utils/message-retry-manager.js +23 -0
- package/lib/Utils/message-retry-manager.js.map +1 -1
- package/lib/Utils/messages-media.d.ts +2 -1
- package/lib/Utils/messages-media.d.ts.map +1 -1
- package/lib/Utils/messages-media.js +19 -7
- package/lib/Utils/messages-media.js.map +1 -1
- package/lib/Utils/messages.d.ts +3 -12
- package/lib/Utils/messages.d.ts.map +1 -1
- package/lib/Utils/messages.js +210 -193
- package/lib/Utils/messages.js.map +1 -1
- package/lib/Utils/noise-handler.d.ts.map +1 -1
- package/lib/Utils/noise-handler.js.map +1 -1
- package/lib/Utils/offline-node-processor.d.ts.map +1 -1
- package/lib/Utils/offline-node-processor.js.map +1 -1
- package/lib/Utils/pre-key-manager.d.ts.map +1 -1
- package/lib/Utils/pre-key-manager.js.map +1 -1
- package/lib/Utils/process-message.d.ts.map +1 -1
- package/lib/Utils/process-message.js +18 -2
- package/lib/Utils/process-message.js.map +1 -1
- package/lib/Utils/reporting-utils.d.ts.map +1 -1
- package/lib/Utils/reporting-utils.js.map +1 -1
- package/lib/Utils/rich-message-utils.d.ts +8 -3
- package/lib/Utils/rich-message-utils.d.ts.map +1 -1
- package/lib/Utils/rich-message-utils.js +1 -1
- package/lib/Utils/rich-message-utils.js.map +1 -1
- package/lib/Utils/signal.d.ts +14 -1
- package/lib/Utils/signal.d.ts.map +1 -1
- package/lib/Utils/signal.js +42 -0
- package/lib/Utils/signal.js.map +1 -1
- package/lib/Utils/stanza-ack.d.ts.map +1 -1
- package/lib/Utils/stanza-ack.js.map +1 -1
- package/lib/Utils/sync-action-utils.d.ts.map +1 -1
- package/lib/Utils/sync-action-utils.js.map +1 -1
- package/lib/Utils/tc-token-utils.d.ts.map +1 -1
- package/lib/Utils/tc-token-utils.js +0 -1
- package/lib/Utils/tc-token-utils.js.map +1 -1
- package/lib/Utils/use-multi-file-auth-state.d.ts.map +1 -1
- package/lib/Utils/use-multi-file-auth-state.js.map +1 -1
- package/lib/Utils/use-single-file-auth-state.d.ts.map +1 -1
- package/lib/Utils/use-single-file-auth-state.js.map +1 -1
- package/lib/Utils/validate-connection.d.ts +1 -1
- package/lib/Utils/validate-connection.d.ts.map +1 -1
- package/lib/Utils/validate-connection.js +4 -8
- package/lib/Utils/validate-connection.js.map +1 -1
- package/lib/WABinary/constants.d.ts.map +1 -1
- package/lib/WABinary/constants.js.map +1 -1
- package/lib/WABinary/decode.d.ts.map +1 -1
- package/lib/WABinary/decode.js.map +1 -1
- package/lib/WABinary/encode.d.ts.map +1 -1
- package/lib/WABinary/encode.js.map +1 -1
- package/lib/WABinary/generic-utils.d.ts +1 -3
- package/lib/WABinary/generic-utils.d.ts.map +1 -1
- package/lib/WABinary/generic-utils.js +6 -7
- package/lib/WABinary/generic-utils.js.map +1 -1
- package/lib/WABinary/index.d.ts.map +1 -1
- package/lib/WABinary/index.js.map +1 -1
- package/lib/WABinary/jid-utils.d.ts.map +1 -1
- package/lib/WABinary/jid-utils.js.map +1 -1
- package/lib/WABinary/types.d.ts.map +1 -1
- package/lib/WABinary/types.js.map +1 -1
- package/lib/WAM/BinaryInfo.d.ts.map +1 -1
- package/lib/WAM/BinaryInfo.js.map +1 -1
- package/lib/WAM/constants.d.ts.map +1 -1
- package/lib/WAM/constants.js.map +1 -1
- package/lib/WAM/encode.d.ts.map +1 -1
- package/lib/WAM/encode.js.map +1 -1
- package/lib/WAM/index.d.ts.map +1 -1
- package/lib/WAM/index.js.map +1 -1
- package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/USyncContactProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/USyncStatusProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/USyncUsernameProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/USyncUsernameProtocol.js +3 -1
- package/lib/WAUSync/Protocols/USyncUsernameProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.js.map +1 -1
- package/lib/WAUSync/Protocols/index.d.ts.map +1 -1
- package/lib/WAUSync/Protocols/index.js.map +1 -1
- package/lib/WAUSync/USyncQuery.d.ts.map +1 -1
- package/lib/WAUSync/USyncQuery.js +1 -1
- package/lib/WAUSync/USyncQuery.js.map +1 -1
- package/lib/WAUSync/USyncUser.d.ts.map +1 -1
- package/lib/WAUSync/USyncUser.js.map +1 -1
- package/lib/WAUSync/index.d.ts.map +1 -1
- package/lib/WAUSync/index.js.map +1 -1
- package/lib/index.d.ts +1 -1
- package/lib/index.d.ts.map +1 -1
- package/lib/index.js +1 -1
- package/lib/index.js.map +1 -1
- package/package.json +34 -5
- package/lib/Types/Newsletter.d.ts.map +0 -1
- 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,
|
|
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,
|
|
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
|
-
|
|
98
|
-
|
|
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
|
-
|
|
101
|
-
|
|
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
|
-
|
|
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 }, '
|
|
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
|
-
|
|
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 }, '
|
|
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 }, '
|
|
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
|
|
275
|
+
const children = getAllBinaryNodeChildren(node);
|
|
152
276
|
const author = node.attrs.participant;
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
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
|
-
|
|
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
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
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
|
-
|
|
221
|
-
|
|
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
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
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
|
-
|
|
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
|
|
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
|
|
673
|
-
const
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
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
|
-
|
|
684
|
-
|
|
685
|
-
|
|
686
|
-
|
|
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
|
|
696
|
-
|
|
697
|
-
|
|
698
|
-
|
|
699
|
-
|
|
700
|
-
|
|
701
|
-
|
|
702
|
-
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
|
|
710
|
-
|
|
711
|
-
|
|
712
|
-
|
|
713
|
-
|
|
714
|
-
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
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 }, '
|
|
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
|
-
|
|
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
|
-
|
|
1256
|
-
|
|
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
|
|
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
|
-
|
|
1421
|
-
|
|
1422
|
-
//
|
|
1423
|
-
//
|
|
1424
|
-
//
|
|
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
|