alicezetion 1.6.7 → 1.6.9

Sign up to get free protection for your applications and to get access to all the features.
Files changed (57) hide show
  1. package/.cache/replit/__replit_disk_meta.json +1 -1
  2. package/.cache/replit/nix/env.json +1 -1
  3. package/index.js +804 -381
  4. package/leiamnash/addExternalModule.js +19 -19
  5. package/leiamnash/addUserToGroup.js +113 -113
  6. package/leiamnash/changeAdminStatus.js +79 -79
  7. package/leiamnash/changeApprovalMode.js +80 -0
  8. package/leiamnash/changeArchivedStatus.js +55 -55
  9. package/leiamnash/changeBio.js +77 -77
  10. package/leiamnash/changeBlockedStatus.js +47 -47
  11. package/leiamnash/changeGroupImage.js +129 -129
  12. package/leiamnash/changeNickname.js +59 -59
  13. package/leiamnash/changeThreadColor.js +71 -71
  14. package/leiamnash/changeThreadEmoji.js +55 -55
  15. package/leiamnash/chat.js +447 -459
  16. package/leiamnash/createNewGroup.js +86 -86
  17. package/leiamnash/createPoll.js +71 -71
  18. package/leiamnash/deleteMessage.js +56 -56
  19. package/leiamnash/deleteThread.js +56 -56
  20. package/leiamnash/forwardAttachment.js +60 -60
  21. package/leiamnash/getCurrentUserID.js +7 -7
  22. package/leiamnash/getEmojiUrl.js +29 -29
  23. package/leiamnash/getFriendsList.js +84 -84
  24. package/leiamnash/getThreadHistory.js +645 -645
  25. package/leiamnash/getThreadHistoryDeprecated.js +93 -93
  26. package/leiamnash/getThreadInfo.js +212 -206
  27. package/leiamnash/getThreadInfoDeprecated.js +80 -80
  28. package/leiamnash/getThreadList.js +238 -238
  29. package/leiamnash/getThreadListDeprecated.js +75 -75
  30. package/leiamnash/getThreadPictures.js +79 -79
  31. package/leiamnash/getUserID.js +66 -66
  32. package/leiamnash/getUserInfo.js +72 -72
  33. package/leiamnash/handleFriendRequest.js +61 -61
  34. package/leiamnash/handleMessageRequest.js +65 -65
  35. package/leiamnash/httpGet.js +52 -52
  36. package/leiamnash/httpPost.js +52 -52
  37. package/leiamnash/listenMqtt.js +1078 -509
  38. package/leiamnash/logout.js +75 -75
  39. package/leiamnash/markAsDelivered.js +58 -58
  40. package/leiamnash/markAsRead.js +80 -80
  41. package/leiamnash/markAsReadAll.js +49 -49
  42. package/leiamnash/markAsSeen.js +59 -59
  43. package/leiamnash/muteThread.js +52 -52
  44. package/leiamnash/removeUserFromGroup.js +79 -79
  45. package/leiamnash/resolvePhotoUrl.js +45 -45
  46. package/leiamnash/searchForThread.js +53 -53
  47. package/leiamnash/sendTypingIndicator.js +103 -103
  48. package/leiamnash/setMessageReaction.js +117 -117
  49. package/leiamnash/setPostReaction.js +76 -76
  50. package/leiamnash/setTitle.js +86 -86
  51. package/leiamnash/threadColors.js +57 -57
  52. package/leiamnash/unfriend.js +52 -52
  53. package/leiamnash/unsendMessage.js +49 -49
  54. package/package.json +72 -72
  55. package/replit.nix +3 -0
  56. package/utils.js +196 -71
  57. package/leiamnash/listen.js +0 -553
