@dyyxyzz/baileys-mod 6.0.53 → 7.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (213) hide show
  1. package/LICENSE +21 -0
  2. package/WAProto/WAProto.proto +5311 -0
  3. package/WAProto/index.js +65801 -141371
  4. package/lib/Defaults/index.js +117 -141
  5. package/lib/Defaults/index.js.bak +123 -0
  6. package/lib/KeyDB/BinarySearch.js +20 -0
  7. package/lib/KeyDB/KeyedDB.js +167 -0
  8. package/lib/KeyDB/index.js +4 -0
  9. package/lib/Signal/Group/ciphertext-message.js +12 -14
  10. package/lib/Signal/Group/group-session-builder.js +10 -42
  11. package/lib/Signal/Group/group_cipher.js +75 -87
  12. package/lib/Signal/Group/index.js +13 -57
  13. package/lib/Signal/Group/keyhelper.js +17 -52
  14. package/lib/Signal/Group/sender-chain-key.js +27 -33
  15. package/lib/Signal/Group/sender-key-distribution-message.js +62 -63
  16. package/lib/Signal/Group/sender-key-message.js +65 -66
  17. package/lib/Signal/Group/sender-key-name.js +45 -44
  18. package/lib/Signal/Group/sender-key-record.js +39 -49
  19. package/lib/Signal/Group/sender-key-state.js +80 -93
  20. package/lib/Signal/Group/sender-message-key.js +27 -28
  21. package/lib/Signal/libsignal.js +313 -163
  22. package/lib/Signal/lid-mapping.js +155 -0
  23. package/lib/Socket/Client/index.js +4 -19
  24. package/lib/Socket/Client/types.js +13 -0
  25. package/lib/Socket/Client/websocket.js +52 -0
  26. package/lib/Socket/Client/websocket.js.bak +53 -0
  27. package/lib/Socket/business.js +359 -242
  28. package/lib/Socket/chats.js +839 -943
  29. package/lib/Socket/communities.js +413 -0
  30. package/lib/Socket/groups.js +304 -309
  31. package/lib/Socket/index.js +15 -10
  32. package/lib/Socket/messages-recv.js +1107 -1054
  33. package/lib/Socket/messages-send.js +639 -449
  34. package/lib/Socket/mex.js +45 -0
  35. package/lib/Socket/newsletter.js +197 -311
  36. package/lib/Socket/socket.js +5 -3
  37. package/lib/Store/index.js +6 -10
  38. package/lib/Store/make-cache-manager-store.js +73 -81
  39. package/lib/Store/make-in-memory-store.js +286 -423
  40. package/lib/Store/make-ordered-dictionary.js +77 -79
  41. package/lib/Store/object-repository.js +24 -26
  42. package/lib/Types/Auth.js +3 -2
  43. package/lib/Types/Bussines.js +3 -0
  44. package/lib/Types/Call.js +3 -2
  45. package/lib/Types/Chat.js +9 -4
  46. package/lib/Types/Contact.js +3 -2
  47. package/lib/Types/Events.js +3 -2
  48. package/lib/Types/GroupMetadata.js +3 -2
  49. package/lib/Types/Label.js +24 -26
  50. package/lib/Types/LabelAssociation.js +6 -8
  51. package/lib/Types/Message.js +12 -9
  52. package/lib/Types/Newsletter.js +33 -38
  53. package/lib/Types/Newsletter.js.bak +33 -0
  54. package/lib/Types/Product.js +3 -2
  55. package/lib/Types/Signal.js +3 -2
  56. package/lib/Types/Socket.js +4 -2
  57. package/lib/Types/State.js +11 -2
  58. package/lib/Types/USync.js +3 -2
  59. package/lib/Types/index.js +27 -41
  60. package/lib/Utils/auth-utils.js +211 -198
  61. package/lib/Utils/baileys-event-stream.js +42 -61
  62. package/lib/Utils/browser-utils.js +5 -1
  63. package/lib/Utils/business.js +213 -214
  64. package/lib/Utils/chat-utils.js +703 -689
  65. package/lib/Utils/crypto.js +112 -133
  66. package/lib/Utils/decode-wa-message.js +252 -183
  67. package/lib/Utils/event-buffer.js +510 -496
  68. package/lib/Utils/generics.js +319 -392
  69. package/lib/Utils/history.js +83 -92
  70. package/lib/Utils/index.js +21 -33
  71. package/lib/Utils/link-preview.js +71 -83
  72. package/lib/Utils/logger.js +5 -7
  73. package/lib/Utils/lt-hash.js +40 -46
  74. package/lib/Utils/make-mutex.js +34 -41
  75. package/lib/Utils/message-retry-manager.js +113 -0
  76. package/lib/Utils/messages-media.js +550 -768
  77. package/lib/Utils/messages.js +371 -270
  78. package/lib/Utils/noise-handler.js +138 -149
  79. package/lib/Utils/pre-key-manager.js +85 -0
  80. package/lib/Utils/process-message.js +323 -303
  81. package/lib/Utils/signal.js +149 -141
  82. package/lib/Utils/use-multi-file-auth-state.js +95 -103
  83. package/lib/Utils/validate-connection.js +183 -214
  84. package/lib/WABinary/constants.js +1298 -35
  85. package/lib/WABinary/decode.js +237 -249
  86. package/lib/WABinary/encode.js +213 -260
  87. package/lib/WABinary/generic-utils.js +56 -65
  88. package/lib/WABinary/index.js +7 -21
  89. package/lib/WABinary/jid-utils.js +89 -58
  90. package/lib/WABinary/types.js +3 -2
  91. package/lib/WAM/BinaryInfo.js +10 -12
  92. package/lib/WAM/constants.js +22851 -15348
  93. package/lib/WAM/encode.js +135 -136
  94. package/lib/WAM/index.js +5 -19
  95. package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -30
  96. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +49 -53
  97. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +27 -28
  98. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +36 -39
  99. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -50
  100. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +26 -20
  101. package/lib/WAUSync/Protocols/index.js +6 -20
  102. package/lib/WAUSync/USyncQuery.js +86 -85
  103. package/lib/WAUSync/USyncUser.js +23 -25
  104. package/lib/WAUSync/index.js +5 -19
  105. package/lib/index.js +19 -34
  106. package/package.json +89 -106
  107. package/engine-requirements.js +0 -10
  108. package/lib/Defaults/baileys-version.json +0 -3
  109. package/lib/Defaults/index.d.ts +0 -53
  110. package/lib/Defaults/phonenumber-mcc.json +0 -223
  111. package/lib/Signal/Group/ciphertext-message.d.ts +0 -9
  112. package/lib/Signal/Group/group-session-builder.d.ts +0 -14
  113. package/lib/Signal/Group/group_cipher.d.ts +0 -17
  114. package/lib/Signal/Group/index.d.ts +0 -11
  115. package/lib/Signal/Group/keyhelper.d.ts +0 -10
  116. package/lib/Signal/Group/queue-job.d.ts +0 -1
  117. package/lib/Signal/Group/queue-job.js +0 -57
  118. package/lib/Signal/Group/sender-chain-key.d.ts +0 -13
  119. package/lib/Signal/Group/sender-key-distribution-message.d.ts +0 -16
  120. package/lib/Signal/Group/sender-key-message.d.ts +0 -18
  121. package/lib/Signal/Group/sender-key-name.d.ts +0 -17
  122. package/lib/Signal/Group/sender-key-record.d.ts +0 -30
  123. package/lib/Signal/Group/sender-key-state.d.ts +0 -38
  124. package/lib/Signal/Group/sender-message-key.d.ts +0 -11
  125. package/lib/Signal/libsignal.d.ts +0 -3
  126. package/lib/Socket/Client/abstract-socket-client.d.ts +0 -17
  127. package/lib/Socket/Client/abstract-socket-client.js +0 -13
  128. package/lib/Socket/Client/index.d.ts +0 -3
  129. package/lib/Socket/Client/mobile-socket-client.d.ts +0 -13
  130. package/lib/Socket/Client/mobile-socket-client.js +0 -65
  131. package/lib/Socket/Client/web-socket-client.d.ts +0 -12
  132. package/lib/Socket/Client/web-socket-client.js +0 -62
  133. package/lib/Socket/business.d.ts +0 -171
  134. package/lib/Socket/chats.d.ts +0 -267
  135. package/lib/Socket/chats.js.bak +0 -981
  136. package/lib/Socket/dugong.d.ts +0 -254
  137. package/lib/Socket/dugong.js +0 -484
  138. package/lib/Socket/groups.d.ts +0 -115
  139. package/lib/Socket/index.d.ts +0 -173
  140. package/lib/Socket/messages-recv.d.ts +0 -161
  141. package/lib/Socket/messages-send.d.ts +0 -149
  142. package/lib/Socket/newsletter.d.ts +0 -134
  143. package/lib/Socket/registration.d.ts +0 -267
  144. package/lib/Socket/registration.js +0 -166
  145. package/lib/Socket/socket.d.ts +0 -43
  146. package/lib/Socket/usync.d.ts +0 -36
  147. package/lib/Socket/usync.js +0 -70
  148. package/lib/Store/index.d.ts +0 -3
  149. package/lib/Store/make-cache-manager-store.d.ts +0 -13
  150. package/lib/Store/make-in-memory-store.d.ts +0 -118
  151. package/lib/Store/make-ordered-dictionary.d.ts +0 -13
  152. package/lib/Store/object-repository.d.ts +0 -10
  153. package/lib/Types/Auth.d.ts +0 -110
  154. package/lib/Types/Call.d.ts +0 -13
  155. package/lib/Types/Chat.d.ts +0 -102
  156. package/lib/Types/Contact.d.ts +0 -19
  157. package/lib/Types/Events.d.ts +0 -157
  158. package/lib/Types/GroupMetadata.d.ts +0 -55
  159. package/lib/Types/Label.d.ts +0 -35
  160. package/lib/Types/LabelAssociation.d.ts +0 -29
  161. package/lib/Types/Message.d.ts +0 -273
  162. package/lib/Types/Newsletter.d.ts +0 -103
  163. package/lib/Types/Product.d.ts +0 -78
  164. package/lib/Types/Signal.d.ts +0 -57
  165. package/lib/Types/Socket.d.ts +0 -111
  166. package/lib/Types/State.d.ts +0 -27
  167. package/lib/Types/USync.d.ts +0 -25
  168. package/lib/Types/index.d.ts +0 -57
  169. package/lib/Utils/auth-utils.d.ts +0 -18
  170. package/lib/Utils/baileys-event-stream.d.ts +0 -16
  171. package/lib/Utils/business.d.ts +0 -22
  172. package/lib/Utils/chat-utils.d.ts +0 -71
  173. package/lib/Utils/crypto.d.ts +0 -41
  174. package/lib/Utils/decode-wa-message.d.ts +0 -19
  175. package/lib/Utils/event-buffer.d.ts +0 -35
  176. package/lib/Utils/generics.d.ts +0 -92
  177. package/lib/Utils/generics.js.bak +0 -433
  178. package/lib/Utils/history.d.ts +0 -15
  179. package/lib/Utils/index.d.ts +0 -17
  180. package/lib/Utils/link-preview.d.ts +0 -21
  181. package/lib/Utils/logger.d.ts +0 -4
  182. package/lib/Utils/lt-hash.d.ts +0 -12
  183. package/lib/Utils/make-mutex.d.ts +0 -7
  184. package/lib/Utils/messages-media.d.ts +0 -116
  185. package/lib/Utils/messages.d.ts +0 -77
  186. package/lib/Utils/noise-handler.d.ts +0 -21
  187. package/lib/Utils/process-message.d.ts +0 -41
  188. package/lib/Utils/signal.d.ts +0 -32
  189. package/lib/Utils/use-multi-file-auth-state.d.ts +0 -13
  190. package/lib/Utils/validate-connection.d.ts +0 -11
  191. package/lib/Utils/validate-connection.js.bak +0 -237
  192. package/lib/WABinary/constants.d.ts +0 -30
  193. package/lib/WABinary/decode.d.ts +0 -7
  194. package/lib/WABinary/encode.d.ts +0 -3
  195. package/lib/WABinary/generic-utils.d.ts +0 -17
  196. package/lib/WABinary/index.d.ts +0 -5
  197. package/lib/WABinary/jid-utils.d.ts +0 -31
  198. package/lib/WABinary/types.d.ts +0 -18
  199. package/lib/WAM/BinaryInfo.d.ts +0 -17
  200. package/lib/WAM/constants.d.ts +0 -38
  201. package/lib/WAM/encode.d.ts +0 -3
  202. package/lib/WAM/index.d.ts +0 -3
  203. package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts +0 -9
  204. package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts +0 -22
  205. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts +0 -12
  206. package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts +0 -12
  207. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts +0 -25
  208. package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts +0 -8
  209. package/lib/WAUSync/Protocols/index.d.ts +0 -4
  210. package/lib/WAUSync/USyncQuery.d.ts +0 -28
  211. package/lib/WAUSync/USyncUser.d.ts +0 -12
  212. package/lib/WAUSync/index.d.ts +0 -3
  213. package/lib/index.d.ts +0 -12
