@neelegirl/baileys 1.5.3 → 1.5.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 (438) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +195 -195
  3. package/WAProto/GenerateStatics.sh +3 -0
  4. package/WAProto/WAProto.proto +478 -1153
  5. package/WAProto/fix-imports.js +29 -0
  6. package/WAProto/index.d.ts +2645 -51971
  7. package/WAProto/index.js +48649 -154035
  8. package/engine-requirements.js +10 -0
  9. package/lib/Defaults/index.d.ts +62 -78
  10. package/lib/Defaults/index.d.ts.map +1 -0
  11. package/lib/Defaults/index.js +115 -149
  12. package/lib/Defaults/index.js.map +1 -0
  13. package/lib/Signal/Group/ciphertext-message.d.ts +10 -0
  14. package/lib/Signal/Group/ciphertext-message.d.ts.map +1 -0
  15. package/lib/Signal/Group/ciphertext-message.js +12 -0
  16. package/lib/Signal/Group/ciphertext-message.js.map +1 -0
  17. package/lib/Signal/{WASignalGroup → Group}/group-session-builder.d.ts +11 -13
  18. package/lib/Signal/Group/group-session-builder.d.ts.map +1 -0
  19. package/lib/Signal/Group/group-session-builder.js +30 -0
  20. package/lib/Signal/Group/group-session-builder.js.map +1 -0
  21. package/lib/Signal/Group/group_cipher.d.ts +17 -0
  22. package/lib/Signal/Group/group_cipher.d.ts.map +1 -0
  23. package/lib/Signal/{WASignalGroup → Group}/group_cipher.js +41 -70
  24. package/lib/Signal/Group/group_cipher.js.map +1 -0
  25. package/lib/Signal/Group/index.d.ts +12 -0
  26. package/lib/Signal/Group/index.d.ts.map +1 -0
  27. package/lib/Signal/Group/index.js +12 -0
  28. package/lib/Signal/Group/index.js.map +1 -0
  29. package/lib/Signal/Group/keyhelper.d.ts +11 -0
  30. package/lib/Signal/Group/keyhelper.d.ts.map +1 -0
  31. package/lib/Signal/Group/keyhelper.js +18 -0
  32. package/lib/Signal/Group/keyhelper.js.map +1 -0
  33. package/lib/Signal/Group/sender-chain-key.d.ts +14 -0
  34. package/lib/Signal/Group/sender-chain-key.d.ts.map +1 -0
  35. package/lib/Signal/Group/sender-chain-key.js +26 -0
  36. package/lib/Signal/Group/sender-chain-key.js.map +1 -0
  37. package/lib/Signal/Group/sender-key-distribution-message.d.ts +17 -0
  38. package/lib/Signal/Group/sender-key-distribution-message.d.ts.map +1 -0
  39. package/lib/Signal/Group/sender-key-distribution-message.js +63 -0
  40. package/lib/Signal/Group/sender-key-distribution-message.js.map +1 -0
  41. package/lib/Signal/Group/sender-key-message.d.ts +19 -0
  42. package/lib/Signal/Group/sender-key-message.d.ts.map +1 -0
  43. package/lib/Signal/Group/sender-key-message.js +66 -0
  44. package/lib/Signal/Group/sender-key-message.js.map +1 -0
  45. package/lib/Signal/Group/sender-key-name.d.ts +18 -0
  46. package/lib/Signal/Group/sender-key-name.d.ts.map +1 -0
  47. package/lib/Signal/Group/sender-key-name.js +48 -0
  48. package/lib/Signal/Group/sender-key-name.js.map +1 -0
  49. package/lib/Signal/Group/sender-key-record.d.ts +31 -0
  50. package/lib/Signal/Group/sender-key-record.d.ts.map +1 -0
  51. package/lib/Signal/Group/sender-key-record.js +41 -0
  52. package/lib/Signal/Group/sender-key-record.js.map +1 -0
  53. package/lib/Signal/Group/sender-key-state.d.ts +39 -0
  54. package/lib/Signal/Group/sender-key-state.d.ts.map +1 -0
  55. package/lib/Signal/Group/sender-key-state.js +84 -0
  56. package/lib/Signal/Group/sender-key-state.js.map +1 -0
  57. package/lib/Signal/Group/sender-message-key.d.ts +12 -0
  58. package/lib/Signal/Group/sender-message-key.d.ts.map +1 -0
  59. package/lib/Signal/{WASignalGroup/sender_message_key.js → Group/sender-message-key.js} +26 -39
  60. package/lib/Signal/Group/sender-message-key.js.map +1 -0
  61. package/lib/Signal/libsignal.d.ts +5 -8
  62. package/lib/Signal/libsignal.d.ts.map +1 -0
  63. package/lib/Signal/libsignal.js +342 -391
  64. package/lib/Signal/libsignal.js.map +1 -0
  65. package/lib/Signal/lid-mapping.d.ts +23 -28
  66. package/lib/Signal/lid-mapping.d.ts.map +1 -0
  67. package/lib/Signal/lid-mapping.js +171 -184
  68. package/lib/Signal/lid-mapping.js.map +1 -0
  69. package/lib/Socket/Client/index.d.ts +3 -2
  70. package/lib/Socket/Client/index.d.ts.map +1 -0
  71. package/lib/Socket/Client/index.js +3 -22
  72. package/lib/Socket/Client/index.js.map +1 -0
  73. package/lib/Socket/Client/types.d.ts +15 -15
  74. package/lib/Socket/Client/types.d.ts.map +1 -0
  75. package/lib/Socket/Client/types.js +8 -15
  76. package/lib/Socket/Client/types.js.map +1 -0
  77. package/lib/Socket/Client/websocket.d.ts +12 -12
  78. package/lib/Socket/Client/websocket.d.ts.map +1 -0
  79. package/lib/Socket/Client/websocket.js +24 -36
  80. package/lib/Socket/Client/websocket.js.map +1 -0
  81. package/lib/Socket/business.d.ts +178 -177
  82. package/lib/Socket/business.d.ts.map +1 -0
  83. package/lib/Socket/business.js +81 -120
  84. package/lib/Socket/business.js.map +1 -0
  85. package/lib/Socket/chats.d.ts +93 -92
  86. package/lib/Socket/chats.d.ts.map +1 -0
  87. package/lib/Socket/chats.js +462 -618
  88. package/lib/Socket/chats.js.map +1 -0
  89. package/lib/Socket/communities.d.ts +244 -223
  90. package/lib/Socket/communities.d.ts.map +1 -0
  91. package/lib/Socket/communities.js +431 -433
  92. package/lib/Socket/communities.js.map +1 -0
  93. package/lib/Socket/groups.d.ts +118 -110
  94. package/lib/Socket/groups.d.ts.map +1 -0
  95. package/lib/Socket/groups.js +148 -181
  96. package/lib/Socket/groups.js.map +1 -0
  97. package/lib/Socket/index.d.ts +222 -182
  98. package/lib/Socket/index.d.ts.map +1 -0
  99. package/lib/Socket/index.js +12 -18
  100. package/lib/Socket/index.js.map +1 -0
  101. package/lib/Socket/messages-recv.d.ts +165 -166
  102. package/lib/Socket/messages-recv.d.ts.map +1 -0
  103. package/lib/Socket/messages-recv.js +753 -1031
  104. package/lib/Socket/messages-recv.js.map +1 -0
  105. package/lib/Socket/messages-send.d.ts +161 -157
  106. package/lib/Socket/messages-send.d.ts.map +1 -0
  107. package/lib/Socket/messages-send.js +564 -1396
  108. package/lib/Socket/messages-send.js.map +1 -0
  109. package/lib/Socket/mex.d.ts +3 -2
  110. package/lib/Socket/mex.d.ts.map +1 -0
  111. package/lib/Socket/mex.js +42 -47
  112. package/lib/Socket/mex.js.map +1 -0
  113. package/lib/Socket/newsletter.d.ts +139 -137
  114. package/lib/Socket/newsletter.d.ts.map +1 -0
  115. package/lib/Socket/newsletter.js +181 -295
  116. package/lib/Socket/newsletter.js.map +1 -0
  117. package/lib/Socket/socket.d.ts +42 -36
  118. package/lib/Socket/socket.d.ts.map +1 -0
  119. package/lib/Socket/socket.js +487 -557
  120. package/lib/Socket/socket.js.map +1 -0
  121. package/lib/Types/Auth.d.ts +87 -97
  122. package/lib/Types/Auth.d.ts.map +1 -0
  123. package/lib/Types/Auth.js +2 -3
  124. package/lib/Types/Auth.js.map +1 -0
  125. package/lib/Types/Bussines.d.ts +25 -0
  126. package/lib/Types/Bussines.d.ts.map +1 -0
  127. package/lib/Types/Bussines.js +2 -3
  128. package/lib/Types/Bussines.js.map +1 -0
  129. package/lib/Types/Call.d.ts +13 -13
  130. package/lib/Types/Call.d.ts.map +1 -0
  131. package/lib/Types/Call.js +2 -3
  132. package/lib/Types/Call.js.map +1 -0
  133. package/lib/Types/Chat.d.ts +77 -97
  134. package/lib/Types/Chat.d.ts.map +1 -0
  135. package/lib/Types/Chat.js +8 -9
  136. package/lib/Types/Chat.js.map +1 -0
  137. package/lib/Types/Contact.d.ts +10 -9
  138. package/lib/Types/Contact.d.ts.map +1 -0
  139. package/lib/Types/Contact.js +2 -3
  140. package/lib/Types/Contact.js.map +1 -0
  141. package/lib/Types/Events.d.ts +151 -175
  142. package/lib/Types/Events.d.ts.map +1 -0
  143. package/lib/Types/Events.js +2 -3
  144. package/lib/Types/Events.js.map +1 -0
  145. package/lib/Types/GroupMetadata.d.ts +49 -48
  146. package/lib/Types/GroupMetadata.d.ts.map +1 -0
  147. package/lib/Types/GroupMetadata.js +2 -3
  148. package/lib/Types/GroupMetadata.js.map +1 -0
  149. package/lib/Types/Label.d.ts +13 -14
  150. package/lib/Types/Label.d.ts.map +1 -0
  151. package/lib/Types/Label.js +24 -30
  152. package/lib/Types/Label.js.map +1 -0
  153. package/lib/Types/LabelAssociation.d.ts +15 -20
  154. package/lib/Types/LabelAssociation.d.ts.map +1 -0
  155. package/lib/Types/LabelAssociation.js +6 -12
  156. package/lib/Types/LabelAssociation.js.map +1 -0
  157. package/lib/Types/Message.d.ts +231 -412
  158. package/lib/Types/Message.d.ts.map +1 -0
  159. package/lib/Types/Message.js +11 -19
  160. package/lib/Types/Message.js.map +1 -0
  161. package/lib/Types/Newsletter.d.ts +130 -104
  162. package/lib/Types/Newsletter.d.ts.map +1 -0
  163. package/lib/Types/Newsletter.js +31 -40
  164. package/lib/Types/Newsletter.js.map +1 -0
  165. package/lib/Types/Product.d.ts +58 -71
  166. package/lib/Types/Product.d.ts.map +1 -0
  167. package/lib/Types/Product.js +2 -3
  168. package/lib/Types/Product.js.map +1 -0
  169. package/lib/Types/Signal.d.ts +60 -82
  170. package/lib/Types/Signal.d.ts.map +1 -0
  171. package/lib/Types/Signal.js +2 -3
  172. package/lib/Types/Signal.js.map +1 -0
  173. package/lib/Types/Socket.d.ts +73 -81
  174. package/lib/Types/Socket.d.ts.map +1 -0
  175. package/lib/Types/Socket.js +3 -3
  176. package/lib/Types/Socket.js.map +1 -0
  177. package/lib/Types/State.d.ts +17 -19
  178. package/lib/Types/State.d.ts.map +1 -0
  179. package/lib/Types/State.js +13 -14
  180. package/lib/Types/State.js.map +1 -0
  181. package/lib/Types/USync.d.ts +8 -8
  182. package/lib/Types/USync.d.ts.map +1 -0
  183. package/lib/Types/USync.js +2 -3
  184. package/lib/Types/USync.js.map +1 -0
  185. package/lib/Types/index.d.ts +47 -62
  186. package/lib/Types/index.d.ts.map +1 -0
  187. package/lib/Types/index.js +26 -50
  188. package/lib/Types/index.js.map +1 -0
  189. package/lib/Utils/auth-utils.d.ts +19 -21
  190. package/lib/Utils/auth-utils.d.ts.map +1 -0
  191. package/lib/Utils/auth-utils.js +257 -528
  192. package/lib/Utils/auth-utils.js.map +1 -0
  193. package/lib/Utils/baileys-event-stream.d.ts +17 -18
  194. package/lib/Utils/baileys-event-stream.d.ts.map +1 -0
  195. package/lib/Utils/baileys-event-stream.js +56 -70
  196. package/lib/Utils/baileys-event-stream.js.map +1 -0
  197. package/lib/Utils/browser-utils.d.ts +4 -0
  198. package/lib/Utils/browser-utils.d.ts.map +1 -0
  199. package/lib/Utils/browser-utils.js +28 -0
  200. package/lib/Utils/browser-utils.js.map +1 -0
  201. package/lib/Utils/business.d.ts +23 -29
  202. package/lib/Utils/business.d.ts.map +1 -0
  203. package/lib/Utils/business.js +231 -255
  204. package/lib/Utils/business.js.map +1 -0
  205. package/lib/Utils/chat-utils.d.ts +70 -82
  206. package/lib/Utils/chat-utils.d.ts.map +1 -0
  207. package/lib/Utils/chat-utils.js +763 -809
  208. package/lib/Utils/chat-utils.js.map +1 -0
  209. package/lib/Utils/crypto.d.ts +41 -56
  210. package/lib/Utils/crypto.d.ts.map +1 -0
  211. package/lib/Utils/crypto.js +142 -189
  212. package/lib/Utils/crypto.js.map +1 -0
  213. package/lib/Utils/decode-wa-message.d.ts +48 -53
  214. package/lib/Utils/decode-wa-message.d.ts.map +1 -0
  215. package/lib/Utils/decode-wa-message.js +279 -323
  216. package/lib/Utils/decode-wa-message.js.map +1 -0
  217. package/lib/Utils/event-buffer.d.ts +34 -39
  218. package/lib/Utils/event-buffer.d.ts.map +1 -0
  219. package/lib/Utils/event-buffer.js +548 -595
  220. package/lib/Utils/event-buffer.js.map +1 -0
  221. package/lib/Utils/generics.d.ts +90 -131
  222. package/lib/Utils/generics.d.ts.map +1 -0
  223. package/lib/Utils/generics.js +381 -630
  224. package/lib/Utils/generics.js.map +1 -0
  225. package/lib/Utils/history.d.ts +19 -23
  226. package/lib/Utils/history.d.ts.map +1 -0
  227. package/lib/Utils/history.js +84 -104
  228. package/lib/Utils/history.js.map +1 -0
  229. package/lib/Utils/index.d.ts +20 -20
  230. package/lib/Utils/index.d.ts.map +1 -0
  231. package/lib/Utils/index.js +19 -40
  232. package/lib/Utils/index.js.map +1 -0
  233. package/lib/Utils/link-preview.d.ts +21 -23
  234. package/lib/Utils/link-preview.d.ts.map +1 -0
  235. package/lib/Utils/link-preview.js +85 -120
  236. package/lib/Utils/link-preview.js.map +1 -0
  237. package/lib/Utils/logger.d.ts +12 -13
  238. package/lib/Utils/logger.d.ts.map +1 -0
  239. package/lib/Utils/logger.js +3 -7
  240. package/lib/Utils/logger.js.map +1 -0
  241. package/lib/Utils/lt-hash.d.ts +13 -14
  242. package/lib/Utils/lt-hash.d.ts.map +1 -0
  243. package/lib/Utils/lt-hash.js +48 -58
  244. package/lib/Utils/lt-hash.js.map +1 -0
  245. package/lib/Utils/make-mutex.d.ts +8 -9
  246. package/lib/Utils/make-mutex.d.ts.map +1 -0
  247. package/lib/Utils/make-mutex.js +40 -49
  248. package/lib/Utils/make-mutex.js.map +1 -0
  249. package/lib/Utils/message-retry-manager.d.ts +82 -88
  250. package/lib/Utils/message-retry-manager.d.ts.map +1 -0
  251. package/lib/Utils/message-retry-manager.js +149 -160
  252. package/lib/Utils/message-retry-manager.js.map +1 -0
  253. package/lib/Utils/messages-media.d.ts +114 -135
  254. package/lib/Utils/messages-media.d.ts.map +1 -0
  255. package/lib/Utils/messages-media.js +663 -869
  256. package/lib/Utils/messages-media.js.map +1 -0
  257. package/lib/Utils/messages.d.ts +76 -105
  258. package/lib/Utils/messages.d.ts.map +1 -0
  259. package/lib/Utils/messages.js +820 -1745
  260. package/lib/Utils/messages.js.map +1 -0
  261. package/lib/Utils/noise-handler.d.ts +20 -21
  262. package/lib/Utils/noise-handler.d.ts.map +1 -0
  263. package/lib/Utils/noise-handler.js +147 -165
  264. package/lib/Utils/noise-handler.js.map +1 -0
  265. package/lib/Utils/pre-key-manager.d.ts +28 -0
  266. package/lib/Utils/pre-key-manager.d.ts.map +1 -0
  267. package/lib/Utils/pre-key-manager.js +106 -0
  268. package/lib/Utils/pre-key-manager.js.map +1 -0
  269. package/lib/Utils/process-message.d.ts +42 -49
  270. package/lib/Utils/process-message.d.ts.map +1 -0
  271. package/lib/Utils/process-message.js +413 -427
  272. package/lib/Utils/process-message.js.map +1 -0
  273. package/lib/Utils/signal.d.ts +34 -42
  274. package/lib/Utils/signal.d.ts.map +1 -0
  275. package/lib/Utils/signal.js +159 -166
  276. package/lib/Utils/signal.js.map +1 -0
  277. package/lib/Utils/use-multi-file-auth-state.d.ts +13 -18
  278. package/lib/Utils/use-multi-file-auth-state.d.ts.map +1 -0
  279. package/lib/Utils/use-multi-file-auth-state.js +121 -238
  280. package/lib/Utils/use-multi-file-auth-state.js.map +1 -0
  281. package/lib/Utils/validate-connection.d.ts +11 -13
  282. package/lib/Utils/validate-connection.d.ts.map +1 -0
  283. package/lib/Utils/validate-connection.js +195 -220
  284. package/lib/Utils/validate-connection.js.map +1 -0
  285. package/lib/WABinary/constants.d.ts +28 -30
  286. package/lib/WABinary/constants.d.ts.map +1 -0
  287. package/lib/WABinary/constants.js +1301 -1316
  288. package/lib/WABinary/constants.js.map +1 -0
  289. package/lib/WABinary/decode.d.ts +7 -9
  290. package/lib/WABinary/decode.d.ts.map +1 -0
  291. package/lib/WABinary/decode.js +238 -288
  292. package/lib/WABinary/decode.js.map +1 -0
  293. package/lib/WABinary/encode.d.ts +3 -3
  294. package/lib/WABinary/encode.d.ts.map +1 -0
  295. package/lib/WABinary/encode.js +216 -265
  296. package/lib/WABinary/encode.js.map +1 -0
  297. package/lib/WABinary/generic-utils.d.ts +15 -28
  298. package/lib/WABinary/generic-utils.d.ts.map +1 -0
  299. package/lib/WABinary/generic-utils.js +102 -142
  300. package/lib/WABinary/generic-utils.js.map +1 -0
  301. package/lib/WABinary/index.d.ts +6 -5
  302. package/lib/WABinary/index.d.ts.map +1 -0
  303. package/lib/WABinary/index.js +6 -25
  304. package/lib/WABinary/index.js.map +1 -0
  305. package/lib/WABinary/jid-utils.d.ts +48 -58
  306. package/lib/WABinary/jid-utils.d.ts.map +1 -0
  307. package/lib/WABinary/jid-utils.js +96 -104
  308. package/lib/WABinary/jid-utils.js.map +1 -0
  309. package/lib/WABinary/types.d.ts +19 -22
  310. package/lib/WABinary/types.d.ts.map +1 -0
  311. package/lib/WABinary/types.js +2 -3
  312. package/lib/WABinary/types.js.map +1 -0
  313. package/lib/WAM/BinaryInfo.d.ts +9 -16
  314. package/lib/WAM/BinaryInfo.d.ts.map +1 -0
  315. package/lib/WAM/BinaryInfo.js +10 -17
  316. package/lib/WAM/BinaryInfo.js.map +1 -0
  317. package/lib/WAM/constants.d.ts +40 -47
  318. package/lib/WAM/constants.d.ts.map +1 -0
  319. package/lib/WAM/constants.js +22853 -15371
  320. package/lib/WAM/constants.js.map +1 -0
  321. package/lib/WAM/encode.d.ts +3 -3
  322. package/lib/WAM/encode.d.ts.map +1 -0
  323. package/lib/WAM/encode.js +150 -164
  324. package/lib/WAM/encode.js.map +1 -0
  325. package/lib/WAM/index.d.ts +4 -3
  326. package/lib/WAM/index.d.ts.map +1 -0
  327. package/lib/WAM/index.js +4 -23
  328. package/lib/WAM/index.js.map +1 -0
  329. package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts +9 -9
  330. package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts.map +1 -0
  331. package/lib/WAUSync/Protocols/USyncContactProtocol.js +12 -19
  332. package/lib/WAUSync/Protocols/USyncContactProtocol.js.map +1 -0
  333. package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts +19 -22
  334. package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts.map +1 -0
  335. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +18 -26
  336. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js.map +1 -0
  337. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts +11 -12
  338. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts.map +1 -0
  339. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +12 -20
  340. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js.map +1 -0
  341. package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts +11 -12
  342. package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts.map +1 -0
  343. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +16 -24
  344. package/lib/WAUSync/Protocols/USyncStatusProtocol.js.map +1 -0
  345. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts +26 -0
  346. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts.map +1 -0
  347. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +51 -0
  348. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js.map +1 -0
  349. package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts +10 -0
  350. package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts.map +1 -0
  351. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +29 -0
  352. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js.map +1 -0
  353. package/lib/WAUSync/Protocols/index.d.ts +5 -6
  354. package/lib/WAUSync/Protocols/index.d.ts.map +1 -0
  355. package/lib/WAUSync/Protocols/index.js +5 -26
  356. package/lib/WAUSync/Protocols/index.js.map +1 -0
  357. package/lib/WAUSync/USyncQuery.d.ts +26 -28
  358. package/lib/WAUSync/USyncQuery.d.ts.map +1 -0
  359. package/lib/WAUSync/USyncQuery.js +64 -62
  360. package/lib/WAUSync/USyncQuery.js.map +1 -0
  361. package/lib/WAUSync/USyncUser.d.ts +11 -10
  362. package/lib/WAUSync/USyncUser.d.ts.map +1 -0
  363. package/lib/WAUSync/USyncUser.js +12 -19
  364. package/lib/WAUSync/USyncUser.js.map +1 -0
  365. package/lib/WAUSync/index.d.ts +4 -3
  366. package/lib/WAUSync/index.d.ts.map +1 -0
  367. package/lib/WAUSync/index.js +4 -23
  368. package/lib/WAUSync/index.js.map +1 -0
  369. package/lib/index.d.ts +12 -13
  370. package/lib/index.d.ts.map +1 -0
  371. package/lib/index.js +11 -33
  372. package/lib/index.js.map +1 -0
  373. package/package.json +99 -96
  374. package/lib/Defaults/baileys-version.json +0 -3
  375. package/lib/Defaults/phonenumber-mcc.json +0 -223
  376. package/lib/Signal/WASignalGroup/GroupProtocol.js +0 -1909
  377. package/lib/Signal/WASignalGroup/ciphertext-message.d.ts +0 -9
  378. package/lib/Signal/WASignalGroup/ciphertext-message.js +0 -19
  379. package/lib/Signal/WASignalGroup/ciphertext_message.js +0 -16
  380. package/lib/Signal/WASignalGroup/generate-proto.sh +0 -1
  381. package/lib/Signal/WASignalGroup/group-session-builder.js +0 -72
  382. package/lib/Signal/WASignalGroup/group.proto +0 -42
  383. package/lib/Signal/WASignalGroup/group_cipher.d.ts +0 -19
  384. package/lib/Signal/WASignalGroup/group_session_builder.js +0 -46
  385. package/lib/Signal/WASignalGroup/index.d.ts +0 -11
  386. package/lib/Signal/WASignalGroup/index.js +0 -61
  387. package/lib/Signal/WASignalGroup/keyhelper.d.ts +0 -16
  388. package/lib/Signal/WASignalGroup/keyhelper.js +0 -66
  389. package/lib/Signal/WASignalGroup/protobufs.js +0 -3
  390. package/lib/Signal/WASignalGroup/queue_job.js +0 -69
  391. package/lib/Signal/WASignalGroup/readme.md +0 -6
  392. package/lib/Signal/WASignalGroup/sender-chain-key.d.ts +0 -14
  393. package/lib/Signal/WASignalGroup/sender-chain-key.js +0 -47
  394. package/lib/Signal/WASignalGroup/sender-key-distribution-message.d.ts +0 -17
  395. package/lib/Signal/WASignalGroup/sender-key-distribution-message.js +0 -71
  396. package/lib/Signal/WASignalGroup/sender-key-message.d.ts +0 -19
  397. package/lib/Signal/WASignalGroup/sender-key-message.js +0 -73
  398. package/lib/Signal/WASignalGroup/sender-key-name.d.ts +0 -19
  399. package/lib/Signal/WASignalGroup/sender-key-name.js +0 -59
  400. package/lib/Signal/WASignalGroup/sender-key-record.d.ts +0 -32
  401. package/lib/Signal/WASignalGroup/sender-key-record.js +0 -58
  402. package/lib/Signal/WASignalGroup/sender-key-state.d.ts +0 -44
  403. package/lib/Signal/WASignalGroup/sender-key-state.js +0 -147
  404. package/lib/Signal/WASignalGroup/sender-message-key.d.ts +0 -11
  405. package/lib/Signal/WASignalGroup/sender-message-key.js +0 -33
  406. package/lib/Signal/WASignalGroup/sender_chain_key.js +0 -50
  407. package/lib/Signal/WASignalGroup/sender_key_distribution_message.js +0 -78
  408. package/lib/Signal/WASignalGroup/sender_key_message.js +0 -92
  409. package/lib/Signal/WASignalGroup/sender_key_name.js +0 -70
  410. package/lib/Signal/WASignalGroup/sender_key_record.js +0 -56
  411. package/lib/Signal/WASignalGroup/sender_key_state.js +0 -129
  412. package/lib/Socket/Client/abstract-socket-client.d.ts +0 -15
  413. package/lib/Socket/Client/abstract-socket-client.js +0 -13
  414. package/lib/Socket/Client/mobile-socket-client.d.ts +0 -12
  415. package/lib/Socket/Client/mobile-socket-client.js +0 -65
  416. package/lib/Socket/usync.d.ts +0 -37
  417. package/lib/Socket/usync.js +0 -83
  418. package/lib/Store/index.d.ts +0 -4
  419. package/lib/Store/index.js +0 -24
  420. package/lib/Store/make-cache-manager-store.d.ts +0 -14
  421. package/lib/Store/make-cache-manager-store.js +0 -90
  422. package/lib/Store/make-in-memory-store.d.ts +0 -123
  423. package/lib/Store/make-in-memory-store.js +0 -429
  424. package/lib/Store/make-ordered-dictionary.d.ts +0 -12
  425. package/lib/Store/make-ordered-dictionary.js +0 -86
  426. package/lib/Store/object-repository.d.ts +0 -10
  427. package/lib/Store/object-repository.js +0 -31
  428. package/lib/Types/Bussiness.d.ts +0 -28
  429. package/lib/Types/MexUpdates.d.ts +0 -9
  430. package/lib/Types/MexUpdates.js +0 -18
  431. package/lib/Utils/use-mongo-file-auth-state.d.ts +0 -6
  432. package/lib/Utils/use-mongo-file-auth-state.js +0 -84
  433. package/lib/Utils/use-single-file-auth-state.d.ts +0 -13
  434. package/lib/Utils/use-single-file-auth-state.js +0 -80
  435. package/lib/WAUSync/Protocols/USyncBotProfileProtocol.d.ts +0 -28
  436. package/lib/WAUSync/Protocols/USyncBotProfileProtocol.js +0 -69
  437. package/lib/WAUSync/Protocols/USyncLIDProtocol.d.ts +0 -10
  438. package/lib/WAUSync/Protocols/USyncLIDProtocol.js +0 -38
