@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.
- package/WAProto/CompanionReg/CompanionReg.d.ts +6 -0
- package/WAProto/CompanionReg/CompanionReg.js +36 -0
- package/WAProto/CompanionReg/CompanionReg.proto +1 -0
- package/WAProto/E2E/E2E.d.ts +434 -6
- package/WAProto/E2E/E2E.js +1427 -2
- package/WAProto/E2E/E2E.proto +33 -0
- package/WAProto/HistorySync/HistorySync.d.ts +434 -6
- package/WAProto/HistorySync/HistorySync.js +1427 -2
- package/WAProto/MdStorageMsgRowOpaqueData/MdStorageMsgRowOpaqueData.d.ts +434 -6
- package/WAProto/MdStorageMsgRowOpaqueData/MdStorageMsgRowOpaqueData.js +1427 -2
- package/WAProto/StatusAttributions/StatusAttributions.d.ts +95 -3
- package/WAProto/StatusAttributions/StatusAttributions.js +270 -2
- package/WAProto/StatusAttributions/StatusAttributions.proto +8 -0
- package/WAProto/Web/Web.d.ts +434 -6
- package/WAProto/Web/Web.js +1427 -2
- package/lib/Defaults/baileys-version.json +1 -1
- package/lib/Socket/business.js +3 -3
- package/lib/Socket/chats.js +7 -7
- package/lib/Socket/groups.js +11 -9
- package/lib/Socket/messages-recv.js +12 -11
- package/lib/Socket/messages-send.js +981 -983
- package/lib/Socket/newsletter.js +3 -3
- package/lib/Socket/socket.js +13 -47
- package/lib/Socket/usync.js +3 -3
- package/lib/Store/make-cache-manager-store.js +9 -17
- package/lib/Store/make-in-memory-store.d.ts +2 -2
- package/lib/Store/make-in-memory-store.js +5 -5
- package/lib/Types/GroupMetadata.d.ts +2 -1
- package/lib/Utils/business.js +4 -0
- package/lib/Utils/decode-wa-message.js +4 -0
- package/lib/Utils/generics.js +10 -9
- package/lib/Utils/messages.js +1329 -1326
- package/lib/Utils/process-message.js +14 -1
- package/lib/index.js +1 -1
- package/package.json +5 -6
- package/WAProto/Adv/JKT48Connect - Valzyy +0 -0
- package/WAProto/Cert/JKT48Connect - Valzyy +0 -0
- package/WAProto/ChatLockSettings/JKT48Connect - Valzyy +0 -0
- package/WAProto/CompanionReg/JKT48Connect - Valzyy +0 -0
- package/WAProto/DeviceCapabilities/JKT48Connect - Valzyy +0 -0
- package/WAProto/E2E/JKT48Connect - Valzyy +0 -0
- package/WAProto/Ephemeral/JKT48Connect - Valzyy +0 -0
- package/WAProto/HistorySync/JKT48Connect - Valzyy +0 -0
- package/WAProto/JKT48Connect - Valzyy +0 -0
- package/WAProto/MdStorageChatRowOpaqueData/JKT48Connect - Valzyy +0 -0
- package/WAProto/MdStorageMsgRowOpaqueData/JKT48Connect - Valzyy +0 -0
- package/WAProto/MmsRetry/JKT48Connect - Valzyy +0 -0
- package/WAProto/Protocol/JKT48Connect - Valzyy +0 -0
- package/WAProto/Reporting/JKT48Connect - Valzyy +0 -0
- package/WAProto/ServerSync/JKT48Connect - Valzyy +0 -0
- package/WAProto/SignalLocalStorageProtocol/JKT48Connect - Valzyy +0 -0
- package/WAProto/SignalWhisperTextProtocol/JKT48Connect - Valzyy +0 -0
- package/WAProto/StatusAttributions/JKT48Connect - Valzyy +0 -0
- package/WAProto/SyncAction/JKT48Connect - Valzyy +0 -0
- package/WAProto/UserPassword/JKT48Connect - Valzyy +0 -0
- package/WAProto/VnameCert/JKT48Connect - Valzyy +0 -0
- package/WAProto/Wa6/JKT48Connect - Valzyy +0 -0
- package/WAProto/Web/JKT48Connect - Valzyy +0 -0
- package/lib/JKT48Connect - Valzyy +0 -0
package/lib/Utils/messages.js
CHANGED
@@ -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.
|
1048
|
-
|| (message === null || message === void 0 ? void 0 : message.viewOnceMessage)
|
1049
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1050
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1051
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1052
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1053
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1054
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1055
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1056
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1057
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1058
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1059
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1060
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1061
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1062
|
-
|| (message === null || message === void 0 ? void 0 : message.
|
1063
|
-
|
1064
|
-
|
1065
|
-
|
1066
|
-
|
1067
|
-
|
1068
|
-
|
1069
|
-
|
1070
|
-
|
1071
|
-
|
1072
|
-
|
1073
|
-
|
1074
|
-
|
1075
|
-
|
1076
|
-
return {
|
1077
|
-
}
|
1078
|
-
else if (msg.
|
1079
|
-
return {
|
1080
|
-
}
|
1081
|
-
else if (msg.
|
1082
|
-
return {
|
1083
|
-
}
|
1084
|
-
else {
|
1085
|
-
return {
|
1086
|
-
|
1087
|
-
|
1088
|
-
|
1089
|
-
|
1090
|
-
|
1091
|
-
|
1092
|
-
|
1093
|
-
|
1094
|
-
|
1095
|
-
|
1096
|
-
if (content === null || content === void 0 ? void 0 : content.
|
1097
|
-
return extractFromTemplateMessage(content.
|
1098
|
-
}
|
1099
|
-
if (
|
1100
|
-
return extractFromTemplateMessage(
|
1101
|
-
}
|
1102
|
-
if ((
|
1103
|
-
return extractFromTemplateMessage((
|
1104
|
-
}
|
1105
|
-
if ((
|
1106
|
-
return extractFromTemplateMessage((
|
1107
|
-
}
|
1108
|
-
if ((
|
1109
|
-
return extractFromTemplateMessage((
|
1110
|
-
}
|
1111
|
-
if ((
|
1112
|
-
return extractFromTemplateMessage((
|
1113
|
-
}
|
1114
|
-
if ((
|
1115
|
-
return extractFromTemplateMessage((
|
1116
|
-
}
|
1117
|
-
if ((
|
1118
|
-
return extractFromTemplateMessage((
|
1119
|
-
}
|
1120
|
-
if ((
|
1121
|
-
return extractFromTemplateMessage((
|
1122
|
-
}
|
1123
|
-
|
1124
|
-
|
1125
|
-
|
1126
|
-
|
1127
|
-
|
1128
|
-
|
1129
|
-
|
1130
|
-
|
1131
|
-
/^
|
1132
|
-
'
|
1133
|
-
|
1134
|
-
|
1135
|
-
|
1136
|
-
|
1137
|
-
|
1138
|
-
|
1139
|
-
|
1140
|
-
|
1141
|
-
|
1142
|
-
}
|
1143
|
-
|
1144
|
-
|
1145
|
-
|
1146
|
-
|
1147
|
-
|
1148
|
-
|
1149
|
-
|
1150
|
-
reactions.
|
1151
|
-
|
1152
|
-
|
1153
|
-
|
1154
|
-
|
1155
|
-
|
1156
|
-
|
1157
|
-
|
1158
|
-
|
1159
|
-
|
1160
|
-
votes.
|
1161
|
-
|
1162
|
-
|
1163
|
-
|
1164
|
-
|
1165
|
-
|
1166
|
-
|
1167
|
-
|
1168
|
-
*
|
1169
|
-
|
1170
|
-
|
1171
|
-
|
1172
|
-
|
1173
|
-
|
1174
|
-
|
1175
|
-
|
1176
|
-
|
1177
|
-
|
1178
|
-
|
1179
|
-
|
1180
|
-
|
1181
|
-
|
1182
|
-
|
1183
|
-
|
1184
|
-
|
1185
|
-
}
|
1186
|
-
|
1187
|
-
|
1188
|
-
|
1189
|
-
|
1190
|
-
|
1191
|
-
|
1192
|
-
|
1193
|
-
|
1194
|
-
|
1195
|
-
|
1196
|
-
|
1197
|
-
|
1198
|
-
}
|
1199
|
-
|
1200
|
-
}
|
1201
|
-
|
1202
|
-
|
1203
|
-
|
1204
|
-
|
1205
|
-
|
1206
|
-
const
|
1207
|
-
|
1208
|
-
|
1209
|
-
|
1210
|
-
|
1211
|
-
|
1212
|
-
|
1213
|
-
|
1214
|
-
|
1215
|
-
}
|
1216
|
-
}
|
1217
|
-
|
1218
|
-
}
|
1219
|
-
|
1220
|
-
|
1221
|
-
|
1222
|
-
|
1223
|
-
|
1224
|
-
|
1225
|
-
|
1226
|
-
|
1227
|
-
|
1228
|
-
//
|
1229
|
-
|
1230
|
-
|
1231
|
-
|
1232
|
-
|
1233
|
-
|
1234
|
-
|
1235
|
-
|
1236
|
-
|
1237
|
-
|
1238
|
-
|
1239
|
-
|
1240
|
-
|
1241
|
-
|
1242
|
-
|
1243
|
-
|
1244
|
-
|
1245
|
-
|
1246
|
-
|
1247
|
-
|
1248
|
-
|
1249
|
-
|
1250
|
-
|
1251
|
-
|
1252
|
-
|
1253
|
-
|
1254
|
-
|
1255
|
-
|
1256
|
-
|
1257
|
-
}
|
1258
|
-
|
1259
|
-
|
1260
|
-
|
1261
|
-
|
1262
|
-
|
1263
|
-
|
1264
|
-
|
1265
|
-
|
1266
|
-
|
1267
|
-
|
1268
|
-
}
|
1269
|
-
|
1270
|
-
|
1271
|
-
|
1272
|
-
|
1273
|
-
|
1274
|
-
|
1275
|
-
|
1276
|
-
|| content?.
|
1277
|
-
|
1278
|
-
|
1279
|
-
|
1280
|
-
|
1281
|
-
}
|
1282
|
-
|
1283
|
-
|
1284
|
-
|
1285
|
-
|
1286
|
-
|
1287
|
-
|
1288
|
-
|
1289
|
-
message
|
1290
|
-
message?.
|
1291
|
-
|
1292
|
-
message
|
1293
|
-
|
1294
|
-
|
1295
|
-
|
1296
|
-
|
1297
|
-
|
1298
|
-
|
1299
|
-
|
1300
|
-
}
|
1301
|
-
}
|
1302
|
-
|
1303
|
-
}
|
1304
|
-
|
1305
|
-
}
|
1306
|
-
|
1307
|
-
|
1308
|
-
|
1309
|
-
|
1310
|
-
|
1311
|
-
|
1312
|
-
|
1313
|
-
|
1314
|
-
|
1315
|
-
|
1316
|
-
|
1317
|
-
|
1318
|
-
|
1319
|
-
|
1320
|
-
|
1321
|
-
|
1322
|
-
|
1323
|
-
|
1324
|
-
|
1325
|
-
|
1326
|
-
|
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
|
}
|