fca-arif-babu 1.0.21 → 8.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 (152) hide show
  1. package/Extra/Balancer.js +49 -0
  2. package/Extra/Bypass/956/index.js +234 -0
  3. package/Extra/Bypass/test/aaaa.json +169 -0
  4. package/Extra/Bypass/test/index.js +188 -0
  5. package/Extra/Database/index.js +469 -0
  6. package/Extra/ExtraAddons.js +82 -0
  7. package/Extra/ExtraFindUID.js +62 -0
  8. package/Extra/ExtraGetThread.js +365 -0
  9. package/Extra/ExtraScreenShot.js +430 -0
  10. package/Extra/ExtraUptimeRobot.js +38 -0
  11. package/Extra/Html/Classic/script.js +119 -0
  12. package/Extra/Html/Classic/style.css +8 -0
  13. package/Extra/Security/AES_256_GCM/index.js +1 -0
  14. package/Extra/Security/Base/Step_1.js +6 -0
  15. package/Extra/Security/Base/Step_2.js +22 -0
  16. package/Extra/Security/Base/Step_3.js +22 -0
  17. package/Extra/Security/Base/index.js +191 -0
  18. package/Extra/Security/Step_1.js +6 -0
  19. package/Extra/Security/Step_2.js +22 -0
  20. package/Extra/Security/Step_3.js +22 -0
  21. package/Extra/Security/index.js +5 -0
  22. package/Extra/Src/Change_Environment.js +24 -0
  23. package/Extra/Src/Check_Update.js +67 -0
  24. package/Extra/Src/History.js +115 -0
  25. package/Extra/Src/Instant_Update.js +65 -0
  26. package/Extra/Src/Last-Run.js +65 -0
  27. package/Extra/Src/Premium.js +81 -0
  28. package/Extra/Src/Release_Memory.js +160 -0
  29. package/Extra/Src/Websocket.js +213 -0
  30. package/Extra/Src/image/62518eafb0670b697788ce4f9a4f71d1.jpg +0 -0
  31. package/Extra/Src/test.js +28 -0
  32. package/Extra/Src/uuid.js +137 -0
  33. package/Func/AcceptAgreement.js +31 -0
  34. package/Func/ClearCache.js +64 -0
  35. package/Func/ReportV1.js +54 -0
  36. package/{LICENSE-MIT → LICENSE.txt} +6 -6
  37. package/Language/index.json +228 -0
  38. package/Main.js +1290 -0
  39. package/README.md +8 -6
  40. package/SECURITY.md +17 -0
  41. package/broadcast.js +44 -0
  42. package/index.js +424 -465
  43. package/logger.js +66 -0
  44. package/package.json +91 -42
  45. package/src/Dev_Horizon_Data.js +125 -0
  46. package/src/Dev_getThreadInfoOLD.js +422 -0
  47. package/src/Dev_shareTest2.js +68 -0
  48. package/src/Dev_shareTest3.js +71 -0
  49. package/src/Premium.js +25 -0
  50. package/src/Screenshot.js +83 -0
  51. package/src/addExternalModule.js +5 -14
  52. package/src/addUserToGroup.js +21 -57
  53. package/src/changeAdminStatus.js +22 -46
  54. package/src/changeArchivedStatus.js +12 -26
  55. package/src/{changeAvatarV2.js → changeAvt.js} +2 -3
  56. package/src/changeBio.js +15 -26
  57. package/src/changeBlockedStatus.js +9 -22
  58. package/src/changeGroupImage.js +26 -55
  59. package/src/changeNickname.js +14 -28
  60. package/src/changeThreadColor.js +19 -22
  61. package/src/changeThreadEmoji.js +12 -25
  62. package/src/createNewGroup.js +18 -36
  63. package/src/createPoll.js +17 -28
  64. package/src/deleteMessage.js +14 -25
  65. package/src/deleteThread.js +13 -26
  66. package/src/editMessage.js +50 -56
  67. package/src/forwardAttachment.js +16 -28
  68. package/src/getAccessToken.js +28 -0
  69. package/src/getCurrentUserID.js +3 -3
  70. package/src/getEmojiUrl.js +8 -10
  71. package/src/getFriendsList.js +15 -25
  72. package/src/getMessage.js +81 -813
  73. package/src/getThreadHistory.js +98 -241
  74. package/src/getThreadInfo.js +286 -89
  75. package/src/getThreadInfoOLD.js +422 -0
  76. package/src/getThreadList.js +158 -215
  77. package/src/getThreadMain.js +220 -0
  78. package/src/getThreadPictures.js +20 -40
  79. package/src/getUID.js +49 -112
  80. package/src/getUserID.js +14 -18
  81. package/src/getUserInfo.js +75 -34
  82. package/src/getUserInfoMain.js +65 -0
  83. package/src/getUserInfoV2.js +32 -0
  84. package/src/getUserInfoV3.js +63 -0
  85. package/src/getUserInfoV4.js +55 -0
  86. package/src/getUserInfoV5.js +61 -0
  87. package/src/handleFriendRequest.js +14 -25
  88. package/src/handleMessageRequest.js +21 -37
  89. package/src/httpGet.js +22 -37
  90. package/src/httpPost.js +20 -36
  91. package/src/httpPostFormData.js +23 -52
  92. package/src/listenMqtt.js +667 -414
  93. package/src/listenMqttV1.js +846 -0
  94. package/src/logout.js +17 -24
  95. package/src/markAsDelivered.js +16 -23
  96. package/src/markAsRead.js +28 -43
  97. package/src/markAsReadAll.js +14 -21
  98. package/src/markAsSeen.js +19 -29
  99. package/src/muteThread.js +13 -18
  100. package/src/removeUserFromGroup.js +18 -48
  101. package/src/resolvePhotoUrl.js +14 -22
  102. package/src/searchForThread.js +13 -23
  103. package/src/sendMessage.js +125 -181
  104. package/src/sendMqttMessage.js +71 -0
  105. package/src/sendTypingIndicator.js +80 -28
  106. package/src/setMessageReaction.js +20 -33
  107. package/src/setPostReaction.js +95 -105
  108. package/src/setTitle.js +19 -35
  109. package/src/shareContact.js +37 -92
  110. package/src/shareLink.js +5 -6
  111. package/src/threadColors.js +17 -109
  112. package/src/unfriend.js +11 -20
  113. package/src/unsendMessage.js +33 -30
  114. package/src/unsendMqttMessage.js +66 -0
  115. package/test/data/shareAttach.js +146 -0
  116. package/test/data/something.mov +1 -0
  117. package/test/data/test.png +1 -0
  118. package/test/data/test.txt +1 -0
  119. package/test/env/.env +1 -0
  120. package/test/example-config.json +18 -0
  121. package/test/test-page.js +140 -0
  122. package/test/test.js +385 -0
  123. package/test/testv2.js +3 -0
  124. package/utils.js +2908 -1247
  125. package/replit.nix +0 -3
  126. package/src/changeAvatar.js +0 -136
  127. package/src/changeBlockedStatusMqtt.js +0 -80
  128. package/src/changeCover.js +0 -72
  129. package/src/changeName.js +0 -78
  130. package/src/changeUsername.js +0 -58
  131. package/src/createCommentPost.js +0 -229
  132. package/src/createPost.js +0 -275
  133. package/src/data/getThreadInfo.json +0 -1
  134. package/src/editMessageOld.js +0 -67
  135. package/src/follow.js +0 -74
  136. package/src/getAccess.js +0 -112
  137. package/src/getAvatarUser.js +0 -78
  138. package/src/getBotInitialData.js +0 -42
  139. package/src/getRegion.js +0 -7
  140. package/src/getThreadHistoryDeprecated.js +0 -93
  141. package/src/getThreadInfoDeprecated.js +0 -80
  142. package/src/getThreadListDeprecated.js +0 -75
  143. package/src/listenNotification.js +0 -85
  144. package/src/pinMessage.js +0 -59
  145. package/src/refreshFb_dtsg.js +0 -89
  146. package/src/searchStickers.js +0 -53
  147. package/src/sendMessageMqtt.js +0 -316
  148. package/src/setMessageReactionMqtt.js +0 -62
  149. package/src/setProfileGuard.js +0 -44
  150. package/src/setStoryReaction.js +0 -64
  151. package/src/stopListenMqtt.js +0 -23
  152. package/src/uploadAttachment.js +0 -94