@@ -1,509 +1,450 @@
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");
12
- const mutationKeys = (keydata) => {
13
- const expanded = (0, crypto_1.hkdf)(keydata, 160, { info: 'WhatsApp Mutation Keys' });
14
- return {
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)
20
- };
1
+ //=======================================================//
2
+ import { getBinaryNodeChild, getBinaryNodeChildren, isJidGroup, jidNormalizedUser } from "../WABinary/index.js";
3
+ import { aesDecrypt, aesEncrypt, hkdf, hmacSign } from "./crypto.js";
4
+ import { LabelAssociationType } from "../Types/LabelAssociation.js";
5
+ import { downloadContentFromMessage } from "./messages-media.js";
6
+ import { LT_HASH_ANTI_TAMPERING } from "./lt-hash.js";
7
+ import { proto } from "../../WAProto/index.js";
8
+ import { toNumber } from "./generics.js";
9
+ import { Boom } from "@hapi/boom";
10
+ //=======================================================//
11
+ const mutationKeys = async (keydata) => {
12
+ const expanded = await hkdf(keydata, 160, { info: "WhatsApp Mutation Keys" });
13
+ return {
14
+ indexKey: expanded.slice(0, 32),
15
+ valueEncryptionKey: expanded.slice(32, 64),
16
+ valueMacKey: expanded.slice(64, 96),
17
+ snapshotMacKey: expanded.slice(96, 128),
18
+ patchMacKey: expanded.slice(128, 160)
19
+ };
21
20
  };
21
+ //=======================================================//
22
22
  const generateMac = (operation, data, keyId, key) => {
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);
23
+ const getKeyData = () => {
24
+ let r;
25
+ switch (operation) {
26
+ case proto.SyncdMutation.SyncdOperation.SET:
27
+ r = 0x01;
28
+ break;
29
+ case 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);
38
+ last.set([keyData.length], last.length - 1);
39
+ const total = Buffer.concat([keyData, data, last]);
40
+ const hmac = hmacSign(total, key, "sha512");
41
+ return hmac.slice(0, 32);
42
42
  };
43
+ //=======================================================//
43
44
  const to64BitNetworkOrder = (e) => {
44
- const buff = Buffer.alloc(8);
45
- buff.writeUint32BE(e, 4);
46
- return buff;
45
+ const buff = Buffer.alloc(8);
46
+ buff.writeUint32BE(e, 4);
47
+ return buff;
47
48
  };
49
+ //=======================================================//
48
50
  const makeLtHashGenerator = ({ indexValueMap, hash }) => {
49
- indexValueMap = { ...indexValueMap };
50
- const addBuffs = [];
51
- const subBuffs = [];
52
- return {
53
- mix: ({ indexMac, valueMac, operation }) => {
54
- const indexMacBase64 = Buffer.from(indexMac).toString('base64');
55
- const prevOp = indexValueMap[indexMacBase64];
56
- if (operation === WAProto_1.proto.SyncdMutation.SyncdOperation.REMOVE) {
57
- if (!prevOp) {
58
- throw new boom_1.Boom('tried remove, but no previous op', { data: { indexMac, valueMac } });
59
- }
60
- // remove from index value mac, since this mutation is erased
61
- delete indexValueMap[indexMacBase64];
62
- }
63
- else {
64
- addBuffs.push(new Uint8Array(valueMac).buffer);
65
- // add this index into the history map
66
- indexValueMap[indexMacBase64] = { valueMac };
67
- }
68
- if (prevOp) {
69
- subBuffs.push(new Uint8Array(prevOp.valueMac).buffer);
70
- }
71
- },
72
- finish: () => {
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);
76
- return {
77
- hash: buffer,
78
- indexValueMap
79
- };
51
+ indexValueMap = { ...indexValueMap };
52
+ const addBuffs = [];
53
+ const subBuffs = [];
54
+ return {
55
+ mix: ({ indexMac, valueMac, operation }) => {
56
+ const indexMacBase64 = Buffer.from(indexMac).toString("base64");
57
+ const prevOp = indexValueMap[indexMacBase64];
58
+ if (operation === proto.SyncdMutation.SyncdOperation.REMOVE) {
59
+ if (!prevOp) {
60
+ throw new Boom("tried remove, but no previous op", { data: { indexMac, valueMac } });
80
61
  }
81
- };
62
+ delete indexValueMap[indexMacBase64];
63
+ }
64
+ else {
65
+ addBuffs.push(new Uint8Array(valueMac).buffer);
66
+ indexValueMap[indexMacBase64] = { valueMac };
67
+ }
68
+ if (prevOp) {
69
+ subBuffs.push(new Uint8Array(prevOp.valueMac).buffer);
70
+ }
71
+ },
72
+ finish: async () => {
73
+ const hashArrayBuffer = new Uint8Array(hash).buffer;
74
+ const result = await LT_HASH_ANTI_TAMPERING.subtractThenAdd(hashArrayBuffer, addBuffs, subBuffs);
75
+ const buffer = Buffer.from(result);
76
+ return {
77
+ hash: buffer,
78
+ indexValueMap
79
+ };
80
+ }
81
+ };
82
82
  };
