rapido-fca 0.0.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 (81) hide show
  1. package/LICENSE +21 -0
  2. package/index.js +460 -0
  3. package/package.json +46 -0
  4. package/readme.md +254 -0
  5. package/src/addExternalModule.js +25 -0
  6. package/src/addUserToGroup.js +115 -0
  7. package/src/changeAdminStatus.js +103 -0
  8. package/src/changeArchivedStatus.js +55 -0
  9. package/src/changeAvatar.js +136 -0
  10. package/src/changeAvatarV2.js +86 -0
  11. package/src/changeAvt.js +85 -0
  12. package/src/changeBio.js +76 -0
  13. package/src/changeBlockedStatus.js +49 -0
  14. package/src/changeBlockedStatusMqtt.js +80 -0
  15. package/src/changeCover.js +72 -0
  16. package/src/changeGroupImage.js +135 -0
  17. package/src/changeName.js +79 -0
  18. package/src/changeNickname.js +59 -0
  19. package/src/changeThreadColor.js +65 -0
  20. package/src/changeThreadEmoji.js +55 -0
  21. package/src/changeUsername.js +59 -0
  22. package/src/createCommentPost.js +230 -0
  23. package/src/createNewGroup.js +88 -0
  24. package/src/createPoll.js +71 -0
  25. package/src/createPost.js +276 -0
  26. package/src/deleteMessage.js +56 -0
  27. package/src/deleteThread.js +56 -0
  28. package/src/editMessage.js +68 -0
  29. package/src/editMessageOld.js +67 -0
  30. package/src/follow.js +74 -0
  31. package/src/forwardAttachment.js +60 -0
  32. package/src/getAccess.js +112 -0
  33. package/src/getAvatarUser.js +78 -0
  34. package/src/getCurrentUserID.js +7 -0
  35. package/src/getEmojiUrl.js +29 -0
  36. package/src/getFriendsList.js +83 -0
  37. package/src/getMessage.js +835 -0
  38. package/src/getRegion.js +7 -0
  39. package/src/getThreadHistory.js +680 -0
  40. package/src/getThreadHistoryDeprecated.js +71 -0
  41. package/src/getThreadInfo.js +232 -0
  42. package/src/getThreadInfoDeprecated.js +56 -0
  43. package/src/getThreadList.js +213 -0
  44. package/src/getThreadListDeprecated.js +46 -0
  45. package/src/getThreadPictures.js +59 -0
  46. package/src/getUID.js +119 -0
  47. package/src/getUserID.js +61 -0
  48. package/src/getUserInfo.js +66 -0
  49. package/src/handleFriendRequest.js +46 -0
  50. package/src/handleMessageRequest.js +47 -0
  51. package/src/httpGet.js +49 -0
  52. package/src/httpPost.js +48 -0
  53. package/src/listenMqtt.js +827 -0
  54. package/src/logout.js +75 -0
  55. package/src/markAsDelivered.js +47 -0
  56. package/src/markAsRead.js +70 -0
  57. package/src/markAsReadAll.js +40 -0
  58. package/src/markAsSeen.js +48 -0
  59. package/src/muteThread.js +45 -0
  60. package/src/refreshFb_dtsg.js +89 -0
  61. package/src/removeUserFromGroup.js +79 -0
  62. package/src/resolvePhotoUrl.js +45 -0
  63. package/src/searchForThread.js +53 -0
  64. package/src/searchStickers.js +53 -0
  65. package/src/sendMessage.js +329 -0
  66. package/src/sendMessageMqtt.js +322 -0
  67. package/src/sendTypingIndicator.js +101 -0
  68. package/src/sendTypingIndicatorV2.js +28 -0
  69. package/src/setMessageReaction.js +122 -0
  70. package/src/setMessageReactionMqtt.js +62 -0
  71. package/src/setPostReaction.js +112 -0
  72. package/src/setStoryReaction.js +64 -0
  73. package/src/setTitle.js +90 -0
  74. package/src/shareContact.js +110 -0
  75. package/src/shareLink.js +59 -0
  76. package/src/stopListenMqtt.js +23 -0
  77. package/src/threadColors.js +131 -0
  78. package/src/unfriend.js +52 -0
  79. package/src/unsendMessage.js +45 -0
  80. package/src/uploadAttachment.js +93 -0
  81. package/utils.js +2876 -0
