@yemo-dev/yebail 1.0.0

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