@@ -1,250 +1,197 @@
1
1
  "use strict";
2
2
 
3
3
  const utils = require("../utils");
4
- // @NethWs3Dev
4
+ const log = require("npmlog");
5
5
 
6
- function formatEventReminders(reminder) {
7
- return {
8
- reminderID: reminder.id,
9
- eventCreatorID: reminder.lightweight_event_creator.id,
10
- time: reminder.time,
11
- eventType: reminder.lightweight_event_type.toLowerCase(),
12
- locationName: reminder.location_name,
13
- // @TODO verify this
14
- locationCoordinates: reminder.location_coordinates,
15
- locationPage: reminder.location_page,
16
- eventStatus: reminder.lightweight_event_status.toLowerCase(),
17
- note: reminder.note,
18
- repeatMode: reminder.repeat_mode.toLowerCase(),
19
- eventTitle: reminder.event_title,
20
- triggerMessage: reminder.trigger_message,
21
- secondsToNotifyBefore: reminder.seconds_to_notify_before,
22
- allowsRsvp: reminder.allows_rsvp,
23
- relatedEvent: reminder.related_event,
24
- members: reminder.event_reminder_members.edges.map(function (member) {
25
- return {
26
- memberID: member.node.id,
27
- state: member.guest_list_state.toLowerCase(),
28
- };
29
- }),
30
- };
6
+ function createProfileUrl(url, username, id) {
7
+ if (url) return url;
8
+ return "https://www.facebook.com/" + (username || utils.formatID(id.toString()));
31
9
  }
32
10
 
33
- function formatThreadGraphQLResponse(messageThread) {
34
- const threadID = messageThread.thread_key.thread_fbid
35
- ? messageThread.thread_key.thread_fbid
36
- : messageThread.thread_key.other_user_id;
37
-
38
- // Remove me
39
- const lastM = messageThread.last_message;
40
- const snippetID =
41
- lastM &&
42
- lastM.nodes &&
43
- lastM.nodes[0] &&
44
- lastM.nodes[0].message_sender &&
45
- lastM.nodes[0].message_sender.messaging_actor
46
- ? lastM.nodes[0].message_sender.messaging_actor.id
47
- : null;
48
- const snippetText =
49
- lastM && lastM.nodes && lastM.nodes[0] ? lastM.nodes[0].snippet : null;
50
- const lastR = messageThread.last_read_receipt;
51
- const lastReadTimestamp =
52
- lastR && lastR.nodes && lastR.nodes[0] && lastR.nodes[0].timestamp_precise
53
- ? lastR.nodes[0].timestamp_precise
54
- : null;
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
+ case "UnavailableMessagingActor":
50
+ return {
51
+ accountType: p["__typename"],
52
+ userID: utils.formatID(p.id.toString()),
53
+ name: p.name,
54
+ url: createProfileUrl(p.url, p.username, p.id), // in this case p.url is null all the time
55
+ profilePicture: p.big_image_src.uri, // in this case it is default facebook photo, we could determine gender using it
56
+ username: (p.username || null), // maybe we could use it to generate profile URL?
57
+ isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
58
+ };
59
+ default:
60
+ log.warn("getThreadList", "Found participant with unsupported typename. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues\n" + JSON.stringify(p, null, 2));
61
+ return {
62
+ accountType: p["__typename"],
63
+ userID: utils.formatID(p.id.toString()),
64
+ name: p.name || `[unknown ${p["__typename"]}]`, // probably it will always be something... but fallback to [unknown], just in case
65
+ };
66
+ }
67
+ });
68
+ }
55
69
 