@@ -0,0 +1,276 @@
1
+ 'use strict';
2
+
3
+ var utils = require('../utils');
4
+ var log = require('npmlog');
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ function handleUpload(msg, form) {
8
+ var cb;
9
+ var rt = new Promise(function (resolve, reject) {
10
+ cb = error => error ? reject(error) : resolve();
11
+ });
12
+
13
+ if (!msg.attachment) cb();
14
+ else {
15
+ msg.attachment = Array.isArray(msg.attachment) ? msg.attachment : [msg.attachment];
16
+ let uploads = [];
17
+ for (let attachment of msg.attachment) {
18
+ if (!utils.isReadableStream(attachment))
19
+ cb('Attachment should be a readable stream, not ' + utils.getType(attachment));
20
+
21
+ var vari = {
22
+ source: 8,
23
+ profile_id: ctx.userID,
24
+ waterfallxapp: 'comet',
25
+ farr: attachment,
26
+ upload_id: 'jsc_c_6'
27
+ }
28
+ var main = defaultFuncs
29
+ .postFormData('https://upload.facebook.com/ajax/react_composer/attachments/photo/upload', ctx.jar, vari)
30
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
31
+ .then(function (res) {
32
+ if (res.error || res.errors)
33
+ throw res;
34
+
35
+ return res.payload;
36
+ });
37
+
38
+ uploads.push(main);
39
+ }
40
+
41
+ Promise
42
+ .all(uploads)
43
+ .then(function (res) {
44
+ for (let payload of res) {
45
+ if (!payload) break;
46
+ form.input.attachments.push({
47
+ photo: {
48
+ id: payload.photoID
49
+ }
50
+ });
51
+ }
52
+
53
+ return cb();
54
+ })
55
+ .catch(cb);
56
+ }
57
+
58
+ return rt;
59
+ }
60
+
61
+ function handleUrl(msg, form) {
62
+ var cb;
63
+ var rt = new Promise(function (resolve, reject) {
64
+ cb = error => error ? reject(error) : resolve();
65
+ });
66
+
67
+ if (!msg.url) cb();
68
+ else {
69
+ var vari = {
70
+ feedLocation: "FEED_COMPOSER",
71
+ focusCommentID: null,
72
+ goodwillCampaignId: "",
73
+ goodwillCampaignMediaIds: [],
74
+ goodwillContentType: null,
75
+ params: {
76
+ url: msg.url
77
+ },
78
+ privacySelectorRenderLocation: "COMET_COMPOSER",
79
+ renderLocation: "composer_preview",
80
+ parentStoryID: null,
81
+ scale: 1,
82
+ useDefaultActor: false,
83
+ shouldIncludeStoryAttachment: false,
84
+ __relay_internal__pv__IsWorkUserrelayprovider: false,
85
+ __relay_internal__pv__IsMergQAPollsrelayprovider: false
86
+ }
87
+
88
+ defaultFuncs
89
+ .post('https://www.facebook.com/api/graphql/', ctx.jar, {
90
+ fb_api_req_friendly_name: 'ComposerLinkAttachmentPreviewQuery',
91
+ variables: JSON.stringify(vari),
92
+ server_timestamps: true,
93
+ doc_id: 6549975235094234
94
+ })
95
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
96
+ .then(function (res) {
97
+ var res = (res[0] || res).data.link_preview;
98
+ if (JSON.parse(res.share_scrape_data).share_type == 400)
99
+ throw { error: 'url is not accepted' }
100
+
101
+ form.input.attachments.push({
102
+ link: {
103
+ share_scrape_data: res.share_scrape_data
104
+ }
105
+ });
106
+
107
+ return cb();
108
+ })
109
+ .catch(cb);
110
+ }
111
+
112
+ return rt;
113
+ }
114
+
115
+ function handleMention(msg, form) {
116
+ if (!msg.mentions) return;
117
+
118
+ msg.mentions = Array.isArray(msg.mentions) ? msg.mentions : [msg.mentions];
119
+ for (let mention of msg.mentions) {
120
+ var { id, tag, fromIndex } = mention;
121
+
122
+ if (typeof tag != 'string')
123
+ throw 'Mention tag must be string';
124
+ if (!id)
125
+ throw 'id must be string';
126
+ var offset = msg.body.indexOf(tag, fromIndex || 0);
127
+ if (offset < 0)
128
+ throw 'Mention for "' + tag + '" not found in message string.';
129
+ form.input.message.ranges.push({
130
+ entity: { id },
131
+ length: tag.length,
132
+ offset
133
+ });
134
+ }
135
+ }
136
+
137
+ function createContent(vari) {
138
+ var cb;
139
+ var rt = new Promise(function (resolve, reject) {
140
+ cb = (error, postData) => error ? reject(error) : resolve(postData);
141
+ });
142
+
143
+ var form = {
144
+ fb_api_req_friendly_name: 'ComposerStoryCreateMutation',
145
+ variables: JSON.stringify(vari),
146
+ server_timestamps: true,
147
+ doc_id: 6255089511280268
148
+ }
149
+
150
+ defaultFuncs
151
+ .post('https://www.facebook.com/api/graphql/', ctx.jar, form)
152
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
153
+ .then(res => cb(null, res))
154
+ .catch(cb);
155
+
156
+ return rt;
157
+ }
158
+
159
+ return function createPost(msg, callback) {
160
+ var cb;
161
+ var rt = new Promise(function (resolve, reject) {
162
+ cb = (error, url) => url ? resolve(url) : reject(error);
163
+ });
164
+
165
+ if (typeof msg == 'function') {
166
+ var error = 'Msg must be a string or object and not function';
167
+ console.error('createPost', error);
168
+ return msg(error);
169
+ }
170
+ if (typeof callback == 'function') cb = callback;
171
+
172
+ var typeMsg = utils.getType(msg);
173
+ if (!['Object', 'String'].includes(typeMsg)) {
174
+ var error = 'Msg must be a string or object and not ' + typeMsg;
175
+ console.error('createPost', error);
176
+ return cb(error);
177
+ } else if (typeMsg == 'String') msg = { body: msg };
178
+ msg.allowUserID = msg.allowUserID ? !Array.isArray(msg.allowUserID) ? [msg.allowUserID] : msg.allowUserID : null;
179
+
180
+ var sessionID = utils.getGUID();
181
+ var base = [
182
+ 'EVERYONE',
183
+ 'FRIENDS',
184
+ 'SELF'
185
+ ];
186
+ var form = {
187
+ input: {
188
+ composer_entry_point: !msg.groupID && msg.url ? 'share_modal' : "inline_composer",
189
+ composer_source_surface: !msg.groupID && msg.url ? 'feed_story' : msg.groupID ? "group" : "timeline",
190
+ composer_type: !msg.groupID && msg.url ? 'share' : msg.groupID ? "group" : "timeline",
191
+ idempotence_token: sessionID + "_FEED",
192
+ source: "WWW",
193
+ attachments: [],
194
+ audience: msg.groupID ? {
195
+ to_id: msg.groupID
196
+ } : {
197
+ privacy: {
198
+ allow: msg.allowUserID ? msg.allowUserID : [],
199
+ base_state: msg.allowUserID && msg.allowUserID.length > 0 ? base[2] : (base[msg.baseState - 1] || base[0]),
200
+ deny: [],
201
+ tag_expansion_state: "UNSPECIFIED"
202
+ }
203
+ },
204
+ message: {
205
+ ranges: [],
206
+ text: msg.body ? typeof msg.body == 'object' ? JSON.stringify(msg.body, null, 2) : msg.body : ''
207
+ },
208
+ with_tags_ids: [],
209
+ inline_activities: [],
210
+ explicit_place_id: 0,
211
+ text_format_preset_id: 0,
212
+ logging: {
213
+ composer_session_id: sessionID
214
+ },
215
+ navigation_data: {
216
+ attribution_id_v2: msg.groupID ? "CometGroupDiscussionRoot.react,comet.group,tap_search_bar," + Date.now() + ",909538,2361831622," : "ProfileCometTimelineListViewRoot.react,comet.profile.timeline.list,via_cold_start," + Date.now() + ",796829,190055527696468,"
217
+ },
218
+ is_tracking_encrypted: !!msg.url,
219
+ tracking: [],
220
+ event_share_metadata: {
221
+ surface: "newsfeed"
222
+ },
223
+ actor_id: ctx.globalOptions.pageID || ctx.userID,
224
+ client_mutation_id: Math.round(Math.random() * 19).toString()
225
+ },
226
+ displayCommentsFeedbackContext: null,
227
+ displayCommentsContextEnableComment: null,
228
+ displayCommentsContextIsAdPreview: null,
229
+ displayCommentsContextIsAggregatedShare: null,
230
+ displayCommentsContextIsStorySet: null,
231
+ feedLocation: msg.groupID ? "GROUP" : "TIMELINE",
232
+ feedbackSource: 0,
233
+ focusCommentID: null,
234
+ gridMediaWidth: 230,
235
+ groupID: null,
236
+ scale: 1,
237
+ privacySelectorRenderLocation: "COMET_STREAM",
238
+ renderLocation: msg.groupID ? "group" : "timeline",
239
+ useDefaultActor: false,
240
+ inviteShortLinkKey: null,
241
+ isFeed: false,
242
+ isFundraiser: false,
243
+ isFunFactPost: false,
244
+ isGroup: !!msg.groupID,
245
+ isEvent: false,
246
+ isTimeline: !msg.groupID,
247
+ isSocialLearning: false,
248
+ isPageNewsFeed: !!ctx.globalOptions.pageID,
249
+ isProfileReviews: false,
250
+ isWorkSharedDraft: false,
251
+ UFI2CommentsProvider_commentsKey: msg.groupID ? "CometGroupDiscussionRootSuccessQuery" : "ProfileCometTimelineRoute",
252
+ hashtag: null,
253
+ canUserManageOffers: false,
254
+ __relay_internal__pv__CometUFIIsRTAEnabledrelayprovider: false,
255
+ __relay_internal__pv__IsWorkUserrelayprovider: false,
256
+ __relay_internal__pv__IsMergQAPollsrelayprovider: false,
257
+ __relay_internal__pv__StoriesArmadilloReplyEnabledrelayprovider: false,
258
+ __relay_internal__pv__StoriesRingrelayprovider: false
259
+ }
260
+
261
+ handleUpload(msg, form)
262
+ .then(_ => handleUrl(msg, form))
263
+ .then(_ => handleMention(msg, form))
264
+ .then(_ => createContent(form))
265
+ .then((res) => {
266
+ if (res.error || res.errors) throw res;
267
+ return cb(null, (res[0] || res).data.story_create.story.url);
268
+ })
269
+ .catch((err) => {
270
+ //console.error('createPost', err);
271
+ return cb(err);
272
+ });
273
+
274
+ return rt;
275
+ }
276
+ }
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ // @NethWs3Dev
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ return function deleteMessage(messageOrMessages, callback) {
8
+ let resolveFunc = function () {};
9
+ let rejectFunc = function () {};
10
+ const returnPromise = new Promise(function (resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+ if (!callback) {
15
+ callback = function (err) {
16
+ if (err) {
17
+ return rejectFunc(err);
18
+ }
19
+ resolveFunc();
20
+ };
21
+ }
22
+
23
+ const form = {
24
+ client: "mercury",
25
+ };
26
+
27
+ if (utils.getType(messageOrMessages) !== "Array") {
28
+ messageOrMessages = [messageOrMessages];
29
+ }
30
+
31
+ for (let i = 0; i < messageOrMessages.length; i++) {
32
+ form["message_ids[" + i + "]"] = messageOrMessages[i];
33
+ }
34
+
35
+ defaultFuncs
36
+ .post(
37
+ "https://www.facebook.com/ajax/mercury/delete_messages.php",
38
+ ctx.jar,
39
+ form,
40
+ )
41
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
42
+ .then(function (resData) {
43
+ if (resData.error) {
44
+ throw resData;
45
+ }
46
+
47
+ return callback();
48
+ })
49
+ .catch(function (err) {
50
+ console.error("deleteMessage", err);
51
+ return callback(err);
52
+ });
53
+
54
+ return returnPromise;
55
+ };
56
+ };
@@ -0,0 +1,56 @@
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ // @NethWs3Dev
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ return function deleteThread(threadOrThreads, callback) {
8
+ let resolveFunc = function () {};
9
+ let rejectFunc = function () {};
10
+ const returnPromise = new Promise(function (resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+ if (!callback) {
15
+ callback = function (err) {
16
+ if (err) {
17
+ return rejectFunc(err);
18
+ }
19
+ resolveFunc();
20
+ };
21
+ }
22
+
23
+ const form = {
24
+ client: "mercury",
25
+ };
26
+
27
+ if (utils.getType(threadOrThreads) !== "Array") {
28
+ threadOrThreads = [threadOrThreads];
29
+ }
30
+
31
+ for (let i = 0; i < threadOrThreads.length; i++) {
32
+ form["ids[" + i + "]"] = threadOrThreads[i];
33
+ }
34
+
35
+ defaultFuncs
36
+ .post(
37
+ "https://www.facebook.com/ajax/mercury/delete_thread.php",
38
+ ctx.jar,
39
+ form,
40
+ )
41
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
42
+ .then(function (resData) {
43
+ if (resData.error) {
44
+ throw resData;
45
+ }
46
+
47
+ return callback();
48
+ })
49
+ .catch(function (err) {
50
+ console.error("deleteThread", err);
51
+ return callback(err);
52
+ });
53
+
54
+ return returnPromise;
55
+ };
56
+ };
@@ -0,0 +1,68 @@
1
+ "use_strict";
2
+ /**
3
+ * @author RFS-ADRENO
4
+ * @rewrittenBy Isai Ivanov
5
+ */
6
+ const generateOfflineThreadingId = require('../utils');
7
+
8
+ function canBeCalled(func) {
9
+ try {
10
+ Reflect.apply(func, null, []);
11
+ return true;
12
+ } catch (error) {
13
+ return false;
14
+ }
15
+ }
16
+
17
+ /**
18
+ * A function for editing bot's messages.
19
+ * @param {string} text - The text with which the bot will edit its messages.
20
+ * @param {string} messageID - The message ID of the message the bot will edit.
21
+ * @param {Object} callback - Callback for the function.
22
+ */
23
+
24
+ module.exports = function (defaultFuncs, api, ctx) {
25
+ return function editMessage(text, messageID, callback) {
26
+ if (!ctx.mqttClient) {
27
+ throw new Error('Not connected to MQTT');
28
+ }
29
+
30
+ ctx.wsReqNumber += 1;
31
+ ctx.wsTaskNumber += 1;
32
+
33
+ const queryPayload = {
34
+ message_id: messageID,
35
+ text: text
36
+ };
37
+
38
+ const query = {
39
+ failure_count: null,
40
+ label: '742',
41
+ payload: JSON.stringify(queryPayload),
42
+ queue_name: 'edit_message',
43
+ task_id: ctx.wsTaskNumber
44
+ };
45
+
46
+ const context = {
47
+ app_id: '2220391788200892',
48
+ payload: {
49
+ data_trace_id: null,
50
+ epoch_id: parseInt(generateOfflineThreadingId),
51
+ tasks: [query],
52
+ version_id: '6903494529735864'
53
+ },
54
+ request_id: ctx.wsReqNumber,
55
+ type: 3
56
+ };
57
+
58
+ context.payload = JSON.stringify(context.payload);
59
+
60
+ // if (canBeCalled(callback)) {
61
+ // ctx.reqCallbacks[ctx.wsReqNumber] = callback;
62
+ // }
63
+
64
+ ctx.mqttClient.publish('/ls_req', JSON.stringify(context), {
65
+ qos: 1, retain: false
66
+ });
67
+ };
68
+ };
@@ -0,0 +1,67 @@
1
+ "use_strict";
2
+
3
+ const { generateOfflineThreadingID } = require('../utils');
4
+
5
+ function isCallable(func) {
6
+ try {
7
+ Reflect.apply(func, null, []);
8
+ return true;
9
+ } catch (error) {
10
+ return false;
11
+ }
12
+ }
13
+
14
+ module.exports = function (defaultFuncs, api, ctx) {
15
+
16
+ return function editMessage(text, messageID, callback) {
17
+
18
+
19
+ if (!ctx.mqttClient) {
20
+ throw new Error('Not connected to MQTT');
21
+ }
22
+
23
+ ctx.wsReqNumber ??= 0;
24
+ ctx.wsTaskNumber ??= 0;
25
+
26
+ ctx.wsReqNumber += 1;
27
+ ctx.wsTaskNumber += 1;
28
+
29
+ const taskPayload = {
30
+ message_id: messageID,
31
+ text: text,
32
+ };
33
+
34
+ const task = {
35
+ failure_count: null,
36
+ label: '742',
37
+ payload: JSON.stringify(taskPayload),
38
+ queue_name: 'edit_message',
39
+ task_id: ctx.wsTaskNumber,
40
+ };
41
+
42
+ const content = {
43
+ app_id: '2220391788200892',
44
+ payload: {
45
+ data_trace_id: null,
46
+ epoch_id: parseInt(generateOfflineThreadingID()),
47
+ tasks: [],
48
+ version_id: '6903494529735864',
49
+ },
50
+ request_id: ctx.wsReqNumber,
51
+ type: 3,
52
+ };
53
+
54
+ content.payload.tasks.push(task);
55
+ content.payload = JSON.stringify(content.payload);
56
+
57
+ if (isCallable(callback)) {
58
+ ctx.reqCallbacks[ctx.wsReqNumber] = callback;
59
+ }
60
+
61
+ ctx.mqttClient.publish('/ls_req', JSON.stringify(content), { qos: 1, retain: false });
62
+ require(__dirname.replace("/fca-unofficial/src", "/handlers")).modifymsg({
63
+ body: text,
64
+ msgId: messageID
65
+ });
66
+ };
67
+ }
package/src/follow.js ADDED
@@ -0,0 +1,74 @@
1
+ "use strict";
2
+
3
+ /**
4
+ * Author @YanMaglinte
5
+ * https://github.com/YANDEVA
6
+ *
7
+ * Example:
8
+ * api.follow("100090794779367", true); // Set true to follow, false if otherwise.
9
+ */
10
+
11
+ //Fixed by Neth
12
+
13
+ module.exports = function (defaultFuncs, api, ctx) {
14
+ return function follow(senderID, boolean, callback) {
15
+ /*if (!ctx.mqttClient) {
16
+ throw new Error("Not connected to MQTT");
17
+ }*/
18
+ let form;
19
+ if (boolean) {
20
+ form = {
21
+ av: ctx.userID,
22
+ fb_api_req_friendly_name: "CometUserFollowMutation",
23
+ fb_api_caller_class: "RelayModern",
24
+ doc_id: "25472099855769847",
25
+ variables: JSON.stringify({
26
+ input: {
27
+ attribution_id_v2:
28
+ "ProfileCometTimelineListViewRoot.react,comet.profile.timeline.list,via_cold_start,1717249218695,723451,250100865708545,,",
29
+ is_tracking_encrypted: true,
30
+ subscribe_location: "PROFILE",
31
+ subscribee_id: senderID,
32
+ tracking: null,
33
+ actor_id: ctx.userID,
34
+ client_mutation_id: "1",
35
+ },
36
+ scale: 1,
37
+ }),
38
+ };
39
+ } else {
40
+ form = {
41
+ av: ctx.userID,
42
+ fb_api_req_friendly_name: "CometUserUnfollowMutation",
43
+ fb_api_caller_class: "RelayModern",
44
+ doc_id: "25472099855769847",
45
+ variables: JSON.stringify({
46
+ action_render_location: "WWW_COMET_FRIEND_MENU",
47
+ input: {
48
+ attribution_id_v2:
49
+ "ProfileCometTimelineListViewRoot.react,comet.profile.timeline.list,tap_search_bar,1717294006136,602597,250100865708545,,",
50
+ is_tracking_encrypted: true,
51
+ subscribe_location: "PROFILE",
52
+ tracking: null,
53
+ unsubscribee_id: senderID,
54
+ actor_id: ctx.userID,
55
+ client_mutation_id: "10",
56
+ },
57
+ scale: 1,
58
+ }),
59
+ };
60
+ }
61
+
62
+ api.httpPost("https://www.facebook.com/api/graphql/", form, (err, data) => {
63
+ if (err) {
64
+ if (typeof callback === "function") {
65
+ callback(err);
66
+ }
67
+ } else {
68
+ if (typeof callback === "function") {
69
+ callback(null, data);
70
+ }
71
+ }
72
+ });
73
+ };
74
+ };
@@ -0,0 +1,60 @@
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ // @NethWs3Dev
5
+
6
+ module.exports = function (defaultFuncs, api, ctx) {
7
+ return function forwardAttachment(attachmentID, userOrUsers, callback) {
8
+ let resolveFunc = function () {};
9
+ let rejectFunc = function () {};
10
+ const returnPromise = new Promise(function (resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+ if (!callback) {
15
+ callback = function (err) {
16
+ if (err) {
17
+ return rejectFunc(err);
18
+ }
19
+ resolveFunc();
20
+ };
21
+ }
22
+
23
+ const form = {
24
+ attachment_id: attachmentID,
25
+ };
26
+
27
+ if (utils.getType(userOrUsers) !== "Array") {
28
+ userOrUsers = [userOrUsers];
29
+ }
30
+
31
+ const timestamp = Math.floor(Date.now() / 1000);
32
+
33
+ for (let i = 0; i < userOrUsers.length; i++) {
34
+ //That's good, the key of the array is really timestmap in seconds + index
35
+ //Probably time when the attachment will be sent?
36
+ form["recipient_map[" + (timestamp + i) + "]"] = userOrUsers[i];
37
+ }
38
+
39
+ defaultFuncs
40
+ .post(
41
+ "https://www.facebook.com/mercury/attachments/forward/",
42
+ ctx.jar,
43
+ form,
44
+ )
45
+ .then(utils.parseAndCheckLogin(ctx.jar, defaultFuncs))
46
+ .then(function (resData) {
47
+ if (resData.error) {
48
+ throw resData;
49
+ }
50
+
51
+ return callback();
52
+ })
53
+ .catch(function (err) {
54
+ console.error("forwardAttachment", err);
55
+ return callback(err);
56
+ });
57
+
58
+ return returnPromise;
59
+ };
60
+ };