@jkt48connect-corp/baileys 7.2.9 → 7.3.1

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