56
- return {
57
- threadID: threadID,
58
- threadName: messageThread.name,
59
- participantIDs: messageThread.all_participants.edges.map(
60
- (d) => d.node.messaging_actor.id,
61
- ),
62
- userInfo: messageThread.all_participants.edges.map((d) => ({
63
- id: d.node.messaging_actor.id,
64
- name: d.node.messaging_actor.name,
65
- firstName: d.node.messaging_actor.short_name,
66
- vanity: d.node.messaging_actor.username,
67
- url: d.node.messaging_actor.url,
68
- thumbSrc: d.node.messaging_actor.big_image_src.uri,
69
- profileUrl: d.node.messaging_actor.big_image_src.uri,
70
- gender: d.node.messaging_actor.gender,
71
- type: d.node.messaging_actor.__typename,
72
- isFriend: d.node.messaging_actor.is_viewer_friend,
73
- isBirthday: !!d.node.messaging_actor.is_birthday, //not sure?
74
- })),
75
- unreadCount: messageThread.unread_count,
76
- messageCount: messageThread.messages_count,
77
- timestamp: messageThread.updated_time_precise,
78
- muteUntil: messageThread.mute_until,
79
- isGroup: messageThread.thread_type == "GROUP",
80
- isSubscribed: messageThread.is_viewer_subscribed,
81
- isArchived: messageThread.has_viewer_archived,
82
- folder: messageThread.folder,
83
- cannotReplyReason: messageThread.cannot_reply_reason,
84
- eventReminders: messageThread.event_reminders
85
- ? messageThread.event_reminders.nodes.map(formatEventReminders)
86
- : null,
87
- emoji: messageThread.customization_info
88
- ? messageThread.customization_info.emoji
89
- : null,
90
- color:
91
- messageThread.customization_info &&
92
- messageThread.customization_info.outgoing_bubble_color
93
- ? messageThread.customization_info.outgoing_bubble_color.slice(2)
94
- : null,
95
- threadTheme: messageThread.thread_theme,
96
- nicknames:
97
- messageThread.customization_info &&
98
- messageThread.customization_info.participant_customizations
99
- ? messageThread.customization_info.participant_customizations.reduce(
100
- function (res, val) {
101
- if (val.nickname) res[val.participant_id] = val.nickname;
102
- return res;
103
- },
104
- {},
105
- )
106
- : {},
107
- adminIDs: messageThread.thread_admins,
108
- approvalMode: Boolean(messageThread.approval_mode),
109
- approvalQueue: messageThread.group_approval_queue.nodes.map((a) => ({
110
- inviterID: a.inviter.id,
111
- requesterID: a.requester.id,
112
- timestamp: a.request_timestamp,
113
- request_source: a.request_source, // @Undocumented
114
- })),
70
+ // "FF8C0077" -> "8C0077"
71
+ function formatColor(color) {
72
+ if (color && color.match(/^(?:[0-9a-fA-F]{8})$/g)) return color.slice(2);
73
+ return color;
74
+ }
115
75
 
116
- // @Undocumented
117
- reactionsMuteMode: messageThread.reactions_mute_mode.toLowerCase(),
118
- mentionsMuteMode: messageThread.mentions_mute_mode.toLowerCase(),
119
- isPinProtected: messageThread.is_pin_protected,
120
- relatedPageThread: messageThread.related_page_thread,
76
+ function getThreadName(t) {
77
+ if (t.name || t.thread_key.thread_fbid) return t.name;
121
78
 
122
- // @Legacy
123
- name: messageThread.name,
124
- snippet: snippetText,
125
- snippetSender: snippetID,
126
- snippetAttachments: [],
127
- serverTimestamp: messageThread.updated_time_precise,
128
- imageSrc: messageThread.image ? messageThread.image.uri : null,
129
- isCanonicalUser: messageThread.is_canonical_neo_user,
130
- isCanonical: messageThread.thread_type != "GROUP",
131
- recipientsLoadable: true,
132
- hasEmailParticipant: false,
133
- readOnly: false,
134
- canReply: messageThread.cannot_reply_reason == null,
135
- lastMessageTimestamp: messageThread.last_message
136
- ? messageThread.last_message.timestamp_precise
137
- : null,
138
- lastMessageType: "message",
139
- lastReadTimestamp: lastReadTimestamp,
140
- threadType: messageThread.thread_type == "GROUP" ? 2 : 1,
79
+ for (let po of t.all_participants.edges) {
80
+ let p = po.node;
81
+ if (p.messaging_actor.id === t.thread_key.other_user_id) return p.messaging_actor.name;
82
+ }
83
+ }
141
84
 
142
- // update in Wed, 13 Jul 2022 19:41:12 +0700
143
- inviteLink: {
144
- enable: messageThread.joinable_mode
145
- ? messageThread.joinable_mode.mode == 1
146
- : false,
147
- link: messageThread.joinable_mode
148
- ? messageThread.joinable_mode.link
149
- : null,
150
- },
151
- };
85
+ function mapNicknames(customizationInfo) {
86
+ return (customizationInfo && customizationInfo.participant_customizations) ? customizationInfo.participant_customizations.map(u => {
87
+ return {
88
+ "userID": u.participant_id,
89
+ "nickname": u.nickname
90
+ };
91
+ }) : [];
152
92
  }
