@jkt48connect-corp/baileys 7.4.5 → 7.4.9

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