@@ -1,1745 +1,820 @@
1
- "use strict"
2
-
3
- var __importDefault = (this && this.__importDefault) || function (mod) {
4
- return (mod && mod.__esModule) ? mod : { "default": mod }
5
- }
6
-
7
- Object.defineProperty(exports, "__esModule", { value: true })
8
-
9
- const boom_1 = require("@hapi/boom")
10
- const axios_1 = __importDefault(require("axios"))
11
- const crypto_1 = require("crypto")
12
- const fs_1 = require("fs")
13
- const WAProto_1 = require("../../WAProto")
14
- const Defaults_1 = require("../Defaults")
15
- const Types_1 = require("../Types")
16
- const WABinary_1 = require("../WABinary")
17
- const crypto_2 = require("./crypto")
18
- const generics_1 = require("./generics")
19
- const messages_media_1 = require("./messages-media")
20
-
21
- const MIMETYPE_MAP = {
22
- image: 'image/jpeg',
23
- video: 'video/mp4',
24
- document: 'application/pdf',
25
- audio: 'audio/ogg codecs=opus',
26
- sticker: 'image/webp',
27
- 'product-catalog-image': 'image/jpeg'
28
- }
29
-
30
- const MessageTypeProto = {
31
- 'image': Types_1.WAProto.Message.ImageMessage,
32
- 'video': Types_1.WAProto.Message.VideoMessage,
33
- 'audio': Types_1.WAProto.Message.AudioMessage,
34
- 'sticker': Types_1.WAProto.Message.StickerMessage,
35
- 'document': Types_1.WAProto.Message.DocumentMessage,
36
- }
37
-
38
- /**
39
- * Uses a regex to test whether the string contains a URL, and returns the URL if it does.
40
- * @param text eg. hello https://google.com
41
- * @returns the URL, eg. https://google.com
42
- */
43
- const extractUrlFromText = (text) => text.match(Defaults_1.URL_REGEX)?.[0]
44
-
45
- const generateLinkPreviewIfRequired = async (text, getUrlInfo, logger) => {
46
- const url = extractUrlFromText(text)
47
-
48
- if (!!getUrlInfo && url) {
49
- try {
50
- const urlInfo = await getUrlInfo(url)
51
- return urlInfo
52
- }
53
-
54
- catch (error) {
55
- logger?.warn({ trace: error.stack }, 'url generation failed')
56
- }
57
- }
58
- }
59
-
60
- const assertColor = async (color) => {
61
- let assertedColor
62
-
63
- if (typeof color === 'number') {
64
- assertedColor = color > 0 ? color : 0xffffffff + Number(color) + 1
65
- }
66
-
67
- else {
68
- let hex = color.trim().replace('#', '')
69
- if (hex.length <= 6) {
70
- hex = 'FF' + hex.padStart(6, '0')
71
- }
72
- assertedColor = parseInt(hex, 16)
73
- return assertedColor
74
- }
75
- }
76
-
77
- const prepareWAMessageMedia = async (message, options) => {
78
- const logger = options.logger
79
- let mediaType
80
-
81
- for (const key of Defaults_1.MEDIA_KEYS) {
82
- if (key in message) {
83
- mediaType = key
84
- }
85
- }
86
-
87
- if (!mediaType) {
88
- throw new boom_1.Boom('Invalid media type', { statusCode: 400 });
89
- }
90
-
91
- const uploadData = {
92
- ...message,
93
- media: message[mediaType]
94
- }
95
-
96
- delete uploadData[mediaType]
97
-
98
- // check if cacheable + generate cache key
99
- const cacheableKey = typeof uploadData.media === 'object' &&
100
- 'url' in uploadData.media &&
101
- !!uploadData.media.url &&
102
- !!options.mediaCache &&
103
- mediaType + ':' + uploadData.media.url.toString()
104
-
105
- if (mediaType === 'document' && !uploadData.fileName) {
106
- uploadData.fileName = 'file'
107
- }
108
-
109
- if (!uploadData.mimetype) {
110
- uploadData.mimetype = MIMETYPE_MAP[mediaType]
111
- }
112
-
113
- if (cacheableKey) {
114
- const mediaBuff = await options.mediaCache.get(cacheableKey)
115
- if (mediaBuff) {
116
- logger?.debug({ cacheableKey }, 'got media cache hit')
117
- const obj = Types_1.WAProto.Message.decode(mediaBuff)
118
- const key = `${mediaType}Message`
119
- Object.assign(obj[key], { ...uploadData, media: undefined })
120
- return obj
121
- }
122
- }
123
-
124
- const isNewsletter = !!options.jid && WABinary_1.isJidNewsletter(options.jid)
125
-
126
- if (isNewsletter) {
127
- logger?.info({ key: cacheableKey }, 'Preparing raw media for newsletter')
128
- const { filePath, fileSha256, fileLength } = await messages_media_1.getRawMediaUploadData(uploadData.media, options.mediaTypeOverride || mediaType, logger)
129
- const fileSha256B64 = fileSha256.toString('base64')
130
- const { mediaUrl, directPath } = await options.upload(filePath, {
131
- fileEncSha256B64: fileSha256B64,
132
- mediaType: mediaType,
133
- timeoutMs: options.mediaUploadTimeoutMs
134
- })
135
-
136
- await fs_1.promises.unlink(filePath)
137
-
138
- const obj = Types_1.WAProto.Message.fromObject({
139
- // todo: add more support here
140
- [`${mediaType}Message`]: MessageTypeProto[mediaType].fromObject({
141
- url: mediaUrl,
142
- directPath,
143
- fileSha256,
144
- fileLength,
145
- ...uploadData,
146
- media: undefined
147
- })
148
- })
149
-
150
- if (uploadData.ptv) {
151
- obj.ptvMessage = obj.videoMessage
152
- delete obj.videoMessage
153
- }
154
-
155
- if (cacheableKey) {
156
- logger?.debug({ cacheableKey }, 'set cache');
157
- await options.mediaCache.set(cacheableKey, Types_1.WAProto.Message.encode(obj).finish())
158
- }
159
-
160
- return obj
161
- }
162
-
163
- const requiresDurationComputation = mediaType === 'audio' && typeof uploadData.seconds === 'undefined'
164
- const requiresThumbnailComputation = (mediaType === 'image' || mediaType === 'video') && typeof uploadData['jpegThumbnail'] === 'undefined'
165
- const requiresWaveformProcessing = mediaType === 'audio' && uploadData.ptt === true
166
- const requiresAudioBackground = options.backgroundColor && mediaType === 'audio' && uploadData.ptt === true
167
- const requiresOriginalForSomeProcessing = requiresDurationComputation || requiresThumbnailComputation
168
-
169
- const { mediaKey, encFilePath, originalFilePath, fileEncSha256, fileSha256, fileLength } = await messages_media_1.encryptedStream(uploadData.media, options.mediaTypeOverride || mediaType, {
170
- logger,
171
- saveOriginalFileIfRequired: requiresOriginalForSomeProcessing,
172
- opts: options.options
173
- })
174
-
175
- const fileEncSha256B64 = fileEncSha256.toString('base64')
176
-
177
- const [{ mediaUrl, directPath }] = await Promise.all([
178
- (async () => {
179
- const result = await options.upload(encFilePath, {
180
- fileEncSha256B64,
181
- mediaType,
182
- timeoutMs: options.mediaUploadTimeoutMs
183
- })
184
- logger?.debug({ mediaType, cacheableKey }, 'uploaded media')
185
- return result
186
- })(),
187
-
188
- (async () => {
189
- try {
190
- if (requiresThumbnailComputation) {
191
- const { thumbnail, originalImageDimensions } = await messages_media_1.generateThumbnail(originalFilePath, mediaType, options)
192
- uploadData.jpegThumbnail = thumbnail
193
- if (!uploadData.width && originalImageDimensions) {
194
- uploadData.width = originalImageDimensions.width
195
- uploadData.height = originalImageDimensions.height
196
-
197
- logger?.debug('set dimensions')
198
- }
199
-
200
- logger?.debug('generated thumbnail');
201
- }
202
-
203
- if (requiresDurationComputation) {
204
- uploadData.seconds = await messages_media_1.getAudioDuration(originalFilePath)
205
- logger?.debug('computed audio duration')
206
- }
207
-
208
- if (requiresWaveformProcessing) {
209
- uploadData.waveform = await messages_media_1.getAudioWaveform(originalFilePath, logger)
210
- logger?.debug('processed waveform')
211
- }
212
-
213
- if (requiresAudioBackground) {
214
- uploadData.backgroundArgb = await assertColor(options.backgroundColor)
215
- logger?.debug('computed backgroundColor audio status')
216
- }
217
- }
218
- catch (error) {
219
- logger?.warn({ trace: error.stack }, 'failed to obtain extra info')
220
- }
221
- })()
222
- ]).finally(async () => {
223
- try {
224
- await fs_1.promises.unlink(encFilePath)
225
-
226
- if (originalFilePath) {
227
- await fs_1.promises.unlink(originalFilePath)
228
- }
229
-
230
- logger?.debug('removed tmp files')
231
- }
232
- catch (error) {
233
- logger?.warn('failed to remove tmp file')
234
- }
235
- })
236
-
237
- const obj = Types_1.WAProto.Message.fromObject({
238
- [`${mediaType}Message`]: MessageTypeProto[mediaType].fromObject({
239
- url: mediaUrl,
240
- directPath,
241
- mediaKey,
242
- fileEncSha256,
243
- fileSha256,
244
- fileLength,
245
- mediaKeyTimestamp: generics_1.unixTimestampSeconds(),
246
- ...uploadData,
247
- media: undefined
248
- })
249
- })
250
-
251
- if (uploadData.ptv) {
252
- obj.ptvMessage = obj.videoMessage
253
- delete obj.videoMessage
254
- }
255
-
256
- if (cacheableKey) {
257
- logger?.debug({ cacheableKey }, 'set cache');
258
- await options.mediaCache.set(cacheableKey, Types_1.WAProto.Message.encode(obj).finish())
259
- }
260
-
261
- return obj
262
- }
263
-
264
- const prepareAlbumMessageContent = async (jid, albums, options) => {
265
- let mediaHandle
266
- let mediaMsg
267
- let message = []
268
-
269
- const albumMsg = generateWAMessageFromContent(jid, {
270
- albumMessage: {
271
- expectedImageCount: albums.filter(item => 'image' in item).length,
272
- expectedVideoCount: albums.filter(item => 'video' in item).length
273
- }
274
- }, options)
275
-
276
- await options.baron.relayMessage(jid, albumMsg.message, {
277
- messageId: albumMsg.key.id
278
- })
279
-
280
- for (const i in albums) {
281
- const media = albums[i]
282
-
283
- if ('image' in media) {
284
- mediaMsg = await generateWAMessage(jid, {
285
- image: media.image,
286
- ...media,
287
- ...options
288
- }, {
289
- userJid: options.userJid,
290
- upload: async (encFilePath, opts) => {
291
- const up = await options.baron.waUploadToServer(encFilePath, { ...opts, newsletter: WABinary_1.isJidNewsletter(jid) })
292
- mediaHandle = up.handle
293
- return up
294
- },
295
- ...options
296
- })
297
- } else if ('video' in message) {
298
- mediaMsg = await generateWAMessage(jid, {
299
- video: media.video,
300
- ...media,
301
- ...options
302
- }, {
303
- userJid: options.userJid,
304
- upload: async (encFilePath, opts) => {
305
- const up = await options.baron.waUploadToServer(encFilePath, { ...opts, newsletter: WABinary_1.isJidNewsletter(jid) })
306
- mediaHandle = up.handle
307
- return up
308
- },
309
- ...options
310
- })
311
- }
312
-
313
- if (mediaMsg) {
314
- mediaMsg.message.messageContextInfo = {
315
- messageSecret: crypto_1.randomBytes(32),
316
- messageAssociation: {
317
- associationType: 1,
318
- parentMessageKey: albumMsg.key
319
- }
320
- }
321
- }
322
-
323
- message.push(mediaMsg)
324
- }
325
-
326
- return message
327
- }
328
-
329
- const prepareDisappearingMessageSettingContent = (expiration) => {
330
- const content = {
331
- ephemeralMessage: {
332
- message: {
333
- protocolMessage: {
334
- type: Types_1.WAProto.Message.ProtocolMessage.Type.EPHEMERAL_SETTING,
335
- ephemeralExpiration: expiration ? expiration : 0
336
- }
337
- }
338
- }
339
- }
340
-
341
- return Types_1.WAProto.Message.fromObject(content)
342
- }
343
-
344
- /**
345
- * Generate forwarded message content like WA does
346
- * @param message the message to forward
347
- * @param options.forceForward will show the message as forwarded even if it is from you
348
- */
349
- const generateForwardMessageContent = (message, forceForward) => {
350
- let content = message.message
351
-
352
- if (!content) {
353
- throw new boom_1.Boom('no content in message', { statusCode: 400 })
354
- }
355
-
356
- // hacky copy
357
- content = normalizeMessageContent(content)
358
- content = WAProto_1.proto.Message.decode(WAProto_1.proto.Message.encode(content).finish())
359
-
360
- let key = Object.keys(content)[0]
361
- let score = content[key].contextInfo?.forwardingScore || 0
362
-
363
- if (forceForward) score += forceForward ? forceForward : 1
364
-
365
- if (key === 'conversation') {
366
- content.extendedTextMessage = { text: content[key] }
367
- delete content.conversation
368
- key = 'extendedTextMessage'
369
- }
370
-
371
- if (score > 0) {
372
- content[key].contextInfo = { forwardingScore: score, isForwarded: true }
373
- }
374
-
375
- else {
376
- content[key].contextInfo = {}
377
- }
378
-
379
- return content
380
- }
381
-
382
- const generateWAMessageContent = async (message, options) => {
383
- let m = {}
384
- if ('text' in message) {
385
- const extContent = { text: message.text }
386
- let urlInfo = message.linkPreview
387
-
388
- if (typeof urlInfo === 'undefined') {
389
- urlInfo = await generateLinkPreviewIfRequired(message.text, options.getUrlInfo, options.logger)
390
- }
391
-
392
- if (urlInfo) {
393
- extContent.canonicalUrl = urlInfo['canonical-url']
394
- extContent.matchedText = urlInfo['matched-text']
395
- extContent.jpegThumbnail = urlInfo.jpegThumbnail
396
- extContent.description = urlInfo.description
397
- extContent.title = urlInfo.title
398
- extContent.previewType = 0
399
- const img = urlInfo.highQualityThumbnail
400
-
401
- if (img) {
402
- extContent.thumbnailDirectPath = img.directPath
403
- extContent.mediaKey = img.mediaKey
404
- extContent.mediaKeyTimestamp = img.mediaKeyTimestamp
405
- extContent.thumbnailWidth = img.width
406
- extContent.thumbnailHeight = img.height
407
- extContent.thumbnailSha256 = img.fileSha256
408
- extContent.thumbnailEncSha256 = img.fileEncSha256
409
- }
410
- }
411
-
412
- if (options.backgroundColor) {
413
- extContent.backgroundArgb = await assertColor(options.backgroundColor)
414
- }
415
-
416
- if (options.textColor) {
417
- extContent.textArgb = await assertColor(options.textColor)
418
- }
419
-
420
- if (options.font) {
421
- extContent.font = options.font
422
- }
423
-
424
- extContent.contextInfo = {
425
- ...(message.contextInfo || {}),
426
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
427
- }
428
-
429
- m.extendedTextMessage = extContent
430
- }
431
-
432
- else if ('contacts' in message) {
433
- const contactLen = message.contacts.contacts.length
434
-
435
- let contactMessage
436
-
437
- if (!contactLen) {
438
- throw new boom_1.Boom('require atleast 1 contact', { statusCode: 400 })
439
- }
440
-
441
- if (contactLen === 1) {
442
- contactMessage = {
443
- contactMessage: Types_1.WAProto.Message.ContactMessage.fromObject(message.contacts.contacts[0])
444
- }
445
- }
446
-
447
- else {
448
- contactMessage = {
449
- contactsArrayMessage: Types_1.WAProto.Message.ContactsArrayMessage.fromObject(message.contacts)
450
- }
451
- }
452
-
453
- const [type] = Object.keys(contactMessage)
454
-
455
- contactMessage[type].contextInfo = {
456
- ...(message.contextInfo || {}),
457
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
458
- }
459
-
460
- m = contactMessage
461
- }
462
-
463
- else if ('location' in message) {
464
- let locationMessage
465
-
466
- if (message.live) {
467
- locationMessage = {
468
- liveLocationMessage: Types_1.WAProto.Message.LiveLocationMessage.fromObject(message.location)
469
- }
470
- }
471
-
472
- else {
473
- locationMessage = {
474
- locationMessage: Types_1.WAProto.Message.LocationMessage.fromObject(message.location)
475
- }
476
- }
477
-
478
- const [type] = Object.keys(locationMessage)
479
-
480
- locationMessage[type].contextInfo = {
481
- ...(message.contextInfo || {}),
482
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
483
- }
484
-
485
- m = locationMessage
486
- }
487
-
488
- else if ('react' in message) {
489
- if (!message.react.senderTimestampMs) {
490
- message.react.senderTimestampMs = Date.now()
491
- }
492
-
493
- m.reactionMessage = Types_1.WAProto.Message.ReactionMessage.fromObject(message.react)
494
- }
495
-
496
- else if ('delete' in message) {
497
- m.protocolMessage = {
498
- key: message.delete,
499
- type: Types_1.WAProto.Message.ProtocolMessage.Type.REVOKE
500
- }
501
- }
502
-
503
- else if ('forward' in message) {
504
- const mess = generateForwardMessageContent(message.forward, message.force)
505
- const [type] = Object.keys(mess)
506
-
507
- mess[type].contextInfo = {
508
- ...(message.contextInfo || {}),
509
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
510
- }
511
-
512
- m = mess
513
- }
514
-
515
- else if ('disappearingMessagesInChat' in message) {
516
- const exp = typeof message.disappearingMessagesInChat === 'boolean' ?
517
- (message.disappearingMessagesInChat ? Defaults_1.WA_DEFAULT_EPHEMERAL : 0) :
518
- message.disappearingMessagesInChat
519
- m = prepareDisappearingMessageSettingContent(exp)
520
- }
521
-
522
- else if ('groupInvite' in message) {
523
- m.groupInviteMessage = {}
524
-
525
- m.groupInviteMessage.inviteCode = message.groupInvite.code
526
- m.groupInviteMessage.inviteExpiration = message.groupInvite.expiration
527
- m.groupInviteMessage.caption = message.groupInvite.caption
528
- m.groupInviteMessage.groupJid = message.groupInvite.jid
529
- m.groupInviteMessage.groupName = message.groupInvite.name
530
- m.groupInviteMessage.contextInfo = message.contextInfo
531
-
532
- if (options.getProfilePicUrl) {
533
- const pfpUrl = await options.getProfilePicUrl(message.groupInvite.jid)
534
- const { thumbnail } = await messages_media_1.generateThumbnail(pfpUrl, 'image')
535
- m.groupInviteMessage.jpegThumbnail = thumbnail
536
- }
537
-
538
- m.groupInviteMessage.contextInfo = {
539
- ...(message.contextInfo || {}),
540
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
541
- }
542
- }
543
-
544
- else if ('adminInvite' in message) {
545
- m.newsletterAdminInviteMessage = {}
546
-
547
- m.newsletterAdminInviteMessage.newsletterJid = message.adminInvite.jid
548
- m.newsletterAdminInviteMessage.newsletterName= message.adminInvite.name
549
- m.newsletterAdminInviteMessage.caption = message.adminInvite.caption
550
- m.newsletterAdminInviteMessage.inviteExpiration = message.adminInvite.expiration
551
- m.newsletterAdminInviteMessage.contextInfo = message.contextInfo
552
-
553
- if (options.getProfilePicUrl) {
554
- const pfpUrl = await options.getProfilePicUrl(message.adminInvite.jid)
555
- const { thumbnail } = await messages_media_1.generateThumbnail(pfpUrl, 'image')
556
- m.newsletterAdminInviteMessage.jpegThumbnail = thumbnail
557
- }
558
-
559
- m.newsletterAdminInviteMessage.contextInfo = {
560
- ...(message.contextInfo || {}),
561
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
562
- }
563
- }
564
-
565
- else if ('pin' in message) {
566
- m.pinInChatMessage = {}
567
- m.messageContextInfo = {}
568
-
569
- m.pinInChatMessage.key = message.pin.key
570
- m.pinInChatMessage.type = message.pin?.type || 1
571
- m.pinInChatMessage.senderTimestampMs = message.pin?.time || Date.now()
572
- m.messageContextInfo.messageAddOnDurationInSecs = message.pin.type === 1 ? message.pin.time || 86400 : 0
573
- m.messageContextInfo.messageAddOnExpiryType = WAProto_1.proto.MessageContextInfo.MessageAddonExpiryType.STATIC
574
- }
575
-
576
- else if ('keep' in message) {
577
- m.keepInChatMessage = {}
578
-
579
- m.keepInChatMessage.key = message.keep.key
580
- m.keepInChatMessage.keepType = message.keep?.type || 1
581
- m.keepInChatMessage.timestampMs = message.keep?.time || Date.now()
582
- }
583
-
584
- else if ('call' in message) {
585
- m.scheduledCallCreationMessage = {}
586
-
587
- m.scheduledCallCreationMessage.scheduledTimestampMs = message.call?.time || Date.now()
588
- m.scheduledCallCreationMessage.callType = message.call?.type || 1
589
- m.scheduledCallCreationMessage.title = message.call?.name || 'Call Creation'
590
-
591
- m.scheduledCallCreationMessage.contextInfo = {
592
- ...(message.contextInfo || {}),
593
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
594
- }
595
- }
596
-
597
- else if ('paymentInvite' in message) {
598
- m.messageContextInfo = {}
599
- m.paymentInviteMessage = {}
600
-
601
- m.paymentInviteMessage.expiryTimestamp = message.paymentInvite?.expiry || 0
602
- m.paymentInviteMessage.serviceType = message.paymentInvite?.type || 2
603
-
604
- m.paymentInviteMessage.contextInfo = {
605
- ...(message.contextInfo || {}),
606
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
607
- }
608
- }
609
-
610
- else if ('buttonReply' in message) {
611
- switch (message.type) {
612
- case 'list':
613
- m.listResponseMessage = {
614
- title: message.buttonReply.title,
615
- description: message.buttonReply.description,
616
- singleSelectReply: {
617
- selectedRowId: message.buttonReply.rowId
618
- },
619
- lisType: WAProto_1.proto.Message.ListResponseMessage.ListType.SINGLE_SELECT
620
- }
621
- break
622
- case 'template':
623
- m.templateButtonReplyMessage = {
624
- selectedDisplayText: message.buttonReply.displayText,
625
- selectedId: message.buttonReply.id,
626
- selectedIndex: message.buttonReply.index
627
- }
628
- break
629
- case 'plain':
630
- m.buttonsResponseMessage = {
631
- selectedButtonId: message.buttonReply.id,
632
- selectedDisplayText: message.buttonReply.displayText,
633
- type: WAProto_1.proto.Message.ButtonsResponseMessage.Type.DISPLAY_TEXT
634
- }
635
- break
636
- case 'interactive':
637
- m.interactiveResponseMessage = {
638
- body: {
639
- text: message.buttonReply.displayText,
640
- format: WAProto_1.proto.Message.InteractiveResponseMessage.Body.Format.EXTENSIONS_1
641
- },
642
- nativeFlowResponseMessage: {
643
- name: message.buttonReply.nativeFlows.name,
644
- paramsJson: message.buttonReply.nativeFlows.paramsJson,
645
- version: message.buttonReply.nativeFlows.version
646
- }
647
- }
648
- break
649
- }
650
- }
651
-
652
- else if ('ptv' in message && message.ptv) {
653
- const { videoMessage } = await prepareWAMessageMedia({ video: message.video }, options)
654
-
655
- m.ptvMessage = videoMessage
656
- }
657
-
658
- else if ('album' in message) {
659
- const imageMessages = message.album.filter(item => 'image' in item)
660
- const videoMessages = message.album.filter(item => 'video' in item)
661
-
662
- m.albumMessage = Types_1.WAProto.Message.AlbumMessage.fromObject({
663
- expectedImageCount: imageMessages.length,
664
- expectedVideoCount: videoMessages.length
665
- })
666
- }
667
-
668
- else if ('order' in message) {
669
- m.orderMessage = Types_1.WAProto.Message.OrderMessage.fromObject(message.order)
670
-
671
- m.orderMessage.contextInfo = {
672
- ...(message.contextInfo || {}),
673
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
674
- }
675
- }
676
-
677
- else if ('event' in message) {
678
- m.eventMessage = Types_1.WAProto.Message.EventMessage.fromObject(message.event)
679
-
680
- if (!message.event.startTime) {
681
- m.eventMessage.startTime = generics_1.unixTimestampSeconds() + 86400
682
- }
683
-
684
- if (options.getCallLink && message.event.call) {
685
- const link = await options.getCallLink(message.event.call, m.eventMessage.startTime)
686
- m.eventMessage.joinLink = link
687
- }
688
-
689
-
690
- m.eventMessage.contextInfo = {
691
- ...(message.contextInfo || {}),
692
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
693
- }
694
- }
695
-
696
- else if ('product' in message) {
697
- const { imageMessage } = await prepareWAMessageMedia({ image: message.product.productImage }, options)
698
-
699
- m.productMessage = Types_1.WAProto.Message.ProductMessage.fromObject({
700
- ...message,
701
- product: {
702
- ...message.product,
703
- productImage: imageMessage,
704
- }
705
- })
706
-
707
- m.productMessage.contextInfo = {
708
- ...(message.contextInfo || {}),
709
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
710
- }
711
- }
712
-
713
- else if ('pollResult' in message) {
714
- if (!Array.isArray(message.pollResult.values)) {
715
- throw new boom_1.Boom('Invalid pollResult values', { statusCode: 400 })
716
- }
717
-
718
- const pollResultSnapshotMessage = {
719
- name: message.pollResult.name,
720
- pollVotes: message.pollResult.values.map(([optionName, optionVoteCount]) => ({
721
- optionName,
722
- optionVoteCount
723
- }))
724
- }
725
-
726
- pollResultSnapshotMessage.contextInfo = {
727
- ...(message.contextInfo || {}),
728
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
729
- }
730
-
731
- m.pollResultSnapshotMessage = pollResultSnapshotMessage
732
- }
733
-
734
- else if ('poll' in message) {
735
- if (!Array.isArray(message.poll.values)) {
736
- throw new boom_1.Boom('Invalid poll values', { statusCode: 400 })
737
- }
738
-
739
- if (message.poll.selectableCount < 0
740
- || message.poll.selectableCount > message.poll.values.length) {
741
- throw new boom_1.Boom(`poll.selectableCount in poll should be >= 0 and <= ${message.poll.values.length}`, { statusCode: 400 })
742
- }
743
-
744
- const pollCreationMessage = {
745
- name: message.poll.name,
746
- selectableOptionsCount: message.poll?.selectableCount || 0,
747
- options: message.poll.values.map(optionName => ({ optionName })),
748
- }
749
-
750
- pollCreationMessage.contextInfo = {
751
- ...(message.contextInfo || {}),
752
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
753
- }
754
-
755
- if(message.poll?.toAnnouncementGroup) {
756
- m.pollCreationMessageV2 = pollCreationMessage
757
- } else {
758
- if(message.poll.selectableCount > 0) {
759
- m.pollCreationMessageV3 = pollCreationMessage
760
- } else {
761
- m.pollCreationMessage = pollCreationMessage
762
- }
763
- }
764
- }
765
-
766
- else if ('payment' in message) {
767
- const requestPaymentMessage = {
768
- amount: {
769
- currencyCode: message.payment?.currency || 'IDR',
770
- offset: message.payment?.offset || 0,
771
- value: message.payment?.amount || 999999999
772
- },
773
- expiryTimestamp: message.payment?.expiry || 0,
774
- amount1000: message.payment?.amount || 999999999 * 1000,
775
- currencyCodeIso4217: message.payment?.currency || 'IDR',
776
- requestFrom: message.payment?.from || '0@s.whatsapp.net',
777
- noteMessage: {
778
- extendedTextMessage: {
779
- text: message.payment?.note || 'Notes'
780
- }
781
- },
782
- background: {
783
- placeholderArgb: message.payment?.image?.placeholderArgb || 4278190080,
784
- textArgb: message.payment?.image?.textArgb || 4294967295,
785
- subtextArgb: message.payment?.image?.subtextArgb || 4294967295,
786
- type: 1
787
- }
788
- }
789
-
790
- requestPaymentMessage.noteMessage.extendedTextMessage.contextInfo = {
791
- ...(message.contextInfo || {}),
792
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
793
- }
794
-
795
- m.requestPaymentMessage = requestPaymentMessage
796
- }
797
-
798
- else if ('stickerPack' in message) {
799
- const { stickers, cover, name, publisher, packId, description } = message.stickerPack
800
-
801
- const { zip } = require('fflate')
802
-
803
- const stickerData = {}
804
- const stickerPromises = stickers.map(async (s, i) => {
805
- const { stream } = await messages_media_1.getStream(s.sticker)
806
- const buffer = await messages_media_1.toBuffer(stream)
807
- const hash = crypto_2.sha256(buffer).toString('base64url')
808
- const fileName = `${i.toString().padStart(2, '0')}_${hash}.webp`
809
- stickerData[fileName] = [new Uint8Array(buffer), { level: 0 }]
810
-
811
- return {
812
- fileName,
813
- mimetype: 'image/webp',
814
- isAnimated: s.isAnimated || false,
815
- isLottie: s.isLottie || false,
816
- emojis: s.emojis || [],
817
- accessibilityLabel: s.accessibilityLabel || ''
818
- }
819
- })
820
-
821
- const stickerMetadata = await Promise.all(stickerPromises)
822
-
823
- const zipBuffer = await new Promise((resolve, reject) => {
824
- zip(stickerData, (err, data) => {
825
- if (err) {
826
- reject(err)
827
- } else {
828
- resolve(Buffer.from(data))
829
- }
830
- })
831
- })
832
-
833
- const coverBuffer = await messages_media_1.toBuffer((await messages_media_1.getStream(cover)).stream)
834
-
835
- const [stickerPackUpload, coverUpload] = await Promise.all([
836
- messages_media_1.encryptedStream(zipBuffer, 'sticker-pack', { logger: options.logger, opts: options.options }),
837
- prepareWAMessageMedia({ image: coverBuffer }, { ...options, mediaTypeOverride: 'image' })
838
- ])
839
-
840
- const stickerPackUploadResult = await options.upload(stickerPackUpload.encFilePath, {
841
- fileEncSha256B64: stickerPackUpload.fileEncSha256.toString('base64'),
842
- mediaType: 'sticker-pack',
843
- timeoutMs: options.mediaUploadTimeoutMs
844
- })
845
-
846
-
847
- const coverImage = coverUpload.imageMessage
848
- const imageDataHash = crypto_2.sha256(coverBuffer).toString('base64')
849
- const stickerPackId = packId || generics_1.generateMessageID()
850
-
851
- m.stickerPackMessage = {
852
- name,
853
- publisher,
854
- stickerPackId,
855
- packDescription: description,
856
- stickerPackOrigin: WAProto_1.proto.Message.StickerPackMessage.StickerPackOrigin.THIRD_PARTY,
857
- stickerPackSize: stickerPackUpload.fileLength,
858
- stickers: stickerMetadata,
859
- fileSha256: stickerPackUpload.fileSha256,
860
- fileEncSha256: stickerPackUpload.fileEncSha256,
861
- mediaKey: stickerPackUpload.mediaKey,
862
- directPath: stickerPackUploadResult.directPath,
863
- fileLength: stickerPackUpload.fileLength,
864
- mediaKeyTimestamp: generics_1.unixTimestampSeconds(),
865
- trayIconFileName: `${stickerPackId}.png`,
866
- imageDataHash,
867
- thumbnailDirectPath: coverImage.directPath,
868
- thumbnailFileSha256: coverImage.fileSha256,
869
- thumbnailFileEncSha256: coverImage.fileEncSha256,
870
- thumbnailHeight: coverImage.height,
871
- thumbnailWidth: coverImage.width
872
- }
873
-
874
- m.stickerPackMessage.contextInfo = {
875
- ...(message.contextInfo || {}),
876
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
877
- }
878
- }
879
-
880
- else if ('sharePhoneNumber' in message) {
881
- m.protocolMessage = {
882
- type: Types_1.WAProto.Message.ProtocolMessage.Type.SHARE_PHONE_NUMBER
883
- }
884
- }
885
-
886
- else if ('requestPhoneNumber' in message) {
887
- m.requestPhoneNumberMessage = {}
888
- }
889
-
890
- else {
891
- const mess = await prepareWAMessageMedia(message, options)
892
- const [type] = Object.keys(mess)
893
-
894
- mess[type].contextInfo = {
895
- ...(message.contextInfo || {}),
896
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
897
- }
898
-
899
- m = mess
900
- }
901
-
902
- if ('sections' in message && !!message.sections) {
903
- const listMessage = {
904
- title: message.title,
905
- buttonText: message.buttonText,
906
- footerText: message.footer,
907
- description: message.text,
908
- sections: message.sections,
909
- listType: WAProto_1.proto.Message.ListMessage.ListType.SINGLE_SELECT
910
- }
911
-
912
- listMessage.contextInfo = {
913
- ...(message.contextInfo || {}),
914
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
915
- }
916
-
917
- m = { listMessage }
918
- }
919
-
920
- else if ('productList' in message && !!message.productList) {
921
- const thumbnail = message.thumbnail ? await messages_media_1.generateThumbnail(message.thumbnail, 'image') : null
922
-
923
- const listMessage = {
924
- title: message.title,
925
- buttonText: message.buttonText,
926
- footerText: message.footer,
927
- description: message.text,
928
- productListInfo: {
929
- productSections: message.productList,
930
- headerImage: {
931
- productId: message.productList[0].products[0].productId,
932
- jpegThumbnail: thumbnail?.thumbnail || null
933
- },
934
- businessOwnerJid: message.businessOwnerJid
935
- },
936
- listType: WAProto_1.proto.Message.ListMessage.ListType.PRODUCT_LIST
937
- }
938
-
939
- listMessage.contextInfo = {
940
- ...(message.contextInfo || {}),
941
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
942
- }
943
-
944
- m = { listMessage }
945
- }
946
-
947
- else if ('buttons' in message && !!message.buttons) {
948
- const buttonsMessage = {
949
- buttons: message.buttons.map(b => ({ ...b, type: WAProto_1.proto.Message.ButtonsMessage.Button.Type.RESPONSE }))
950
- }
951
-
952
- if ('text' in message) {
953
- buttonsMessage.contentText = message.text
954
- buttonsMessage.headerType = WAProto_1.proto.Message.ButtonsMessage.HeaderType.EMPTY
955
- }
956
-
957
- else {
958
- if ('caption' in message) {
959
- buttonsMessage.contentText = message.caption
960
- }
961
-
962
- const type = Object.keys(m)[0].replace('Message', '').toUpperCase()
963
-
964
- buttonsMessage.headerType = WAProto_1.proto.Message.ButtonsMessage.HeaderType[type]
965
-
966
- Object.assign(buttonsMessage, m)
967
- }
968
-
969
- if ('footer' in message && !!message.footer) {
970
- buttonsMessage.footerText = message.footer
971
- }
972
-
973
- if ('title' in message && !!message.title) {
974
- buttonsMessage.text = message.title
975
- buttonsMessage.headerType = WAProto_1.proto.Message.ButtonsMessage.HeaderType.TEXT
976
- }
977
-
978
- buttonsMessage.contextInfo = {
979
- ...(message.contextInfo || {}),
980
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
981
- }
982
-
983
- m = { buttonsMessage }
984
- }
985
-
986
- else if ('templateButtons' in message && !!message.templateButtons) {
987
- const hydratedTemplate = {
988
- hydratedButtons: message.templateButtons
989
- }
990
-
991
- if ('text' in message) {
992
- hydratedTemplate.hydratedContentText = message.text
993
- }
994
-
995
- else {
996
- if ('caption' in message) {
997
- hydratedTemplate.hydratedContentText = message.caption
998
- }
999
-
1000
- Object.assign(msg, m)
1001
- }
1002
-
1003
- if ('footer' in message && !!message.footer) {
1004
- hydratedTemplate.hydratedFooterText = message.footer
1005
- }
1006
-
1007
- hydratedTemplate.contextInfo = {
1008
- ...(message.contextInfo || {}),
1009
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
1010
- }
1011
-
1012
- m = { templateMessage: { hydratedTemplate }}
1013
- }
1014
-
1015
- else if ('interactiveButtons' in message && !!message.interactiveButtons) {
1016
- const interactiveMessage = {
1017
- nativeFlowMessage: {
1018
- buttons: message.interactiveButtons
1019
- }
1020
- }
1021
-
1022
- if ('text' in message) {
1023
- interactiveMessage.body = {
1024
- text: message.text
1025
- },
1026
- interactiveMessage.header = {
1027
- title: message.title,
1028
- subtitle: message.subtitle,
1029
- hasMediaAttachment: false
1030
- }
1031
- }
1032
-
1033
- else {
1034
- if ('caption' in message) {
1035
- interactiveMessage.body = {
1036
- text: message.caption
1037
- }
1038
-
1039
- interactiveMessage.header = {
1040
- title: message.title,
1041
- subtitle: message.subtitle,
1042
- hasMediaAttachment: message.hasMediaAttachment ? message.hasMediaAttachment : false,
1043
- ...Object.assign(interactiveMessage, m)
1044
- }
1045
- }
1046
- }
1047
-
1048
- if ('footer' in message && !!message.footer) {
1049
- interactiveMessage.footer = {
1050
- text: message.footer
1051
- }
1052
- }
1053
-
1054
- interactiveMessage.contextInfo = {
1055
- ...(message.contextInfo || {}),
1056
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
1057
- }
1058
-
1059
- m = { interactiveMessage }
1060
- }
1061
-
1062
- else if ('shop' in message && !!message.shop) {
1063
- const interactiveMessage = {
1064
- shopStorefrontMessage: {
1065
- surface: message.shop.surface,
1066
- id: message.shop.id
1067
- }
1068
- }
1069
-
1070
- if ('text' in message) {
1071
- interactiveMessage.body = {
1072
- text: message.text
1073
- },
1074
- interactiveMessage.header = {
1075
- title: message.title,
1076
- subtitle: message.subtitle,
1077
- hasMediaAttachment: false
1078
- }
1079
- }
1080
-
1081
- else {
1082
- if ('caption' in message) {
1083
- interactiveMessage.body = {
1084
- text: message.caption
1085
- }
1086
-
1087
- interactiveMessage.header = {
1088
- title: message.title,
1089
- subtitle: message.subtitle,
1090
- hasMediaAttachment: message.hasMediaAttachment ? message.hasMediaAttachment : false,
1091
- ...Object.assign(interactiveMessage, m)
1092
- }
1093
- }
1094
- }
1095
-
1096
- if ('footer' in message && !!message.footer) {
1097
- interactiveMessage.footer = {
1098
- text: message.footer
1099
- }
1100
- }
1101
-
1102
- interactiveMessage.contextInfo = {
1103
- ...(message.contextInfo || {}),
1104
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
1105
- }
1106
-
1107
- m = { interactiveMessage }
1108
- }
1109
-
1110
- else if ('collection' in message && !!message.collection) {
1111
- const interactiveMessage = {
1112
- collectionMessage: {
1113
- bizJid: message.collection.bizJid,
1114
- id: message.collection.id,
1115
- messageVersion: message?.collection?.version
1116
- }
1117
- }
1118
-
1119
- if ('text' in message) {
1120
- interactiveMessage.body = {
1121
- text: message.text
1122
- },
1123
- interactiveMessage.header = {
1124
- title: message.title,
1125
- subtitle: message.subtitle,
1126
- hasMediaAttachment: false
1127
- }
1128
- }
1129
-
1130
- else {
1131
- if ('caption' in message) {
1132
- interactiveMessage.body = {
1133
- text: message.caption
1134
- }
1135
- interactiveMessage.header = {
1136
- title: message.title,
1137
- subtitle: message.subtitle,
1138
- hasMediaAttachment: message.hasMediaAttachment ? message.hasMediaAttachment : false,
1139
- ...Object.assign(interactiveMessage, m)
1140
- }
1141
- }
1142
- }
1143
-
1144
- if ('footer' in message && !message.footer) {
1145
- interactiveMessage.footer = {
1146
- text: message.footer
1147
- }
1148
- }
1149
-
1150
- interactiveMessage.contextInfo = {
1151
- ...(message.contextInfo || {}),
1152
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
1153
- }
1154
-
1155
- m = { interactiveMessage }
1156
- }
1157
-
1158
- else if ('cards' in message && !!message.cards) {
1159
- const slides = await Promise.all(message.cards.map(async (slide) => {
1160
- const { image, video, product, title, body, footer, buttons } = slide
1161
- let header
1162
-
1163
- if (product) {
1164
- const { imageMessage } = await prepareWAMessageMedia({ image: product.productImage, ...options }, options)
1165
- header = {
1166
- productMessage: {
1167
- product: {
1168
- ...product,
1169
- productImage: imageMessage,
1170
- },
1171
- ...slide
1172
- }
1173
- }
1174
- }
1175
-
1176
- else if (image) {
1177
- header = await prepareWAMessageMedia({ image: image, ...options }, options)
1178
- }
1179
-
1180
- else if (video) {
1181
- header = await prepareWAMessageMedia({ video: video, ...options }, options)
1182
- }
1183
-
1184
- const msg = {
1185
- header: {
1186
- title,
1187
- hasMediaAttachment: true,
1188
- ...header
1189
- },
1190
- body: {
1191
- text: body
1192
- },
1193
- footer: {
1194
- text: footer
1195
- },
1196
- nativeFlowMessage: {
1197
- buttons,
1198
- }
1199
- }
1200
-
1201
- return msg
1202
- }))
1203
-
1204
- const interactiveMessage = {
1205
- carouselMessage: {
1206
- cards: slides
1207
- }
1208
- }
1209
-
1210
- if ('text' in message) {
1211
- interactiveMessage.body = {
1212
- text: message.text
1213
- },
1214
- interactiveMessage.header = {
1215
- title: message.title,
1216
- subtitle: message.subtitle,
1217
- hasMediaAttachment: false
1218
- }
1219
- }
1220
-
1221
- if ('footer' in message && !!message.footer) {
1222
- interactiveMessage.footer = {
1223
- text: message.footer
1224
- }
1225
- }
1226
-
1227
- interactiveMessage.contextInfo = {
1228
- ...(message.contextInfo || {}),
1229
- ...(message.mentions ? { mentionedJid: message.mentions } : {})
1230
- }
1231
-
1232
- m = { interactiveMessage }
1233
- }
1234
-
1235
- if ('ephemeral' in message && !!message.ephemeral) {
1236
- m = { ephemeralMessage: { message: m } }
1237
- }
1238
-
1239
- if ('viewOnce' in message && !!message.viewOnce) {
1240
- m = { viewOnceMessageV2: { message: m } }
1241
- }
1242
-
1243
- if ('viewOnceExt' in message && !!message.viewOnceExt) {
1244
- m = { viewOnceMessageV2Extension: { message: m } }
1245
- }
1246
-
1247
- if ('edit' in message) {
1248
- m = {
1249
- protocolMessage: {
1250
- key: message.edit,
1251
- editedMessage: m,
1252
- timestampMs: Date.now(),
1253
- type: Types_1.WAProto.Message.ProtocolMessage.Type.MESSAGE_EDIT
1254
- }
1255
- }
1256
- }
1257
-
1258
- return Types_1.WAProto.Message.fromObject(m)
1259
- }
1260
-
1261
- const generateWAMessageFromContent = (jid, message, options) => {
1262
- if (!options.timestamp) {
1263
- options.timestamp = new Date()
1264
- }
1265
-
1266
- const innerMessage = normalizeMessageContent(message)
1267
- const key = getContentType(innerMessage)
1268
- const timestamp = generics_1.unixTimestampSeconds(options.timestamp)
1269
- const threadId = []
1270
- const { quoted, userJid } = options
1271
-
1272
- if (quoted && !WABinary_1.isJidNewsletter(jid)) {
1273
- const participant = quoted.key.fromMe
1274
- ? userJid
1275
- : quoted.participant || quoted.key.participant || quoted.key.remoteJid
1276
-
1277
- let quotedMsg = normalizeMessageContent(quoted.message)
1278
- const msgType = getContentType(quotedMsg)
1279
-
1280
- quotedMsg = WAProto_1.proto.Message.fromObject({ [msgType]: quotedMsg[msgType] })
1281
-
1282
- const quotedContent = quotedMsg[msgType]
1283
-
1284
- if (typeof quotedContent === 'object' && quotedContent && 'contextInfo' in quotedContent) {
1285
- delete quotedContent.contextInfo
1286
- }
1287
-
1288
- let requestPayment
1289
-
1290
- if (key === 'requestPaymentMessage') {
1291
- if (innerMessage?.requestPaymentMessage && innerMessage?.requestPaymentMessage?.noteMessage?.extendedTextMessage) {
1292
- requestPayment = innerMessage?.requestPaymentMessage?.noteMessage?.extendedTextMessage
1293
- } else if (innerMessage?.requestPaymentMessage && innerMessage?.requestPaymentMessage?.noteMessage?.stickerMessage) {
1294
- requestPayment = innerMessage.requestPaymentMessage?.noteMessage?.stickerMessage
1295
- }
1296
- }
1297
-
1298
- const contextInfo = (key === 'requestPaymentMessage' ? requestPayment?.contextInfo : innerMessage[key].contextInfo) || {}
1299
-
1300
- contextInfo.participant = WABinary_1.jidNormalizedUser(participant)
1301
- contextInfo.stanzaId = quoted.key.id
1302
- contextInfo.quotedMessage = quotedMsg
1303
-
1304
- if (jid !== quoted.key.remoteJid) {
1305
- contextInfo.remoteJid = quoted.key.remoteJid
1306
- }
1307
-
1308
- if (contextInfo.quotedMessage) {
1309
- contextInfo.quotedType = WAProto_1.proto.ContextInfo.QuotedType.EXPLICIT
1310
- }
1311
-
1312
- if (contextInfo.quotedMessage && WABinary_1.isJidGroup(jid)) {
1313
- threadId.push({
1314
- threadType: WAProto_1.proto.ThreadID.ThreadType.VIEW_REPLIES,
1315
- threadKey: {
1316
- remoteJid: quoted?.key?.remoteJid,
1317
- fromMe: quoted?.key?.fromMe,
1318
- id: generics_1.generateMessageID(),
1319
- ...(quoted?.key?.fromMe ? {} : { participant: quoted?.key?.participant })
1320
- }
1321
- })
1322
- }
1323
-
1324
- if (key === 'requestPaymentMessage' && requestPayment) {
1325
- requestPayment.contextInfo = contextInfo
1326
- } else {
1327
- innerMessage[key].contextInfo = contextInfo
1328
- }
1329
- }
1330
-
1331
- if (key !== 'protocolMessage' &&
1332
- key !== 'ephemeralMessage' &&
1333
- !WABinary_1.isJidNewsletter(jid)) {
1334
- message.messageContextInfo = {
1335
- threadId: threadId.length > 0 ? threadId : [],
1336
- messageSecret: crypto_1.randomBytes(32),
1337
- ...message.messageContextInfo
1338
- }
1339
- innerMessage[key].contextInfo = {
1340
- ...(innerMessage[key].contextInfo || {}),
1341
- expiration: options.ephemeralExpiration ? options.ephemeralExpiration : 0
1342
- }
1343
- }
1344
-
1345
- message = Types_1.WAProto.Message.fromObject(message)
1346
-
1347
- const messageJSON = {
1348
- key: {
1349
- remoteJid: jid,
1350
- fromMe: true,
1351
- id: options?.messageId || generics_1.generateMessageID()
1352
- },
1353
- message: message,
1354
- messageTimestamp: timestamp,
1355
- messageStubParameters: [],
1356
- participant: WABinary_1.isJidGroup(jid) || WABinary_1.isJidStatusBroadcast(jid) ? userJid : undefined,
1357
- status: Types_1.WAMessageStatus.PENDING
1358
- }
1359
-
1360
- return Types_1.WAProto.WebMessageInfo.fromObject(messageJSON)
1361
- }
1362
-
1363
- const generateWAMessage = async (jid, content, options) => {
1364
- options.logger = options?.logger?.child({ msgId: options.messageId })
1365
-
1366
- return generateWAMessageFromContent(jid, await generateWAMessageContent(content, { newsletter: WABinary_1.isJidNewsletter(jid), ...options }), options)
1367
- }
1368
-
1369
- const getContentType = (content) => {
1370
- if (content) {
1371
- const keys = Object.keys(content)
1372
- const key = keys.find(k => (k === 'conversation' || k.endsWith('Message') || k.endsWith('V2') || k.endsWith('V3') || k.endsWith('V4') || k.endsWith('V5')) && k !== 'senderKeyDistributionMessage' && k !== 'messageContextInfo')
1373
-
1374
- return key
1375
- }
1376
- }
1377
-
1378
- /**
1379
- * Normalizes ephemeral, view once messages to regular message content
1380
- * Eg. image messages in ephemeral messages, in view once messages etc.
1381
- * @param content
1382
- * @returns
1383
- */
1384
- const normalizeMessageContent = (content) => {
1385
- if (!content) {
1386
- return undefined
1387
- }
1388
-
1389
- for (let i = 0; i < 5; i++) {
1390
- const inner = getFutureProofMessage(content)
1391
- if (!inner) {
1392
- break
1393
- }
1394
-
1395
- content = inner.message
1396
- }
1397
-
1398
- return content
1399
-
1400
- function getFutureProofMessage(message) {
1401
- return (
1402
- (message?.editedMessage)
1403
- || (message?.statusAddYours)
1404
- || (message?.botTaskMessage)
1405
- || (message?.eventCoverImage)
1406
- || (message?.questionMessage)
1407
- || (message?.viewOnceMessage)
1408
- || (message?.botInvokeMessage)
1409
- || (message?.ephemeralMessage)
1410
- || (message?.limitSharingMessage)
1411
- || (message?.viewOnceMessageV2)
1412
- || (message?.lottieStickerMessage)
1413
- || (message?.groupStatusMessage)
1414
- || (message?.questionReplyMessage)
1415
- || (message?.botForwardedMessage)
1416
- || (message?.statusMentionMessage)
1417
- || (message?.groupStatusMessageV2)
1418
- || (message?.pollCreationMessageV4)
1419
- || (message?.pollCreationMessageV5)
1420
- || (message?.associatedChildMessage)
1421
- || (message?.groupMentionedMessage)
1422
- || (message?.groupStatusMentionMessage)
1423
- || (message?.viewOnceMessageV2Extension)
1424
- || (message?.documentWithCaptionMessage)
1425
- || (message?.pollCreationOptionImageMessage))
1426
- }
1427
- }
1428
-
1429
- /**
1430
- * Extract the true message content from a message
1431
- * Eg. extracts the inner message from a disappearing message/view once message
1432
- */
1433
- const extractMessageContent = (content) => {
1434
- const extractFromButtonsMessage = (msg) => {
1435
- const header = typeof msg.header === 'object' && msg.header !== null
1436
-
1437
- if (header ? msg.header?.imageMessage : msg.imageMessage) {
1438
- return { imageMessage: header ? msg.header.imageMessage : msg.imageMessage }
1439
- }
1440
-
1441
- else if (header ? msg.header?.documentMessage : msg.documentMessage) {
1442
- return { documentMessage: header ? msg.header.documentMessage : msg.documentMessage }
1443
- }
1444
-
1445
- else if (header ? msg.header?.videoMessage : msg.videoMessage) {
1446
- return { videoMessage: header ? msg.header.videoMessage: msg.videoMessage }
1447
- }
1448
-
1449
- else if (header ? msg.header?.locationMessage : msg.locationMessage) {
1450
- return { locationMessage: header ? msg.header.locationMessage : msg.locationMessage }
1451
- }
1452
-
1453
- else if (header ? msg.header?.productMessage : msg.productMessage) {
1454
- return { productMessage: header ? msg.header.productMessage : msg.productMessage }
1455
- }
1456
-
1457
- else {
1458
- return {
1459
- conversation: 'contentText' in msg
1460
- ? msg.contentText
1461
- : ('hydratedContentText' in msg ? msg.hydratedContentText : 'body' in msg ? msg.body.text : '')
1462
- }
1463
- }
1464
- }
1465
-
1466
- content = normalizeMessageContent(content)
1467
-
1468
- if (content?.buttonsMessage) {
1469
- return extractFromButtonsMessage(content.buttonsMessage)
1470
- }
1471
-
1472
- if (content?.interactiveMessage) {
1473
- return extractFromButtonsMessage(content.interactiveMessage)
1474
- }
1475
-
1476
- if (content?.templateMessage?.interactiveMessageTemplate) {
1477
- return extractFromButtonsMessage(content?.templateMessage?.interactiveMessageTemplate)
1478
- }
1479
-
1480
- if (content?.templateMessage?.hydratedFourRowTemplate) {
1481
- return extractFromButtonsMessage(content?.templateMessage?.hydratedFourRowTemplate)
1482
- }
1483
-
1484
- if (content?.templateMessage?.hydratedTemplate) {
1485
- return extractFromButtonsMessage(content?.templateMessage?.hydratedTemplate)
1486
- }
1487
-
1488
- if (content?.templateMessage?.fourRowTemplate) {
1489
- return extractFromButtonsMessage(content?.templateMessage?.fourRowTemplate)
1490
- }
1491
-
1492
- return content
1493
- }
1494
-
1495
- /**
1496
- * Returns the device predicted by message ID
1497
- */
1498
- const getDevice = (id) => /^3A.{18}$/.test(id) ? 'ios' :
1499
- /^3E.{20}$/.test(id) ? 'web' :
1500
- /^(.{21}|.{32})$/.test(id) ? 'android' :
1501
- /^(3F|.{18}$)/.test(id) ? 'desktop' :
1502
- 'baileys'
1503
-
1504
- /** Upserts a receipt in the message */
1505
- const updateMessageWithReceipt = (msg, receipt) => {
1506
- msg.userReceipt = msg.userReceipt || []
1507
- const recp = msg.userReceipt.find(m => m.userJid === receipt.userJid)
1508
-
1509
- if (recp) {
1510
- Object.assign(recp, receipt)
1511
- }
1512
-
1513
- else {
1514
- msg.userReceipt.push(receipt)
1515
- }
1516
- }
1517
-
1518
- /** Update the message with a new reaction */
1519
- const updateMessageWithReaction = (msg, reaction) => {
1520
- const authorID = generics_1.getKeyAuthor(reaction.key)
1521
- const reactions = (msg.reactions || [])
1522
- .filter(r => generics_1.getKeyAuthor(r.key) !== authorID)
1523
-
1524
- reaction.text = reaction.text || ''
1525
- reactions.push(reaction)
1526
- msg.reactions = reactions
1527
- }
1528
-
1529
- /** Update the message with a new poll update */
1530
- const updateMessageWithPollUpdate = (msg, update) => {
1531
- const authorID = generics_1.getKeyAuthor(update.pollUpdateMessageKey)
1532
- const votes = (msg.pollUpdates || [])
1533
- .filter(r => generics_1.getKeyAuthor(r.pollUpdateMessageKey) !== authorID)
1534
-
1535
- if (update.vote?.selectedOptions?.length) {
1536
- votes.push(update)
1537
- }
1538
-
1539
- msg.pollUpdates = votes
1540
- }
1541
-
1542
- /**
1543
- * Aggregates all poll updates in a poll.
1544
- * @param msg the poll creation message
1545
- * @param meId your jid
1546
- * @returns A list of options & their voters
1547
- */
1548
- function getAggregateVotesInPollMessage({ message, pollUpdates }, meId) {
1549
- message = normalizeMessageContent(message)
1550
-
1551
- const opts = message?.pollCreationMessage?.options || message?.pollCreationMessageV2?.options || message?.pollCreationMessageV3?.options || []
1552
-
1553
- const voteHashMap = opts.reduce((acc, opt) => {
1554
- const hash = crypto_2.sha256(Buffer.from(opt.optionName || '')).toString()
1555
- acc[hash] = {
1556
- name: opt.optionName || '',
1557
- voters: []
1558
- }
1559
-
1560
- return acc
1561
- }, {})
1562
-
1563
- for (const update of pollUpdates || []) {
1564
- const { vote } = update
1565
-
1566
- if (!vote) {
1567
- continue
1568
- }
1569
-
1570
- for (const option of vote.selectedOptions || []) {
1571
- const hash = option.toString()
1572
- let data = voteHashMap[hash]
1573
-
1574
- if (!data) {
1575
- voteHashMap[hash] = {
1576
- name: 'Unknown',
1577
- voters: []
1578
- }
1579
-
1580
- data = voteHashMap[hash]
1581
- }
1582
-
1583
- voteHashMap[hash].voters.push(generics_1.getKeyAuthor(update.pollUpdateMessageKey, meId))
1584
- }
1585
- }
1586
-
1587
- return Object.values(voteHashMap)
1588
- }
1589
-
1590
- /** Given a list of message keys, aggregates them by chat & sender. Useful for sending read receipts in bulk */
1591
- const aggregateMessageKeysNotFromMe = (keys) => {
1592
- const keyMap = {}
1593
-
1594
- for (const { remoteJid, id, participant, fromMe } of keys) {
1595
- if (!fromMe) {
1596
- const uqKey = `${remoteJid}:${participant || ''}`
1597
-
1598
- if (!keyMap[uqKey]) {
1599
- keyMap[uqKey] = {
1600
- jid: remoteJid,
1601
- participant: participant,
1602
- messageIds: []
1603
- }
1604
- }
1605
-
1606
- keyMap[uqKey].messageIds.push(id)
1607
- }
1608
- }
1609
-
1610
- return Object.values(keyMap)
1611
- }
1612
-
1613
- const REUPLOAD_REQUIRED_STATUS = [410, 404]
1614
-
1615
- /**
1616
- * Downloads the given message. Throws an error if it's not a media message
1617
- */
1618
- const downloadMediaMessage = async (message, type, options, ctx) => {
1619
- const result = await downloadMsg().catch(async (error) => {
1620
- if (ctx && axios_1.default.isAxiosError(error) && // check if the message requires a reupload
1621
- REUPLOAD_REQUIRED_STATUS.includes(error.response?.status)) {
1622
- ctx.logger.info({ key: message.key }, 'sending reupload media request...')
1623
-
1624
- // request reupload
1625
- message = await ctx.reuploadRequest(message)
1626
-
1627
- const result = await downloadMsg()
1628
-
1629
- return result
1630
- }
1631
-
1632
- throw error
1633
- })
1634
-
1635
- return result
1636
-
1637
- async function downloadMsg() {
1638
- const mContent = extractMessageContent(message.message)
1639
-
1640
- if (!mContent) {
1641
- throw new boom_1.Boom('No message present', { statusCode: 400, data: message })
1642
- }
1643
-
1644
- const contentType = getContentType(mContent)
1645
- let mediaType = contentType?.replace('Message', '')
1646
- const media = contentType === 'productMessage' ? mContent[contentType]?.product?.productImage : mContent[contentType]
1647
-
1648
- if (!media || typeof media !== 'object' || (!('url' in media) && !('thumbnailDirectPath' in media))) {
1649
- throw new boom_1.Boom(`"${contentType}" message is not a media message`)
1650
- }
1651
-
1652
- let download
1653
-
1654
- if ('thumbnailDirectPath' in media && !('url' in media)) {
1655
- download = {
1656
- directPath: media.thumbnailDirectPath,
1657
- mediaKey: media.mediaKey
1658
- }
1659
- mediaType = 'thumbnail-link'
1660
- }
1661
-
1662
- else {
1663
- download = media
1664
- }
1665
-
1666
- const stream = await messages_media_1.downloadContentFromMessage(download, mediaType, options)
1667
-
1668
- if (type === 'buffer') {
1669
- const bufferArray = []
1670
-
1671
- for await (const chunk of stream) {
1672
- bufferArray.push(chunk)
1673
- }
1674
-
1675
- return Buffer.concat(bufferArray)
1676
- }
1677
-
1678
- return stream
1679
- }
1680
- }
1681
-
1682
- /** Checks whether the given message is a media message if it is returns the inner content */
1683
- const assertMediaContent = (content) => {
1684
- content = extractMessageContent(content)
1685
-
1686
- const mediaContent = content?.documentMessage
1687
- || content?.imageMessage
1688
- || content?.videoMessage
1689
- || content?.audioMessage
1690
- || content?.stickerMessage
1691
-
1692
- if (!mediaContent) {
1693
- throw new boom_1.Boom('given message is not a media message', { statusCode: 400, data: content })
1694
- }
1695
- return mediaContent
1696
- }
1697
-
1698
- /**
1699
- * this is an experimental patch to make buttons work
1700
- * Don't know how it works, but it does for now
1701
- */
1702
- const patchMessageForMdIfRequired = (message) => {
1703
- if (message?.buttonsMessage ||
1704
- message?.templateMessage ||
1705
- message?.listMessage ||
1706
- message?.interactiveMessage?.nativeFlowMesaage
1707
- ) {
1708
- message = {
1709
- viewOnceMessageV2Extension: {
1710
- message: {
1711
- messageContextInfo: {
1712
- deviceListMetadataVersion: 2,
1713
- deviceListMetadata: {}
1714
- },
1715
- ...message
1716
- }
1717
- }
1718
- }
1719
- }
1720
- return message
1721
- }
1722
-
1723
- module.exports = {
1724
- extractUrlFromText,
1725
- generateLinkPreviewIfRequired,
1726
- prepareWAMessageMedia,
1727
- prepareAlbumMessageContent,
1728
- prepareDisappearingMessageSettingContent,
1729
- generateForwardMessageContent,
1730
- generateWAMessageContent,
1731
- generateWAMessageFromContent,
1732
- generateWAMessage,
1733
- getContentType,
1734
- normalizeMessageContent,
1735
- extractMessageContent,
1736
- getDevice,
1737
- updateMessageWithReceipt,
1738
- updateMessageWithReaction,
1739
- updateMessageWithPollUpdate,
1740
- getAggregateVotesInPollMessage,
1741
- aggregateMessageKeysNotFromMe,
1742
- downloadMediaMessage,
1743
- assertMediaContent,
1744
- patchMessageForMdIfRequired
1745
- }
1
+ import { Boom } from '@hapi/boom';
2
+ import { randomBytes } from 'crypto';
3
+ import { promises as fs } from 'fs';
4
+ import {} from 'stream';
5
+ import { proto } from '../../WAProto/index.js';
6
+ import { CALL_AUDIO_PREFIX, CALL_VIDEO_PREFIX, MEDIA_KEYS, URL_REGEX, WA_DEFAULT_EPHEMERAL } from '../Defaults/index.js';
7
+ import { WAMessageStatus, WAProto } from '../Types/index.js';
8
+ import { isJidGroup, isJidNewsletter, isJidStatusBroadcast, jidNormalizedUser } from '../WABinary/index.js';
9
+ import { sha256 } from './crypto.js';
10
+ import { generateMessageIDV2, getKeyAuthor, unixTimestampSeconds } from './generics.js';
11
+ import { downloadContentFromMessage, encryptedStream, generateThumbnail, getAudioDuration, getAudioWaveform, getRawMediaUploadData } from './messages-media.js';
12
+ const MIMETYPE_MAP = {
13
+ image: 'image/jpeg',
14
+ video: 'video/mp4',
15
+ document: 'application/pdf',
16
+ audio: 'audio/ogg; codecs=opus',
17
+ sticker: 'image/webp',
18
+ 'product-catalog-image': 'image/jpeg'
19
+ };
20
+ const MessageTypeProto = {
21
+ image: WAProto.Message.ImageMessage,
22
+ video: WAProto.Message.VideoMessage,
23
+ audio: WAProto.Message.AudioMessage,
24
+ sticker: WAProto.Message.StickerMessage,
25
+ document: WAProto.Message.DocumentMessage
26
+ };
27
+ /**
28
+ * Uses a regex to test whether the string contains a URL, and returns the URL if it does.
29
+ * @param text eg. hello https://google.com
30
+ * @returns the URL, eg. https://google.com
31
+ */
32
+ export const extractUrlFromText = (text) => text.match(URL_REGEX)?.[0];
33
+ export const generateLinkPreviewIfRequired = async (text, getUrlInfo, logger) => {
34
+ const url = extractUrlFromText(text);
35
+ if (!!getUrlInfo && url) {
36
+ try {
37
+ const urlInfo = await getUrlInfo(url);
38
+ return urlInfo;
39
+ }
40
+ catch (error) {
41
+ // ignore if fails
42
+ logger?.warn({ trace: error.stack }, 'url generation failed');
43
+ }
44
+ }
45
+ };
46
+ const assertColor = async (color) => {
47
+ let assertedColor;
48
+ if (typeof color === 'number') {
49
+ assertedColor = color > 0 ? color : 0xffffffff + Number(color) + 1;
50
+ }
51
+ else {
52
+ let hex = color.trim().replace('#', '');
53
+ if (hex.length <= 6) {
54
+ hex = 'FF' + hex.padStart(6, '0');
55
+ }
56
+ assertedColor = parseInt(hex, 16);
57
+ return assertedColor;
58
+ }
59
+ };
60
+ export const prepareWAMessageMedia = async (message, options) => {
61
+ const logger = options.logger;
62
+ let mediaType;
63
+ for (const key of MEDIA_KEYS) {
64
+ if (key in message) {
65
+ mediaType = key;
66
+ }
67
+ }
68
+ if (!mediaType) {
69
+ throw new Boom('Invalid media type', { statusCode: 400 });
70
+ }
71
+ const uploadData = {
72
+ ...message,
73
+ media: message[mediaType]
74
+ };
75
+ delete uploadData[mediaType];
76
+ // check if cacheable + generate cache key
77
+ const cacheableKey = typeof uploadData.media === 'object' &&
78
+ 'url' in uploadData.media &&
79
+ !!uploadData.media.url &&
80
+ !!options.mediaCache &&
81
+ mediaType + ':' + uploadData.media.url.toString();
82
+ if (mediaType === 'document' && !uploadData.fileName) {
83
+ uploadData.fileName = 'file';
84
+ }
85
+ if (!uploadData.mimetype) {
86
+ uploadData.mimetype = MIMETYPE_MAP[mediaType];
87
+ }
88
+ if (cacheableKey) {
89
+ const mediaBuff = await options.mediaCache.get(cacheableKey);
90
+ if (mediaBuff) {
91
+ logger?.debug({ cacheableKey }, 'got media cache hit');
92
+ const obj = proto.Message.decode(mediaBuff);
93
+ const key = `${mediaType}Message`;
94
+ Object.assign(obj[key], { ...uploadData, media: undefined });
95
+ return obj;
96
+ }
97
+ }
98
+ const isNewsletter = !!options.jid && isJidNewsletter(options.jid);
99
+ if (isNewsletter) {
100
+ logger?.info({ key: cacheableKey }, 'Preparing raw media for newsletter');
101
+ const { filePath, fileSha256, fileLength } = await getRawMediaUploadData(uploadData.media, options.mediaTypeOverride || mediaType, logger);
102
+ const fileSha256B64 = fileSha256.toString('base64');
103
+ const { mediaUrl, directPath } = await options.upload(filePath, {
104
+ fileEncSha256B64: fileSha256B64,
105
+ mediaType: mediaType,
106
+ timeoutMs: options.mediaUploadTimeoutMs
107
+ });
108
+ await fs.unlink(filePath);
109
+ const obj = WAProto.Message.fromObject({
110
+ // todo: add more support here
111
+ [`${mediaType}Message`]: MessageTypeProto[mediaType].fromObject({
112
+ url: mediaUrl,
113
+ directPath,
114
+ fileSha256,
115
+ fileLength,
116
+ ...uploadData,
117
+ media: undefined
118
+ })
119
+ });
120
+ if (uploadData.ptv) {
121
+ obj.ptvMessage = obj.videoMessage;
122
+ delete obj.videoMessage;
123
+ }
124
+ if (obj.stickerMessage) {
125
+ obj.stickerMessage.stickerSentTs = Date.now();
126
+ }
127
+ if (cacheableKey) {
128
+ logger?.debug({ cacheableKey }, 'set cache');
129
+ await options.mediaCache.set(cacheableKey, WAProto.Message.encode(obj).finish());
130
+ }
131
+ return obj;
132
+ }
133
+ const requiresDurationComputation = mediaType === 'audio' && typeof uploadData.seconds === 'undefined';
134
+ const requiresThumbnailComputation = (mediaType === 'image' || mediaType === 'video') && typeof uploadData['jpegThumbnail'] === 'undefined';
135
+ const requiresWaveformProcessing = mediaType === 'audio' && uploadData.ptt === true;
136
+ const requiresAudioBackground = options.backgroundColor && mediaType === 'audio' && uploadData.ptt === true;
137
+ const requiresOriginalForSomeProcessing = requiresDurationComputation || requiresThumbnailComputation;
138
+ const { mediaKey, encFilePath, originalFilePath, fileEncSha256, fileSha256, fileLength } = await encryptedStream(uploadData.media, options.mediaTypeOverride || mediaType, {
139
+ logger,
140
+ saveOriginalFileIfRequired: requiresOriginalForSomeProcessing,
141
+ opts: options.options
142
+ });
143
+ const fileEncSha256B64 = fileEncSha256.toString('base64');
144
+ const [{ mediaUrl, directPath }] = await Promise.all([
145
+ (async () => {
146
+ const result = await options.upload(encFilePath, {
147
+ fileEncSha256B64,
148
+ mediaType,
149
+ timeoutMs: options.mediaUploadTimeoutMs
150
+ });
151
+ logger?.debug({ mediaType, cacheableKey }, 'uploaded media');
152
+ return result;
153
+ })(),
154
+ (async () => {
155
+ try {
156
+ if (requiresThumbnailComputation) {
157
+ const { thumbnail, originalImageDimensions } = await generateThumbnail(originalFilePath, mediaType, options);
158
+ uploadData.jpegThumbnail = thumbnail;
159
+ if (!uploadData.width && originalImageDimensions) {
160
+ uploadData.width = originalImageDimensions.width;
161
+ uploadData.height = originalImageDimensions.height;
162
+ logger?.debug('set dimensions');
163
+ }
164
+ logger?.debug('generated thumbnail');
165
+ }
166
+ if (requiresDurationComputation) {
167
+ uploadData.seconds = await getAudioDuration(originalFilePath);
168
+ logger?.debug('computed audio duration');
169
+ }
170
+ if (requiresWaveformProcessing) {
171
+ uploadData.waveform = await getAudioWaveform(originalFilePath, logger);
172
+ logger?.debug('processed waveform');
173
+ }
174
+ if (requiresAudioBackground) {
175
+ uploadData.backgroundArgb = await assertColor(options.backgroundColor);
176
+ logger?.debug('computed backgroundColor audio status');
177
+ }
178
+ }
179
+ catch (error) {
180
+ logger?.warn({ trace: error.stack }, 'failed to obtain extra info');
181
+ }
182
+ })()
183
+ ]).finally(async () => {
184
+ try {
185
+ await fs.unlink(encFilePath);
186
+ if (originalFilePath) {
187
+ await fs.unlink(originalFilePath);
188
+ }
189
+ logger?.debug('removed tmp files');
190
+ }
191
+ catch (error) {
192
+ logger?.warn('failed to remove tmp file');
193
+ }
194
+ });
195
+ const obj = WAProto.Message.fromObject({
196
+ [`${mediaType}Message`]: MessageTypeProto[mediaType].fromObject({
197
+ url: mediaUrl,
198
+ directPath,
199
+ mediaKey,
200
+ fileEncSha256,
201
+ fileSha256,
202
+ fileLength,
203
+ mediaKeyTimestamp: unixTimestampSeconds(),
204
+ ...uploadData,
205
+ media: undefined
206
+ })
207
+ });
208
+ if (uploadData.ptv) {
209
+ obj.ptvMessage = obj.videoMessage;
210
+ delete obj.videoMessage;
211
+ }
212
+ if (cacheableKey) {
213
+ logger?.debug({ cacheableKey }, 'set cache');
214
+ await options.mediaCache.set(cacheableKey, WAProto.Message.encode(obj).finish());
215
+ }
216
+ return obj;
217
+ };
218
+ export const prepareDisappearingMessageSettingContent = (ephemeralExpiration) => {
219
+ ephemeralExpiration = ephemeralExpiration || 0;
220
+ const content = {
221
+ ephemeralMessage: {
222
+ message: {
223
+ protocolMessage: {
224
+ type: WAProto.Message.ProtocolMessage.Type.EPHEMERAL_SETTING,
225
+ ephemeralExpiration
226
+ }
227
+ }
228
+ }
229
+ };
230
+ return WAProto.Message.fromObject(content);
231
+ };
232
+ /**
233
+ * Generate forwarded message content like WA does
234
+ * @param message the message to forward
235
+ * @param options.forceForward will show the message as forwarded even if it is from you
236
+ */
237
+ export const generateForwardMessageContent = (message, forceForward) => {
238
+ let content = message.message;
239
+ if (!content) {
240
+ throw new Boom('no content in message', { statusCode: 400 });
241
+ }
242
+ // hacky copy
243
+ content = normalizeMessageContent(content);
244
+ content = proto.Message.decode(proto.Message.encode(content).finish());
245
+ let key = Object.keys(content)[0];
246
+ let score = content?.[key]?.contextInfo?.forwardingScore || 0;
247
+ score += message.key.fromMe && !forceForward ? 0 : 1;
248
+ if (key === 'conversation') {
249
+ content.extendedTextMessage = { text: content[key] };
250
+ delete content.conversation;
251
+ key = 'extendedTextMessage';
252
+ }
253
+ const key_ = content?.[key];
254
+ if (score > 0) {
255
+ key_.contextInfo = { forwardingScore: score, isForwarded: true };
256
+ }
257
+ else {
258
+ key_.contextInfo = {};
259
+ }
260
+ return content;
261
+ };
262
+ export const generateWAMessageContent = async (message, options) => {
263
+ var _a, _b;
264
+ let m = {};
265
+ if ('text' in message) {
266
+ const extContent = { text: message.text };
267
+ let urlInfo = message.linkPreview;
268
+ if (typeof urlInfo === 'undefined') {
269
+ urlInfo = await generateLinkPreviewIfRequired(message.text, options.getUrlInfo, options.logger);
270
+ }
271
+ if (urlInfo) {
272
+ extContent.matchedText = urlInfo['matched-text'];
273
+ extContent.jpegThumbnail = urlInfo.jpegThumbnail;
274
+ extContent.description = urlInfo.description;
275
+ extContent.title = urlInfo.title;
276
+ extContent.previewType = 0;
277
+ const img = urlInfo.highQualityThumbnail;
278
+ if (img) {
279
+ extContent.thumbnailDirectPath = img.directPath;
280
+ extContent.mediaKey = img.mediaKey;
281
+ extContent.mediaKeyTimestamp = img.mediaKeyTimestamp;
282
+ extContent.thumbnailWidth = img.width;
283
+ extContent.thumbnailHeight = img.height;
284
+ extContent.thumbnailSha256 = img.fileSha256;
285
+ extContent.thumbnailEncSha256 = img.fileEncSha256;
286
+ }
287
+ }
288
+ if (options.backgroundColor) {
289
+ extContent.backgroundArgb = await assertColor(options.backgroundColor);
290
+ }
291
+ if (options.font) {
292
+ extContent.font = options.font;
293
+ }
294
+ m.extendedTextMessage = extContent;
295
+ }
296
+ else if ('contacts' in message) {
297
+ const contactLen = message.contacts.contacts.length;
298
+ if (!contactLen) {
299
+ throw new Boom('require atleast 1 contact', { statusCode: 400 });
300
+ }
301
+ if (contactLen === 1) {
302
+ m.contactMessage = WAProto.Message.ContactMessage.create(message.contacts.contacts[0]);
303
+ }
304
+ else {
305
+ m.contactsArrayMessage = WAProto.Message.ContactsArrayMessage.create(message.contacts);
306
+ }
307
+ }
308
+ else if ('location' in message) {
309
+ m.locationMessage = WAProto.Message.LocationMessage.create(message.location);
310
+ }
311
+ else if ('react' in message) {
312
+ if (!message.react.senderTimestampMs) {
313
+ message.react.senderTimestampMs = Date.now();
314
+ }
315
+ m.reactionMessage = WAProto.Message.ReactionMessage.create(message.react);
316
+ }
317
+ else if ('delete' in message) {
318
+ m.protocolMessage = {
319
+ key: message.delete,
320
+ type: WAProto.Message.ProtocolMessage.Type.REVOKE
321
+ };
322
+ }
323
+ else if ('forward' in message) {
324
+ m = generateForwardMessageContent(message.forward, message.force);
325
+ }
326
+ else if ('disappearingMessagesInChat' in message) {
327
+ const exp = typeof message.disappearingMessagesInChat === 'boolean'
328
+ ? message.disappearingMessagesInChat
329
+ ? WA_DEFAULT_EPHEMERAL
330
+ : 0
331
+ : message.disappearingMessagesInChat;
332
+ m = prepareDisappearingMessageSettingContent(exp);
333
+ }
334
+ else if ('groupInvite' in message) {
335
+ m.groupInviteMessage = {};
336
+ m.groupInviteMessage.inviteCode = message.groupInvite.inviteCode;
337
+ m.groupInviteMessage.inviteExpiration = message.groupInvite.inviteExpiration;
338
+ m.groupInviteMessage.caption = message.groupInvite.text;
339
+ m.groupInviteMessage.groupJid = message.groupInvite.jid;
340
+ m.groupInviteMessage.groupName = message.groupInvite.subject;
341
+ //TODO: use built-in interface and get disappearing mode info etc.
342
+ //TODO: cache / use store!?
343
+ if (options.getProfilePicUrl) {
344
+ const pfpUrl = await options.getProfilePicUrl(message.groupInvite.jid, 'preview');
345
+ if (pfpUrl) {
346
+ const resp = await fetch(pfpUrl, { method: 'GET', dispatcher: options?.options?.dispatcher });
347
+ if (resp.ok) {
348
+ const buf = Buffer.from(await resp.arrayBuffer());
349
+ m.groupInviteMessage.jpegThumbnail = buf;
350
+ }
351
+ }
352
+ }
353
+ }
354
+ else if ('pin' in message) {
355
+ m.pinInChatMessage = {};
356
+ m.messageContextInfo = {};
357
+ m.pinInChatMessage.key = message.pin;
358
+ m.pinInChatMessage.type = message.type;
359
+ m.pinInChatMessage.senderTimestampMs = Date.now();
360
+ m.messageContextInfo.messageAddOnDurationInSecs = message.type === 1 ? message.time || 86400 : 0;
361
+ }
362
+ else if ('buttonReply' in message) {
363
+ switch (message.type) {
364
+ case 'template':
365
+ m.templateButtonReplyMessage = {
366
+ selectedDisplayText: message.buttonReply.displayText,
367
+ selectedId: message.buttonReply.id,
368
+ selectedIndex: message.buttonReply.index
369
+ };
370
+ break;
371
+ case 'plain':
372
+ m.buttonsResponseMessage = {
373
+ selectedButtonId: message.buttonReply.id,
374
+ selectedDisplayText: message.buttonReply.displayText,
375
+ type: proto.Message.ButtonsResponseMessage.Type.DISPLAY_TEXT
376
+ };
377
+ break;
378
+ }
379
+ }
380
+ else if ('ptv' in message && message.ptv) {
381
+ const { videoMessage } = await prepareWAMessageMedia({ video: message.video }, options);
382
+ m.ptvMessage = videoMessage;
383
+ }
384
+ else if ('product' in message) {
385
+ const { imageMessage } = await prepareWAMessageMedia({ image: message.product.productImage }, options);
386
+ m.productMessage = WAProto.Message.ProductMessage.create({
387
+ ...message,
388
+ product: {
389
+ ...message.product,
390
+ productImage: imageMessage
391
+ }
392
+ });
393
+ }
394
+ else if ('listReply' in message) {
395
+ m.listResponseMessage = { ...message.listReply };
396
+ }
397
+ else if ('event' in message) {
398
+ m.eventMessage = {};
399
+ const startTime = Math.floor(message.event.startDate.getTime() / 1000);
400
+ if (message.event.call && options.getCallLink) {
401
+ const token = await options.getCallLink(message.event.call, { startTime });
402
+ m.eventMessage.joinLink = (message.event.call === 'audio' ? CALL_AUDIO_PREFIX : CALL_VIDEO_PREFIX) + token;
403
+ }
404
+ m.messageContextInfo = {
405
+ // encKey
406
+ messageSecret: message.event.messageSecret || randomBytes(32)
407
+ };
408
+ m.eventMessage.name = message.event.name;
409
+ m.eventMessage.description = message.event.description;
410
+ m.eventMessage.startTime = startTime;
411
+ m.eventMessage.endTime = message.event.endDate ? message.event.endDate.getTime() / 1000 : undefined;
412
+ m.eventMessage.isCanceled = message.event.isCancelled ?? false;
413
+ m.eventMessage.extraGuestsAllowed = message.event.extraGuestsAllowed;
414
+ m.eventMessage.isScheduleCall = message.event.isScheduleCall ?? false;
415
+ m.eventMessage.location = message.event.location;
416
+ }
417
+ else if ('poll' in message) {
418
+ (_a = message.poll).selectableCount || (_a.selectableCount = 0);
419
+ (_b = message.poll).toAnnouncementGroup || (_b.toAnnouncementGroup = false);
420
+ if (!Array.isArray(message.poll.values)) {
421
+ throw new Boom('Invalid poll values', { statusCode: 400 });
422
+ }
423
+ if (message.poll.selectableCount < 0 || message.poll.selectableCount > message.poll.values.length) {
424
+ throw new Boom(`poll.selectableCount in poll should be >= 0 and <= ${message.poll.values.length}`, {
425
+ statusCode: 400
426
+ });
427
+ }
428
+ m.messageContextInfo = {
429
+ // encKey
430
+ messageSecret: message.poll.messageSecret || randomBytes(32)
431
+ };
432
+ const pollCreationMessage = {
433
+ name: message.poll.name,
434
+ selectableOptionsCount: message.poll.selectableCount,
435
+ options: message.poll.values.map(optionName => ({ optionName }))
436
+ };
437
+ if (message.poll.toAnnouncementGroup) {
438
+ // poll v2 is for community announcement groups (single select and multiple)
439
+ m.pollCreationMessageV2 = pollCreationMessage;
440
+ }
441
+ else {
442
+ if (message.poll.selectableCount === 1) {
443
+ //poll v3 is for single select polls
444
+ m.pollCreationMessageV3 = pollCreationMessage;
445
+ }
446
+ else {
447
+ // poll for multiple choice polls
448
+ m.pollCreationMessage = pollCreationMessage;
449
+ }
450
+ }
451
+ }
452
+ else if ('sharePhoneNumber' in message) {
453
+ m.protocolMessage = {
454
+ type: proto.Message.ProtocolMessage.Type.SHARE_PHONE_NUMBER
455
+ };
456
+ }
457
+ else if ('requestPhoneNumber' in message) {
458
+ m.requestPhoneNumberMessage = {};
459
+ }
460
+ else if ('limitSharing' in message) {
461
+ m.protocolMessage = {
462
+ type: proto.Message.ProtocolMessage.Type.LIMIT_SHARING,
463
+ limitSharing: {
464
+ sharingLimited: message.limitSharing === true,
465
+ trigger: 1,
466
+ limitSharingSettingTimestamp: Date.now(),
467
+ initiatedByMe: true
468
+ }
469
+ };
470
+ }
471
+ else {
472
+ m = await prepareWAMessageMedia(message, options);
473
+ }
474
+ if ('viewOnce' in message && !!message.viewOnce) {
475
+ m = { viewOnceMessage: { message: m } };
476
+ }
477
+ if ('mentions' in message && message.mentions?.length) {
478
+ const messageType = Object.keys(m)[0];
479
+ const key = m[messageType];
480
+ if ('contextInfo' in key && !!key.contextInfo) {
481
+ key.contextInfo.mentionedJid = message.mentions;
482
+ }
483
+ else if (key) {
484
+ key.contextInfo = {
485
+ mentionedJid: message.mentions
486
+ };
487
+ }
488
+ }
489
+ if ('edit' in message) {
490
+ m = {
491
+ protocolMessage: {
492
+ key: message.edit,
493
+ editedMessage: m,
494
+ timestampMs: Date.now(),
495
+ type: WAProto.Message.ProtocolMessage.Type.MESSAGE_EDIT
496
+ }
497
+ };
498
+ }
499
+ if ('contextInfo' in message && !!message.contextInfo) {
500
+ const messageType = Object.keys(m)[0];
501
+ const key = m[messageType];
502
+ if ('contextInfo' in key && !!key.contextInfo) {
503
+ key.contextInfo = { ...key.contextInfo, ...message.contextInfo };
504
+ }
505
+ else if (key) {
506
+ key.contextInfo = message.contextInfo;
507
+ }
508
+ }
509
+ return WAProto.Message.create(m);
510
+ };
511
+ export const generateWAMessageFromContent = (jid, message, options) => {
512
+ // set timestamp to now
513
+ // if not specified
514
+ if (!options.timestamp) {
515
+ options.timestamp = new Date();
516
+ }
517
+ const innerMessage = normalizeMessageContent(message);
518
+ const key = getContentType(innerMessage);
519
+ const timestamp = unixTimestampSeconds(options.timestamp);
520
+ const { quoted, userJid } = options;
521
+ if (quoted && !isJidNewsletter(jid)) {
522
+ const participant = quoted.key.fromMe
523
+ ? userJid // TODO: Add support for LIDs
524
+ : quoted.participant || quoted.key.participant || quoted.key.remoteJid;
525
+ let quotedMsg = normalizeMessageContent(quoted.message);
526
+ const msgType = getContentType(quotedMsg);
527
+ // strip any redundant properties
528
+ quotedMsg = proto.Message.create({ [msgType]: quotedMsg[msgType] });
529
+ const quotedContent = quotedMsg[msgType];
530
+ if (typeof quotedContent === 'object' && quotedContent && 'contextInfo' in quotedContent) {
531
+ delete quotedContent.contextInfo;
532
+ }
533
+ const contextInfo = ('contextInfo' in innerMessage[key] && innerMessage[key]?.contextInfo) || {};
534
+ contextInfo.participant = jidNormalizedUser(participant);
535
+ contextInfo.stanzaId = quoted.key.id;
536
+ contextInfo.quotedMessage = quotedMsg;
537
+ // if a participant is quoted, then it must be a group
538
+ // hence, remoteJid of group must also be entered
539
+ if (jid !== quoted.key.remoteJid) {
540
+ contextInfo.remoteJid = quoted.key.remoteJid;
541
+ }
542
+ if (contextInfo && innerMessage[key]) {
543
+ /* @ts-ignore */
544
+ innerMessage[key].contextInfo = contextInfo;
545
+ }
546
+ }
547
+ if (
548
+ // if we want to send a disappearing message
549
+ !!options?.ephemeralExpiration &&
550
+ // and it's not a protocol message -- delete, toggle disappear message
551
+ key !== 'protocolMessage' &&
552
+ // already not converted to disappearing message
553
+ key !== 'ephemeralMessage' &&
554
+ // newsletters don't support ephemeral messages
555
+ !isJidNewsletter(jid)) {
556
+ /* @ts-ignore */
557
+ innerMessage[key].contextInfo = {
558
+ ...(innerMessage[key].contextInfo || {}),
559
+ expiration: options.ephemeralExpiration || WA_DEFAULT_EPHEMERAL
560
+ //ephemeralSettingTimestamp: options.ephemeralOptions.eph_setting_ts?.toString()
561
+ };
562
+ }
563
+ message = WAProto.Message.create(message);
564
+ const messageJSON = {
565
+ key: {
566
+ remoteJid: jid,
567
+ fromMe: true,
568
+ id: options?.messageId || generateMessageIDV2()
569
+ },
570
+ message: message,
571
+ messageTimestamp: timestamp,
572
+ messageStubParameters: [],
573
+ participant: isJidGroup(jid) || isJidStatusBroadcast(jid) ? userJid : undefined, // TODO: Add support for LIDs
574
+ status: WAMessageStatus.PENDING
575
+ };
576
+ return WAProto.WebMessageInfo.fromObject(messageJSON);
577
+ };
578
+ export const generateWAMessage = async (jid, content, options) => {
579
+ // ensure msg ID is with every log
580
+ options.logger = options?.logger?.child({ msgId: options.messageId });
581
+ // Pass jid in the options to generateWAMessageContent
582
+ return generateWAMessageFromContent(jid, await generateWAMessageContent(content, { ...options, jid }), options);
583
+ };
584
+ /** Get the key to access the true type of content */
585
+ export const getContentType = (content) => {
586
+ if (content) {
587
+ const keys = Object.keys(content);
588
+ const key = keys.find(k => (k === 'conversation' || k.includes('Message')) && k !== 'senderKeyDistributionMessage');
589
+ return key;
590
+ }
591
+ };
592
+ /**
593
+ * Normalizes ephemeral, view once messages to regular message content
594
+ * Eg. image messages in ephemeral messages, in view once messages etc.
595
+ * @param content
596
+ * @returns
597
+ */
598
+ export const normalizeMessageContent = (content) => {
599
+ if (!content) {
600
+ return undefined;
601
+ }
602
+ // set max iterations to prevent an infinite loop
603
+ for (let i = 0; i < 5; i++) {
604
+ const inner = getFutureProofMessage(content);
605
+ if (!inner) {
606
+ break;
607
+ }
608
+ content = inner.message;
609
+ }
610
+ return content;
611
+ function getFutureProofMessage(message) {
612
+ return (message?.ephemeralMessage ||
613
+ message?.viewOnceMessage ||
614
+ message?.documentWithCaptionMessage ||
615
+ message?.viewOnceMessageV2 ||
616
+ message?.viewOnceMessageV2Extension ||
617
+ message?.editedMessage);
618
+ }
619
+ };
620
+ /**
621
+ * Extract the true message content from a message
622
+ * Eg. extracts the inner message from a disappearing message/view once message
623
+ */
624
+ export const extractMessageContent = (content) => {
625
+ const extractFromTemplateMessage = (msg) => {
626
+ if (msg.imageMessage) {
627
+ return { imageMessage: msg.imageMessage };
628
+ }
629
+ else if (msg.documentMessage) {
630
+ return { documentMessage: msg.documentMessage };
631
+ }
632
+ else if (msg.videoMessage) {
633
+ return { videoMessage: msg.videoMessage };
634
+ }
635
+ else if (msg.locationMessage) {
636
+ return { locationMessage: msg.locationMessage };
637
+ }
638
+ else {
639
+ return {
640
+ conversation: 'contentText' in msg ? msg.contentText : 'hydratedContentText' in msg ? msg.hydratedContentText : ''
641
+ };
642
+ }
643
+ };
644
+ content = normalizeMessageContent(content);
645
+ if (content?.buttonsMessage) {
646
+ return extractFromTemplateMessage(content.buttonsMessage);
647
+ }
648
+ if (content?.templateMessage?.hydratedFourRowTemplate) {
649
+ return extractFromTemplateMessage(content?.templateMessage?.hydratedFourRowTemplate);
650
+ }
651
+ if (content?.templateMessage?.hydratedTemplate) {
652
+ return extractFromTemplateMessage(content?.templateMessage?.hydratedTemplate);
653
+ }
654
+ if (content?.templateMessage?.fourRowTemplate) {
655
+ return extractFromTemplateMessage(content?.templateMessage?.fourRowTemplate);
656
+ }
657
+ return content;
658
+ };
659
+ /**
660
+ * Returns the device predicted by message ID
661
+ */
662
+ export const getDevice = (id) => /^3A.{18}$/.test(id)
663
+ ? 'ios'
664
+ : /^3E.{20}$/.test(id)
665
+ ? 'web'
666
+ : /^(.{21}|.{32})$/.test(id)
667
+ ? 'android'
668
+ : /^(3F|.{18}$)/.test(id)
669
+ ? 'desktop'
670
+ : 'unknown';
671
+ /** Upserts a receipt in the message */
672
+ export const updateMessageWithReceipt = (msg, receipt) => {
673
+ msg.userReceipt = msg.userReceipt || [];
674
+ const recp = msg.userReceipt.find(m => m.userJid === receipt.userJid);
675
+ if (recp) {
676
+ Object.assign(recp, receipt);
677
+ }
678
+ else {
679
+ msg.userReceipt.push(receipt);
680
+ }
681
+ };
682
+ /** Update the message with a new reaction */
683
+ export const updateMessageWithReaction = (msg, reaction) => {
684
+ const authorID = getKeyAuthor(reaction.key);
685
+ const reactions = (msg.reactions || []).filter(r => getKeyAuthor(r.key) !== authorID);
686
+ reaction.text = reaction.text || '';
687
+ reactions.push(reaction);
688
+ msg.reactions = reactions;
689
+ };
690
+ /** Update the message with a new poll update */
691
+ export const updateMessageWithPollUpdate = (msg, update) => {
692
+ const authorID = getKeyAuthor(update.pollUpdateMessageKey);
693
+ const reactions = (msg.pollUpdates || []).filter(r => getKeyAuthor(r.pollUpdateMessageKey) !== authorID);
694
+ if (update.vote?.selectedOptions?.length) {
695
+ reactions.push(update);
696
+ }
697
+ msg.pollUpdates = reactions;
698
+ };
699
+ /**
700
+ * Aggregates all poll updates in a poll.
701
+ * @param msg the poll creation message
702
+ * @param meId your jid
703
+ * @returns A list of options & their voters
704
+ */
705
+ export function getAggregateVotesInPollMessage({ message, pollUpdates }, meId) {
706
+ const opts = message?.pollCreationMessage?.options ||
707
+ message?.pollCreationMessageV2?.options ||
708
+ message?.pollCreationMessageV3?.options ||
709
+ [];
710
+ const voteHashMap = opts.reduce((acc, opt) => {
711
+ const hash = sha256(Buffer.from(opt.optionName || '')).toString();
712
+ acc[hash] = {
713
+ name: opt.optionName || '',
714
+ voters: []
715
+ };
716
+ return acc;
717
+ }, {});
718
+ for (const update of pollUpdates || []) {
719
+ const { vote } = update;
720
+ if (!vote) {
721
+ continue;
722
+ }
723
+ for (const option of vote.selectedOptions || []) {
724
+ const hash = option.toString();
725
+ let data = voteHashMap[hash];
726
+ if (!data) {
727
+ voteHashMap[hash] = {
728
+ name: 'Unknown',
729
+ voters: []
730
+ };
731
+ data = voteHashMap[hash];
732
+ }
733
+ voteHashMap[hash].voters.push(getKeyAuthor(update.pollUpdateMessageKey, meId));
734
+ }
735
+ }
736
+ return Object.values(voteHashMap);
737
+ }
738
+ /** Given a list of message keys, aggregates them by chat & sender. Useful for sending read receipts in bulk */
739
+ export const aggregateMessageKeysNotFromMe = (keys) => {
740
+ const keyMap = {};
741
+ for (const { remoteJid, id, participant, fromMe } of keys) {
742
+ if (!fromMe) {
743
+ const uqKey = `${remoteJid}:${participant || ''}`;
744
+ if (!keyMap[uqKey]) {
745
+ keyMap[uqKey] = {
746
+ jid: remoteJid,
747
+ participant: participant,
748
+ messageIds: []
749
+ };
750
+ }
751
+ keyMap[uqKey].messageIds.push(id);
752
+ }
753
+ }
754
+ return Object.values(keyMap);
755
+ };
756
+ const REUPLOAD_REQUIRED_STATUS = [410, 404];
757
+ /**
758
+ * Downloads the given message. Throws an error if it's not a media message
759
+ */
760
+ export const downloadMediaMessage = async (message, type, options, ctx) => {
761
+ const result = await downloadMsg().catch(async (error) => {
762
+ if (ctx &&
763
+ typeof error?.status === 'number' && // treat errors with status as HTTP failures requiring reupload
764
+ REUPLOAD_REQUIRED_STATUS.includes(error.status)) {
765
+ ctx.logger.info({ key: message.key }, 'sending reupload media request...');
766
+ // request reupload
767
+ message = await ctx.reuploadRequest(message);
768
+ const result = await downloadMsg();
769
+ return result;
770
+ }
771
+ throw error;
772
+ });
773
+ return result;
774
+ async function downloadMsg() {
775
+ const mContent = extractMessageContent(message.message);
776
+ if (!mContent) {
777
+ throw new Boom('No message present', { statusCode: 400, data: message });
778
+ }
779
+ const contentType = getContentType(mContent);
780
+ let mediaType = contentType?.replace('Message', '');
781
+ const media = mContent[contentType];
782
+ if (!media || typeof media !== 'object' || (!('url' in media) && !('thumbnailDirectPath' in media))) {
783
+ throw new Boom(`"${contentType}" message is not a media message`);
784
+ }
785
+ let download;
786
+ if ('thumbnailDirectPath' in media && !('url' in media)) {
787
+ download = {
788
+ directPath: media.thumbnailDirectPath,
789
+ mediaKey: media.mediaKey
790
+ };
791
+ mediaType = 'thumbnail-link';
792
+ }
793
+ else {
794
+ download = media;
795
+ }
796
+ const stream = await downloadContentFromMessage(download, mediaType, options);
797
+ if (type === 'buffer') {
798
+ const bufferArray = [];
799
+ for await (const chunk of stream) {
800
+ bufferArray.push(chunk);
801
+ }
802
+ return Buffer.concat(bufferArray);
803
+ }
804
+ return stream;
805
+ }
806
+ };
807
+ /** Checks whether the given message is a media message; if it is returns the inner content */
808
+ export const assertMediaContent = (content) => {
809
+ content = extractMessageContent(content);
810
+ const mediaContent = content?.documentMessage ||
811
+ content?.imageMessage ||
812
+ content?.videoMessage ||
813
+ content?.audioMessage ||
814
+ content?.stickerMessage;
815
+ if (!mediaContent) {
816
+ throw new Boom('given message is not a media message', { statusCode: 400, data: content });
817
+ }
818
+ return mediaContent;
819
+ };
820
+ //# sourceMappingURL=messages.js.map