153
93
 
154
94
  function formatThreadList(data) {
155
- // console.log(JSON.stringify(data.find(t => t.thread_key.thread_fbid === "5095817367161431"), null, 2));
156
- return data.map((t) => formatThreadGraphQLResponse(t));
95
+ return data.map(t => {
96
+ let lastMessageNode = (t.last_message && t.last_message.nodes && t.last_message.nodes.length > 0) ? t.last_message.nodes[0] : null;
97
+ return {
98
+ threadID: t.thread_key ? utils.formatID(t.thread_key.thread_fbid || t.thread_key.other_user_id) : null, // shall never be null
99
+ name: getThreadName(t),
100
+ unreadCount: t.unread_count,
101
+ messageCount: t.messages_count,
102
+ imageSrc: t.image ? t.image.uri : null,
103
+ emoji: t.customization_info ? t.customization_info.emoji : null,
104
+ color: formatColor(t.customization_info ? t.customization_info.outgoing_bubble_color : null),
105
+ nicknames: mapNicknames(t.customization_info),
106
+ muteUntil: t.mute_until,
107
+ participants: formatParticipants(t.all_participants),
108
+ adminIDs: t.thread_admins.map(a => a.id),
109
+ folder: t.folder,
110
+ isGroup: t.thread_type === "GROUP",
111
+ // rtc_call_data: t.rtc_call_data, // TODO: format and document this
112
+ // isPinProtected: t.is_pin_protected, // feature from future? always false (2018-04-04)
113
+ customizationEnabled: t.customization_enabled, // false for ONE_TO_ONE with Page or ReducedMessagingActor
114
+ participantAddMode: t.participant_add_mode_as_string, // "ADD" if "GROUP" and null if "ONE_TO_ONE"
115
+ montageThread: t.montage_thread ? Buffer.from(t.montage_thread.id, "base64").toString() : null, // base64 encoded string "message_thread:0000000000000000"
116
+ // it is not userID nor any other ID known to me...
117
+ // can somebody inspect it? where is it used?
118
+ // probably Messenger Day uses it
119
+ reactionsMuteMode: t.reactions_mute_mode,
120
+ mentionsMuteMode: t.mentions_mute_mode,
121
+ isArchived: t.has_viewer_archived,
122
+ isSubscribed: t.is_viewer_subscribed,
123
+ timestamp: t.updated_time_precise, // in miliseconds
124
+ // isCanonicalUser: t.is_canonical_neo_user, // is it always false?
125
+ // TODO: how about putting snippet in another object? current implementation does not handle every possibile message type etc.
126
+ snippet: lastMessageNode ? lastMessageNode.snippet : null,
127
+ snippetAttachments: lastMessageNode ? lastMessageNode.extensible_attachment : null, // TODO: not sure if it works
128
+ snippetSender: lastMessageNode ? utils.formatID((lastMessageNode.message_sender.messaging_actor.id || "").toString()) : null,
129
+ lastMessageTimestamp: lastMessageNode ? lastMessageNode.timestamp_precise : null, // timestamp in miliseconds
130
+ lastReadTimestamp: (t.last_read_receipt && t.last_read_receipt.nodes.length > 0)
131
+ ? (t.last_read_receipt.nodes[0] ? t.last_read_receipt.nodes[0].timestamp_precise : null)
132
+ : null, // timestamp in miliseconds
133
+ cannotReplyReason: t.cannot_reply_reason, // TODO: inspect possible values
134
+ approvalMode: Boolean(t.approval_mode),
135
+
136
+ // @Legacy
137
+ participantIDs: formatParticipants(t.all_participants).map(participant => participant.userID),
138
+ threadType: t.thread_type === "GROUP" ? 2 : 1 // "GROUP" or "ONE_TO_ONE"
139
+ };
140
+ });
157
141
  }
