alicezetion 1.6.7 → 1.6.8

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 (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
@@ -1,80 +1,80 @@
1
- "use strict";
2
-
3
- var utils = require("../utils");
4
- var log = require("npmlog");
5
-
6
- module.exports = function(defaultFuncs, api, ctx) {
7
- return function getThreadInfo(threadID, callback) {
8
- var resolveFunc = function(){};
9
- var rejectFunc = function(){};
10
- var returnPromise = new Promise(function (resolve, reject) {
11
- resolveFunc = resolve;
12
- rejectFunc = reject;
13
- });
14
-
15
- if (!callback) {
16
- callback = function (err, friendList) {
17
- if (err) {
18
- return rejectFunc(err);
19
- }
20
- resolveFunc(friendList);
21
- };
22
- }
23
-
24
- var form = {
25
- client: "mercury"
26
- };
27
-
28
- api.getUserInfo(threadID, function(err, userRes) {
29
- if (err) {
30
- return callback(err);
31
- }
32
- var key = Object.keys(userRes).length > 0 ? "user_ids" : "thread_fbids";
33
- form["threads[" + key + "][0]"] = threadID;
34
-
35
- if (ctx.globalOptions.pageId)
36
- form.request_user_id = ctx.globalOptions.pageId;
37
-
38
- defaultFuncs
39
- .post(
40
- "https://www.facebook.com/ajax/mercury/thread_info.php",
41
- ctx.jar,
42
- form
43
- )
44
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
45
- .then(function(resData) {
46
- if (resData.error) {
47
- throw resData;
48
- } else if (!resData.payload) {
49
- throw {
50
- error: "Could not retrieve thread Info."
51
- };
52
- }
53
- var threadData = resData.payload.threads[0];
54
- var userData = userRes[threadID];
55
-
56
- if (threadData == null) {
57
- throw {
58
- error: "ThreadData is null"
59
- };
60
- }
61
-
62
- threadData.name =
63
- userData != null && userData.name != null
64
- ? userData.name
65
- : threadData.name;
66
- threadData.image_src =
67
- userData != null && userData.thumbSrc != null
68
- ? userData.thumbSrc
69
- : threadData.image_src;
70
-
71
- callback(null, utils.formatThread(threadData));
72
- })
73
- .catch(function(err) {
74
- log.error("getThreadInfo", err);
75
- return callback(err);
76
- });
77
- });
78
- return returnPromise;
79
- };
80
- };
1
+ "use strict";
2
+
3
+ var utils = require("../utils");
4
+ var log = require("npmlog");
5
+
6
+ module.exports = function(defaultFuncs, api, ctx) {
7
+ return function getThreadInfo(threadID, callback) {
8
+ var resolveFunc = function(){};
9
+ var rejectFunc = function(){};
10
+ var returnPromise = new Promise(function (resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+
15
+ if (!callback) {
16
+ callback = function (err, friendList) {
17
+ if (err) {
18
+ return rejectFunc(err);
19
+ }
20
+ resolveFunc(friendList);
21
+ };
22
+ }
23
+
24
+ var form = {
25
+ client: "mercury"
26
+ };
27
+
28
+ api.getUserInfo(threadID, function(err, userRes) {
29
+ if (err) {
30
+ return callback(err);
31
+ }
32
+ var key = Object.keys(userRes).length > 0 ? "user_ids" : "thread_fbids";
33
+ form["threads[" + key + "][0]"] = threadID;
34
+
35
+ if (ctx.globalOptions.pageId)
36
+ form.request_user_id = ctx.globalOptions.pageId;
37
+
38
+ defaultFuncs
39
+ .post(
40
+ "https://www.facebook.com/ajax/mercury/thread_info.php",
41
+ ctx.jar,
42
+ form
43
+ )
44
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
45
+ .then(function(resData) {
46
+ if (resData.error) {
47
+ throw resData;
48
+ } else if (!resData.payload) {
49
+ throw {
50
+ error: "Could not retrieve thread Info."
51
+ };
52
+ }
53
+ var threadData = resData.payload.threads[0];
54
+ var userData = userRes[threadID];
55
+
56
+ if (threadData == null) {
57
+ throw {
58
+ error: "ThreadData is null"
59
+ };
60
+ }
61
+
62
+ threadData.name =
63
+ userData != null && userData.name != null
64
+ ? userData.name
65
+ : threadData.name;
66
+ threadData.image_src =
67
+ userData != null && userData.thumbSrc != null
68
+ ? userData.thumbSrc
69
+ : threadData.image_src;
70
+
71
+ callback(null, utils.formatThread(threadData));
72
+ })
73
+ .catch(function(err) {
74
+ log.error("getThreadInfo", err);
75
+ return callback(err);
76
+ });
77
+ });
78
+ return returnPromise;
79
+ };
80
+ };
@@ -1,238 +1,238 @@
1
- "use strict";
2
-
3
- const utils = require("../utils");
4
- const log = require("npmlog");
5
-
6
- function createProfileUrl(url, username, id) {
7
- if (url) return url;
8
- return "https://www.facebook.com/" + (username || utils.formatID(id.toString()));
9
- }
10
-
11
- function formatParticipants(participants) {
12
- return participants.edges.map((p)=>{
13
- p = p.node.messaging_actor;
14
- switch (p["__typename"]) {
15
- case "User":
16
- return {
17
- accountType: p["__typename"],
18
- userID: utils.formatID(p.id.toString()), // do we need .toString()? when it is not a string?
19
- name: p.name,
20
- shortName: p.short_name,
21
- gender: p.gender,
22
- url: p.url, // how about making it profileURL
23
- profilePicture: p.big_image_src.uri,
24
- username: (p.username||null),
25
- // TODO: maybe better names for these?
26
- isViewerFriend: p.is_viewer_friend, // true/false
27
- isMessengerUser: p.is_messenger_user, // true/false
28
- isVerified: p.is_verified, // true/false
29
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
30
- isViewerCoworker: p.is_viewer_coworker, // true/false
31
- isEmployee: p.is_employee // null? when it is something other? can someone check?
32
- };
33
- case "Page":
34
- return {
35
- accountType: p["__typename"],
36
- userID: utils.formatID(p.id.toString()), // or maybe... pageID?
37
- name: p.name,
38
- url: p.url,
39
- profilePicture: p.big_image_src.uri,
40
- username: (p.username||null),
41
- // uhm... better names maybe?
42
- acceptsMessengerUserFeedback: p.accepts_messenger_user_feedback, // true/false
43
- isMessengerUser: p.is_messenger_user, // true/false
44
- isVerified: p.is_verified, // true/false
45
- isMessengerPlatformBot: p.is_messenger_platform_bot, // true/false
46
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
47
- };
48
- case "ReducedMessagingActor":
49
- return {
50
- accountType: p["__typename"],
51
- userID: utils.formatID(p.id.toString()),
52
- name: p.name,
53
- url: createProfileUrl(p.url, p.username, p.id), // in this case p.url is null all the time
54
- profilePicture: p.big_image_src.uri, // in this case it is default facebook photo, we could determine gender using it
55
- username: (p.username||null), // maybe we could use it to generate profile URL?
56
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
57
- };
58
- case "UnavailableMessagingActor":
59
- return {
60
- accountType: p["__typename"],
61
- userID: utils.formatID(p.id.toString()),
62
- name: p.name, // "Facebook User" in user's language
63
- url: createProfileUrl(p.url, p.username, p.id), // in this case p.url is null all the time
64
- profilePicture: p.big_image_src.uri, // default male facebook photo
65
- username: (p.username||null), // maybe we could use it to generate profile URL?
66
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
67
- };
68
- default:
69
- log.warn("getThreadList", "Found participant with unsupported typename. Please open an issue at https://github.com/Schmavery/fca-unofficial/issues\n" + JSON.stringify(p, null, 2));
70
- return {
71
- accountType: p["__typename"],
72
- userID: utils.formatID(p.id.toString()),
73
- name: p.name || `[unknown ${p["__typename"]}]`, // probably it will always be something... but fallback to [unknown], just in case
74
- };
75
- }
76
- });
77
- }
78
-
79
- // "FF8C0077" -> "8C0077"
80
- function formatColor(color) {
81
- if (color && color.match(/^(?:[0-9a-fA-F]{8})$/g)) {
82
- return color.slice(2);
83
- }
84
- return color;
85
- }
86
-
87
- function getThreadName(t) {
88
- if (t.name || t.thread_key.thread_fbid) return t.name;
89
-
90
- for (let po of t.all_participants.edges) {
91
- let p = po.node;
92
- if (p.messaging_actor.id === t.thread_key.other_user_id) return p.messaging_actor.name;
93
- }
94
- }
95
-
96
- function mapNicknames(customizationInfo) {
97
- return (customizationInfo && customizationInfo.participant_customizations) ? customizationInfo.participant_customizations.map(u => {
98
- return {
99
- "userID": u.participant_id,
100
- "nickname": u.nickname
101
- };
102
- }):[];
103
- }
104
-
105
- function formatThreadList(data) {
106
- return data.map(t => {
107
- let lastMessageNode = (t.last_message&&t.last_message.nodes&&t.last_message.nodes.length>0)?t.last_message.nodes[0]:null;
108
- return {
109
- threadID: t.thread_key?utils.formatID(t.thread_key.thread_fbid || t.thread_key.other_user_id):null, // shall never be null
110
- name: getThreadName(t),
111
- unreadCount: t.unread_count,
112
- messageCount: t.messages_count,
113
- imageSrc: t.image?t.image.uri:null,
114
- emoji: t.customization_info?t.customization_info.emoji:null,
115
- color: formatColor(t.customization_info?t.customization_info.outgoing_bubble_color:null),
116
- nicknames: mapNicknames(t.customization_info),
117
- muteUntil: t.mute_until,
118
- participants: formatParticipants(t.all_participants),
119
- adminIDs: t.thread_admins.map(a => a.id),
120
- folder: t.folder,
121
- isGroup: t.thread_type === "GROUP",
122
- // rtc_call_data: t.rtc_call_data, // TODO: format and document this
123
- // isPinProtected: t.is_pin_protected, // feature from future? always false (2018-04-04)
124
- customizationEnabled: t.customization_enabled, // false for ONE_TO_ONE with Page or ReducedMessagingActor
125
- participantAddMode: t.participant_add_mode_as_string, // "ADD" if "GROUP" and null if "ONE_TO_ONE"
126
- montageThread: t.montage_thread?Buffer.from(t.montage_thread.id,"base64").toString():null, // base64 encoded string "message_thread:0000000000000000"
127
- // it is not userID nor any other ID known to me...
128
- // can somebody inspect it? where is it used?
129
- // probably Messenger Day uses it
130
- reactionsMuteMode: t.reactions_mute_mode,
131
- mentionsMuteMode: t.mentions_mute_mode,
132
- isArchived: t.has_viewer_archived,
133
- isSubscribed: t.is_viewer_subscribed,
134
- timestamp: t.updated_time_precise, // in miliseconds
135
- // isCanonicalUser: t.is_canonical_neo_user, // is it always false?
136
- // TODO: how about putting snippet in another object? current implementation does not handle every possibile message type etc.
137
- snippet: lastMessageNode?lastMessageNode.snippet:null,
138
- snippetAttachments: lastMessageNode?lastMessageNode.extensible_attachment:null, // TODO: not sure if it works
139
- snippetSender: lastMessageNode?utils.formatID((lastMessageNode.message_sender.messaging_actor.id || "").toString()):null,
140
- lastMessageTimestamp: lastMessageNode?lastMessageNode.timestamp_precise:null, // timestamp in miliseconds
141
- lastReadTimestamp: (t.last_read_receipt&&t.last_read_receipt.nodes.length>0)
142
- ? (t.last_read_receipt.nodes[0]?t.last_read_receipt.nodes[0].timestamp_precise:null)
143
- : null, // timestamp in miliseconds
144
- cannotReplyReason: t.cannot_reply_reason, // TODO: inspect possible values
145
- approvalMode: Boolean(t.approval_mode),
146
-
147
- // @Legacy
148
- participantIDs: formatParticipants(t.all_participants).map(participant => participant.userID),
149
- threadType: t.thread_type === "GROUP" ? 2 : 1 // "GROUP" or "ONE_TO_ONE"
150
- };
151
- });
152
- }
153
-
154
- module.exports = function(defaultFuncs, api, ctx) {
155
- return function getThreadList(limit, timestamp, tags, callback) {
156
- if (!callback && (utils.getType(tags) === "Function" || utils.getType(tags) === "AsyncFunction")) {
157
- callback = tags;
158
- tags = [""];
159
- }
160
- if (utils.getType(limit) !== "Number" || !Number.isInteger(limit) || limit <= 0) {
161
- throw {error: "getThreadList: limit must be a positive integer"};
162
- }
163
- if (utils.getType(timestamp) !== "Null" &&
164
- (utils.getType(timestamp) !== "Number" || !Number.isInteger(timestamp))) {
165
- throw {error: "getThreadList: timestamp must be an integer or null"};
166
- }
167
- if (utils.getType(tags) === "String") {
168
- tags = [tags];
169
- }
170
- if (utils.getType(tags) !== "Array") {
171
- throw {error: "getThreadList: tags must be an array"};
172
- }
173
-
174
- var resolveFunc = function(){};
175
- var rejectFunc = function(){};
176
- var returnPromise = new Promise(function (resolve, reject) {
177
- resolveFunc = resolve;
178
- rejectFunc = reject;
179
- });
180
-
181
- if (utils.getType(callback) !== "Function" && utils.getType(callback) !== "AsyncFunction") {
182
- callback = function (err, data) {
183
- if (err) {
184
- return rejectFunc(err);
185
- }
186
- resolveFunc(data);
187
- };
188
- }
189
-
190
- const form = {
191
- "av": ctx.globalOptions.pageID,
192
- "queries": JSON.stringify({
193
- "o0": {
194
- // This doc_id was valid on 2020-07-20
195
- "doc_id": "3336396659757871",
196
- "query_params": {
197
- "limit": limit+(timestamp?1:0),
198
- "before": timestamp,
199
- "tags": tags,
200
- "includeDeliveryReceipts": true,
201
- "includeSeqID": false
202
- }
203
- }
204
- }),
205
- "batch_name": "MessengerGraphQLThreadlistFetcher"
206
- };
207
-
208
- defaultFuncs
209
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
210
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
211
- .then((resData) => {
212
- if (resData[resData.length - 1].error_results > 0) {
213
- throw resData[0].o0.errors;
214
- }
215
-
216
- if (resData[resData.length - 1].successful_results === 0) {
217
- throw {error: "getThreadList: there was no successful_results", res: resData};
218
- }
219
-
220
- // When we ask for threads using timestamp from the previous request,
221
- // we are getting the last thread repeated as the first thread in this response.
222
- // .shift() gets rid of it
223
- // It is also the reason for increasing limit by 1 when timestamp is set
224
- // this way user asks for 10 threads, we are asking for 11,
225
- // but after removing the duplicated one, it is again 10
226
- if (timestamp) {
227
- resData[0].o0.data.viewer.message_threads.nodes.shift();
228
- }
229
- callback(null, formatThreadList(resData[0].o0.data.viewer.message_threads.nodes));
230
- })
231
- .catch((err) => {
232
- log.error("getThreadList", err);
233
- return callback(err);
234
- });
235
-
236
- return returnPromise;
237
- };
238
- };
1
+ "use strict";
2
+
3
+ const utils = require("../utils");
4
+ const log = require("npmlog");
5
+
6
+ function createProfileUrl(url, username, id) {
7
+ if (url) return url;
8
+ return "https://www.facebook.com/" + (username || utils.formatID(id.toString()));
9
+ }
10
+
11
+ function formatParticipants(participants) {
12
+ return participants.edges.map((p)=>{
13
+ p = p.node.messaging_actor;
14
+ switch (p["__typename"]) {
15
+ case "User":
16
+ return {
17
+ accountType: p["__typename"],
18
+ userID: utils.formatID(p.id.toString()), // do we need .toString()? when it is not a string?
19
+ name: p.name,
20
+ shortName: p.short_name,
21
+ gender: p.gender,
22
+ url: p.url, // how about making it profileURL
23
+ profilePicture: p.big_image_src.uri,
24
+ username: (p.username||null),
25
+ // TODO: maybe better names for these?
26
+ isViewerFriend: p.is_viewer_friend, // true/false
27
+ isMessengerUser: p.is_messenger_user, // true/false
28
+ isVerified: p.is_verified, // true/false
29
+ isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
30
+ isViewerCoworker: p.is_viewer_coworker, // true/false
31
+ isEmployee: p.is_employee // null? when it is something other? can someone check?
32
+ };
33
+ case "Page":
34
+ return {
35
+ accountType: p["__typename"],
36
+ userID: utils.formatID(p.id.toString()), // or maybe... pageID?
37
+ name: p.name,
38
+ url: p.url,
39
+ profilePicture: p.big_image_src.uri,
40
+ username: (p.username||null),
41
+ // uhm... better names maybe?
42
+ acceptsMessengerUserFeedback: p.accepts_messenger_user_feedback, // true/false
43
+ isMessengerUser: p.is_messenger_user, // true/false
44
+ isVerified: p.is_verified, // true/false
45
+ isMessengerPlatformBot: p.is_messenger_platform_bot, // true/false
46
+ isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
47
+ };
48
+ case "ReducedMessagingActor":
49
+ return {
50
+ accountType: p["__typename"],
51
+ userID: utils.formatID(p.id.toString()),
52
+ name: p.name,
53
+ url: createProfileUrl(p.url, p.username, p.id), // in this case p.url is null all the time
54
+ profilePicture: p.big_image_src.uri, // in this case it is default facebook photo, we could determine gender using it
55
+ username: (p.username||null), // maybe we could use it to generate profile URL?
56
+ isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
57
+ };
58
+ case "UnavailableMessagingActor":
59
+ return {
60
+ accountType: p["__typename"],
61
+ userID: utils.formatID(p.id.toString()),
62
+ name: p.name, // "Facebook User" in user's language
63
+ url: createProfileUrl(p.url, p.username, p.id), // in this case p.url is null all the time
64
+ profilePicture: p.big_image_src.uri, // default male facebook photo
65
+ username: (p.username||null), // maybe we could use it to generate profile URL?
66
+ isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
67
+ };
68
+ default:
69
+ log.warn("getThreadList", "Found participant with unsupported typename. Please open an issue at https://github.com/Schmavery/fca-unofficial/issues\n" + JSON.stringify(p, null, 2));
70
+ return {
71
+ accountType: p["__typename"],
72
+ userID: utils.formatID(p.id.toString()),
73
+ name: p.name || `[unknown ${p["__typename"]}]`, // probably it will always be something... but fallback to [unknown], just in case
74
+ };
75
+ }
76
+ });
77
+ }
78
+
79
+ // "FF8C0077" -> "8C0077"
80
+ function formatColor(color) {
81
+ if (color && color.match(/^(?:[0-9a-fA-F]{8})$/g)) {
82
+ return color.slice(2);
83
+ }
84
+ return color;
85
+ }
86
+
87
+ function getThreadName(t) {
88
+ if (t.name || t.thread_key.thread_fbid) return t.name;
89
+
90
+ for (let po of t.all_participants.edges) {
91
+ let p = po.node;
92
+ if (p.messaging_actor.id === t.thread_key.other_user_id) return p.messaging_actor.name;
93
+ }
94
+ }
95
+
96
+ function mapNicknames(customizationInfo) {
97
+ return (customizationInfo && customizationInfo.participant_customizations) ? customizationInfo.participant_customizations.map(u => {
98
+ return {
99
+ "userID": u.participant_id,
100
+ "nickname": u.nickname
101
+ };
102
+ }):[];
103
+ }
104
+
105
+ function formatThreadList(data) {
106
+ return data.map(t => {
107
+ let lastMessageNode = (t.last_message&&t.last_message.nodes&&t.last_message.nodes.length>0)?t.last_message.nodes[0]:null;
108
+ return {
109
+ threadID: t.thread_key?utils.formatID(t.thread_key.thread_fbid || t.thread_key.other_user_id):null, // shall never be null
110
+ name: getThreadName(t),
111
+ unreadCount: t.unread_count,
112
+ messageCount: t.messages_count,
113
+ imageSrc: t.image?t.image.uri:null,
114
+ emoji: t.customization_info?t.customization_info.emoji:null,
115
+ color: formatColor(t.customization_info?t.customization_info.outgoing_bubble_color:null),
116
+ nicknames: mapNicknames(t.customization_info),
117
+ muteUntil: t.mute_until,
118
+ participants: formatParticipants(t.all_participants),
119
+ adminIDs: t.thread_admins.map(a => a.id),
120
+ folder: t.folder,
121
+ isGroup: t.thread_type === "GROUP",
122
+ // rtc_call_data: t.rtc_call_data, // TODO: format and document this
123
+ // isPinProtected: t.is_pin_protected, // feature from future? always false (2018-04-04)
124
+ customizationEnabled: t.customization_enabled, // false for ONE_TO_ONE with Page or ReducedMessagingActor
125
+ participantAddMode: t.participant_add_mode_as_string, // "ADD" if "GROUP" and null if "ONE_TO_ONE"
126
+ montageThread: t.montage_thread?Buffer.from(t.montage_thread.id,"base64").toString():null, // base64 encoded string "message_thread:0000000000000000"
127
+ // it is not userID nor any other ID known to me...
128
+ // can somebody inspect it? where is it used?
129
+ // probably Messenger Day uses it
130
+ reactionsMuteMode: t.reactions_mute_mode,
131
+ mentionsMuteMode: t.mentions_mute_mode,
132
+ isArchived: t.has_viewer_archived,
133
+ isSubscribed: t.is_viewer_subscribed,
134
+ timestamp: t.updated_time_precise, // in miliseconds
135
+ // isCanonicalUser: t.is_canonical_neo_user, // is it always false?
136
+ // TODO: how about putting snippet in another object? current implementation does not handle every possibile message type etc.
137
+ snippet: lastMessageNode?lastMessageNode.snippet:null,
138
+ snippetAttachments: lastMessageNode?lastMessageNode.extensible_attachment:null, // TODO: not sure if it works
139
+ snippetSender: lastMessageNode?utils.formatID((lastMessageNode.message_sender.messaging_actor.id || "").toString()):null,
140
+ lastMessageTimestamp: lastMessageNode?lastMessageNode.timestamp_precise:null, // timestamp in miliseconds
141
+ lastReadTimestamp: (t.last_read_receipt&&t.last_read_receipt.nodes.length>0)
142
+ ? (t.last_read_receipt.nodes[0]?t.last_read_receipt.nodes[0].timestamp_precise:null)
143
+ : null, // timestamp in miliseconds
144
+ cannotReplyReason: t.cannot_reply_reason, // TODO: inspect possible values
145
+ approvalMode: Boolean(t.approval_mode),
146
+
147
+ // @Legacy
148
+ participantIDs: formatParticipants(t.all_participants).map(participant => participant.userID),
149
+ threadType: t.thread_type === "GROUP" ? 2 : 1 // "GROUP" or "ONE_TO_ONE"
150
+ };
151
+ });
152
+ }
153
+
154
+ module.exports = function(defaultFuncs, api, ctx) {
155
+ return function getThreadList(limit, timestamp, tags, callback) {
156
+ if (!callback && (utils.getType(tags) === "Function" || utils.getType(tags) === "AsyncFunction")) {
157
+ callback = tags;
158
+ tags = [""];
159
+ }
160
+ if (utils.getType(limit) !== "Number" || !Number.isInteger(limit) || limit <= 0) {
161
+ throw {error: "getThreadList: limit must be a positive integer"};
162
+ }
163
+ if (utils.getType(timestamp) !== "Null" &&
164
+ (utils.getType(timestamp) !== "Number" || !Number.isInteger(timestamp))) {
165
+ throw {error: "getThreadList: timestamp must be an integer or null"};
166
+ }
167
+ if (utils.getType(tags) === "String") {
168
+ tags = [tags];
169
+ }
170
+ if (utils.getType(tags) !== "Array") {
171
+ throw {error: "getThreadList: tags must be an array"};
172
+ }
173
+
174
+ var resolveFunc = function(){};
175
+ var rejectFunc = function(){};
176
+ var returnPromise = new Promise(function (resolve, reject) {
177
+ resolveFunc = resolve;
178
+ rejectFunc = reject;
179
+ });
180
+
181
+ if (utils.getType(callback) !== "Function" && utils.getType(callback) !== "AsyncFunction") {
182
+ callback = function (err, data) {
183
+ if (err) {
184
+ return rejectFunc(err);
185
+ }
186
+ resolveFunc(data);
187
+ };
188
+ }
189
+
190
+ const form = {
191
+ "av": ctx.globalOptions.pageID,
192
+ "queries": JSON.stringify({
193
+ "o0": {
194
+ // This doc_id was valid on 2020-07-20
195
+ "doc_id": "3336396659757871",
196
+ "query_params": {
197
+ "limit": limit+(timestamp?1:0),
198
+ "before": timestamp,
199
+ "tags": tags,
200
+ "includeDeliveryReceipts": true,
201
+ "includeSeqID": false
202
+ }
203
+ }
204
+ }),
205
+ "batch_name": "MessengerGraphQLThreadlistFetcher"
206
+ };
207
+
208
+ defaultFuncs
209
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
210
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
211
+ .then((resData) => {
212
+ if (resData[resData.length - 1].error_results > 0) {
213
+ throw resData[0].o0.errors;
214
+ }
215
+
216
+ if (resData[resData.length - 1].successful_results === 0) {
217
+ throw {error: "getThreadList: there was no successful_results", res: resData};
218
+ }
219
+
220
+ // When we ask for threads using timestamp from the previous request,
221
+ // we are getting the last thread repeated as the first thread in this response.
222
+ // .shift() gets rid of it
223
+ // It is also the reason for increasing limit by 1 when timestamp is set
224
+ // this way user asks for 10 threads, we are asking for 11,
225
+ // but after removing the duplicated one, it is again 10
226
+ if (timestamp) {
227
+ resData[0].o0.data.viewer.message_threads.nodes.shift();
228
+ }
229
+ callback(null, formatThreadList(resData[0].o0.data.viewer.message_threads.nodes));
230
+ })
231
+ .catch((err) => {
232
+ log.error("getThreadList", err);
233
+ return callback(err);
234
+ });
235
+
236
+ return returnPromise;
237
+ };
238
+ };