@alannxd/baileys 6.0.5 → 6.0.6
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/WAProto/fix-import.js +29 -0
- package/WAProto/index.js +160 -201
- package/engine-requirements.js +1 -1
- package/lib/Defaults/baileys-version.json +3 -0
- package/lib/Defaults/index.d.ts +15 -37
- package/lib/Defaults/index.js +136 -119
- package/lib/Defaults/phonenumber-mcc.json +223 -0
- package/lib/Signal/Group/ciphertext-message.d.ts +0 -1
- package/lib/Signal/Group/ciphertext-message.js +5 -2
- package/lib/Signal/Group/group-session-builder.d.ts +3 -4
- package/lib/Signal/Group/group-session-builder.js +41 -7
- package/lib/Signal/Group/group_cipher.d.ts +4 -4
- package/lib/Signal/Group/group_cipher.js +51 -37
- package/lib/Signal/Group/index.d.ts +11 -12
- package/lib/Signal/Group/index.js +57 -12
- package/lib/Signal/Group/keyhelper.d.ts +1 -2
- package/lib/Signal/Group/keyhelper.js +44 -7
- package/lib/Signal/Group/queue-job.d.ts +1 -0
- package/lib/Signal/Group/queue-job.js +57 -0
- package/lib/Signal/Group/sender-chain-key.d.ts +2 -3
- package/lib/Signal/Group/sender-chain-key.js +15 -7
- package/lib/Signal/Group/sender-key-distribution-message.d.ts +1 -2
- package/lib/Signal/Group/sender-key-distribution-message.js +11 -8
- package/lib/Signal/Group/sender-key-message.d.ts +1 -2
- package/lib/Signal/Group/sender-key-message.js +12 -9
- package/lib/Signal/Group/sender-key-name.d.ts +0 -1
- package/lib/Signal/Group/sender-key-name.js +5 -2
- package/lib/Signal/Group/sender-key-record.d.ts +2 -3
- package/lib/Signal/Group/sender-key-record.js +21 -9
- package/lib/Signal/Group/sender-key-state.d.ts +6 -7
- package/lib/Signal/Group/sender-key-state.js +42 -27
- package/lib/Signal/Group/sender-message-key.d.ts +0 -1
- package/lib/Signal/Group/sender-message-key.js +7 -4
- package/lib/Signal/libsignal.d.ts +3 -5
- package/lib/Signal/libsignal.js +90 -347
- package/lib/Socket/Client/{types.d.ts → abstract-socket-client.d.ts} +5 -4
- package/lib/Socket/Client/abstract-socket-client.js +13 -0
- package/lib/Socket/Client/index.d.ts +3 -3
- package/lib/Socket/Client/index.js +19 -3
- package/lib/Socket/Client/mobile-socket-client.d.ts +13 -0
- package/lib/Socket/Client/mobile-socket-client.js +65 -0
- package/lib/Socket/Client/{websocket.d.ts → web-socket-client.d.ts} +2 -3
- package/lib/Socket/Client/web-socket-client.js +62 -0
- package/lib/Socket/business.d.ts +108 -154
- package/lib/Socket/business.js +43 -162
- package/lib/Socket/chats.d.ts +239 -96
- package/lib/Socket/chats.js +427 -627
- package/lib/Socket/communities.d.ts +146 -239
- package/lib/Socket/communities.js +80 -90
- package/lib/Socket/groups.d.ts +57 -104
- package/lib/Socket/groups.js +161 -154
- package/lib/Socket/index.d.ts +115 -202
- package/lib/Socket/index.js +10 -11
- package/lib/Socket/luxu.d.ts +266 -22
- package/lib/Socket/luxu.js +465 -422
- package/lib/Socket/messages-recv.d.ts +84 -136
- package/lib/Socket/messages-recv.js +615 -1421
- package/lib/Socket/messages-send.d.ts +126 -142
- package/lib/Socket/messages-send.js +671 -878
- package/lib/Socket/newsletter.d.ts +85 -121
- package/lib/Socket/newsletter.js +272 -147
- package/lib/Socket/registration.d.ts +267 -0
- package/lib/Socket/registration.js +166 -0
- package/lib/Socket/socket.d.ts +19 -34
- package/lib/Socket/socket.js +313 -544
- package/lib/Socket/usync.d.ts +36 -0
- package/lib/Socket/usync.js +70 -0
- package/lib/Store/index.d.ts +3 -10
- package/lib/Store/index.js +10 -10
- package/lib/Store/make-cache-manager-store.d.ts +11 -17
- package/lib/Store/make-cache-manager-store.js +41 -43
- package/lib/Store/make-in-memory-store.d.ts +118 -39
- package/lib/Store/make-in-memory-store.js +341 -112
- package/lib/Store/make-ordered-dictionary.d.ts +10 -11
- package/lib/Store/make-ordered-dictionary.js +20 -14
- package/lib/Store/object-repository.d.ts +9 -10
- package/lib/Store/object-repository.js +6 -11
- package/lib/Types/Auth.d.ts +12 -19
- package/lib/Types/Auth.js +2 -2
- package/lib/Types/Call.d.ts +1 -3
- package/lib/Types/Call.js +2 -2
- package/lib/Types/Chat.d.ts +13 -35
- package/lib/Types/Chat.js +4 -8
- package/lib/Types/Contact.d.ts +1 -8
- package/lib/Types/Contact.js +2 -2
- package/lib/Types/Events.d.ts +17 -116
- package/lib/Types/Events.js +2 -2
- package/lib/Types/GroupMetadata.d.ts +5 -21
- package/lib/Types/GroupMetadata.js +2 -2
- package/lib/Types/Label.d.ts +0 -12
- package/lib/Types/Label.js +5 -3
- package/lib/Types/LabelAssociation.d.ts +0 -1
- package/lib/Types/LabelAssociation.js +5 -3
- package/lib/Types/Message.d.ts +58 -105
- package/lib/Types/Message.js +9 -11
- package/lib/Types/Newsletter.d.ts +103 -0
- package/lib/Types/Newsletter.js +38 -0
- package/lib/Types/Product.d.ts +1 -2
- package/lib/Types/Product.js +2 -2
- package/lib/Types/Signal.d.ts +2 -32
- package/lib/Types/Signal.js +2 -2
- package/lib/Types/Socket.d.ts +25 -50
- package/lib/Types/Socket.js +2 -3
- package/lib/Types/State.d.ts +2 -72
- package/lib/Types/State.js +2 -56
- package/lib/Types/USync.d.ts +2 -3
- package/lib/Types/USync.js +2 -2
- package/lib/Types/index.d.ts +14 -22
- package/lib/Types/index.js +31 -15
- package/lib/Utils/auth-utils.d.ts +6 -12
- package/lib/Utils/auth-utils.js +143 -239
- package/lib/Utils/baileys-event-stream.d.ts +16 -0
- package/lib/Utils/baileys-event-stream.js +63 -0
- package/lib/Utils/business.d.ts +2 -3
- package/lib/Utils/business.js +69 -66
- package/lib/Utils/chat-utils.d.ts +23 -52
- package/lib/Utils/chat-utils.js +253 -396
- package/lib/Utils/crypto.d.ts +22 -18
- package/lib/Utils/crypto.js +90 -57
- package/lib/Utils/decode-wa-message.d.ts +8 -55
- package/lib/Utils/decode-wa-message.js +84 -203
- package/lib/Utils/event-buffer.d.ts +8 -9
- package/lib/Utils/event-buffer.js +77 -185
- package/lib/Utils/generics.d.ts +29 -28
- package/lib/Utils/generics.js +210 -180
- package/lib/Utils/history.d.ts +9 -18
- package/lib/Utils/history.js +55 -93
- package/lib/Utils/index.d.ts +17 -22
- package/lib/Utils/index.js +33 -22
- package/lib/Utils/link-preview.d.ts +5 -5
- package/lib/Utils/link-preview.js +24 -16
- package/lib/Utils/logger.d.ts +3 -11
- package/lib/Utils/logger.js +7 -3
- package/lib/Utils/lt-hash.d.ts +12 -8
- package/lib/Utils/lt-hash.js +46 -3
- package/lib/Utils/make-mutex.d.ts +2 -4
- package/lib/Utils/make-mutex.js +34 -24
- package/lib/Utils/messages-media.d.ts +44 -61
- package/lib/Utils/messages-media.js +482 -451
- package/lib/Utils/messages.d.ts +18 -32
- package/lib/Utils/messages.js +369 -458
- package/lib/Utils/noise-handler.d.ts +14 -13
- package/lib/Utils/noise-handler.js +99 -145
- package/lib/Utils/process-message.d.ts +12 -31
- package/lib/Utils/process-message.js +150 -459
- package/lib/Utils/signal.d.ts +5 -20
- package/lib/Utils/signal.js +72 -120
- package/lib/Utils/use-multi-file-auth-state.d.ts +2 -2
- package/lib/Utils/use-multi-file-auth-state.js +27 -29
- package/lib/Utils/validate-connection.d.ts +7 -7
- package/lib/Utils/validate-connection.js +99 -73
- package/lib/WABinary/constants.d.ts +27 -25
- package/lib/WABinary/constants.js +20 -1281
- package/lib/WABinary/decode.d.ts +5 -5
- package/lib/WABinary/decode.js +42 -52
- package/lib/WABinary/encode.d.ts +3 -3
- package/lib/WABinary/encode.js +155 -110
- package/lib/WABinary/generic-utils.d.ts +7 -8
- package/lib/WABinary/generic-utils.js +49 -48
- package/lib/WABinary/index.d.ts +5 -6
- package/lib/WABinary/index.js +21 -6
- package/lib/WABinary/jid-utils.d.ts +8 -25
- package/lib/WABinary/jid-utils.js +40 -74
- package/lib/WABinary/types.d.ts +1 -2
- package/lib/WABinary/types.js +2 -2
- package/lib/WAM/BinaryInfo.d.ts +11 -3
- package/lib/WAM/BinaryInfo.js +5 -2
- package/lib/WAM/constants.d.ts +3 -5
- package/lib/WAM/constants.js +11958 -19461
- package/lib/WAM/encode.d.ts +3 -3
- package/lib/WAM/encode.js +22 -17
- package/lib/WAM/index.d.ts +3 -4
- package/lib/WAM/index.js +19 -4
- package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts +3 -4
- package/lib/WAUSync/Protocols/USyncContactProtocol.js +13 -33
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts +2 -3
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +14 -11
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts +2 -3
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +12 -9
- package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts +2 -3
- package/lib/WAUSync/Protocols/USyncStatusProtocol.js +13 -9
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts +3 -4
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +22 -20
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts +3 -5
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +8 -13
- package/lib/WAUSync/Protocols/index.d.ts +4 -6
- package/lib/WAUSync/Protocols/index.js +20 -6
- package/lib/WAUSync/USyncQuery.d.ts +4 -6
- package/lib/WAUSync/USyncQuery.js +35 -44
- package/lib/WAUSync/USyncUser.d.ts +5 -10
- package/lib/WAUSync/USyncUser.js +5 -10
- package/lib/WAUSync/index.js +19 -4
- package/lib/index.d.ts +9 -10
- package/lib/index.js +34 -12
- package/package.json +50 -83
- package/WAProto/GenerateStatics.sh +0 -3
- package/WAProto/WAProto.proto +0 -5479
- package/WAProto/fix-imports.js +0 -85
- package/WAProto/index.d.ts +0 -14017
- package/lib/Signal/lid-mapping.d.ts +0 -23
- package/lib/Signal/lid-mapping.js +0 -277
- package/lib/Socket/Client/types.js +0 -11
- package/lib/Socket/Client/websocket.js +0 -54
- package/lib/Socket/mex.d.ts +0 -3
- package/lib/Socket/mex.js +0 -42
- package/lib/Store/keyed-db.d.ts +0 -22
- package/lib/Store/keyed-db.js +0 -108
- package/lib/Types/Bussines.d.ts +0 -25
- package/lib/Types/Bussines.js +0 -2
- package/lib/Types/Mex.d.ts +0 -141
- package/lib/Types/Mex.js +0 -37
- package/lib/Utils/browser-utils.d.ts +0 -4
- package/lib/Utils/browser-utils.js +0 -28
- package/lib/Utils/companion-reg-client-utils.d.ts +0 -17
- package/lib/Utils/companion-reg-client-utils.js +0 -35
- package/lib/Utils/identity-change-handler.d.ts +0 -44
- package/lib/Utils/identity-change-handler.js +0 -50
- package/lib/Utils/message-retry-manager.d.ts +0 -115
- package/lib/Utils/message-retry-manager.js +0 -265
- package/lib/Utils/offline-node-processor.d.ts +0 -17
- package/lib/Utils/offline-node-processor.js +0 -40
- package/lib/Utils/pre-key-manager.d.ts +0 -28
- package/lib/Utils/pre-key-manager.js +0 -106
- package/lib/Utils/reporting-utils.d.ts +0 -11
- package/lib/Utils/reporting-utils.js +0 -258
- package/lib/Utils/stanza-ack.d.ts +0 -11
- package/lib/Utils/stanza-ack.js +0 -38
- package/lib/Utils/sync-action-utils.d.ts +0 -19
- package/lib/Utils/sync-action-utils.js +0 -49
- package/lib/Utils/tc-token-utils.d.ts +0 -37
- package/lib/Utils/tc-token-utils.js +0 -163
- package/lib/WAUSync/Protocols/USyncUsernameProtocol.d.ts +0 -10
- package/lib/WAUSync/Protocols/USyncUsernameProtocol.js +0 -25
- package/lib/WAUSync/index.d.ts +0 -4
package/lib/Utils/chat-utils.js
CHANGED
|
@@ -1,44 +1,51 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.processSyncAction = exports.chatModificationToAppPatch = exports.decodePatches = exports.decodeSyncdSnapshot = exports.downloadExternalPatch = exports.downloadExternalBlob = exports.extractSyncdPatches = exports.decodeSyncdPatch = exports.decodeSyncdMutations = exports.encodeSyncdPatch = exports.newLTHashState = void 0;
|
|
4
|
+
const boom_1 = require("@hapi/boom");
|
|
5
|
+
const WAProto_1 = require("../../WAProto");
|
|
6
|
+
const LabelAssociation_1 = require("../Types/LabelAssociation");
|
|
7
|
+
const WABinary_1 = require("../WABinary");
|
|
8
|
+
const crypto_1 = require("./crypto");
|
|
9
|
+
const generics_1 = require("./generics");
|
|
10
|
+
const lt_hash_1 = require("./lt-hash");
|
|
11
|
+
const messages_media_1 = require("./messages-media");
|
|
11
12
|
const mutationKeys = (keydata) => {
|
|
12
|
-
const
|
|
13
|
+
const expanded = (0, crypto_1.hkdf)(keydata, 160, { info: 'WhatsApp Mutation Keys' });
|
|
13
14
|
return {
|
|
14
|
-
indexKey:
|
|
15
|
-
valueEncryptionKey:
|
|
16
|
-
valueMacKey:
|
|
17
|
-
snapshotMacKey:
|
|
18
|
-
patchMacKey:
|
|
15
|
+
indexKey: expanded.slice(0, 32),
|
|
16
|
+
valueEncryptionKey: expanded.slice(32, 64),
|
|
17
|
+
valueMacKey: expanded.slice(64, 96),
|
|
18
|
+
snapshotMacKey: expanded.slice(96, 128),
|
|
19
|
+
patchMacKey: expanded.slice(128, 160)
|
|
19
20
|
};
|
|
20
21
|
};
|
|
21
22
|
const generateMac = (operation, data, keyId, key) => {
|
|
22
|
-
const
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
23
|
+
const getKeyData = () => {
|
|
24
|
+
let r;
|
|
25
|
+
switch (operation) {
|
|
26
|
+
case WAProto_1.proto.SyncdMutation.SyncdOperation.SET:
|
|
27
|
+
r = 0x01;
|
|
28
|
+
break;
|
|
29
|
+
case WAProto_1.proto.SyncdMutation.SyncdOperation.REMOVE:
|
|
30
|
+
r = 0x02;
|
|
31
|
+
break;
|
|
32
|
+
}
|
|
33
|
+
const buff = Buffer.from([r]);
|
|
34
|
+
return Buffer.concat([buff, Buffer.from(keyId, 'base64')]);
|
|
35
|
+
};
|
|
36
|
+
const keyData = getKeyData();
|
|
37
|
+
const last = Buffer.alloc(8); // 8 bytes
|
|
38
|
+
last.set([keyData.length], last.length - 1);
|
|
39
|
+
const total = Buffer.concat([keyData, data, last]);
|
|
40
|
+
const hmac = (0, crypto_1.hmacSign)(total, key, 'sha512');
|
|
41
|
+
return hmac.slice(0, 32);
|
|
35
42
|
};
|
|
36
43
|
const to64BitNetworkOrder = (e) => {
|
|
37
44
|
const buff = Buffer.alloc(8);
|
|
38
45
|
buff.writeUint32BE(e, 4);
|
|
39
46
|
return buff;
|
|
40
47
|
};
|
|
41
|
-
|
|
48
|
+
const makeLtHashGenerator = ({ indexValueMap, hash }) => {
|
|
42
49
|
indexValueMap = { ...indexValueMap };
|
|
43
50
|
const addBuffs = [];
|
|
44
51
|
const subBuffs = [];
|
|
@@ -46,85 +53,71 @@ export const makeLtHashGenerator = ({ indexValueMap, hash }) => {
|
|
|
46
53
|
mix: ({ indexMac, valueMac, operation }) => {
|
|
47
54
|
const indexMacBase64 = Buffer.from(indexMac).toString('base64');
|
|
48
55
|
const prevOp = indexValueMap[indexMacBase64];
|
|
49
|
-
if (operation === proto.SyncdMutation.SyncdOperation.REMOVE) {
|
|
56
|
+
if (operation === WAProto_1.proto.SyncdMutation.SyncdOperation.REMOVE) {
|
|
50
57
|
if (!prevOp) {
|
|
51
|
-
|
|
52
|
-
// The missing REMOVE will cause an LTHash mismatch, which is handled
|
|
53
|
-
// by the MAC validation layer (snapshot recovery or retry).
|
|
54
|
-
return;
|
|
58
|
+
throw new boom_1.Boom('tried remove, but no previous op', { data: { indexMac, valueMac } });
|
|
55
59
|
}
|
|
56
60
|
// remove from index value mac, since this mutation is erased
|
|
57
61
|
delete indexValueMap[indexMacBase64];
|
|
58
62
|
}
|
|
59
63
|
else {
|
|
60
|
-
addBuffs.push(valueMac);
|
|
64
|
+
addBuffs.push(new Uint8Array(valueMac).buffer);
|
|
61
65
|
// add this index into the history map
|
|
62
66
|
indexValueMap[indexMacBase64] = { valueMac };
|
|
63
67
|
}
|
|
64
68
|
if (prevOp) {
|
|
65
|
-
subBuffs.push(prevOp.valueMac);
|
|
69
|
+
subBuffs.push(new Uint8Array(prevOp.valueMac).buffer);
|
|
66
70
|
}
|
|
67
71
|
},
|
|
68
72
|
finish: () => {
|
|
69
|
-
const
|
|
73
|
+
const hashArrayBuffer = new Uint8Array(hash).buffer;
|
|
74
|
+
const result = lt_hash_1.LT_HASH_ANTI_TAMPERING.subtractThenAdd(hashArrayBuffer, addBuffs, subBuffs);
|
|
75
|
+
const buffer = Buffer.from(result);
|
|
70
76
|
return {
|
|
71
|
-
hash:
|
|
77
|
+
hash: buffer,
|
|
72
78
|
indexValueMap
|
|
73
79
|
};
|
|
74
80
|
}
|
|
75
81
|
};
|
|
76
82
|
};
|
|
77
83
|
const generateSnapshotMac = (lthash, version, name, key) => {
|
|
78
|
-
const total = Buffer.concat([
|
|
79
|
-
|
|
84
|
+
const total = Buffer.concat([
|
|
85
|
+
lthash,
|
|
86
|
+
to64BitNetworkOrder(version),
|
|
87
|
+
Buffer.from(name, 'utf-8')
|
|
88
|
+
]);
|
|
89
|
+
return (0, crypto_1.hmacSign)(total, key, 'sha256');
|
|
80
90
|
};
|
|
81
91
|
const generatePatchMac = (snapshotMac, valueMacs, version, type, key) => {
|
|
82
|
-
const total = Buffer.concat([
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
}
|
|
90
|
-
return state;
|
|
91
|
-
};
|
|
92
|
-
export const MAX_SYNC_ATTEMPTS = 2;
|
|
93
|
-
/**
|
|
94
|
-
* Check if an error is a missing app state sync key.
|
|
95
|
-
* WA Web treats these as "Blocked" (waits for key arrival), not fatal.
|
|
96
|
-
* In Baileys we retry with a snapshot which may use a different key.
|
|
97
|
-
*/
|
|
98
|
-
export const isMissingKeyError = (error) => {
|
|
99
|
-
return error?.data?.isMissingKey === true;
|
|
100
|
-
};
|
|
101
|
-
/**
|
|
102
|
-
* Determines if an app state sync error is unrecoverable.
|
|
103
|
-
* TypeError indicates a WASM crash; otherwise we give up after MAX_SYNC_ATTEMPTS.
|
|
104
|
-
* Missing keys are NOT checked here — they are handled separately as "Blocked".
|
|
105
|
-
*/
|
|
106
|
-
export const isAppStateSyncIrrecoverable = (error, attempts) => {
|
|
107
|
-
return attempts >= MAX_SYNC_ATTEMPTS || error?.name === 'TypeError';
|
|
92
|
+
const total = Buffer.concat([
|
|
93
|
+
snapshotMac,
|
|
94
|
+
...valueMacs,
|
|
95
|
+
to64BitNetworkOrder(version),
|
|
96
|
+
Buffer.from(type, 'utf-8')
|
|
97
|
+
]);
|
|
98
|
+
return (0, crypto_1.hmacSign)(total, key);
|
|
108
99
|
};
|
|
109
|
-
|
|
100
|
+
const newLTHashState = () => ({ version: 0, hash: Buffer.alloc(128), indexValueMap: {} });
|
|
101
|
+
exports.newLTHashState = newLTHashState;
|
|
102
|
+
const encodeSyncdPatch = async ({ type, index, syncAction, apiVersion, operation }, myAppStateKeyId, state, getAppStateSyncKey) => {
|
|
110
103
|
const key = !!myAppStateKeyId ? await getAppStateSyncKey(myAppStateKeyId) : undefined;
|
|
111
104
|
if (!key) {
|
|
112
|
-
throw new Boom(`myAppStateKey ("${myAppStateKeyId}") not present`, {
|
|
105
|
+
throw new boom_1.Boom(`myAppStateKey ("${myAppStateKeyId}") not present`, { statusCode: 404 });
|
|
113
106
|
}
|
|
114
107
|
const encKeyId = Buffer.from(myAppStateKeyId, 'base64');
|
|
115
108
|
state = { ...state, indexValueMap: { ...state.indexValueMap } };
|
|
116
109
|
const indexBuffer = Buffer.from(JSON.stringify(index));
|
|
117
|
-
const dataProto = proto.SyncActionData.fromObject({
|
|
110
|
+
const dataProto = WAProto_1.proto.SyncActionData.fromObject({
|
|
118
111
|
index: indexBuffer,
|
|
119
112
|
value: syncAction,
|
|
120
113
|
padding: new Uint8Array(0),
|
|
121
114
|
version: apiVersion
|
|
122
115
|
});
|
|
123
|
-
const encoded = proto.SyncActionData.encode(dataProto).finish();
|
|
116
|
+
const encoded = WAProto_1.proto.SyncActionData.encode(dataProto).finish();
|
|
124
117
|
const keyValue = mutationKeys(key.keyData);
|
|
125
|
-
const encValue = aesEncrypt(encoded, keyValue.valueEncryptionKey);
|
|
118
|
+
const encValue = (0, crypto_1.aesEncrypt)(encoded, keyValue.valueEncryptionKey);
|
|
126
119
|
const valueMac = generateMac(operation, encValue, encKeyId, keyValue.valueMacKey);
|
|
127
|
-
const indexMac = hmacSign(indexBuffer, keyValue.indexKey);
|
|
120
|
+
const indexMac = (0, crypto_1.hmacSign)(indexBuffer, keyValue.indexKey);
|
|
128
121
|
// update LT hash
|
|
129
122
|
const generator = makeLtHashGenerator(state);
|
|
130
123
|
generator.mix({ indexMac, valueMac, operation });
|
|
@@ -154,52 +147,33 @@ export const encodeSyncdPatch = async ({ type, index, syncAction, apiVersion, op
|
|
|
154
147
|
state.indexValueMap[base64Index] = { valueMac };
|
|
155
148
|
return { patch, state };
|
|
156
149
|
};
|
|
157
|
-
|
|
150
|
+
exports.encodeSyncdPatch = encodeSyncdPatch;
|
|
151
|
+
const decodeSyncdMutations = async (msgMutations, initialState, getAppStateSyncKey, onMutation, validateMacs) => {
|
|
158
152
|
const ltGenerator = makeLtHashGenerator(initialState);
|
|
159
|
-
const derivedKeyCache = new Map();
|
|
160
153
|
// indexKey used to HMAC sign record.index.blob
|
|
161
154
|
// valueEncryptionKey used to AES-256-CBC encrypt record.value.blob[0:-32]
|
|
162
155
|
// the remaining record.value.blob[0:-32] is the mac, it the HMAC sign of key.keyId + decoded proto data + length of bytes in keyId
|
|
163
156
|
for (const msgMutation of msgMutations) {
|
|
164
157
|
// if it's a syncdmutation, get the operation property
|
|
165
158
|
// otherwise, if it's only a record -- it'll be a SET mutation
|
|
166
|
-
const operation = 'operation' in msgMutation ? msgMutation.operation : proto.SyncdMutation.SyncdOperation.SET;
|
|
167
|
-
const record = 'record' in msgMutation && !!msgMutation.record ? msgMutation.record : msgMutation;
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
catch (err) {
|
|
173
|
-
// Missing-key errors must propagate so the orchestrator can park the
|
|
174
|
-
// collection (Blocked) and retry when APP_STATE_SYNC_KEY_SHARE arrives.
|
|
175
|
-
// Other errors → individual record corruption, skip and keep going.
|
|
176
|
-
if (isMissingKeyError(err))
|
|
177
|
-
throw err;
|
|
178
|
-
continue;
|
|
179
|
-
}
|
|
180
|
-
const content = record.value.blob;
|
|
181
|
-
const encContent = content.subarray(0, -32);
|
|
182
|
-
const ogValueMac = content.subarray(-32);
|
|
159
|
+
const operation = 'operation' in msgMutation ? msgMutation.operation : WAProto_1.proto.SyncdMutation.SyncdOperation.SET;
|
|
160
|
+
const record = ('record' in msgMutation && !!msgMutation.record) ? msgMutation.record : msgMutation;
|
|
161
|
+
const key = await getKey(record.keyId.id);
|
|
162
|
+
const content = Buffer.from(record.value.blob);
|
|
163
|
+
const encContent = content.slice(0, -32);
|
|
164
|
+
const ogValueMac = content.slice(-32);
|
|
183
165
|
if (validateMacs) {
|
|
184
166
|
const contentHmac = generateMac(operation, encContent, record.keyId.id, key.valueMacKey);
|
|
185
167
|
if (Buffer.compare(contentHmac, ogValueMac) !== 0) {
|
|
186
|
-
|
|
187
|
-
continue;
|
|
168
|
+
throw new boom_1.Boom('HMAC content verification failed');
|
|
188
169
|
}
|
|
189
170
|
}
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
result = aesDecrypt(encContent, key.valueEncryptionKey);
|
|
193
|
-
}
|
|
194
|
-
catch {
|
|
195
|
-
// decrypt failed — skip this record instead of aborting
|
|
196
|
-
continue;
|
|
197
|
-
}
|
|
198
|
-
const syncAction = proto.SyncActionData.decode(result);
|
|
171
|
+
const result = (0, crypto_1.aesDecrypt)(encContent, key.valueEncryptionKey);
|
|
172
|
+
const syncAction = WAProto_1.proto.SyncActionData.decode(result);
|
|
199
173
|
if (validateMacs) {
|
|
200
|
-
const hmac = hmacSign(syncAction.index, key.indexKey);
|
|
174
|
+
const hmac = (0, crypto_1.hmacSign)(syncAction.index, key.indexKey);
|
|
201
175
|
if (Buffer.compare(hmac, record.index.blob) !== 0) {
|
|
202
|
-
throw new Boom('HMAC index verification failed');
|
|
176
|
+
throw new boom_1.Boom('HMAC index verification failed');
|
|
203
177
|
}
|
|
204
178
|
}
|
|
205
179
|
const indexStr = Buffer.from(syncAction.index).toString();
|
|
@@ -213,46 +187,40 @@ export const decodeSyncdMutations = async (msgMutations, initialState, getAppSta
|
|
|
213
187
|
return ltGenerator.finish();
|
|
214
188
|
async function getKey(keyId) {
|
|
215
189
|
const base64Key = Buffer.from(keyId).toString('base64');
|
|
216
|
-
const cached = derivedKeyCache.get(base64Key);
|
|
217
|
-
if (cached) {
|
|
218
|
-
return cached;
|
|
219
|
-
}
|
|
220
190
|
const keyEnc = await getAppStateSyncKey(base64Key);
|
|
221
191
|
if (!keyEnc) {
|
|
222
|
-
throw new Boom(`failed to find key "${base64Key}" to decode mutation`, {
|
|
223
|
-
data: { isMissingKey: true, msgMutations }
|
|
224
|
-
});
|
|
192
|
+
throw new boom_1.Boom(`failed to find key "${base64Key}" to decode mutation`, { statusCode: 404, data: { msgMutations } });
|
|
225
193
|
}
|
|
226
|
-
|
|
227
|
-
derivedKeyCache.set(base64Key, keys);
|
|
228
|
-
return keys;
|
|
194
|
+
return mutationKeys(keyEnc.keyData);
|
|
229
195
|
}
|
|
230
196
|
};
|
|
231
|
-
|
|
197
|
+
exports.decodeSyncdMutations = decodeSyncdMutations;
|
|
198
|
+
const decodeSyncdPatch = async (msg, name, initialState, getAppStateSyncKey, onMutation, validateMacs) => {
|
|
232
199
|
if (validateMacs) {
|
|
233
200
|
const base64Key = Buffer.from(msg.keyId.id).toString('base64');
|
|
234
201
|
const mainKeyObj = await getAppStateSyncKey(base64Key);
|
|
235
202
|
if (!mainKeyObj) {
|
|
236
|
-
throw new Boom(`failed to find key "${base64Key}" to decode patch`, {
|
|
203
|
+
throw new boom_1.Boom(`failed to find key "${base64Key}" to decode patch`, { statusCode: 404, data: { msg } });
|
|
237
204
|
}
|
|
238
205
|
const mainKey = mutationKeys(mainKeyObj.keyData);
|
|
239
206
|
const mutationmacs = msg.mutations.map(mutation => mutation.record.value.blob.slice(-32));
|
|
240
|
-
const patchMac = generatePatchMac(msg.snapshotMac, mutationmacs, toNumber(msg.version.version), name, mainKey.patchMacKey);
|
|
207
|
+
const patchMac = generatePatchMac(msg.snapshotMac, mutationmacs, (0, generics_1.toNumber)(msg.version.version), name, mainKey.patchMacKey);
|
|
241
208
|
if (Buffer.compare(patchMac, msg.patchMac) !== 0) {
|
|
242
|
-
throw new Boom('Invalid patch mac');
|
|
209
|
+
throw new boom_1.Boom('Invalid patch mac');
|
|
243
210
|
}
|
|
244
211
|
}
|
|
245
|
-
const result = await decodeSyncdMutations(msg.mutations, initialState, getAppStateSyncKey, onMutation, validateMacs);
|
|
212
|
+
const result = await (0, exports.decodeSyncdMutations)(msg.mutations, initialState, getAppStateSyncKey, onMutation, validateMacs);
|
|
246
213
|
return result;
|
|
247
214
|
};
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
const
|
|
215
|
+
exports.decodeSyncdPatch = decodeSyncdPatch;
|
|
216
|
+
const extractSyncdPatches = async (result, options) => {
|
|
217
|
+
const syncNode = (0, WABinary_1.getBinaryNodeChild)(result, 'sync');
|
|
218
|
+
const collectionNodes = (0, WABinary_1.getBinaryNodeChildren)(syncNode, 'collection');
|
|
251
219
|
const final = {};
|
|
252
220
|
await Promise.all(collectionNodes.map(async (collectionNode) => {
|
|
253
|
-
const patchesNode = getBinaryNodeChild(collectionNode, 'patches');
|
|
254
|
-
const patches = getBinaryNodeChildren(patchesNode || collectionNode, 'patch');
|
|
255
|
-
const snapshotNode = getBinaryNodeChild(collectionNode, 'snapshot');
|
|
221
|
+
const patchesNode = (0, WABinary_1.getBinaryNodeChild)(collectionNode, 'patches');
|
|
222
|
+
const patches = (0, WABinary_1.getBinaryNodeChildren)(patchesNode || collectionNode, 'patch');
|
|
223
|
+
const snapshotNode = (0, WABinary_1.getBinaryNodeChild)(collectionNode, 'snapshot');
|
|
256
224
|
const syncds = [];
|
|
257
225
|
const name = collectionNode.attrs.name;
|
|
258
226
|
const hasMorePatches = collectionNode.attrs.has_more_patches === 'true';
|
|
@@ -261,16 +229,16 @@ export const extractSyncdPatches = async (result, options) => {
|
|
|
261
229
|
if (!Buffer.isBuffer(snapshotNode)) {
|
|
262
230
|
snapshotNode.content = Buffer.from(Object.values(snapshotNode.content));
|
|
263
231
|
}
|
|
264
|
-
const blobRef = proto.ExternalBlobReference.decode(snapshotNode.content);
|
|
265
|
-
const data = await downloadExternalBlob(blobRef, options);
|
|
266
|
-
snapshot = proto.SyncdSnapshot.decode(data);
|
|
232
|
+
const blobRef = WAProto_1.proto.ExternalBlobReference.decode(snapshotNode.content);
|
|
233
|
+
const data = await (0, exports.downloadExternalBlob)(blobRef, options);
|
|
234
|
+
snapshot = WAProto_1.proto.SyncdSnapshot.decode(data);
|
|
267
235
|
}
|
|
268
236
|
for (let { content } of patches) {
|
|
269
237
|
if (content) {
|
|
270
238
|
if (!Buffer.isBuffer(content)) {
|
|
271
239
|
content = Buffer.from(Object.values(content));
|
|
272
240
|
}
|
|
273
|
-
const syncd = proto.SyncdPatch.decode(content);
|
|
241
|
+
const syncd = WAProto_1.proto.SyncdPatch.decode(content);
|
|
274
242
|
if (!syncd.version) {
|
|
275
243
|
syncd.version = { version: +collectionNode.attrs.version + 1 };
|
|
276
244
|
}
|
|
@@ -281,27 +249,32 @@ export const extractSyncdPatches = async (result, options) => {
|
|
|
281
249
|
}));
|
|
282
250
|
return final;
|
|
283
251
|
};
|
|
284
|
-
|
|
285
|
-
|
|
252
|
+
exports.extractSyncdPatches = extractSyncdPatches;
|
|
253
|
+
const downloadExternalBlob = async (blob, options) => {
|
|
254
|
+
const stream = await (0, messages_media_1.downloadContentFromMessage)(blob, 'md-app-state', { options });
|
|
286
255
|
const bufferArray = [];
|
|
287
256
|
for await (const chunk of stream) {
|
|
288
257
|
bufferArray.push(chunk);
|
|
289
258
|
}
|
|
290
259
|
return Buffer.concat(bufferArray);
|
|
291
260
|
};
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
const
|
|
261
|
+
exports.downloadExternalBlob = downloadExternalBlob;
|
|
262
|
+
const downloadExternalPatch = async (blob, options) => {
|
|
263
|
+
const buffer = await (0, exports.downloadExternalBlob)(blob, options);
|
|
264
|
+
const syncData = WAProto_1.proto.SyncdMutations.decode(buffer);
|
|
295
265
|
return syncData;
|
|
296
266
|
};
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
newState
|
|
267
|
+
exports.downloadExternalPatch = downloadExternalPatch;
|
|
268
|
+
const decodeSyncdSnapshot = async (name, snapshot, getAppStateSyncKey, minimumVersionNumber, validateMacs = true) => {
|
|
269
|
+
const newState = (0, exports.newLTHashState)();
|
|
270
|
+
newState.version = (0, generics_1.toNumber)(snapshot.version.version);
|
|
300
271
|
const mutationMap = {};
|
|
301
|
-
const areMutationsRequired = typeof minimumVersionNumber === 'undefined'
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
272
|
+
const areMutationsRequired = typeof minimumVersionNumber === 'undefined'
|
|
273
|
+
|| newState.version > minimumVersionNumber;
|
|
274
|
+
const { hash, indexValueMap } = await (0, exports.decodeSyncdMutations)(snapshot.records, newState, getAppStateSyncKey, areMutationsRequired
|
|
275
|
+
? (mutation) => {
|
|
276
|
+
var _a;
|
|
277
|
+
const index = (_a = mutation.syncAction.index) === null || _a === void 0 ? void 0 : _a.toString();
|
|
305
278
|
mutationMap[index] = mutation;
|
|
306
279
|
}
|
|
307
280
|
: () => { }, validateMacs);
|
|
@@ -311,17 +284,12 @@ export const decodeSyncdSnapshot = async (name, snapshot, getAppStateSyncKey, mi
|
|
|
311
284
|
const base64Key = Buffer.from(snapshot.keyId.id).toString('base64');
|
|
312
285
|
const keyEnc = await getAppStateSyncKey(base64Key);
|
|
313
286
|
if (!keyEnc) {
|
|
314
|
-
throw new Boom(`failed to find key "${base64Key}" to decode mutation
|
|
287
|
+
throw new boom_1.Boom(`failed to find key "${base64Key}" to decode mutation`);
|
|
315
288
|
}
|
|
316
289
|
const result = mutationKeys(keyEnc.keyData);
|
|
317
290
|
const computedSnapshotMac = generateSnapshotMac(newState.hash, newState.version, name, result.snapshotMacKey);
|
|
318
291
|
if (Buffer.compare(snapshot.mac, computedSnapshotMac) !== 0) {
|
|
319
|
-
|
|
320
|
-
// records (poisoned server-side snapshot); the aggregate client hash diverges
|
|
321
|
-
// from the server-computed mac. Fall through with a warning so the session stays
|
|
322
|
-
// alive with partial state, symmetric to how decodePatches handles its own
|
|
323
|
-
// LTHash mismatch a few lines below.
|
|
324
|
-
logger?.warn({ name, version: newState.version }, 'LTHash verification failed on snapshot, continuing with partial state');
|
|
292
|
+
throw new boom_1.Boom(`failed to verify LTHash at ${newState.version} of ${name} from snapshot`);
|
|
325
293
|
}
|
|
326
294
|
}
|
|
327
295
|
return {
|
|
@@ -329,51 +297,45 @@ export const decodeSyncdSnapshot = async (name, snapshot, getAppStateSyncKey, mi
|
|
|
329
297
|
mutationMap
|
|
330
298
|
};
|
|
331
299
|
};
|
|
332
|
-
|
|
300
|
+
exports.decodeSyncdSnapshot = decodeSyncdSnapshot;
|
|
301
|
+
const decodePatches = async (name, syncds, initial, getAppStateSyncKey, options, minimumVersionNumber, logger, validateMacs = true) => {
|
|
302
|
+
var _a;
|
|
333
303
|
const newState = {
|
|
334
304
|
...initial,
|
|
335
305
|
indexValueMap: { ...initial.indexValueMap }
|
|
336
306
|
};
|
|
337
307
|
const mutationMap = {};
|
|
338
|
-
for (
|
|
308
|
+
for (let i = 0; i < syncds.length; i++) {
|
|
309
|
+
const syncd = syncds[i];
|
|
339
310
|
const { version, keyId, snapshotMac } = syncd;
|
|
340
311
|
if (syncd.externalMutations) {
|
|
341
|
-
logger
|
|
342
|
-
const ref = await downloadExternalPatch(syncd.externalMutations, options);
|
|
343
|
-
logger
|
|
344
|
-
syncd.mutations
|
|
312
|
+
logger === null || logger === void 0 ? void 0 : logger.trace({ name, version }, 'downloading external patch');
|
|
313
|
+
const ref = await (0, exports.downloadExternalPatch)(syncd.externalMutations, options);
|
|
314
|
+
logger === null || logger === void 0 ? void 0 : logger.debug({ name, version, mutations: ref.mutations.length }, 'downloaded external patch');
|
|
315
|
+
(_a = syncd.mutations) === null || _a === void 0 ? void 0 : _a.push(...ref.mutations);
|
|
345
316
|
}
|
|
346
|
-
const patchVersion = toNumber(version.version);
|
|
317
|
+
const patchVersion = (0, generics_1.toNumber)(version.version);
|
|
347
318
|
newState.version = patchVersion;
|
|
348
319
|
const shouldMutate = typeof minimumVersionNumber === 'undefined' || patchVersion > minimumVersionNumber;
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
: () => { }, validateMacs);
|
|
357
|
-
}
|
|
358
|
-
catch (err) {
|
|
359
|
-
if (isMissingKeyError(err))
|
|
360
|
-
throw err;
|
|
361
|
-
logger?.warn({ name, version: patchVersion, error: err.message }, 'failed to decode patch, skipping');
|
|
362
|
-
continue;
|
|
363
|
-
}
|
|
320
|
+
const decodeResult = await (0, exports.decodeSyncdPatch)(syncd, name, newState, getAppStateSyncKey, shouldMutate
|
|
321
|
+
? mutation => {
|
|
322
|
+
var _a;
|
|
323
|
+
const index = (_a = mutation.syncAction.index) === null || _a === void 0 ? void 0 : _a.toString();
|
|
324
|
+
mutationMap[index] = mutation;
|
|
325
|
+
}
|
|
326
|
+
: (() => { }), true);
|
|
364
327
|
newState.hash = decodeResult.hash;
|
|
365
328
|
newState.indexValueMap = decodeResult.indexValueMap;
|
|
366
329
|
if (validateMacs) {
|
|
367
330
|
const base64Key = Buffer.from(keyId.id).toString('base64');
|
|
368
331
|
const keyEnc = await getAppStateSyncKey(base64Key);
|
|
369
332
|
if (!keyEnc) {
|
|
370
|
-
throw new Boom(`failed to find key "${base64Key}" to decode mutation
|
|
333
|
+
throw new boom_1.Boom(`failed to find key "${base64Key}" to decode mutation`);
|
|
371
334
|
}
|
|
372
335
|
const result = mutationKeys(keyEnc.keyData);
|
|
373
336
|
const computedSnapshotMac = generateSnapshotMac(newState.hash, newState.version, name, result.snapshotMacKey);
|
|
374
337
|
if (Buffer.compare(snapshotMac, computedSnapshotMac) !== 0) {
|
|
375
|
-
|
|
376
|
-
break;
|
|
338
|
+
throw new boom_1.Boom(`failed to verify LTHash at ${newState.version} of ${name}`);
|
|
377
339
|
}
|
|
378
340
|
}
|
|
379
341
|
// clear memory used up by the mutations
|
|
@@ -381,31 +343,31 @@ export const decodePatches = async (name, syncds, initial, getAppStateSyncKey, o
|
|
|
381
343
|
}
|
|
382
344
|
return { state: newState, mutationMap };
|
|
383
345
|
};
|
|
384
|
-
|
|
385
|
-
|
|
346
|
+
exports.decodePatches = decodePatches;
|
|
347
|
+
const chatModificationToAppPatch = (mod, jid) => {
|
|
348
|
+
const OP = WAProto_1.proto.SyncdMutation.SyncdOperation;
|
|
386
349
|
const getMessageRange = (lastMessages) => {
|
|
387
350
|
let messageRange;
|
|
388
351
|
if (Array.isArray(lastMessages)) {
|
|
389
352
|
const lastMsg = lastMessages[lastMessages.length - 1];
|
|
390
353
|
messageRange = {
|
|
391
|
-
lastMessageTimestamp: lastMsg
|
|
392
|
-
messages: lastMessages
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
: undefined
|
|
354
|
+
lastMessageTimestamp: lastMsg === null || lastMsg === void 0 ? void 0 : lastMsg.messageTimestamp,
|
|
355
|
+
messages: (lastMessages === null || lastMessages === void 0 ? void 0 : lastMessages.length) ? lastMessages.map(m => {
|
|
356
|
+
var _a, _b;
|
|
357
|
+
if (!((_a = m.key) === null || _a === void 0 ? void 0 : _a.id) || !((_b = m.key) === null || _b === void 0 ? void 0 : _b.remoteJid)) {
|
|
358
|
+
throw new boom_1.Boom('Incomplete key', { statusCode: 400, data: m });
|
|
359
|
+
}
|
|
360
|
+
if ((0, WABinary_1.isJidGroup)(m.key.remoteJid) && !m.key.fromMe && !m.key.participant) {
|
|
361
|
+
throw new boom_1.Boom('Expected not from me message to have participant', { statusCode: 400, data: m });
|
|
362
|
+
}
|
|
363
|
+
if (!m.messageTimestamp || !(0, generics_1.toNumber)(m.messageTimestamp)) {
|
|
364
|
+
throw new boom_1.Boom('Missing timestamp in last message list', { statusCode: 400, data: m });
|
|
365
|
+
}
|
|
366
|
+
if (m.key.participant) {
|
|
367
|
+
m.key.participant = (0, WABinary_1.jidNormalizedUser)(m.key.participant);
|
|
368
|
+
}
|
|
369
|
+
return m;
|
|
370
|
+
}) : undefined
|
|
409
371
|
};
|
|
410
372
|
}
|
|
411
373
|
else {
|
|
@@ -456,33 +418,25 @@ export const chatModificationToAppPatch = (mod, jid) => {
|
|
|
456
418
|
operation: OP.SET
|
|
457
419
|
};
|
|
458
420
|
}
|
|
459
|
-
else if ('deleteForMe' in mod) {
|
|
460
|
-
const { timestamp, key, deleteMedia } = mod.deleteForMe;
|
|
461
|
-
patch = {
|
|
462
|
-
syncAction: {
|
|
463
|
-
deleteMessageForMeAction: {
|
|
464
|
-
deleteMedia,
|
|
465
|
-
messageTimestamp: timestamp
|
|
466
|
-
}
|
|
467
|
-
},
|
|
468
|
-
index: ['deleteMessageForMe', jid, key.id, key.fromMe ? '1' : '0', '0'],
|
|
469
|
-
type: 'regular_high',
|
|
470
|
-
apiVersion: 3,
|
|
471
|
-
operation: OP.SET
|
|
472
|
-
};
|
|
473
|
-
}
|
|
474
421
|
else if ('clear' in mod) {
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
422
|
+
if (mod.clear === 'all') {
|
|
423
|
+
throw new boom_1.Boom('not supported');
|
|
424
|
+
}
|
|
425
|
+
else {
|
|
426
|
+
const key = mod.clear.messages[0];
|
|
427
|
+
patch = {
|
|
428
|
+
syncAction: {
|
|
429
|
+
deleteMessageForMeAction: {
|
|
430
|
+
deleteMedia: false,
|
|
431
|
+
messageTimestamp: key.timestamp
|
|
432
|
+
}
|
|
433
|
+
},
|
|
434
|
+
index: ['deleteMessageForMe', jid, key.id, key.fromMe ? '1' : '0', '0'],
|
|
435
|
+
type: 'regular_high',
|
|
436
|
+
apiVersion: 3,
|
|
437
|
+
operation: OP.SET
|
|
438
|
+
};
|
|
439
|
+
}
|
|
486
440
|
}
|
|
487
441
|
else if ('pin' in mod) {
|
|
488
442
|
patch = {
|
|
@@ -497,28 +451,6 @@ export const chatModificationToAppPatch = (mod, jid) => {
|
|
|
497
451
|
operation: OP.SET
|
|
498
452
|
};
|
|
499
453
|
}
|
|
500
|
-
else if ('contact' in mod) {
|
|
501
|
-
patch = {
|
|
502
|
-
syncAction: {
|
|
503
|
-
contactAction: mod.contact || {}
|
|
504
|
-
},
|
|
505
|
-
index: ['contact', jid],
|
|
506
|
-
type: 'critical_unblock_low',
|
|
507
|
-
apiVersion: 2,
|
|
508
|
-
operation: mod.contact ? OP.SET : OP.REMOVE
|
|
509
|
-
};
|
|
510
|
-
}
|
|
511
|
-
else if ('disableLinkPreviews' in mod) {
|
|
512
|
-
patch = {
|
|
513
|
-
syncAction: {
|
|
514
|
-
privacySettingDisableLinkPreviewsAction: mod.disableLinkPreviews || {}
|
|
515
|
-
},
|
|
516
|
-
index: ['setting_disableLinkPreviews'],
|
|
517
|
-
type: 'regular',
|
|
518
|
-
apiVersion: 8,
|
|
519
|
-
operation: OP.SET
|
|
520
|
-
};
|
|
521
|
-
}
|
|
522
454
|
else if ('star' in mod) {
|
|
523
455
|
const key = mod.star.messages[0];
|
|
524
456
|
patch = {
|
|
@@ -537,7 +469,7 @@ export const chatModificationToAppPatch = (mod, jid) => {
|
|
|
537
469
|
patch = {
|
|
538
470
|
syncAction: {
|
|
539
471
|
deleteChatAction: {
|
|
540
|
-
messageRange: getMessageRange(mod.lastMessages)
|
|
472
|
+
messageRange: getMessageRange(mod.lastMessages),
|
|
541
473
|
}
|
|
542
474
|
},
|
|
543
475
|
index: ['deleteChat', jid, '1'],
|
|
@@ -556,90 +488,64 @@ export const chatModificationToAppPatch = (mod, jid) => {
|
|
|
556
488
|
index: ['setting_pushName'],
|
|
557
489
|
type: 'critical_block',
|
|
558
490
|
apiVersion: 1,
|
|
559
|
-
operation: OP.SET
|
|
560
|
-
};
|
|
561
|
-
}
|
|
562
|
-
else if ('quickReply' in mod) {
|
|
563
|
-
patch = {
|
|
564
|
-
syncAction: {
|
|
565
|
-
quickReplyAction: {
|
|
566
|
-
count: 0,
|
|
567
|
-
deleted: mod.quickReply.deleted || false,
|
|
568
|
-
keywords: [],
|
|
569
|
-
message: mod.quickReply.message || '',
|
|
570
|
-
shortcut: mod.quickReply.shortcut || ''
|
|
571
|
-
}
|
|
572
|
-
},
|
|
573
|
-
index: ['quick_reply', mod.quickReply.timestamp || String(Math.floor(Date.now() / 1000))],
|
|
574
|
-
type: 'regular',
|
|
575
|
-
apiVersion: 2,
|
|
576
|
-
operation: OP.SET
|
|
577
|
-
};
|
|
578
|
-
}
|
|
579
|
-
else if ('addLabel' in mod) {
|
|
580
|
-
patch = {
|
|
581
|
-
syncAction: {
|
|
582
|
-
labelEditAction: {
|
|
583
|
-
name: mod.addLabel.name,
|
|
584
|
-
color: mod.addLabel.color,
|
|
585
|
-
predefinedId: mod.addLabel.predefinedId,
|
|
586
|
-
deleted: mod.addLabel.deleted
|
|
587
|
-
}
|
|
588
|
-
},
|
|
589
|
-
index: ['label_edit', mod.addLabel.id],
|
|
590
|
-
type: 'regular',
|
|
591
|
-
apiVersion: 3,
|
|
592
|
-
operation: OP.SET
|
|
491
|
+
operation: OP.SET,
|
|
593
492
|
};
|
|
594
493
|
}
|
|
595
494
|
else if ('addChatLabel' in mod) {
|
|
596
495
|
patch = {
|
|
597
496
|
syncAction: {
|
|
598
497
|
labelAssociationAction: {
|
|
599
|
-
labeled: true
|
|
498
|
+
labeled: true,
|
|
600
499
|
}
|
|
601
500
|
},
|
|
602
|
-
index: [LabelAssociationType.Chat, mod.addChatLabel.labelId, jid],
|
|
501
|
+
index: [LabelAssociation_1.LabelAssociationType.Chat, mod.addChatLabel.labelId, jid],
|
|
603
502
|
type: 'regular',
|
|
604
503
|
apiVersion: 3,
|
|
605
|
-
operation: OP.SET
|
|
504
|
+
operation: OP.SET,
|
|
606
505
|
};
|
|
607
506
|
}
|
|
608
507
|
else if ('removeChatLabel' in mod) {
|
|
609
508
|
patch = {
|
|
610
509
|
syncAction: {
|
|
611
510
|
labelAssociationAction: {
|
|
612
|
-
labeled: false
|
|
511
|
+
labeled: false,
|
|
613
512
|
}
|
|
614
513
|
},
|
|
615
|
-
index: [LabelAssociationType.Chat, mod.removeChatLabel.labelId, jid],
|
|
514
|
+
index: [LabelAssociation_1.LabelAssociationType.Chat, mod.removeChatLabel.labelId, jid],
|
|
616
515
|
type: 'regular',
|
|
617
516
|
apiVersion: 3,
|
|
618
|
-
operation: OP.SET
|
|
517
|
+
operation: OP.SET,
|
|
619
518
|
};
|
|
620
519
|
}
|
|
621
520
|
else if ('addMessageLabel' in mod) {
|
|
622
521
|
patch = {
|
|
623
522
|
syncAction: {
|
|
624
523
|
labelAssociationAction: {
|
|
625
|
-
labeled: true
|
|
524
|
+
labeled: true,
|
|
626
525
|
}
|
|
627
526
|
},
|
|
628
|
-
index: [
|
|
527
|
+
index: [
|
|
528
|
+
LabelAssociation_1.LabelAssociationType.Message,
|
|
529
|
+
mod.addMessageLabel.labelId,
|
|
530
|
+
jid,
|
|
531
|
+
mod.addMessageLabel.messageId,
|
|
532
|
+
'0',
|
|
533
|
+
'0'
|
|
534
|
+
],
|
|
629
535
|
type: 'regular',
|
|
630
536
|
apiVersion: 3,
|
|
631
|
-
operation: OP.SET
|
|
537
|
+
operation: OP.SET,
|
|
632
538
|
};
|
|
633
539
|
}
|
|
634
540
|
else if ('removeMessageLabel' in mod) {
|
|
635
541
|
patch = {
|
|
636
542
|
syncAction: {
|
|
637
543
|
labelAssociationAction: {
|
|
638
|
-
labeled: false
|
|
544
|
+
labeled: false,
|
|
639
545
|
}
|
|
640
546
|
},
|
|
641
547
|
index: [
|
|
642
|
-
LabelAssociationType.Message,
|
|
548
|
+
LabelAssociation_1.LabelAssociationType.Message,
|
|
643
549
|
mod.removeMessageLabel.labelId,
|
|
644
550
|
jid,
|
|
645
551
|
mod.removeMessageLabel.messageId,
|
|
@@ -648,30 +554,34 @@ export const chatModificationToAppPatch = (mod, jid) => {
|
|
|
648
554
|
],
|
|
649
555
|
type: 'regular',
|
|
650
556
|
apiVersion: 3,
|
|
651
|
-
operation: OP.SET
|
|
557
|
+
operation: OP.SET,
|
|
652
558
|
};
|
|
653
559
|
}
|
|
654
560
|
else {
|
|
655
|
-
throw new Boom('not supported');
|
|
561
|
+
throw new boom_1.Boom('not supported');
|
|
656
562
|
}
|
|
657
563
|
patch.syncAction.timestamp = Date.now();
|
|
658
564
|
return patch;
|
|
659
565
|
};
|
|
660
|
-
|
|
566
|
+
exports.chatModificationToAppPatch = chatModificationToAppPatch;
|
|
567
|
+
const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) => {
|
|
568
|
+
var _a, _b, _c, _d;
|
|
661
569
|
const isInitialSync = !!initialSyncOpts;
|
|
662
|
-
const accountSettings = initialSyncOpts
|
|
663
|
-
logger
|
|
570
|
+
const accountSettings = initialSyncOpts === null || initialSyncOpts === void 0 ? void 0 : initialSyncOpts.accountSettings;
|
|
571
|
+
logger === null || logger === void 0 ? void 0 : logger.trace({ syncAction, initialSync: !!initialSyncOpts }, 'processing sync action');
|
|
664
572
|
const { syncAction: { value: action }, index: [type, id, msgId, fromMe] } = syncAction;
|
|
665
|
-
if (action
|
|
573
|
+
if (action === null || action === void 0 ? void 0 : action.muteAction) {
|
|
666
574
|
ev.emit('chats.update', [
|
|
667
575
|
{
|
|
668
576
|
id,
|
|
669
|
-
muteEndTime:
|
|
577
|
+
muteEndTime: ((_a = action.muteAction) === null || _a === void 0 ? void 0 : _a.muted)
|
|
578
|
+
? (0, generics_1.toNumber)(action.muteAction.muteEndTimestamp)
|
|
579
|
+
: null,
|
|
670
580
|
conditional: getChatUpdateConditional(id, undefined)
|
|
671
581
|
}
|
|
672
582
|
]);
|
|
673
583
|
}
|
|
674
|
-
else if (action
|
|
584
|
+
else if ((action === null || action === void 0 ? void 0 : action.archiveChatAction) || type === 'archive' || type === 'unarchive') {
|
|
675
585
|
// okay so we've to do some annoying computation here
|
|
676
586
|
// when we're initially syncing the app state
|
|
677
587
|
// there are a few cases we need to handle
|
|
@@ -682,38 +592,36 @@ export const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) =
|
|
|
682
592
|
// we compare the timestamp of latest message from the other person to determine this
|
|
683
593
|
// 2. if the account unarchiveChats setting is false -- then it doesn't matter,
|
|
684
594
|
// it'll always take an app state action to mark in unarchived -- which we'll get anyway
|
|
685
|
-
const archiveAction = action
|
|
686
|
-
const isArchived = archiveAction
|
|
595
|
+
const archiveAction = action === null || action === void 0 ? void 0 : action.archiveChatAction;
|
|
596
|
+
const isArchived = archiveAction
|
|
597
|
+
? archiveAction.archived
|
|
598
|
+
: type === 'archive';
|
|
687
599
|
// // basically we don't need to fire an "archive" update if the chat is being marked unarchvied
|
|
688
600
|
// // this only applies for the initial sync
|
|
689
601
|
// if(isInitialSync && !isArchived) {
|
|
690
602
|
// isArchived = false
|
|
691
603
|
// }
|
|
692
|
-
const msgRange = !accountSettings
|
|
604
|
+
const msgRange = !(accountSettings === null || accountSettings === void 0 ? void 0 : accountSettings.unarchiveChats) ? undefined : archiveAction === null || archiveAction === void 0 ? void 0 : archiveAction.messageRange;
|
|
693
605
|
// logger?.debug({ chat: id, syncAction }, 'message range archive')
|
|
694
|
-
ev.emit('chats.update', [
|
|
695
|
-
{
|
|
606
|
+
ev.emit('chats.update', [{
|
|
696
607
|
id,
|
|
697
608
|
archived: isArchived,
|
|
698
609
|
conditional: getChatUpdateConditional(id, msgRange)
|
|
699
|
-
}
|
|
700
|
-
]);
|
|
610
|
+
}]);
|
|
701
611
|
}
|
|
702
|
-
else if (action
|
|
612
|
+
else if (action === null || action === void 0 ? void 0 : action.markChatAsReadAction) {
|
|
703
613
|
const markReadAction = action.markChatAsReadAction;
|
|
704
614
|
// basically we don't need to fire an "read" update if the chat is being marked as read
|
|
705
615
|
// because the chat is read by default
|
|
706
616
|
// this only applies for the initial sync
|
|
707
617
|
const isNullUpdate = isInitialSync && markReadAction.read;
|
|
708
|
-
ev.emit('chats.update', [
|
|
709
|
-
{
|
|
618
|
+
ev.emit('chats.update', [{
|
|
710
619
|
id,
|
|
711
|
-
unreadCount: isNullUpdate ? null : !!markReadAction
|
|
712
|
-
conditional: getChatUpdateConditional(id, markReadAction
|
|
713
|
-
}
|
|
714
|
-
]);
|
|
620
|
+
unreadCount: isNullUpdate ? null : !!(markReadAction === null || markReadAction === void 0 ? void 0 : markReadAction.read) ? 0 : -1,
|
|
621
|
+
conditional: getChatUpdateConditional(id, markReadAction === null || markReadAction === void 0 ? void 0 : markReadAction.messageRange)
|
|
622
|
+
}]);
|
|
715
623
|
}
|
|
716
|
-
else if (action
|
|
624
|
+
else if ((action === null || action === void 0 ? void 0 : action.deleteMessageForMeAction) || type === 'deleteMessageForMe') {
|
|
717
625
|
ev.emit('messages.delete', {
|
|
718
626
|
keys: [
|
|
719
627
|
{
|
|
@@ -724,35 +632,37 @@ export const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) =
|
|
|
724
632
|
]
|
|
725
633
|
});
|
|
726
634
|
}
|
|
727
|
-
else if (action
|
|
728
|
-
|
|
729
|
-
|
|
730
|
-
|
|
731
|
-
|
|
732
|
-
|
|
733
|
-
|
|
635
|
+
else if (action === null || action === void 0 ? void 0 : action.contactAction) {
|
|
636
|
+
ev.emit('contacts.upsert', [{
|
|
637
|
+
id: id,
|
|
638
|
+
name: action.contactAction.fullName,
|
|
639
|
+
lid: action.contactAction.lidJid || undefined,
|
|
640
|
+
jid: (0, WABinary_1.isJidUser)(id) ? id : undefined
|
|
641
|
+
}]);
|
|
642
|
+
}
|
|
643
|
+
else if (action === null || action === void 0 ? void 0 : action.pushNameSetting) {
|
|
644
|
+
const name = (_b = action === null || action === void 0 ? void 0 : action.pushNameSetting) === null || _b === void 0 ? void 0 : _b.name;
|
|
645
|
+
if (name && (me === null || me === void 0 ? void 0 : me.name) !== name) {
|
|
734
646
|
ev.emit('creds.update', { me: { ...me, name } });
|
|
735
647
|
}
|
|
736
648
|
}
|
|
737
|
-
else if (action
|
|
738
|
-
ev.emit('chats.update', [
|
|
739
|
-
{
|
|
649
|
+
else if (action === null || action === void 0 ? void 0 : action.pinAction) {
|
|
650
|
+
ev.emit('chats.update', [{
|
|
740
651
|
id,
|
|
741
|
-
pinned: action.pinAction
|
|
652
|
+
pinned: ((_c = action.pinAction) === null || _c === void 0 ? void 0 : _c.pinned) ? (0, generics_1.toNumber)(action.timestamp) : null,
|
|
742
653
|
conditional: getChatUpdateConditional(id, undefined)
|
|
743
|
-
}
|
|
744
|
-
]);
|
|
654
|
+
}]);
|
|
745
655
|
}
|
|
746
|
-
else if (action
|
|
656
|
+
else if (action === null || action === void 0 ? void 0 : action.unarchiveChatsSetting) {
|
|
747
657
|
const unarchiveChats = !!action.unarchiveChatsSetting.unarchiveChats;
|
|
748
658
|
ev.emit('creds.update', { accountSettings: { unarchiveChats } });
|
|
749
|
-
logger
|
|
659
|
+
logger === null || logger === void 0 ? void 0 : logger.info(`archive setting updated => '${action.unarchiveChatsSetting.unarchiveChats}'`);
|
|
750
660
|
if (accountSettings) {
|
|
751
661
|
accountSettings.unarchiveChats = unarchiveChats;
|
|
752
662
|
}
|
|
753
663
|
}
|
|
754
|
-
else if (action
|
|
755
|
-
let starred = action
|
|
664
|
+
else if ((action === null || action === void 0 ? void 0 : action.starAction) || type === 'star') {
|
|
665
|
+
let starred = (_d = action === null || action === void 0 ? void 0 : action.starAction) === null || _d === void 0 ? void 0 : _d.starred;
|
|
756
666
|
if (typeof starred !== 'boolean') {
|
|
757
667
|
starred = syncAction.index[syncAction.index.length - 1] === '1';
|
|
758
668
|
}
|
|
@@ -763,99 +673,46 @@ export const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) =
|
|
|
763
673
|
}
|
|
764
674
|
]);
|
|
765
675
|
}
|
|
766
|
-
else if (action
|
|
676
|
+
else if ((action === null || action === void 0 ? void 0 : action.deleteChatAction) || type === 'deleteChat') {
|
|
767
677
|
if (!isInitialSync) {
|
|
768
678
|
ev.emit('chats.delete', [id]);
|
|
769
679
|
}
|
|
770
680
|
}
|
|
771
|
-
else if (action
|
|
681
|
+
else if (action === null || action === void 0 ? void 0 : action.labelEditAction) {
|
|
772
682
|
const { name, color, deleted, predefinedId } = action.labelEditAction;
|
|
773
683
|
ev.emit('labels.edit', {
|
|
774
|
-
id
|
|
684
|
+
id,
|
|
775
685
|
name: name,
|
|
776
686
|
color: color,
|
|
777
687
|
deleted: deleted,
|
|
778
688
|
predefinedId: predefinedId ? String(predefinedId) : undefined
|
|
779
689
|
});
|
|
780
690
|
}
|
|
781
|
-
else if (action
|
|
691
|
+
else if (action === null || action === void 0 ? void 0 : action.labelAssociationAction) {
|
|
782
692
|
ev.emit('labels.association', {
|
|
783
|
-
type: action.labelAssociationAction.labeled
|
|
784
|
-
|
|
693
|
+
type: action.labelAssociationAction.labeled
|
|
694
|
+
? 'add'
|
|
695
|
+
: 'remove',
|
|
696
|
+
association: type === LabelAssociation_1.LabelAssociationType.Chat
|
|
785
697
|
? {
|
|
786
|
-
type: LabelAssociationType.Chat,
|
|
698
|
+
type: LabelAssociation_1.LabelAssociationType.Chat,
|
|
787
699
|
chatId: syncAction.index[2],
|
|
788
700
|
labelId: syncAction.index[1]
|
|
789
701
|
}
|
|
790
702
|
: {
|
|
791
|
-
type: LabelAssociationType.Message,
|
|
703
|
+
type: LabelAssociation_1.LabelAssociationType.Message,
|
|
792
704
|
chatId: syncAction.index[2],
|
|
793
705
|
messageId: syncAction.index[3],
|
|
794
706
|
labelId: syncAction.index[1]
|
|
795
707
|
}
|
|
796
708
|
});
|
|
797
709
|
}
|
|
798
|
-
else if (action?.localeSetting?.locale) {
|
|
799
|
-
ev.emit('settings.update', { setting: 'locale', value: action.localeSetting.locale });
|
|
800
|
-
}
|
|
801
|
-
else if (action?.timeFormatAction) {
|
|
802
|
-
ev.emit('settings.update', { setting: 'timeFormat', value: action.timeFormatAction });
|
|
803
|
-
}
|
|
804
|
-
else if (action?.pnForLidChatAction) {
|
|
805
|
-
if (action.pnForLidChatAction.pnJid) {
|
|
806
|
-
ev.emit('lid-mapping.update', { lid: id, pn: action.pnForLidChatAction.pnJid });
|
|
807
|
-
}
|
|
808
|
-
}
|
|
809
|
-
else if (action?.privacySettingRelayAllCalls) {
|
|
810
|
-
ev.emit('settings.update', {
|
|
811
|
-
setting: 'privacySettingRelayAllCalls',
|
|
812
|
-
value: action.privacySettingRelayAllCalls
|
|
813
|
-
});
|
|
814
|
-
}
|
|
815
|
-
else if (action?.statusPrivacy) {
|
|
816
|
-
ev.emit('settings.update', { setting: 'statusPrivacy', value: action.statusPrivacy });
|
|
817
|
-
}
|
|
818
|
-
else if (action?.lockChatAction) {
|
|
819
|
-
ev.emit('chats.lock', { id: id, locked: !!action.lockChatAction.locked });
|
|
820
|
-
}
|
|
821
|
-
else if (action?.privacySettingDisableLinkPreviewsAction) {
|
|
822
|
-
ev.emit('settings.update', {
|
|
823
|
-
setting: 'disableLinkPreviews',
|
|
824
|
-
value: action.privacySettingDisableLinkPreviewsAction
|
|
825
|
-
});
|
|
826
|
-
}
|
|
827
|
-
else if (action?.notificationActivitySettingAction?.notificationActivitySetting) {
|
|
828
|
-
ev.emit('settings.update', {
|
|
829
|
-
setting: 'notificationActivitySetting',
|
|
830
|
-
value: action.notificationActivitySettingAction.notificationActivitySetting
|
|
831
|
-
});
|
|
832
|
-
}
|
|
833
|
-
else if (action?.lidContactAction) {
|
|
834
|
-
ev.emit('contacts.upsert', [
|
|
835
|
-
{
|
|
836
|
-
id: id,
|
|
837
|
-
name: action.lidContactAction.fullName ||
|
|
838
|
-
action.lidContactAction.firstName ||
|
|
839
|
-
action.lidContactAction.username ||
|
|
840
|
-
undefined,
|
|
841
|
-
username: action.lidContactAction.username || undefined,
|
|
842
|
-
lid: id,
|
|
843
|
-
phoneNumber: undefined
|
|
844
|
-
}
|
|
845
|
-
]);
|
|
846
|
-
}
|
|
847
|
-
else if (action?.privacySettingChannelsPersonalisedRecommendationAction) {
|
|
848
|
-
ev.emit('settings.update', {
|
|
849
|
-
setting: 'channelsPersonalisedRecommendation',
|
|
850
|
-
value: action.privacySettingChannelsPersonalisedRecommendationAction
|
|
851
|
-
});
|
|
852
|
-
}
|
|
853
710
|
else {
|
|
854
|
-
logger
|
|
711
|
+
logger === null || logger === void 0 ? void 0 : logger.debug({ syncAction, id }, 'unprocessable update');
|
|
855
712
|
}
|
|
856
713
|
function getChatUpdateConditional(id, msgRange) {
|
|
857
714
|
return isInitialSync
|
|
858
|
-
? data => {
|
|
715
|
+
? (data) => {
|
|
859
716
|
const chat = data.historySets.chats[id] || data.chatUpserts[id];
|
|
860
717
|
if (chat) {
|
|
861
718
|
return msgRange ? isValidPatchBasedOnMessageRange(chat, msgRange) : true;
|
|
@@ -864,9 +721,9 @@ export const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) =
|
|
|
864
721
|
: undefined;
|
|
865
722
|
}
|
|
866
723
|
function isValidPatchBasedOnMessageRange(chat, msgRange) {
|
|
867
|
-
const lastMsgTimestamp = Number(msgRange
|
|
868
|
-
const chatLastMsgTimestamp = Number(chat
|
|
724
|
+
const lastMsgTimestamp = Number((msgRange === null || msgRange === void 0 ? void 0 : msgRange.lastMessageTimestamp) || (msgRange === null || msgRange === void 0 ? void 0 : msgRange.lastSystemMessageTimestamp) || 0);
|
|
725
|
+
const chatLastMsgTimestamp = Number((chat === null || chat === void 0 ? void 0 : chat.lastMessageRecvTimestamp) || 0);
|
|
869
726
|
return lastMsgTimestamp >= chatLastMsgTimestamp;
|
|
870
727
|
}
|
|
871
728
|
};
|
|
872
|
-
|
|
729
|
+
exports.processSyncAction = processSyncAction;
|