158
142
 
159
143
  module.exports = function (defaultFuncs, api, ctx) {
160
144
  return function getThreadList(limit, timestamp, tags, callback) {
161
- if (
162
- !callback &&
163
- (utils.getType(tags) === "Function" ||
164
- utils.getType(tags) === "AsyncFunction")
165
- ) {
145
+ if (!callback && (utils.getType(tags) === "Function" || utils.getType(tags) === "AsyncFunction")) {
166
146
  callback = tags;
167
147
  tags = [""];
168
148
  }
169
- if (
170
- utils.getType(limit) !== "Number" ||
171
- !Number.isInteger(limit) ||
172
- limit <= 0
173
- ) {
174
- throw new utils.CustomError({
175
- error: "getThreadList: limit must be a positive integer",
176
- });
177
- }
178
- if (
179
- utils.getType(timestamp) !== "Null" &&
180
- (utils.getType(timestamp) !== "Number" || !Number.isInteger(timestamp))
181
- ) {
182
- throw new utils.CustomError({
183
- error: "getThreadList: timestamp must be an integer or null",
184
- });
185
- }
186
- if (utils.getType(tags) === "String") {
187
- tags = [tags];
188
- }
189
- if (utils.getType(tags) !== "Array") {
190
- throw new utils.CustomError({
191
- error: "getThreadList: tags must be an array",
192
- message: "getThreadList: tags must be an array",
193
- });
194
- }
149
+ if (utils.getType(limit) !== "Number" || !Number.isInteger(limit) || limit <= 0) throw { error: "getThreadList: limit must be a positive integer" };
150
+
151
+ if (utils.getType(timestamp) !== "Null" && (utils.getType(timestamp) !== "Number" || !Number.isInteger(timestamp))) throw { error: "getThreadList: timestamp must be an integer or null" };
152
+
153
+ if (utils.getType(tags) === "String") tags = [tags];
154
+ if (utils.getType(tags) !== "Array") throw { error: "getThreadList: tags must be an array" };
195
155
 
196
- let resolveFunc = function () {};
197
- let rejectFunc = function () {};
198
- const returnPromise = new Promise(function (resolve, reject) {
156
+ var resolveFunc = function () { };
157
+ var rejectFunc = function () { };
158
+ var returnPromise = new Promise(function (resolve, reject) {
199
159
  resolveFunc = resolve;
200
160
  rejectFunc = reject;
201
161
  });
202
162
 
203
- if (
204
- utils.getType(callback) !== "Function" &&
205
- utils.getType(callback) !== "AsyncFunction"
206
- ) {
163
+ if (utils.getType(callback) !== "Function" && utils.getType(callback) !== "AsyncFunction") {
207
164
  callback = function (err, data) {
208
- if (err) {
209
- return rejectFunc(err);
210
- }
165
+ if (err) return rejectFunc(err);
211
166
  resolveFunc(data);
212
167
  };
213
168
  }
214
169
 
215
170
  const form = {
216
- av: ctx.userID,
217
- queries: JSON.stringify({
218
- o0: {
171
+ "av": ctx.globalOptions.pageID,
172
+ "queries": JSON.stringify({
173
+ "o0": {
219
174
  // This doc_id was valid on 2020-07-20
220
- // "doc_id": "3336396659757871",
221
- doc_id: "3426149104143726",
222
- query_params: {
223
- limit: limit + (timestamp ? 1 : 0),
224
- before: timestamp,
225
- tags: tags,
226
- includeDeliveryReceipts: true,
227
- includeSeqID: false,
228
- },
229
- },
175
+ "doc_id": "3336396659757871",
176
+ "query_params": {
177
+ "limit": limit + (timestamp ? 1 : 0),
178
+ "before": timestamp,
179
+ "tags": tags,
180
+ "includeDeliveryReceipts": true,
181
+ "includeSeqID": false
182
+ }
183
+ }
230
184
  }),
231
- batch_name: "MessengerGraphQLThreadlistFetcher",
185
+ "batch_name": "MessengerGraphQLThreadlistFetcher"
232
186
  };
233
187
 
234
188
  defaultFuncs
235
189
  .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
236
190
  .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
237
191
  .then((resData) => {
238
- if (resData[resData.length - 1].error_results > 0) {
239
- throw new utils.CustomError(resData[0].o0.errors);
240
- }
192
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
241
193
 
242
- if (resData[resData.length - 1].successful_results === 0) {
243
- throw new utils.CustomError({
244
- error: "getThreadList: there was no successful_results",
245
- res: resData,
246
- });
247
- }
194
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "getThreadList: there was no successful_results", res: resData };
248
195
 
249
196
  // When we ask for threads using timestamp from the previous request,
250
197
  // we are getting the last thread repeated as the first thread in this response.
@@ -252,16 +199,12 @@ module.exports = function (defaultFuncs, api, ctx) {
252
199
  // It is also the reason for increasing limit by 1 when timestamp is set
253
200
  // this way user asks for 10 threads, we are asking for 11,
254
201
  // but after removing the duplicated one, it is again 10
255
- if (timestamp) {
256
- resData[0].o0.data.viewer.message_threads.nodes.shift();
257
- }
258
- callback(
259
- null,
260
- formatThreadList(resData[0].o0.data.viewer.message_threads.nodes),
261
- );
202
+ if (timestamp) resData[0].o0.data.viewer.message_threads.nodes.shift();
203
+
204
+ callback(null, formatThreadList(resData[0].o0.data.viewer.message_threads.nodes));
262
205
  })
263
206
  .catch((err) => {
264
- console.error("getThreadList", err);
207
+ log.error("getThreadList", "Lỗi: getThreadList Có Thể Do Bạn Spam Quá Nhiều, Hãy Thử Lại !");
265
208
  return callback(err);
266
209
  });
267
210
 
@@ -0,0 +1,220 @@
1
+ "use strict";
2
+
3
+ var utils = require("../utils");
4
+ var log = require("npmlog");
5
+ // tương lai đi rồi fix ahahha
6
+ function formatEventReminders(reminder) {
7
+ return {
8
+ reminderID: reminder.id,
9
+ eventCreatorID: reminder.lightweight_event_creator.id,
10
+ time: reminder.time,
11
+ eventType: reminder.lightweight_event_type.toLowerCase(),
12
+ locationName: reminder.location_name,
13
+ // @TODO verify this
14
+ locationCoordinates: reminder.location_coordinates,
15
+ locationPage: reminder.location_page,
16
+ eventStatus: reminder.lightweight_event_status.toLowerCase(),
17
+ note: reminder.note,
18
+ repeatMode: reminder.repeat_mode.toLowerCase(),
19
+ eventTitle: reminder.event_title,
20
+ triggerMessage: reminder.trigger_message,
21
+ secondsToNotifyBefore: reminder.seconds_to_notify_before,
22
+ allowsRsvp: reminder.allows_rsvp,
23
+ relatedEvent: reminder.related_event,
24
+ members: reminder.event_reminder_members.edges.map(function(member) {
25
+ return {
26
+ memberID: member.node.id,
27
+ state: member.guest_list_state.toLowerCase()
28
+ };
29
+ })
30
+ };
31
+ }
32
+
33
+ function formatThreadGraphQLResponse(data) {
34
+ try{
35
+ var messageThread = data.message_thread;
36
+ } catch (err){
37
+ console.error("GetThreadInfoGraphQL", "Can't get this thread info!");
38
+ return {err: err};
39
+ }
40
+ var threadID = messageThread.thread_key.thread_fbid
41
+ ? messageThread.thread_key.thread_fbid
42
+ : messageThread.thread_key.other_user_id;
43
+
44
+ // Remove me
45
+ var lastM = messageThread.last_message;
46
+ var snippetID =
47
+ lastM &&
48
+ lastM.nodes &&
49
+ lastM.nodes[0] &&
50
+ lastM.nodes[0].message_sender &&
51
+ lastM.nodes[0].message_sender.messaging_actor
52
+ ? lastM.nodes[0].message_sender.messaging_actor.id
53
+ : null;
54
+ var snippetText =
55
+ lastM && lastM.nodes && lastM.nodes[0] ? lastM.nodes[0].snippet : null;
56
+ var lastR = messageThread.last_read_receipt;
57
+ var lastReadTimestamp =
58
+ lastR && lastR.nodes && lastR.nodes[0] && lastR.nodes[0].timestamp_precise
59
+ ? lastR.nodes[0].timestamp_precise
60
+ : null;
61
+
62
+ return {
63
+ threadID: threadID,
64
+ threadName: messageThread.name,
65
+ participantIDs: messageThread.all_participants.edges.map(d => d.node.messaging_actor.id),
66
+ userInfo: messageThread.all_participants.edges.map(d => ({
67
+ id: d.node.messaging_actor.id,
68
+ name: d.node.messaging_actor.name,
69
+ firstName: d.node.messaging_actor.short_name,
70
+ vanity: d.node.messaging_actor.username,
71
+ thumbSrc: d.node.messaging_actor.big_image_src.uri,
72
+ profileUrl: d.node.messaging_actor.big_image_src.uri,
73
+ gender: d.node.messaging_actor.gender,
74
+ type: d.node.messaging_actor.__typename,
75
+ isFriend: d.node.messaging_actor.is_viewer_friend,
76
+ isBirthday: !!d.node.messaging_actor.is_birthday //not sure?
77
+ })),
78
+ unreadCount: messageThread.unread_count,
79
+ messageCount: messageThread.messages_count,
80
+ timestamp: messageThread.updated_time_precise,
81
+ muteUntil: messageThread.mute_until,
82
+ isGroup: messageThread.thread_type == "GROUP",
83
+ isSubscribed: messageThread.is_viewer_subscribed,
84
+ isArchived: messageThread.has_viewer_archived,
85
+ folder: messageThread.folder,
86
+ cannotReplyReason: messageThread.cannot_reply_reason,
87
+ eventReminders: messageThread.event_reminders
88
+ ? messageThread.event_reminders.nodes.map(formatEventReminders)
89
+ : null,
90
+ emoji: messageThread.customization_info
91
+ ? messageThread.customization_info.emoji
92
+ : null,
93
+ color:
94
+ messageThread.customization_info &&
95
+ messageThread.customization_info.outgoing_bubble_color
96
+ ? messageThread.customization_info.outgoing_bubble_color.slice(2)
97
+ : null,
98
+ nicknames:
99
+ messageThread.customization_info &&
100
+ messageThread.customization_info.participant_customizations
101
+ ? messageThread.customization_info.participant_customizations.reduce(
102
+ function(res, val) {
103
+ if (val.nickname) res[val.participant_id] = val.nickname;
104
+ return res;
105
+ },
106
+ {}
107
+ )
108
+ : {},
109
+ adminIDs: messageThread.thread_admins,
110
+ approvalMode: Boolean(messageThread.approval_mode),
111
+ approvalQueue: messageThread.group_approval_queue.nodes.map(a => ({
112
+ inviterID: a.inviter.id,
113
+ requesterID: a.requester.id,
114
+ timestamp: a.request_timestamp,
115
+ request_source: a.request_source // @Undocumented
116
+ })),
117
+
118
+ // @Undocumented
119
+ reactionsMuteMode: messageThread.reactions_mute_mode.toLowerCase(),
120
+ mentionsMuteMode: messageThread.mentions_mute_mode.toLowerCase(),
121
+ isPinProtected: messageThread.is_pin_protected,
122
+ relatedPageThread: messageThread.related_page_thread,
123
+
124
+ // @Legacy
125
+ name: messageThread.name,
126
+ snippet: snippetText,
127
+ snippetSender: snippetID,
128
+ snippetAttachments: [],
129
+ serverTimestamp: messageThread.updated_time_precise,
130
+ imageSrc: messageThread.image ? messageThread.image.uri : null,
131
+ isCanonicalUser: messageThread.is_canonical_neo_user,
132
+ isCanonical: messageThread.thread_type != "GROUP",
133
+ recipientsLoadable: true,
134
+ hasEmailParticipant: false,
135
+ readOnly: false,
136
+ canReply: messageThread.cannot_reply_reason == null,
137
+ lastMessageTimestamp: messageThread.last_message
138
+ ? messageThread.last_message.timestamp_precise
139
+ : null,
140
+ lastMessageType: "message",
141
+ lastReadTimestamp: lastReadTimestamp,
142
+ threadType: messageThread.thread_type == "GROUP" ? 2 : 1,
143
+ TimeCreate: Date.now(),
144
+ TimeUpdate: Date.now()
145
+ };
146
+ }
147
+
148
+ module.exports = function(defaultFuncs, api, ctx) {
149
+ return function getThreadInfoGraphQL(threadID, callback) {
150
+ var resolveFunc = function(){};
151
+ var rejectFunc = function(){};
152
+ var returnPromise = new Promise(function (resolve, reject) {
153
+ resolveFunc = resolve;
154
+ rejectFunc = reject;
155
+ });
156
+
157
+ if (utils.getType(callback) != "Function" && utils.getType(callback) != "AsyncFunction") {
158
+ callback = function (err, data) {
159
+ if (err) {
160
+ return rejectFunc(err);
161
+ }
162
+ resolveFunc(data);
163
+ };
164
+ }
165
+
166
+ //! được tìm thấy vào giữa tháng 8/2022 bởi @KanzuWakazaki - đã được chia sẻ cho @D-Jukie và Horizon Team Public group 🤴
167
+ //* những code tương tự muliti thread như này đều có thể là copy idea 🐧
168
+ //* đã áp dụng vào fca mới(cloud - fca(private)) vào cuối tháng 8/2022 bởi @IteralingCode(Hidden Member( always :) )) - Synthetic 4 - @Horizon Team
169
+ //*cập nhật dự án bị bỏ rơi này vào ngày 19/11/2022 bởi @KanzuWakazaki(Owner) - Synthetic 1 - @Horizon Team nhằm đáp ứng nhu cầu của client !
170
+
171
+ if (utils.getType(threadID) !== "Array") threadID = [threadID];
172
+
173
+ var Form = {};
174
+ var ThreadInfo = [];
175
+
176
+ threadID.map(function (x,y) {
177
+ Form["o" + y] = {
178
+ doc_id: "3449967031715030",
179
+ query_params: {
180
+ id: x,
181
+ message_limit: 0,
182
+ load_messages: false,
183
+ load_read_receipts: false,
184
+ before: null
185
+ }
186
+ };
187
+ });
188
+
189
+ var form = {
190
+ queries: JSON.stringify(Form),
191
+ batch_name: "MessengerGraphQLThreadFetcher"
192
+ };
193
+
194
+ defaultFuncs
195
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
196
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
197
+ .then(function(resData) {
198
+ if (resData.error) {
199
+ callback(null,resData.error);
200
+ throw resData;
201
+ }
202
+ resData = resData.splice(0, resData.length - 1);
203
+ resData.sort((a, b) => { return Object.keys(a)[0].localeCompare(Object.keys(b)[0]); });
204
+ resData.map(function (x,y) {
205
+ ThreadInfo.push(formatThreadGraphQLResponse(x["o"+y].data));
206
+ });
207
+ if (Object.keys(resData).length == 1) {
208
+ callback(null, ThreadInfo[0]);
209
+ } else {
210
+ callback(null, ThreadInfo);
211
+ }
212
+ })
213
+ .catch(function(err) {
214
+
215
+ log.error("getThreadInfoGraphQL", "Lỗi: getThreadInfoGraphQL Có Thể Do Bạn Spam Quá Nhiều, Hãy Thử Lại !");
216
+ return callback(err);
217
+ });
218
+ return returnPromise;
219
+ };
220
+ };