@neelegirl/baileys 1.5.5 → 1.5.7

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