package/leiamnash/chat.js CHANGED
@@ -1,459 +1,447 @@
1
- "use strict";
2
-
3
- var utils = require("../utils");
4
- var log = require("npmlog");
5
- var bluebird = require("bluebird");
6
-
7
- var allowedProperties = {
8
- attachment: true,
9
- url: true,
10
- sticker: true,
11
- emoji: true,
12
- emojiSize: true,
13
- body: true,
14
- mentions: true,
15
- location: true,
16
- };
17
-
18
- module.exports = function (defaultFuncs, api, ctx) {
19
- function uploadAttachment(attachments, callback) {
20
- var uploads = [];
21
-
22
- // create an array of promises
23
- for (var i = 0; i < attachments.length; i++) {
24
- if (!utils.isReadableStream(attachments[i])) {
25
- throw {
26
- error:
27
- "Attachment should be a readable stream and not " +
28
- utils.getType(attachments[i]) +
29
- "."
30
- };
31
- }
32
-
33
- var form = {
34
- upload_1024: attachments[i],
35
- voice_clip: "true"
36
- };
37
-
38
- uploads.push(
39
- defaultFuncs
40
- .postFormData(
41
- "https://upload.facebook.com/ajax/mercury/upload.php",
42
- ctx.jar,
43
- form,
44
- {}
45
- )
46
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
47
- .then(function (resData) {
48
- if (resData.error) {
49
- throw resData;
50
- }
51
-
52
- // We have to return the data unformatted unless we want to change it
53
- // back in sendMessage.
54
- return resData.payload.metadata[0];
55
- })
56
- );
57
- }
58
-
59
- // resolve all promises
60
- bluebird
61
- .all(uploads)
62
- .then(function (resData) {
63
- callback(null, resData);
64
- })
65
- .catch(function (err) {
66
- log.error("uploadAttachment", err);
67
- return callback(err);
68
- });
69
- }
70
-
71
- function getUrl(url, callback) {
72
- var form = {
73
- image_height: 960,
74
- image_width: 960,
75
- uri: url
76
- };
77
-
78
- defaultFuncs
79
- .post(
80
- "https://www.facebook.com/message_share_attachment/fromURI/",
81
- ctx.jar,
82
- form
83
- )
84
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
85
- .then(function (resData) {
86
- if (resData.error) {
87
- return callback(resData);
88
- }
89
-
90
- if (!resData.payload) {
91
- return callback({ error: "Invalid url" });
92
- }
93
-
94
- callback(null, resData.payload.share_data.share_params);
95
- })
96
- .catch(function (err) {
97
- log.error("getUrl", err);
98
- return callback(err);
99
- });
100
- }
101
-
102
- function sendContent(form, threadID, isSingleUser, messageAndOTID, callback) {
103
- // There are three cases here:
104
- // 1. threadID is of type array, where we're starting a new group chat with users
105
- // specified in the array.
106
- // 2. User is sending a message to a specific user.
107
- // 3. No additional form params and the message goes to an existing group chat.
108
- if (utils.getType(threadID) === "Array") {
109
- for (var i = 0; i < threadID.length; i++) {
110
- form["specific_to_list[" + i + "]"] = "fbid:" + threadID[i];
111
- }
112
- form["specific_to_list[" + threadID.length + "]"] = "fbid:" + ctx.userID;
113
- form["client_thread_id"] = "root:" + messageAndOTID;
114
- log.info("sendMessage", "Sending message to multiple users: " + threadID);
115
- } else {
116
- // This means that threadID is the id of a user, and the chat
117
- // is a single person chat
118
- if (isSingleUser) {
119
- form["specific_to_list[0]"] = "fbid:" + threadID;
120
- form["specific_to_list[1]"] = "fbid:" + ctx.userID;
121
- form["other_user_fbid"] = threadID;
122
- } else {
123
- form["thread_fbid"] = threadID;
124
- }
125
- }
126
-
127
- if (ctx.globalOptions.pageID) {
128
- form["author"] = "fbid:" + ctx.globalOptions.pageID;
129
- form["specific_to_list[1]"] = "fbid:" + ctx.globalOptions.pageID;
130
- form["creator_info[creatorID]"] = ctx.userID;
131
- form["creator_info[creatorType]"] = "direct_admin";
132
- form["creator_info[labelType]"] = "sent_message";
133
- form["creator_info[pageID]"] = ctx.globalOptions.pageID;
134
- form["request_user_id"] = ctx.globalOptions.pageID;
135
- form["creator_info[profileURI]"] =
136
- "https://www.facebook.com/profile.php?id=" + ctx.userID;
137
- }
138
-
139
- defaultFuncs
140
- .post("https://www.facebook.com/messaging/send/", ctx.jar, form)
141
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
142
- .then(function (resData) {
143
- if (!resData) {
144
- return callback({ error: "Send message failed." });
145
- }
146
-
147
- if (resData.error) {
148
- if (resData.error === 1545012) {
149
- log.warn(
150
- "sendMessage",
151
- "Got error 1545012. This might mean that you're not part of the conversation " +
152
- threadID
153
- );
154
- }
155
- return callback(resData);
156
- }
157
-
158
- var messageInfo = resData.payload.actions.reduce(function (p, v) {
159
- return (
160
- {
161
- threadID: v.thread_fbid,
162
- messageID: v.message_id,
163
- timestamp: v.timestamp
164
- } || p
165
- );
166
- }, null);
167
-
168
- return callback(null, messageInfo);
169
- })
170
- .catch(function (err) {
171
- log.error("sendMessage", err);
172
- if (utils.getType(err) == "Object" && err.error === "Not logged in.") {
173
- ctx.loggedIn = false;
174
- }
175
- return callback(err);
176
- });
177
- }
178
-
179
- function send(form, threadID, messageAndOTID, callback, isGroup) {
180
- // We're doing a query to this to check if the given id is the id of
181
- // a user or of a group chat. The form will be different depending
182
- // on that.
183
- if (utils.getType(threadID) === "Array") {
184
- sendContent(form, threadID, false, messageAndOTID, callback);
185
- } else {
186
- if (utils.getType(isGroup) != "Boolean") {
187
- api.getUserInfo(threadID, function (err, res) {
188
- if (err) {
189
- return callback(err);
190
- }
191
- sendContent(
192
- form,
193
- threadID,
194
- Object.keys(res).length > 0,
195
- messageAndOTID,
196
- callback
197
- );
198
- });
199
- } else {
200
- sendContent(form, threadID, !isGroup, messageAndOTID, callback);
201
- }
202
- }
203
- }
204
-
205
- function handleUrl(msg, form, callback, cb) {
206
- if (msg.url) {
207
- form["shareable_attachment[share_type]"] = "100";
208
- getUrl(msg.url, function (err, params) {
209
- if (err) {
210
- return callback(err);
211
- }
212
-
213
- form["shareable_attachment[share_params]"] = params;
214
- cb();
215
- });
216
- } else {
217
- cb();
218
- }
219
- }
220
-
221
- function handleLocation(msg, form, callback, cb) {
222
- if (msg.location) {
223
- if (msg.location.latitude == null || msg.location.longitude == null) {
224
- return callback({ error: "location property needs both latitude and longitude" });
225
- }
226
-
227
- form["location_attachment[coordinates][latitude]"] = msg.location.latitude;
228
- form["location_attachment[coordinates][longitude]"] = msg.location.longitude;
229
- form["location_attachment[is_current_location]"] = !!msg.location.current;
230
- }
231
-
232
- cb();
233
- }
234
-
235
- function handleSticker(msg, form, callback, cb) {
236
- if (msg.sticker) {
237
- form["sticker_id"] = msg.sticker;
238
- }
239
- cb();
240
- }
241
-
242
- function handleEmoji(msg, form, callback, cb) {
243
- if (msg.emojiSize != null && msg.emoji == null) {
244
- return callback({ error: "emoji property is empty" });
245
- }
246
- if (msg.emoji) {
247
- if (msg.emojiSize == null) {
248
- msg.emojiSize = "medium";
249
- }
250
- if (
251
- msg.emojiSize != "small" &&
252
- msg.emojiSize != "medium" &&
253
- msg.emojiSize != "large"
254
- ) {
255
- return callback({ error: "emojiSize property is invalid" });
256
- }
257
- if (form["body"] != null && form["body"] != "") {
258
- return callback({ error: "body is not empty" });
259
- }
260
- form["body"] = msg.emoji;
261
- form["tags[0]"] = "hot_emoji_size:" + msg.emojiSize;
262
- }
263
- cb();
264
- }
265
-
266
- function handleAttachment(msg, form, callback, cb) {
267
- if (msg.attachment) {
268
- form["image_ids"] = [];
269
- form["gif_ids"] = [];
270
- form["file_ids"] = [];
271
- form["video_ids"] = [];
272
- form["audio_ids"] = [];
273
-
274
- if (utils.getType(msg.attachment) !== "Array") {
275
- msg.attachment = [msg.attachment];
276
- }
277
-
278
- uploadAttachment(msg.attachment, function (err, files) {
279
- if (err) {
280
- return callback(err);
281
- }
282
-
283
- files.forEach(function (file) {
284
- var key = Object.keys(file);
285
- var type = key[0]; // image_id, file_id, etc
286
- form["" + type + "s"].push(file[type]); // push the id
287
- });
288
- cb();
289
- });
290
- } else {
291
- cb();
292
- }
293
- }
294
-
295
- function handleMention(msg, form, callback, cb) {
296
- if (msg.mentions) {
297
- for (let i = 0; i < msg.mentions.length; i++) {
298
- const mention = msg.mentions[i];
299
-
300
- const tag = mention.tag;
301
- if (typeof tag !== "string") {
302
- return callback({ error: "Mention tags must be strings." });
303
- }
304
-
305
- const offset = msg.body.indexOf(tag, mention.fromIndex || 0);
306
-
307
- if (offset < 0) {
308
- log.warn(
309
- "handleMention",
310
- 'Mention for "' + tag + '" not found in message string.'
311
- );
312
- }
313
-
314
- if (mention.id == null) {
315
- log.warn("handleMention", "Mention id should be non-null.");
316
- }
317
-
318
- const id = mention.id || 0;
319
- form["profile_xmd[" + i + "][offset]"] = offset;
320
- form["profile_xmd[" + i + "][length]"] = tag.length;
321
- form["profile_xmd[" + i + "][id]"] = id;
322
- form["profile_xmd[" + i + "][type]"] = "p";
323
- }
324
- }
325
- cb();
326
- }
327
-
328
- return function sendMessage(msg, threadID, callback, replyToMessage, isGroup) {
329
- typeof isGroup == "undefined" ? isGroup = null : "";
330
- if (
331
- !callback &&
332
- (utils.getType(threadID) === "Function" ||
333
- utils.getType(threadID) === "AsyncFunction")
334
- ) {
335
- return threadID({ error: "Pass a threadID as a second argument." });
336
- }
337
- if (
338
- !replyToMessage &&
339
- utils.getType(callback) === "String"
340
- ) {
341
- replyToMessage = callback;
342
- callback = function () { };
343
- }
344
-
345
- var resolveFunc = function(){};
346
- var rejectFunc = function(){};
347
- var returnPromise = new Promise(function (resolve, reject) {
348
- resolveFunc = resolve;
349
- rejectFunc = reject;
350
- });
351
-
352
- if (!callback) {
353
- callback = function (err, friendList) {
354
- if (err) {
355
- return rejectFunc(err);
356
- }
357
- resolveFunc(friendList);
358
- };
359
- }
360
-
361
- var msgType = utils.getType(msg);
362
- var threadIDType = utils.getType(threadID);
363
- var messageIDType = utils.getType(replyToMessage);
364
-
365
- if (msgType !== "String" && msgType !== "Object") {
366
- return callback({
367
- error:
368
- "Message should be of type string or object and not " + msgType + "."
369
- });
370
- }
371
-
372
- // Changing this to accomodate an array of users
373
- if (
374
- threadIDType !== "Array" &&
375
- threadIDType !== "Number" &&
376
- threadIDType !== "String"
377
- ) {
378
- return callback({
379
- error:
380
- "ThreadID should be of type number, string, or array and not " +
381
- threadIDType +
382
- "."
383
- });
384
- }
385
-
386
- if (replyToMessage && messageIDType !== 'String') {
387
- return callback({
388
- error:
389
- "MessageID should be of type string and not " +
390
- threadIDType +
391
- "."
392
- });
393
- }
394
-
395
- if (msgType === "String") {
396
- msg = { body: msg };
397
- }
398
-
399
- var disallowedProperties = Object.keys(msg).filter(
400
- prop => !allowedProperties[prop]
401
- );
402
- if (disallowedProperties.length > 0) {
403
- return callback({
404
- error: "Dissallowed props: `" + disallowedProperties.join(", ") + "`"
405
- });
406
- }
407
-
408
- var messageAndOTID = utils.generateOfflineThreadingID();
409
-
410
- var form = {
411
- client: "mercury",
412
- action_type: "ma-type:user-generated-message",
413
- author: "fbid:" + ctx.userID,
414
- timestamp: Date.now(),
415
- timestamp_absolute: "Today",
416
- timestamp_relative: utils.generateTimestampRelative(),
417
- timestamp_time_passed: "0",
418
- is_unread: false,
419
- is_cleared: false,
420
- is_forward: false,
421
- is_filtered_content: false,
422
- is_filtered_content_bh: false,
423
- is_filtered_content_account: false,
424
- is_filtered_content_quasar: false,
425
- is_filtered_content_invalid_app: false,
426
- is_spoof_warning: false,
427
- source: "source:chat:web",
428
- "source_tags[0]": "source:chat",
429
- body: msg.body ? msg.body.toString() : "",
430
- html_body: false,
431
- ui_push_phase: "V3",
432
- status: "0",
433
- offline_threading_id: messageAndOTID,
434
- message_id: messageAndOTID,
435
- threading_id: utils.generateThreadingID(ctx.clientID),
436
- "ephemeral_ttl_mode:": "0",
437
- manual_retry_cnt: "0",
438
- has_attachment: !!(msg.attachment || msg.url || msg.sticker),
439
- signatureID: utils.getSignatureID(),
440
- replied_to_message_id: replyToMessage
441
- };
442
-
443
- handleLocation(msg, form, callback, () =>
444
- handleSticker(msg, form, callback, () =>
445
- handleAttachment(msg, form, callback, () =>
446
- handleUrl(msg, form, callback, () =>
447
- handleEmoji(msg, form, callback, () =>
448
- handleMention(msg, form, callback, () =>
449
- send(form, threadID, messageAndOTID, callback, isGroup)
450
- )
451
- )
452
- )
453
- )
454
- )
455
- );
456
-
457
- return returnPromise;
458
- };
459
- };
1
+ "use strict";
2
+
3
+ var utils = require("../utils");
4
+ var log = require("npmlog");
5
+ var bluebird = require("bluebird");
6
+
7
+ var allowedProperties = {
8
+ attachment: true,
9
+ url: true,
10
+ sticker: true,
11
+ emoji: true,
12
+ emojiSize: true,
13
+ body: true,
14
+ mentions: true,
15
+ location: true,
16
+ };
17
+
18
+ module.exports = function (defaultFuncs, api, ctx) {
19
+ function uploadAttachment(attachments, callback) {
20
+ var uploads = [];
21
+
22
+ // create an array of promises
23
+ for (var i = 0; i < attachments.length; i++) {
24
+ if (!utils.isReadableStream(attachments[i])) {
25
+ throw {
26
+ error:
27
+ "Attachment should be a readable stream and not " +
28
+ utils.getType(attachments[i]) +
29
+ "."
30
+ };
31
+ }
32
+
33
+ var form = {
34
+ upload_1024: attachments[i],
35
+ voice_clip: "true"
36
+ };
37
+
38
+ uploads.push(
39
+ defaultFuncs
40
+ .postFormData(
41
+ "https://upload.facebook.com/ajax/mercury/upload.php",
42
+ ctx.jar,
43
+ form,
44
+ {}
45
+ )
46
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
47
+ .then(function (resData) {
48
+ if (resData.error) {
49
+ throw resData;
50
+ }
51
+
52
+ // We have to return the data unformatted unless we want to change it
53
+ // back in sendMessage.
54
+ return resData.payload.metadata[0];
55
+ })
56
+ );
57
+ }
58
+
59
+ // resolve all promises
60
+ bluebird
61
+ .all(uploads)
62
+ .then(function (resData) {
63
+ callback(null, resData);
64
+ })
65
+ .catch(function (err) {
66
+ log.error("uploadAttachment", err);
67
+ return callback(err);
68
+ });
69
+ }
70
+
71
+ function getUrl(url, callback) {
72
+ var form = {
73
+ image_height: 960,
74
+ image_width: 960,
75
+ uri: url
76
+ };
77
+
78
+ defaultFuncs
79
+ .post(
80
+ "https://www.facebook.com/message_share_attachment/fromURI/",
81
+ ctx.jar,
82
+ form
83
+ )
84
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
85
+ .then(function (resData) {
86
+ if (resData.error) {
87
+ return callback(resData);
88
+ }
89
+
90
+ if (!resData.payload) {
91
+ return callback({ error: "Invalid url" });
92
+ }
93
+
94
+ callback(null, resData.payload.share_data.share_params);
95
+ })
96
+ .catch(function (err) {
97
+ log.error("getUrl", err);
98
+ return callback(err);
99
+ });
100
+ }
101
+
102
+ function sendContent(form, threadID, isSingleUser, messageAndOTID, callback) {
103
+ // There are three cases here:
104
+ // 1. threadID is of type array, where we're starting a new group chat with users
105
+ // specified in the array.
106
+ // 2. User is sending a message to a specific user.
107
+ // 3. No additional form params and the message goes to an existing group chat.
108
+ if (utils.getType(threadID) === "Array") {
109
+ for (var i = 0; i < threadID.length; i++) {
110
+ form["specific_to_list[" + i + "]"] = "fbid:" + threadID[i];
111
+ }
112
+ form["specific_to_list[" + threadID.length + "]"] = "fbid:" + ctx.userID;
113
+ form["client_thread_id"] = "root:" + messageAndOTID;
114
+ log.info("sendMessage", "Sending message to multiple users: " + threadID);
115
+ } else {
116
+ // This means that threadID is the id of a user, and the chat
117
+ // is a single person chat
118
+ if (isSingleUser) {
119
+ form["specific_to_list[0]"] = "fbid:" + threadID;
120
+ form["specific_to_list[1]"] = "fbid:" + ctx.userID;
121
+ form["other_user_fbid"] = threadID;
122
+ } else {
123
+ form["thread_fbid"] = threadID;
124
+ }
125
+ }
126
+
127
+ if (ctx.globalOptions.pageID) {
128
+ form["author"] = "fbid:" + ctx.globalOptions.pageID;
129
+ form["specific_to_list[1]"] = "fbid:" + ctx.globalOptions.pageID;
130
+ form["creator_info[creatorID]"] = ctx.userID;
131
+ form["creator_info[creatorType]"] = "direct_admin";
132
+ form["creator_info[labelType]"] = "sent_message";
133
+ form["creator_info[pageID]"] = ctx.globalOptions.pageID;
134
+ form["request_user_id"] = ctx.globalOptions.pageID;
135
+ form["creator_info[profileURI]"] =
136
+ "https://www.facebook.com/profile.php?id=" + ctx.userID;
137
+ }
138
+
139
+ defaultFuncs
140
+ .post("https://www.facebook.com/messaging/send/", ctx.jar, form)
141
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
142
+ .then(function (resData) {
143
+ if (!resData) {
144
+ return callback({ error: "Send message failed." });
145
+ }
146
+
147
+ if (resData.error) {
148
+ if (resData.error === 1545012) {
149
+ log.warn(
150
+ "sendMessage",
151
+ "Got error 1545012. This might mean that you're not part of the conversation " +
152
+ threadID
153
+ );
154
+ }
155
+ return callback(resData);
156
+ }
157
+
158
+ var messageInfo = resData.payload.actions.reduce(function (p, v) {
159
+ return (
160
+ {
161
+ threadID: v.thread_fbid,
162
+ messageID: v.message_id,
163
+ timestamp: v.timestamp
164
+ } || p
165
+ );
166
+ }, null);
167
+
168
+ return callback(null, messageInfo);
169
+ })
170
+ .catch(function (err) {
171
+ log.error("sendMessage", err);
172
+ if (utils.getType(err) == "Object" && err.error === "Not logged in.") {
173
+ ctx.loggedIn = false;
174
+ }
175
+ return callback(err);
176
+ });
177
+ }
178
+
179
+ function send(form, threadID, messageAndOTID, callback, isGroup) {
180
+ // We're doing a query to this to check if the given id is the id of
181
+ // a user or of a group chat. The form will be different depending
182
+ // on that.
183
+ if (utils.getType(threadID) === "Array") {
184
+ sendContent(form, threadID, false, messageAndOTID, callback);
185
+ } else {
186
+ if (utils.getType(isGroup) != "Boolean")
187
+ sendContent(form, threadID, threadID.length <= 15, messageAndOTID, callback);
188
+ else
189
+ sendContent(form, threadID, !isGroup, messageAndOTID, callback);
190
+ }
191
+ }
192
+
193
+ function handleUrl(msg, form, callback, cb) {
194
+ if (msg.url) {
195
+ form["shareable_attachment[share_type]"] = "100";
196
+ getUrl(msg.url, function (err, params) {
197
+ if (err) {
198
+ return callback(err);
199
+ }
200
+
201
+ form["shareable_attachment[share_params]"] = params;
202
+ cb();
203
+ });
204
+ } else {
205
+ cb();
206
+ }
207
+ }
208
+
209
+ function handleLocation(msg, form, callback, cb) {
210
+ if (msg.location) {
211
+ if (msg.location.latitude == null || msg.location.longitude == null) {
212
+ return callback({ error: "location property needs both latitude and longitude" });
213
+ }
214
+
215
+ form["location_attachment[coordinates][latitude]"] = msg.location.latitude;
216
+ form["location_attachment[coordinates][longitude]"] = msg.location.longitude;
217
+ form["location_attachment[is_current_location]"] = !!msg.location.current;
218
+ }
219
+
220
+ cb();
221
+ }
222
+
223
+ function handleSticker(msg, form, callback, cb) {
224
+ if (msg.sticker) {
225
+ form["sticker_id"] = msg.sticker;
226
+ }
227
+ cb();
228
+ }
229
+
230
+ function handleEmoji(msg, form, callback, cb) {
231
+ if (msg.emojiSize != null && msg.emoji == null) {
232
+ return callback({ error: "emoji property is empty" });
233
+ }
234
+ if (msg.emoji) {
235
+ if (msg.emojiSize == null) {
236
+ msg.emojiSize = "medium";
237
+ }
238
+ if (
239
+ msg.emojiSize != "small" &&
240
+ msg.emojiSize != "medium" &&
241
+ msg.emojiSize != "large"
242
+ ) {
243
+ return callback({ error: "emojiSize property is invalid" });
244
+ }
245
+ if (form["body"] != null && form["body"] != "") {
246
+ return callback({ error: "body is not empty" });
247
+ }
248
+ form["body"] = msg.emoji;
249
+ form["tags[0]"] = "hot_emoji_size:" + msg.emojiSize;
250
+ }
251
+ cb();
252
+ }
253
+
254
+ function handleAttachment(msg, form, callback, cb) {
255
+ if (msg.attachment) {
256
+ form["image_ids"] = [];
257
+ form["gif_ids"] = [];
258
+ form["file_ids"] = [];
259
+ form["video_ids"] = [];
260
+ form["audio_ids"] = [];
261
+
262
+ if (utils.getType(msg.attachment) !== "Array") {
263
+ msg.attachment = [msg.attachment];
264
+ }
265
+
266
+ uploadAttachment(msg.attachment, function (err, files) {
267
+ if (err) {
268
+ return callback(err);
269
+ }
270
+
271
+ files.forEach(function (file) {
272
+ var key = Object.keys(file);
273
+ var type = key[0]; // image_id, file_id, etc
274
+ form["" + type + "s"].push(file[type]); // push the id
275
+ });
276
+ cb();
277
+ });
278
+ } else {
279
+ cb();
280
+ }
281
+ }
282
+
283
+ function handleMention(msg, form, callback, cb) {
284
+ if (msg.mentions) {
285
+ for (let i = 0; i < msg.mentions.length; i++) {
286
+ const mention = msg.mentions[i];
287
+
288
+ const tag = mention.tag;
289
+ if (typeof tag !== "string") {
290
+ return callback({ error: "Mention tags must be strings." });
291
+ }
292
+
293
+ const offset = msg.body.indexOf(tag, mention.fromIndex || 0);
294
+
295
+ if (offset < 0) {
296
+ log.warn(
297
+ "handleMention",
298
+ 'Mention for "' + tag + '" not found in message string.'
299
+ );
300
+ }
301
+
302
+ if (mention.id == null) {
303
+ log.warn("handleMention", "Mention id should be non-null.");
304
+ }
305
+
306
+ const id = mention.id || 0;
307
+ const emptyChar = '\u200E';
308
+ form["body"] = emptyChar + msg.body;
309
+ form["profile_xmd[" + i + "][offset]"] = offset + 1;
310
+ form["profile_xmd[" + i + "][length]"] = tag.length;
311
+ form["profile_xmd[" + i + "][id]"] = id;
312
+ form["profile_xmd[" + i + "][type]"] = "p";
313
+ }
314
+ }
315
+ cb();
316
+ }
317
+
318
+ return function sendMessage(msg, threadID, callback, replyToMessage, isGroup) {
319
+ typeof isGroup == "undefined" ? isGroup = null : "";
320
+ if (
321
+ !callback &&
322
+ (utils.getType(threadID) === "Function" ||
323
+ utils.getType(threadID) === "AsyncFunction")
324
+ ) {
325
+ return threadID({ error: "Pass a threadID as a second argument." });
326
+ }
327
+ if (
328
+ !replyToMessage &&
329
+ utils.getType(callback) === "String"
330
+ ) {
331
+ replyToMessage = callback;
332
+ callback = undefined;
333
+ }
334
+
335
+ var resolveFunc = function () { };
336
+ var rejectFunc = function () { };
337
+ var returnPromise = new Promise(function (resolve, reject) {
338
+ resolveFunc = resolve;
339
+ rejectFunc = reject;
340
+ });
341
+
342
+ if (!callback) {
343
+ callback = function (err, data) {
344
+ if (err) return rejectFunc(err);
345
+ resolveFunc(data);
346
+ };
347
+ }
348
+
349
+ var msgType = utils.getType(msg);
350
+ var threadIDType = utils.getType(threadID);
351
+ var messageIDType = utils.getType(replyToMessage);
352
+
353
+ if (msgType !== "String" && msgType !== "Object") {
354
+ return callback({
355
+ error:
356
+ "Message should be of type string or object and not " + msgType + "."
357
+ });
358
+ }
359
+
360
+ // Changing this to accomodate an array of users
361
+ if (
362
+ threadIDType !== "Array" &&
363
+ threadIDType !== "Number" &&
364
+ threadIDType !== "String"
365
+ ) {
366
+ return callback({
367
+ error:
368
+ "ThreadID should be of type number, string, or array and not " +
369
+ threadIDType +
370
+ "."
371
+ });
372
+ }
373
+
374
+ if (replyToMessage && messageIDType !== 'String') {
375
+ return callback({
376
+ error:
377
+ "MessageID should be of type string and not " +
378
+ threadIDType +
379
+ "."
380
+ });
381
+ }
382
+
383
+ if (msgType === "String") {
384
+ msg = { body: msg };
385
+ }
386
+
387
+ var disallowedProperties = Object.keys(msg).filter(
388
+ prop => !allowedProperties[prop]
389
+ );
390
+ if (disallowedProperties.length > 0) {
391
+ return callback({
392
+ error: "Dissallowed props: `" + disallowedProperties.join(", ") + "`"
393
+ });
394
+ }
395
+
396
+ var messageAndOTID = utils.generateOfflineThreadingID();
397
+
398
+ var form = {
399
+ client: "mercury",
400
+ action_type: "ma-type:user-generated-message",
401
+ author: "fbid:" + ctx.userID,
402
+ timestamp: Date.now(),
403
+ timestamp_absolute: "Today",
404
+ timestamp_relative: utils.generateTimestampRelative(),
405
+ timestamp_time_passed: "0",
406
+ is_unread: false,
407
+ is_cleared: false,
408
+ is_forward: false,
409
+ is_filtered_content: false,
410
+ is_filtered_content_bh: false,
411
+ is_filtered_content_account: false,
412
+ is_filtered_content_quasar: false,
413
+ is_filtered_content_invalid_app: false,
414
+ is_spoof_warning: false,
415
+ source: "source:chat:web",
416
+ "source_tags[0]": "source:chat",
417
+ body: msg.body ? msg.body.toString() : "",
418
+ html_body: false,
419
+ ui_push_phase: "V3",
420
+ status: "0",
421
+ offline_threading_id: messageAndOTID,
422
+ message_id: messageAndOTID,
423
+ threading_id: utils.generateThreadingID(ctx.clientID),
424
+ "ephemeral_ttl_mode:": "0",
425
+ manual_retry_cnt: "0",
426
+ has_attachment: !!(msg.attachment || msg.url || msg.sticker),
427
+ signatureID: utils.getSignatureID(),
428
+ replied_to_message_id: replyToMessage
429
+ };
430
+
431
+ handleLocation(msg, form, callback, () =>
432
+ handleSticker(msg, form, callback, () =>
433
+ handleAttachment(msg, form, callback, () =>
434
+ handleUrl(msg, form, callback, () =>
435
+ handleEmoji(msg, form, callback, () =>
436
+ handleMention(msg, form, callback, () =>
437
+ send(form, threadID, messageAndOTID, callback, isGroup)
438
+ )
439
+ )
440
+ )
441
+ )
442
+ )
443
+ );
444
+
445
+ return returnPromise;
446
+ };
447
+ };