83
+ //=======================================================//
83
84
  const generateSnapshotMac = (lthash, version, name, key) => {
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');
85
+ const total = Buffer.concat([lthash, to64BitNetworkOrder(version), Buffer.from(name, "utf-8")]);
86
+ return hmacSign(total, key, "sha256");
90
87
  };
88
+ //=======================================================//
91
89
  const generatePatchMac = (snapshotMac, valueMacs, version, type, key) => {
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);
90
+ const total = Buffer.concat([snapshotMac, ...valueMacs, to64BitNetworkOrder(version), Buffer.from(type, "utf-8")]);
91
+ return hmacSign(total, key);
92
+ };
93
+ //=======================================================//
94
+ export const newLTHashState = () => ({ version: 0, hash: Buffer.alloc(128), indexValueMap: {} });
95
+ export const encodeSyncdPatch = async ({ type, index, syncAction, apiVersion, operation }, myAppStateKeyId, state, getAppStateSyncKey) => {
96
+ const key = !!myAppStateKeyId ? await getAppStateSyncKey(myAppStateKeyId) : undefined;
97
+ if (!key) {
98
+ throw new Boom(`myAppStateKey ("${myAppStateKeyId}") not present`, { statusCode: 404 });
99
+ }
100
+ const encKeyId = Buffer.from(myAppStateKeyId, "base64");
101
+ state = { ...state, indexValueMap: { ...state.indexValueMap } };
102
+ const indexBuffer = Buffer.from(JSON.stringify(index));
103
+ const dataProto = proto.SyncActionData.fromObject({
104
+ index: indexBuffer,
105
+ value: syncAction,
106
+ padding: new Uint8Array(0),
107
+ version: apiVersion
108
+ });
109
+ const encoded = proto.SyncActionData.encode(dataProto).finish();
110
+ const keyValue = await mutationKeys(key.keyData);
111
+ const encValue = aesEncrypt(encoded, keyValue.valueEncryptionKey);
112
+ const valueMac = generateMac(operation, encValue, encKeyId, keyValue.valueMacKey);
113
+ const indexMac = hmacSign(indexBuffer, keyValue.indexKey);
114
+ const generator = makeLtHashGenerator(state);
115
+ generator.mix({ indexMac, valueMac, operation });
116
+ Object.assign(state, await generator.finish());
117
+ state.version += 1;
118
+ const snapshotMac = generateSnapshotMac(state.hash, state.version, type, keyValue.snapshotMacKey);
119
+ const patch = {
120
+ patchMac: generatePatchMac(snapshotMac, [valueMac], state.version, type, keyValue.patchMacKey),
121
+ snapshotMac: snapshotMac,
122
+ keyId: { id: encKeyId },
123
+ mutations: [
124
+ {
125
+ operation: operation,
126
+ record: {
127
+ index: {
128
+ blob: indexMac
129
+ },
130
+ value: {
131
+ blob: Buffer.concat([encValue, valueMac])
132
+ },
133
+ keyId: { id: encKeyId }
134
+ }
135
+ }
136
+ ]
137
+ };
138
+ const base64Index = indexMac.toString("base64");
139
+ state.indexValueMap[base64Index] = { valueMac };
140
+ return { patch, state };
99
141
  };
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) => {
103
- const key = !!myAppStateKeyId ? await getAppStateSyncKey(myAppStateKeyId) : undefined;
104
- if (!key) {
105
- throw new boom_1.Boom(`myAppStateKey ("${myAppStateKeyId}") not present`, { statusCode: 404 });
142
+ //=======================================================//
143
+ export const decodeSyncdMutations = async (msgMutations, initialState, getAppStateSyncKey, onMutation, validateMacs) => {
144
+ const ltGenerator = makeLtHashGenerator(initialState);
145
+ for (const msgMutation of msgMutations) {
146
+ const operation = "operation" in msgMutation ? msgMutation.operation : proto.SyncdMutation.SyncdOperation.SET;
147
+ const record = "record" in msgMutation && !!msgMutation.record ? msgMutation.record : msgMutation;
148
+ const key = await getKey(record.keyId.id);
149
+ const content = Buffer.from(record.value.blob);
150
+ const encContent = content.slice(0, -32);
151
+ const ogValueMac = content.slice(-32);
152
+ if (validateMacs) {
153
+ const contentHmac = generateMac(operation, encContent, record.keyId.id, key.valueMacKey);
154
+ if (Buffer.compare(contentHmac, ogValueMac) !== 0) {
155
+ throw new Boom("HMAC content verification failed");
156
+ }
106
157
  }
107
- const encKeyId = Buffer.from(myAppStateKeyId, 'base64');
108
- state = { ...state, indexValueMap: { ...state.indexValueMap } };
109
- const indexBuffer = Buffer.from(JSON.stringify(index));
110
- const dataProto = WAProto_1.proto.SyncActionData.fromObject({
111
- index: indexBuffer,
112
- value: syncAction,
113
- padding: new Uint8Array(0),
114
- version: apiVersion
158
+ const result = aesDecrypt(encContent, key.valueEncryptionKey);
159
+ const syncAction = proto.SyncActionData.decode(result);
160
+ if (validateMacs) {
161
+ const hmac = hmacSign(syncAction.index, key.indexKey);
162
+ if (Buffer.compare(hmac, record.index.blob) !== 0) {
163
+ throw new Boom("HMAC index verification failed");
164
+ }
165
+ }
166
+ const indexStr = Buffer.from(syncAction.index).toString();
167
+ onMutation({ syncAction, index: JSON.parse(indexStr) });
168
+ ltGenerator.mix({
169
+ indexMac: record.index.blob,
170
+ valueMac: ogValueMac,
171
+ operation: operation
115
172
  });
116
- const encoded = WAProto_1.proto.SyncActionData.encode(dataProto).finish();
117
- const keyValue = mutationKeys(key.keyData);
118
- const encValue = (0, crypto_1.aesEncrypt)(encoded, keyValue.valueEncryptionKey);
119
- const valueMac = generateMac(operation, encValue, encKeyId, keyValue.valueMacKey);
120
- const indexMac = (0, crypto_1.hmacSign)(indexBuffer, keyValue.indexKey);
121
- // update LT hash
122
- const generator = makeLtHashGenerator(state);
123
- generator.mix({ indexMac, valueMac, operation });
124
- Object.assign(state, generator.finish());
125
- state.version += 1;
126
- const snapshotMac = generateSnapshotMac(state.hash, state.version, type, keyValue.snapshotMacKey);
127
- const patch = {
128
- patchMac: generatePatchMac(snapshotMac, [valueMac], state.version, type, keyValue.patchMacKey),
129
- snapshotMac: snapshotMac,
130
- keyId: { id: encKeyId },
131
- mutations: [
132
- {
133
- operation: operation,
134
- record: {
135
- index: {
136
- blob: indexMac
137
- },
138
- value: {
139
- blob: Buffer.concat([encValue, valueMac])
140
- },
141
- keyId: { id: encKeyId }
142
- }
143
- }
144
- ]
145
- };
146
- const base64Index = indexMac.toString('base64');
147
- state.indexValueMap[base64Index] = { valueMac };
148
- return { patch, state };
173
+ }
174
+ return await ltGenerator.finish();
175
+ async function getKey(keyId) {
176
+ const base64Key = Buffer.from(keyId).toString("base64");
177
+ const keyEnc = await getAppStateSyncKey(base64Key);
178
+ if (!keyEnc) {
179
+ throw new Boom(`failed to find key "${base64Key}" to decode mutation`, {
180
+ statusCode: 404,
181
+ data: { msgMutations }
182
+ });
183
+ }
184
+ return mutationKeys(keyEnc.keyData);
185
+ }
149
186
  };
150
- exports.encodeSyncdPatch = encodeSyncdPatch;
151
- const decodeSyncdMutations = async (msgMutations, initialState, getAppStateSyncKey, onMutation, validateMacs) => {
152
- const ltGenerator = makeLtHashGenerator(initialState);
153
- // indexKey used to HMAC sign record.index.blob
154
- // valueEncryptionKey used to AES-256-CBC encrypt record.value.blob[0:-32]
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
156
- for (const msgMutation of msgMutations) {
157
- // if it's a syncdmutation, get the operation property
158
- // otherwise, if it's only a record -- it'll be a SET mutation
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);
165
- if (validateMacs) {
166
- const contentHmac = generateMac(operation, encContent, record.keyId.id, key.valueMacKey);
167
- if (Buffer.compare(contentHmac, ogValueMac) !== 0) {
168
- throw new boom_1.Boom('HMAC content verification failed');
169
- }
170
- }
171
- const result = (0, crypto_1.aesDecrypt)(encContent, key.valueEncryptionKey);
172
- const syncAction = WAProto_1.proto.SyncActionData.decode(result);
173
- if (validateMacs) {
174
- const hmac = (0, crypto_1.hmacSign)(syncAction.index, key.indexKey);
175
- if (Buffer.compare(hmac, record.index.blob) !== 0) {
176
- throw new boom_1.Boom('HMAC index verification failed');
177
- }
178
- }
179
- const indexStr = Buffer.from(syncAction.index).toString();
180
- onMutation({ syncAction, index: JSON.parse(indexStr) });
181
- ltGenerator.mix({
182
- indexMac: record.index.blob,
183
- valueMac: ogValueMac,
184
- operation: operation
185
- });
186
- }
187
- return ltGenerator.finish();
188
- async function getKey(keyId) {
189
- const base64Key = Buffer.from(keyId).toString('base64');
190
- const keyEnc = await getAppStateSyncKey(base64Key);
191
- if (!keyEnc) {
192
- throw new boom_1.Boom(`failed to find key "${base64Key}" to decode mutation`, { statusCode: 404, data: { msgMutations } });
193
- }
194
- return mutationKeys(keyEnc.keyData);
195
- }
187
+ //=======================================================//
188
+ export const decodeSyncdPatch = async (msg, name, initialState, getAppStateSyncKey, onMutation, validateMacs) => {
189
+ if (validateMacs) {
190
+ const base64Key = Buffer.from(msg.keyId.id).toString("base64");
191
+ const mainKeyObj = await getAppStateSyncKey(base64Key);
192
+ if (!mainKeyObj) {
193
+ throw new Boom(`failed to find key "${base64Key}" to decode patch`, { statusCode: 404, data: { msg } });
194
+ }
195
+ const mainKey = await mutationKeys(mainKeyObj.keyData);
196
+ const mutationmacs = msg.mutations.map(mutation => mutation.record.value.blob.slice(-32));
197
+ const patchMac = generatePatchMac(msg.snapshotMac, mutationmacs, toNumber(msg.version.version), name, mainKey.patchMacKey);
198
+ if (Buffer.compare(patchMac, msg.patchMac) !== 0) {
199
+ throw new Boom("Invalid patch mac");
200
+ }
201
+ }
202
+ const result = await decodeSyncdMutations(msg.mutations, initialState, getAppStateSyncKey, onMutation, validateMacs);
203
+ return result;
196
204
  };
197
- exports.decodeSyncdMutations = decodeSyncdMutations;
198
- const decodeSyncdPatch = async (msg, name, initialState, getAppStateSyncKey, onMutation, validateMacs) => {
199
- if (validateMacs) {
200
- const base64Key = Buffer.from(msg.keyId.id).toString('base64');
201
- const mainKeyObj = await getAppStateSyncKey(base64Key);
202
- if (!mainKeyObj) {
203
- throw new boom_1.Boom(`failed to find key "${base64Key}" to decode patch`, { statusCode: 404, data: { msg } });
205
+ //=======================================================//
206
+ export const extractSyncdPatches = async (result, options) => {
207
+ const syncNode = getBinaryNodeChild(result, "sync");
208
+ const collectionNodes = getBinaryNodeChildren(syncNode, "collection");
209
+ const final = {};
210
+ await Promise.all(collectionNodes.map(async (collectionNode) => {
211
+ const patchesNode = getBinaryNodeChild(collectionNode, "patches");
212
+ const patches = getBinaryNodeChildren(patchesNode || collectionNode, "patch");
213
+ const snapshotNode = getBinaryNodeChild(collectionNode, "snapshot");
214
+ const syncds = [];
215
+ const name = collectionNode.attrs.name;
216
+ const hasMorePatches = collectionNode.attrs.has_more_patches === "true";
217
+ let snapshot = undefined;
218
+ if (snapshotNode && !!snapshotNode.content) {
219
+ if (!Buffer.isBuffer(snapshotNode)) {
220
+ snapshotNode.content = Buffer.from(Object.values(snapshotNode.content));
221
+ }
222
+ const blobRef = proto.ExternalBlobReference.decode(snapshotNode.content);
223
+ const data = await downloadExternalBlob(blobRef, options);
224
+ snapshot = proto.SyncdSnapshot.decode(data);
225
+ }
226
+ for (let { content } of patches) {
227
+ if (content) {
228
+ if (!Buffer.isBuffer(content)) {
229
+ content = Buffer.from(Object.values(content));
204
230
  }
205
- const mainKey = mutationKeys(mainKeyObj.keyData);
206
- const mutationmacs = msg.mutations.map(mutation => mutation.record.value.blob.slice(-32));
207
- const patchMac = generatePatchMac(msg.snapshotMac, mutationmacs, (0, generics_1.toNumber)(msg.version.version), name, mainKey.patchMacKey);
208
- if (Buffer.compare(patchMac, msg.patchMac) !== 0) {
209
- throw new boom_1.Boom('Invalid patch mac');
231
+ const syncd = proto.SyncdPatch.decode(content);
232
+ if (!syncd.version) {
233
+ syncd.version = { version: +collectionNode.attrs.version + 1 };
210
234
  }
235
+ syncds.push(syncd);
236
+ }
211
237
  }
212
- const result = await (0, exports.decodeSyncdMutations)(msg.mutations, initialState, getAppStateSyncKey, onMutation, validateMacs);
213
- return result;
238
+ final[name] = { patches: syncds, hasMorePatches, snapshot };
239
+ }));
240
+ return final;
214
241
  };
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');
219
- const final = {};
220
- await Promise.all(collectionNodes.map(async (collectionNode) => {
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');
224
- const syncds = [];
225
- const name = collectionNode.attrs.name;
226
- const hasMorePatches = collectionNode.attrs.has_more_patches === 'true';
227
- let snapshot = undefined;
228
- if (snapshotNode && !!snapshotNode.content) {
229
- if (!Buffer.isBuffer(snapshotNode)) {
230
- snapshotNode.content = Buffer.from(Object.values(snapshotNode.content));
231
- }
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);
235
- }
236
- for (let { content } of patches) {
237
- if (content) {
238
- if (!Buffer.isBuffer(content)) {
239
- content = Buffer.from(Object.values(content));
240
- }
241
- const syncd = WAProto_1.proto.SyncdPatch.decode(content);
242
- if (!syncd.version) {
243
- syncd.version = { version: +collectionNode.attrs.version + 1 };
244
- }
245
- syncds.push(syncd);
246
- }
247
- }
248
- final[name] = { patches: syncds, hasMorePatches, snapshot };
249
- }));
250
- return final;
242
+ //=======================================================//
243
+ export const downloadExternalBlob = async (blob, options) => {
244
+ const stream = await downloadContentFromMessage(blob, "md-app-state", { options });
245
+ const bufferArray = [];
246
+ for await (const chunk of stream) {
247
+ bufferArray.push(chunk);
248
+ }
249
+ return Buffer.concat(bufferArray);
251
250
  };
252
- exports.extractSyncdPatches = extractSyncdPatches;
253
- const downloadExternalBlob = async (blob, options) => {
254
- const stream = await (0, messages_media_1.downloadContentFromMessage)(blob, 'md-app-state', { options });
255
- const bufferArray = [];
256
- for await (const chunk of stream) {
257
- bufferArray.push(chunk);
258
- }
259
- return Buffer.concat(bufferArray);
251
+ //=======================================================//
252
+ export const downloadExternalPatch = async (blob, options) => {
253
+ const buffer = await downloadExternalBlob(blob, options);
254
+ const syncData = proto.SyncdMutations.decode(buffer);
255
+ return syncData;
260
256
  };
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);
265
- return syncData;
257
+ //=======================================================//
258
+ export const decodeSyncdSnapshot = async (name, snapshot, getAppStateSyncKey, minimumVersionNumber, validateMacs = true) => {
259
+ const newState = newLTHashState();
260
+ newState.version = toNumber(snapshot.version.version);
261
+ const mutationMap = {};
262
+ const areMutationsRequired = typeof minimumVersionNumber === "undefined" || newState.version > minimumVersionNumber;
263
+ const { hash, indexValueMap } = await decodeSyncdMutations(snapshot.records, newState, getAppStateSyncKey, areMutationsRequired
264
+ ? mutation => {
265
+ const index = mutation.syncAction.index?.toString();
266
+ mutationMap[index] = mutation;
267
+ }
268
+ : () => { }, validateMacs);
269
+ newState.hash = hash;
270
+ newState.indexValueMap = indexValueMap;
271
+ if (validateMacs) {
272
+ const base64Key = Buffer.from(snapshot.keyId.id).toString("base64");
273
+ const keyEnc = await getAppStateSyncKey(base64Key);
274
+ if (!keyEnc) {
275
+ throw new Boom(`failed to find key "${base64Key}" to decode mutation`);
276
+ }
277
+ const result = await mutationKeys(keyEnc.keyData);
278
+ const computedSnapshotMac = generateSnapshotMac(newState.hash, newState.version, name, result.snapshotMacKey);
279
+ if (Buffer.compare(snapshot.mac, computedSnapshotMac) !== 0) {
280
+ throw new Boom(`failed to verify LTHash at ${newState.version} of ${name} from snapshot`);
281
+ }
282
+ }
283
+ return {
284
+ state: newState,
285
+ mutationMap
286
+ };
266
287
  };
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);
271
- const mutationMap = {};
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();
278
- mutationMap[index] = mutation;
279
- }
280
- : () => { }, validateMacs);
281
- newState.hash = hash;
282
- newState.indexValueMap = indexValueMap;
288
+ //=======================================================//
289
+ export const decodePatches = async (name, syncds, initial, getAppStateSyncKey, options, minimumVersionNumber, logger, validateMacs = true) => {
290
+ const newState = {
291
+ ...initial,
292
+ indexValueMap: { ...initial.indexValueMap }
293
+ };
294
+ const mutationMap = {};
295
+ for (const syncd of syncds) {
296
+ const { version, keyId, snapshotMac } = syncd;
297
+ if (syncd.externalMutations) {
298
+ logger?.trace({ name, version }, "downloading external patch");
299
+ const ref = await downloadExternalPatch(syncd.externalMutations, options);
300
+ logger?.debug({ name, version, mutations: ref.mutations.length }, "downloaded external patch");
301
+ syncd.mutations?.push(...ref.mutations);
302
+ }
303
+ const patchVersion = toNumber(version.version);
304
+ newState.version = patchVersion;
305
+ const shouldMutate = typeof minimumVersionNumber === "undefined" || patchVersion > minimumVersionNumber;
306
+ const decodeResult = await decodeSyncdPatch(syncd, name, newState, getAppStateSyncKey, shouldMutate
307
+ ? mutation => {
308
+ const index = mutation.syncAction.index?.toString();
309
+ mutationMap[index] = mutation;
310
+ }
311
+ : () => { }, true);
312
+ newState.hash = decodeResult.hash;
313
+ newState.indexValueMap = decodeResult.indexValueMap;
283
314
  if (validateMacs) {
284
- const base64Key = Buffer.from(snapshot.keyId.id).toString('base64');
285
- const keyEnc = await getAppStateSyncKey(base64Key);
286
- if (!keyEnc) {
287
- throw new boom_1.Boom(`failed to find key "${base64Key}" to decode mutation`);
288
- }
289
- const result = mutationKeys(keyEnc.keyData);
290
- const computedSnapshotMac = generateSnapshotMac(newState.hash, newState.version, name, result.snapshotMacKey);
291
- if (Buffer.compare(snapshot.mac, computedSnapshotMac) !== 0) {
292
- throw new boom_1.Boom(`failed to verify LTHash at ${newState.version} of ${name} from snapshot`);
293
- }
294
- }
295
- return {
296
- state: newState,
297
- mutationMap
298
- };
315
+ const base64Key = Buffer.from(keyId.id).toString("base64");
316
+ const keyEnc = await getAppStateSyncKey(base64Key);
317
+ if (!keyEnc) {
318
+ throw new Boom(`failed to find key "${base64Key}" to decode mutation`);
319
+ }
320
+ const result = await mutationKeys(keyEnc.keyData);
321
+ const computedSnapshotMac = generateSnapshotMac(newState.hash, newState.version, name, result.snapshotMacKey);
322
+ if (Buffer.compare(snapshotMac, computedSnapshotMac) !== 0) {
323
+ throw new Boom(`failed to verify LTHash at ${newState.version} of ${name}`);
324
+ }
325
+ }
326
+ syncd.mutations = [];
327
+ }
328
+ return { state: newState, mutationMap };
299
329
  };
300
- exports.decodeSyncdSnapshot = decodeSyncdSnapshot;
301
- const decodePatches = async (name, syncds, initial, getAppStateSyncKey, options, minimumVersionNumber, logger, validateMacs = true) => {
302
- var _a;
303
- const newState = {
304
- ...initial,
305
- indexValueMap: { ...initial.indexValueMap }
306
- };
307
- const mutationMap = {};
308
- for (let i = 0; i < syncds.length; i++) {
309
- const syncd = syncds[i];
310
- const { version, keyId, snapshotMac } = syncd;
311
- if (syncd.externalMutations) {
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);
316
- }
317
- const patchVersion = (0, generics_1.toNumber)(version.version);
318
- newState.version = patchVersion;
319
- const shouldMutate = typeof minimumVersionNumber === 'undefined' || patchVersion > minimumVersionNumber;
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;
330
+ //=======================================================//
331
+ export const chatModificationToAppPatch = (mod, jid) => {
332
+ const OP = proto.SyncdMutation.SyncdOperation;
333
+ const getMessageRange = (lastMessages) => {
334
+ let messageRange;
335
+ if (Array.isArray(lastMessages)) {
336
+ const lastMsg = lastMessages[lastMessages.length - 1];
337
+ messageRange = {
338
+ lastMessageTimestamp: lastMsg?.messageTimestamp,
339
+ messages: lastMessages?.length
340
+ ? lastMessages.map(m => {
341
+ if (!m.key?.id || !m.key?.remoteJid) {
342
+ throw new Boom("Incomplete key", { statusCode: 400, data: m });
325
343
  }
326
- : (() => { }), true);
327
- newState.hash = decodeResult.hash;
328
- newState.indexValueMap = decodeResult.indexValueMap;
329
- if (validateMacs) {
330
- const base64Key = Buffer.from(keyId.id).toString('base64');
331
- const keyEnc = await getAppStateSyncKey(base64Key);
332
- if (!keyEnc) {
333
- throw new boom_1.Boom(`failed to find key "${base64Key}" to decode mutation`);
344
+ if (isJidGroup(m.key.remoteJid) && !m.key.fromMe && !m.key.participant) {
345
+ throw new Boom("Expected not from me message to have participant", { statusCode: 400, data: m });
334
346
  }
335
- const result = mutationKeys(keyEnc.keyData);
336
- const computedSnapshotMac = generateSnapshotMac(newState.hash, newState.version, name, result.snapshotMacKey);
337
- if (Buffer.compare(snapshotMac, computedSnapshotMac) !== 0) {
338
- throw new boom_1.Boom(`failed to verify LTHash at ${newState.version} of ${name}`);
347
+ if (!m.messageTimestamp || !toNumber(m.messageTimestamp)) {
348
+ throw new Boom("Missing timestamp in last message list", { statusCode: 400, data: m });
339
349
  }
340
- }
341
- // clear memory used up by the mutations
342
- syncd.mutations = [];
350
+ if (m.key.participant) {
351
+ m.key.participant = jidNormalizedUser(m.key.participant);
352
+ }
353
+ return m;
354
+ })
355
+ : undefined
356
+ };
343
357
  }
344
- return { state: newState, mutationMap };
345
- };
346
- exports.decodePatches = decodePatches;
347
- const chatModificationToAppPatch = (mod, jid) => {
348
- const OP = WAProto_1.proto.SyncdMutation.SyncdOperation;
349
- const getMessageRange = (lastMessages) => {
350
- let messageRange;
351
- if (Array.isArray(lastMessages)) {
352
- const lastMsg = lastMessages[lastMessages.length - 1];
353
- messageRange = {
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
371
- };
358
+ else {
359
+ messageRange = lastMessages;
360
+ }
361
+ return messageRange;
362
+ };
363
+ let patch;
364
+ if ("mute" in mod) {
365
+ patch = {
366
+ syncAction: {
367
+ muteAction: {
368
+ muted: !!mod.mute,
369
+ muteEndTimestamp: mod.mute || undefined
372
370
  }
373
- else {
374
- messageRange = lastMessages;
371
+ },
372
+ index: ["mute", jid],
373
+ type: "regular_high",
374
+ apiVersion: 2,
375
+ operation: OP.SET
376
+ };
377
+ }
378
+ else if ("archive" in mod) {
379
+ patch = {
380
+ syncAction: {
381
+ archiveChatAction: {
382
+ archived: !!mod.archive,
383
+ messageRange: getMessageRange(mod.lastMessages)
375
384
  }
376
- return messageRange;
385
+ },
386
+ index: ["archive", jid],
387
+ type: "regular_low",
388
+ apiVersion: 3,
389
+ operation: OP.SET
377
390
  };
378
- let patch;
379
- if ('mute' in mod) {
380
- patch = {
381
- syncAction: {
382
- muteAction: {
383
- muted: !!mod.mute,
384
- muteEndTimestamp: mod.mute || undefined
385
- }
386
- },
387
- index: ['mute', jid],
388
- type: 'regular_high',
389
- apiVersion: 2,
390
- operation: OP.SET
391
- };
392
- }
393
- else if ('archive' in mod) {
394
- patch = {
395
- syncAction: {
396
- archiveChatAction: {
397
- archived: !!mod.archive,
398
- messageRange: getMessageRange(mod.lastMessages)
399
- }
400
- },
401
- index: ['archive', jid],
402
- type: 'regular_low',
403
- apiVersion: 3,
404
- operation: OP.SET
405
- };
406
- }
407
- else if ('markRead' in mod) {
408
- patch = {
409
- syncAction: {
410
- markChatAsReadAction: {
411
- read: mod.markRead,
412
- messageRange: getMessageRange(mod.lastMessages)
413
- }
414
- },
415
- index: ['markChatAsRead', jid],
416
- type: 'regular_low',
417
- apiVersion: 3,
418
- operation: OP.SET
419
- };
420
- }
421
- else if ('clear' in mod) {
422
- if (mod.clear === 'all') {
423
- throw new boom_1.Boom('not supported');
391
+ }
392
+ else if ("markRead" in mod) {
393
+ patch = {
394
+ syncAction: {
395
+ markChatAsReadAction: {
396
+ read: mod.markRead,
397
+ messageRange: getMessageRange(mod.lastMessages)
424
398
  }
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
- };
399
+ },
400
+ index: ["markChatAsRead", jid],
401
+ type: "regular_low",
402
+ apiVersion: 3,
403
+ operation: OP.SET
404
+ };
405
+ }
406
+ else if ("deleteForMe" in mod) {
407
+ const { timestamp, key, deleteMedia } = mod.deleteForMe;
408
+ patch = {
409
+ syncAction: {
410
+ deleteMessageForMeAction: {
411
+ deleteMedia,
412
+ messageTimestamp: timestamp
439
413
  }
440
- }
441
- else if ('pin' in mod) {
442
- patch = {
443
- syncAction: {
444
- pinAction: {
445
- pinned: !!mod.pin
446
- }
447
- },
448
- index: ['pin_v1', jid],
449
- type: 'regular_low',
450
- apiVersion: 5,
451
- operation: OP.SET
452
- };
453
- }
454
- else if ('star' in mod) {
455
- const key = mod.star.messages[0];
456
- patch = {
457
- syncAction: {
458
- starAction: {
459
- starred: !!mod.star.star
460
- }
461
- },
462
- index: ['star', jid, key.id, key.fromMe ? '1' : '0', '0'],
463
- type: 'regular_low',
464
- apiVersion: 2,
465
- operation: OP.SET
466
- };
467
- }
468
- else if ('delete' in mod) {
469
- patch = {
470
- syncAction: {
471
- deleteChatAction: {
472
- messageRange: getMessageRange(mod.lastMessages),
473
- }
474
- },
475
- index: ['deleteChat', jid, '1'],
476
- type: 'regular_high',
477
- apiVersion: 6,
478
- operation: OP.SET
479
- };
480
- }
481
- else if ('pushNameSetting' in mod) {
482
- patch = {
483
- syncAction: {
484
- pushNameSetting: {
485
- name: mod.pushNameSetting
486
- }
487
- },
488
- index: ['setting_pushName'],
489
- type: 'critical_block',
490
- apiVersion: 1,
491
- operation: OP.SET,
492
- };
493
- }
494
- else if ('addChatLabel' in mod) {
495
- patch = {
496
- syncAction: {
497
- labelAssociationAction: {
498
- labeled: true,
499
- }
500
- },
501
- index: [LabelAssociation_1.LabelAssociationType.Chat, mod.addChatLabel.labelId, jid],
502
- type: 'regular',
503
- apiVersion: 3,
504
- operation: OP.SET,
505
- };
506
- } else if ("contact" in mod) {
414
+ },
415
+ index: ["deleteMessageForMe", jid, key.id, key.fromMe ? "1" : "0", "0"],
416
+ type: "regular_high",
417
+ apiVersion: 3,
418
+ operation: OP.SET
419
+ };
420
+ }
421
+ else if ("clear" in mod) {
422
+ patch = {
423
+ syncAction: {
424
+ clearChatAction: {
425
+ messageRange: getMessageRange(mod.lastMessages)
426
+ }
427
+ },
428
+ index: ["clearChat", jid, "1", "0"],
429
+ type: "regular_high",
430
+ apiVersion: 6,
431
+ operation: OP.SET
432
+ };
433
+ }
434
+ else if ("pin" in mod) {
435
+ patch = {
436
+ syncAction: {
437
+ pinAction: {
438
+ pinned: !!mod.pin
439
+ }
440
+ },
441
+ index: ["pin_v1", jid],
442
+ type: "regular_low",
443
+ apiVersion: 5,
444
+ operation: OP.SET
445
+ };
446
+ }
447
+ else if ("contact" in mod) {
507
448
  patch = {
508
449
  syncAction: {
509
450
  contactAction: mod.contact || {}
@@ -513,226 +454,299 @@ const chatModificationToAppPatch = (mod, jid) => {
513
454
  apiVersion: 2,
514
455
  operation: mod.contact ? OP.SET : OP.REMOVE
515
456
  };
516
- } else if ('removeChatLabel' in mod) {
517
- patch = {
518
- syncAction: {
519
- labelAssociationAction: {
520
- labeled: false,
521
- }
522
- },
523
- index: [LabelAssociation_1.LabelAssociationType.Chat, mod.removeChatLabel.labelId, jid],
524
- type: 'regular',
525
- apiVersion: 3,
526
- operation: OP.SET,
527
- };
528
- }
529
- else if ('addMessageLabel' in mod) {
530
- patch = {
531
- syncAction: {
532
- labelAssociationAction: {
533
- labeled: true,
534
- }
535
- },
536
- index: [
537
- LabelAssociation_1.LabelAssociationType.Message,
538
- mod.addMessageLabel.labelId,
539
- jid,
540
- mod.addMessageLabel.messageId,
541
- '0',
542
- '0'
543
- ],
544
- type: 'regular',
545
- apiVersion: 3,
546
- operation: OP.SET,
547
- };
548
- }
549
- else if ('removeMessageLabel' in mod) {
550
- patch = {
551
- syncAction: {
552
- labelAssociationAction: {
553
- labeled: false,
554
- }
555
- },
556
- index: [
557
- LabelAssociation_1.LabelAssociationType.Message,
558
- mod.removeMessageLabel.labelId,
559
- jid,
560
- mod.removeMessageLabel.messageId,
561
- '0',
562
- '0'
563
- ],
564
- type: 'regular',
565
- apiVersion: 3,
566
- operation: OP.SET,
567
- };
568
- }
569
- else {
570
- throw new boom_1.Boom('not supported');
571
- }
572
- patch.syncAction.timestamp = Date.now();
573
- return patch;
457
+ }
458
+ else if ("disableLinkPreviews" in mod) {
459
+ patch = {
460
+ syncAction: {
461
+ privacySettingDisableLinkPreviewsAction: mod.disableLinkPreviews || {}
462
+ },
463
+ index: ["setting_disableLinkPreviews"],
464
+ type: "regular",
465
+ apiVersion: 8,
466
+ operation: OP.SET
467
+ };
468
+ }
469
+ else if ("star" in mod) {
470
+ const key = mod.star.messages[0];
471
+ patch = {
472
+ syncAction: {
473
+ starAction: {
474
+ starred: !!mod.star.star
475
+ }
476
+ },
477
+ index: ["star", jid, key.id, key.fromMe ? "1" : "0", "0"],
478
+ type: "regular_low",
479
+ apiVersion: 2,
480
+ operation: OP.SET
481
+ };
482
+ }
483
+ else if ("delete" in mod) {
484
+ patch = {
485
+ syncAction: {
486
+ deleteChatAction: {
487
+ messageRange: getMessageRange(mod.lastMessages)
488
+ }
489
+ },
490
+ index: ["deleteChat", jid, "1"],
491
+ type: "regular_high",
492
+ apiVersion: 6,
493
+ operation: OP.SET
494
+ };
495
+ }
496
+ else if ("pushNameSetting" in mod) {
497
+ patch = {
498
+ syncAction: {
499
+ pushNameSetting: {
500
+ name: mod.pushNameSetting
501
+ }
502
+ },
503
+ index: ["setting_pushName"],
504
+ type: "critical_block",
505
+ apiVersion: 1,
506
+ operation: OP.SET
507
+ };
508
+ }
509
+ else if ("quickReply" in mod) {
510
+ patch = {
511
+ syncAction: {
512
+ quickReplyAction: {
513
+ count: 0,
514
+ deleted: mod.quickReply.deleted || false,
515
+ keywords: [],
516
+ message: mod.quickReply.message || "",
517
+ shortcut: mod.quickReply.shortcut || ""
518
+ }
519
+ },
520
+ index: ["quick_reply", mod.quickReply.timestamp || String(Math.floor(Date.now() / 1000))],
521
+ type: "regular",
522
+ apiVersion: 2,
523
+ operation: OP.SET
524
+ };
525
+ }
526
+ else if ("addLabel" in mod) {
527
+ patch = {
528
+ syncAction: {
529
+ labelEditAction: {
530
+ name: mod.addLabel.name,
531
+ color: mod.addLabel.color,
532
+ predefinedId: mod.addLabel.predefinedId,
533
+ deleted: mod.addLabel.deleted
534
+ }
535
+ },
536
+ index: ["label_edit", mod.addLabel.id],
537
+ type: "regular",
538
+ apiVersion: 3,
539
+ operation: OP.SET
540
+ };
541
+ }
542
+ else if ("addChatLabel" in mod) {
543
+ patch = {
544
+ syncAction: {
545
+ labelAssociationAction: {
546
+ labeled: true
547
+ }
548
+ },
549
+ index: [LabelAssociationType.Chat, mod.addChatLabel.labelId, jid],
550
+ type: "regular",
551
+ apiVersion: 3,
552
+ operation: OP.SET
553
+ };
554
+ }
555
+ else if ("removeChatLabel" in mod) {
556
+ patch = {
557
+ syncAction: {
558
+ labelAssociationAction: {
559
+ labeled: false
560
+ }
561
+ },
562
+ index: [LabelAssociationType.Chat, mod.removeChatLabel.labelId, jid],
563
+ type: "regular",
564
+ apiVersion: 3,
565
+ operation: OP.SET
566
+ };
567
+ }
568
+ else if ("addMessageLabel" in mod) {
569
+ patch = {
570
+ syncAction: {
571
+ labelAssociationAction: {
572
+ labeled: true
573
+ }
574
+ },
575
+ index: [LabelAssociationType.Message, mod.addMessageLabel.labelId, jid, mod.addMessageLabel.messageId, "0", "0"],
576
+ type: "regular",
577
+ apiVersion: 3,
578
+ operation: OP.SET
579
+ };
580
+ }
581
+ else if ("removeMessageLabel" in mod) {
582
+ patch = {
583
+ syncAction: {
584
+ labelAssociationAction: {
585
+ labeled: false
586
+ }
587
+ },
588
+ index: [
589
+ LabelAssociationType.Message,
590
+ mod.removeMessageLabel.labelId,
591
+ jid,
592
+ mod.removeMessageLabel.messageId,
593
+ "0",
594
+ "0"
595
+ ],
596
+ type: "regular",
597
+ apiVersion: 3,
598
+ operation: OP.SET
599
+ };
600
+ }
601
+ else {
602
+ throw new Boom("not supported");
603
+ }
604
+ patch.syncAction.timestamp = Date.now();
605
+ return patch;
574
606
  };
575
- exports.chatModificationToAppPatch = chatModificationToAppPatch;
576
- const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) => {
577
- var _a, _b, _c, _d;
578
- const isInitialSync = !!initialSyncOpts;
579
- const accountSettings = initialSyncOpts === null || initialSyncOpts === void 0 ? void 0 : initialSyncOpts.accountSettings;
580
- logger === null || logger === void 0 ? void 0 : logger.trace({ syncAction, initialSync: !!initialSyncOpts }, 'processing sync action');
581
- const { syncAction: { value: action }, index: [type, id, msgId, fromMe] } = syncAction;
582
- if (action === null || action === void 0 ? void 0 : action.muteAction) {
583
- ev.emit('chats.update', [
584
- {
585
- id,
586
- muteEndTime: ((_a = action.muteAction) === null || _a === void 0 ? void 0 : _a.muted)
587
- ? (0, generics_1.toNumber)(action.muteAction.muteEndTimestamp)
588
- : null,
589
- conditional: getChatUpdateConditional(id, undefined)
590
- }
591
- ]);
592
- }
593
- else if ((action === null || action === void 0 ? void 0 : action.archiveChatAction) || type === 'archive' || type === 'unarchive') {
594
- // okay so we've to do some annoying computation here
595
- // when we're initially syncing the app state
596
- // there are a few cases we need to handle
597
- // 1. if the account unarchiveChats setting is true
598
- // a. if the chat is archived, and no further messages have been received -- simple, keep archived
599
- // b. if the chat was archived, and the user received messages from the other person afterwards
600
- // then the chat should be marked unarchved --
601
- // we compare the timestamp of latest message from the other person to determine this
602
- // 2. if the account unarchiveChats setting is false -- then it doesn't matter,
603
- // it'll always take an app state action to mark in unarchived -- which we'll get anyway
604
- const archiveAction = action === null || action === void 0 ? void 0 : action.archiveChatAction;
605
- const isArchived = archiveAction
606
- ? archiveAction.archived
607
- : type === 'archive';
608
- // // basically we don't need to fire an "archive" update if the chat is being marked unarchvied
609
- // // this only applies for the initial sync
610
- // if(isInitialSync && !isArchived) {
611
- // isArchived = false
612
- // }
613
- const msgRange = !(accountSettings === null || accountSettings === void 0 ? void 0 : accountSettings.unarchiveChats) ? undefined : archiveAction === null || archiveAction === void 0 ? void 0 : archiveAction.messageRange;
614
- // logger?.debug({ chat: id, syncAction }, 'message range archive')
615
- ev.emit('chats.update', [{
616
- id,
617
- archived: isArchived,
618
- conditional: getChatUpdateConditional(id, msgRange)
619
- }]);
620
- }
621
- else if (action === null || action === void 0 ? void 0 : action.markChatAsReadAction) {
622
- const markReadAction = action.markChatAsReadAction;
623
- // basically we don't need to fire an "read" update if the chat is being marked as read
624
- // because the chat is read by default
625
- // this only applies for the initial sync
626
- const isNullUpdate = isInitialSync && markReadAction.read;
627
- ev.emit('chats.update', [{
628
- id,
629
- unreadCount: isNullUpdate ? null : !!(markReadAction === null || markReadAction === void 0 ? void 0 : markReadAction.read) ? 0 : -1,
630
- conditional: getChatUpdateConditional(id, markReadAction === null || markReadAction === void 0 ? void 0 : markReadAction.messageRange)
631
- }]);
632
- }
633
- else if ((action === null || action === void 0 ? void 0 : action.deleteMessageForMeAction) || type === 'deleteMessageForMe') {
634
- ev.emit('messages.delete', {
635
- keys: [
636
- {
637
- remoteJid: id,
638
- id: msgId,
639
- fromMe: fromMe === '1'
640
- }
641
- ]
642
- });
643
- }
644
- else if (action === null || action === void 0 ? void 0 : action.contactAction) {
645
- ev.emit('contacts.upsert', [{
646
- id: id,
647
- name: action.contactAction.fullName,
648
- lid: action.contactAction.lidJid || undefined,
649
- jid: (0, WABinary_1.isJidUser)(id) ? id : undefined
650
- }]);
651
- }
652
- else if (action === null || action === void 0 ? void 0 : action.pushNameSetting) {
653
- const name = (_b = action === null || action === void 0 ? void 0 : action.pushNameSetting) === null || _b === void 0 ? void 0 : _b.name;
654
- if (name && (me === null || me === void 0 ? void 0 : me.name) !== name) {
655
- ev.emit('creds.update', { me: { ...me, name } });
607
+ //=======================================================//
608
+ export const processSyncAction = (syncAction, ev, me, initialSyncOpts, logger) => {
609
+ const isInitialSync = !!initialSyncOpts;
610
+ const accountSettings = initialSyncOpts?.accountSettings;
611
+ logger?.trace({ syncAction, initialSync: !!initialSyncOpts }, "processing sync action");
612
+ const { syncAction: { value: action }, index: [type, id, msgId, fromMe] } = syncAction;
613
+ if (action?.muteAction) {
614
+ ev.emit("chats.update", [
615
+ {
616
+ id,
617
+ muteEndTime: action.muteAction?.muted ? toNumber(action.muteAction.muteEndTimestamp) : null,
618
+ conditional: getChatUpdateConditional(id, undefined)
619
+ }
620
+ ]);
621
+ }
622
+ else if (action?.archiveChatAction || type === "archive" || type === "unarchive") {
623
+ const archiveAction = action?.archiveChatAction;
624
+ const isArchived = archiveAction ? archiveAction.archived : type === "archive";
625
+ const msgRange = !accountSettings?.unarchiveChats ? undefined : archiveAction?.messageRange;
626
+ ev.emit("chats.update", [
627
+ {
628
+ id,
629
+ archived: isArchived,
630
+ conditional: getChatUpdateConditional(id, msgRange)
631
+ }
632
+ ]);
633
+ }
634
+ else if (action?.markChatAsReadAction) {
635
+ const markReadAction = action.markChatAsReadAction;
636
+ const isNullUpdate = isInitialSync && markReadAction.read;
637
+ ev.emit("chats.update", [
638
+ {
639
+ id,
640
+ unreadCount: isNullUpdate ? null : !!markReadAction?.read ? 0 : -1,
641
+ conditional: getChatUpdateConditional(id, markReadAction?.messageRange)
642
+ }
643
+ ]);
644
+ }
645
+ else if (action?.deleteMessageForMeAction || type === "deleteMessageForMe") {
646
+ ev.emit("messages.delete", {
647
+ keys: [
648
+ {
649
+ remoteJid: id,
650
+ id: msgId,
651
+ fromMe: fromMe === "1"
656
652
  }
657
- }
658
- else if (action === null || action === void 0 ? void 0 : action.pinAction) {
659
- ev.emit('chats.update', [{
660
- id,
661
- pinned: ((_c = action.pinAction) === null || _c === void 0 ? void 0 : _c.pinned) ? (0, generics_1.toNumber)(action.timestamp) : null,
662
- conditional: getChatUpdateConditional(id, undefined)
663
- }]);
664
- }
665
- else if (action === null || action === void 0 ? void 0 : action.unarchiveChatsSetting) {
666
- const unarchiveChats = !!action.unarchiveChatsSetting.unarchiveChats;
667
- ev.emit('creds.update', { accountSettings: { unarchiveChats } });
668
- logger === null || logger === void 0 ? void 0 : logger.info(`archive setting updated => '${action.unarchiveChatsSetting.unarchiveChats}'`);
669
- if (accountSettings) {
670
- accountSettings.unarchiveChats = unarchiveChats;
653
+ ]
654
+ });
655
+ }
656
+ else if (action?.contactAction) {
657
+ ev.emit("contacts.upsert", [
658
+ {
659
+ id: id,
660
+ name: action.contactAction.fullName,
661
+ lid: action.contactAction.lidJid || undefined,
662
+ jid: action.contactAction.pnJid || undefined
663
+ }
664
+ ]);
665
+ }
666
+ else if (action?.pushNameSetting) {
667
+ const name = action?.pushNameSetting?.name;
668
+ if (name && me?.name !== name) {
669
+ ev.emit("creds.update", { me: { ...me, name } });
670
+ }
671
+ }
672
+ else if (action?.pinAction) {
673
+ ev.emit("chats.update", [
674
+ {
675
+ id,
676
+ pinned: action.pinAction?.pinned ? toNumber(action.timestamp) : null,
677
+ conditional: getChatUpdateConditional(id, undefined)
678
+ }
679
+ ]);
680
+ }
681
+ else if (action?.unarchiveChatsSetting) {
682
+ const unarchiveChats = !!action.unarchiveChatsSetting.unarchiveChats;
683
+ ev.emit("creds.update", { accountSettings: { unarchiveChats } });
684
+ logger?.info(`archive setting updated => "${action.unarchiveChatsSetting.unarchiveChats}"`);
685
+ if (accountSettings) {
686
+ accountSettings.unarchiveChats = unarchiveChats;
687
+ }
688
+ }
689
+ else if (action?.starAction || type === "star") {
690
+ let starred = action?.starAction?.starred;
691
+ if (typeof starred !== "boolean") {
692
+ starred = syncAction.index[syncAction.index.length - 1] === "1";
693
+ }
694
+ ev.emit("messages.update", [
695
+ {
696
+ key: { remoteJid: id, id: msgId, fromMe: fromMe === "1" },
697
+ update: { starred }
698
+ }
699
+ ]);
700
+ }
701
+ else if (action?.deleteChatAction || type === "deleteChat") {
702
+ if (!isInitialSync) {
703
+ ev.emit("chats.delete", [id]);
704
+ }
705
+ }
706
+ else if (action?.labelEditAction) {
707
+ const { name, color, deleted, predefinedId } = action.labelEditAction;
708
+ ev.emit("labels.edit", {
709
+ id: id,
710
+ name: name,
711
+ color: color,
712
+ deleted: deleted,
713
+ predefinedId: predefinedId ? String(predefinedId) : undefined
714
+ });
715
+ }
716
+ else if (action?.labelAssociationAction) {
717
+ ev.emit("labels.association", {
718
+ type: action.labelAssociationAction.labeled ? "add" : "remove",
719
+ association: type === LabelAssociationType.Chat
720
+ ? {
721
+ type: LabelAssociationType.Chat,
722
+ chatId: syncAction.index[2],
723
+ labelId: syncAction.index[1]
671
724
  }
672
- }
673
- else if ((action === null || action === void 0 ? void 0 : action.starAction) || type === 'star') {
674
- let starred = (_d = action === null || action === void 0 ? void 0 : action.starAction) === null || _d === void 0 ? void 0 : _d.starred;
675
- if (typeof starred !== 'boolean') {
676
- starred = syncAction.index[syncAction.index.length - 1] === '1';
725
+ : {
726
+ type: LabelAssociationType.Message,
727
+ chatId: syncAction.index[2],
728
+ messageId: syncAction.index[3],
729
+ labelId: syncAction.index[1]
677
730
  }
678
- ev.emit('messages.update', [
679
- {
680
- key: { remoteJid: id, id: msgId, fromMe: fromMe === '1' },
681
- update: { starred }
682
- }
683
- ]);
684
- }
685
- else if ((action === null || action === void 0 ? void 0 : action.deleteChatAction) || type === 'deleteChat') {
686
- if (!isInitialSync) {
687
- ev.emit('chats.delete', [id]);
731
+ });
732
+ }
733
+ else {
734
+ logger?.debug({ syncAction, id }, "unprocessable update");
735
+ }
736
+ function getChatUpdateConditional(id, msgRange) {
737
+ return isInitialSync
738
+ ? data => {
739
+ const chat = data.historySets.chats[id] || data.chatUpserts[id];
740
+ if (chat) {
741
+ return msgRange ? isValidPatchBasedOnMessageRange(chat, msgRange) : true;
688
742
  }
689
- }
690
- else if (action === null || action === void 0 ? void 0 : action.labelEditAction) {
691
- const { name, color, deleted, predefinedId } = action.labelEditAction;
692
- ev.emit('labels.edit', {
693
- id,
694
- name: name,
695
- color: color,
696
- deleted: deleted,
697
- predefinedId: predefinedId ? String(predefinedId) : undefined
698
- });
699
- }
700
- else if (action === null || action === void 0 ? void 0 : action.labelAssociationAction) {
701
- ev.emit('labels.association', {
702
- type: action.labelAssociationAction.labeled
703
- ? 'add'
704
- : 'remove',
705
- association: type === LabelAssociation_1.LabelAssociationType.Chat
706
- ? {
707
- type: LabelAssociation_1.LabelAssociationType.Chat,
708
- chatId: syncAction.index[2],
709
- labelId: syncAction.index[1]
710
- }
711
- : {
712
- type: LabelAssociation_1.LabelAssociationType.Message,
713
- chatId: syncAction.index[2],
714
- messageId: syncAction.index[3],
715
- labelId: syncAction.index[1]
716
- }
717
- });
718
- }
719
- else {
720
- logger === null || logger === void 0 ? void 0 : logger.debug({ syncAction, id }, 'unprocessable update');
721
- }
722
- function getChatUpdateConditional(id, msgRange) {
723
- return isInitialSync
724
- ? (data) => {
725
- const chat = data.historySets.chats[id] || data.chatUpserts[id];
726
- if (chat) {
727
- return msgRange ? isValidPatchBasedOnMessageRange(chat, msgRange) : true;
728
- }
729
- }
730
- : undefined;
731
- }
732
- function isValidPatchBasedOnMessageRange(chat, msgRange) {
733
- const lastMsgTimestamp = Number((msgRange === null || msgRange === void 0 ? void 0 : msgRange.lastMessageTimestamp) || (msgRange === null || msgRange === void 0 ? void 0 : msgRange.lastSystemMessageTimestamp) || 0);
734
- const chatLastMsgTimestamp = Number((chat === null || chat === void 0 ? void 0 : chat.lastMessageRecvTimestamp) || 0);
735
- return lastMsgTimestamp >= chatLastMsgTimestamp;
736
- }
743
+ }
744
+ : undefined;
745
+ }
746
+ function isValidPatchBasedOnMessageRange(chat, msgRange) {
747
+ const lastMsgTimestamp = Number(msgRange?.lastMessageTimestamp || msgRange?.lastSystemMessageTimestamp || 0);
748
+ const chatLastMsgTimestamp = Number(chat?.lastMessageRecvTimestamp || 0);
749
+ return lastMsgTimestamp >= chatLastMsgTimestamp;
750
+ }
737
751
  };
738
- exports.processSyncAction = processSyncAction;
752
+ //=======================================================//