@sixcore/baileys 1.0.0

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