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.
- package/Extra/Balancer.js +49 -0
- package/Extra/Bypass/956/index.js +234 -0
- package/Extra/Bypass/test/aaaa.json +169 -0
- package/Extra/Bypass/test/index.js +188 -0
- package/Extra/Database/index.js +469 -0
- package/Extra/ExtraAddons.js +82 -0
- package/Extra/ExtraFindUID.js +62 -0
- package/Extra/ExtraGetThread.js +365 -0
- package/Extra/ExtraScreenShot.js +430 -0
- package/Extra/ExtraUptimeRobot.js +38 -0
- package/Extra/Html/Classic/script.js +119 -0
- package/Extra/Html/Classic/style.css +8 -0
- package/Extra/Security/AES_256_GCM/index.js +1 -0
- package/Extra/Security/Base/Step_1.js +6 -0
- package/Extra/Security/Base/Step_2.js +22 -0
- package/Extra/Security/Base/Step_3.js +22 -0
- package/Extra/Security/Base/index.js +191 -0
- package/Extra/Security/Step_1.js +6 -0
- package/Extra/Security/Step_2.js +22 -0
- package/Extra/Security/Step_3.js +22 -0
- package/Extra/Security/index.js +5 -0
- package/Extra/Src/Change_Environment.js +24 -0
- package/Extra/Src/Check_Update.js +67 -0
- package/Extra/Src/History.js +115 -0
- package/Extra/Src/Instant_Update.js +65 -0
- package/Extra/Src/Last-Run.js +65 -0
- package/Extra/Src/Premium.js +81 -0
- package/Extra/Src/Release_Memory.js +160 -0
- package/Extra/Src/Websocket.js +213 -0
- package/Extra/Src/image/62518eafb0670b697788ce4f9a4f71d1.jpg +0 -0
- package/Extra/Src/test.js +28 -0
- package/Extra/Src/uuid.js +137 -0
- package/Func/AcceptAgreement.js +31 -0
- package/Func/ClearCache.js +64 -0
- package/Func/ReportV1.js +54 -0
- package/{LICENSE-MIT → LICENSE.txt} +6 -6
- package/Language/index.json +228 -0
- package/Main.js +1290 -0
- package/README.md +8 -6
- package/SECURITY.md +17 -0
- package/broadcast.js +44 -0
- package/index.js +424 -465
- package/logger.js +66 -0
- package/package.json +91 -42
- package/src/Dev_Horizon_Data.js +125 -0
- package/src/Dev_getThreadInfoOLD.js +422 -0
- package/src/Dev_shareTest2.js +68 -0
- package/src/Dev_shareTest3.js +71 -0
- package/src/Premium.js +25 -0
- package/src/Screenshot.js +83 -0
- package/src/addExternalModule.js +5 -14
- package/src/addUserToGroup.js +21 -57
- package/src/changeAdminStatus.js +22 -46
- package/src/changeArchivedStatus.js +12 -26
- package/src/{changeAvatarV2.js → changeAvt.js} +2 -3
- package/src/changeBio.js +15 -26
- package/src/changeBlockedStatus.js +9 -22
- package/src/changeGroupImage.js +26 -55
- package/src/changeNickname.js +14 -28
- package/src/changeThreadColor.js +19 -22
- package/src/changeThreadEmoji.js +12 -25
- package/src/createNewGroup.js +18 -36
- package/src/createPoll.js +17 -28
- package/src/deleteMessage.js +14 -25
- package/src/deleteThread.js +13 -26
- package/src/editMessage.js +50 -56
- package/src/forwardAttachment.js +16 -28
- package/src/getAccessToken.js +28 -0
- package/src/getCurrentUserID.js +3 -3
- package/src/getEmojiUrl.js +8 -10
- package/src/getFriendsList.js +15 -25
- package/src/getMessage.js +81 -813
- package/src/getThreadHistory.js +98 -241
- package/src/getThreadInfo.js +286 -89
- package/src/getThreadInfoOLD.js +422 -0
- package/src/getThreadList.js +158 -215
- package/src/getThreadMain.js +220 -0
- package/src/getThreadPictures.js +20 -40
- package/src/getUID.js +49 -112
- package/src/getUserID.js +14 -18
- package/src/getUserInfo.js +75 -34
- package/src/getUserInfoMain.js +65 -0
- package/src/getUserInfoV2.js +32 -0
- package/src/getUserInfoV3.js +63 -0
- package/src/getUserInfoV4.js +55 -0
- package/src/getUserInfoV5.js +61 -0
- package/src/handleFriendRequest.js +14 -25
- package/src/handleMessageRequest.js +21 -37
- package/src/httpGet.js +22 -37
- package/src/httpPost.js +20 -36
- package/src/httpPostFormData.js +23 -52
- package/src/listenMqtt.js +667 -414
- package/src/listenMqttV1.js +846 -0
- package/src/logout.js +17 -24
- package/src/markAsDelivered.js +16 -23
- package/src/markAsRead.js +28 -43
- package/src/markAsReadAll.js +14 -21
- package/src/markAsSeen.js +19 -29
- package/src/muteThread.js +13 -18
- package/src/removeUserFromGroup.js +18 -48
- package/src/resolvePhotoUrl.js +14 -22
- package/src/searchForThread.js +13 -23
- package/src/sendMessage.js +125 -181
- package/src/sendMqttMessage.js +71 -0
- package/src/sendTypingIndicator.js +80 -28
- package/src/setMessageReaction.js +20 -33
- package/src/setPostReaction.js +95 -105
- package/src/setTitle.js +19 -35
- package/src/shareContact.js +37 -92
- package/src/shareLink.js +5 -6
- package/src/threadColors.js +17 -109
- package/src/unfriend.js +11 -20
- package/src/unsendMessage.js +33 -30
- package/src/unsendMqttMessage.js +66 -0
- package/test/data/shareAttach.js +146 -0
- package/test/data/something.mov +1 -0
- package/test/data/test.png +1 -0
- package/test/data/test.txt +1 -0
- package/test/env/.env +1 -0
- package/test/example-config.json +18 -0
- package/test/test-page.js +140 -0
- package/test/test.js +385 -0
- package/test/testv2.js +3 -0
- package/utils.js +2908 -1247
- package/replit.nix +0 -3
- package/src/changeAvatar.js +0 -136
- package/src/changeBlockedStatusMqtt.js +0 -80
- package/src/changeCover.js +0 -72
- package/src/changeName.js +0 -78
- package/src/changeUsername.js +0 -58
- package/src/createCommentPost.js +0 -229
- package/src/createPost.js +0 -275
- package/src/data/getThreadInfo.json +0 -1
- package/src/editMessageOld.js +0 -67
- package/src/follow.js +0 -74
- package/src/getAccess.js +0 -112
- package/src/getAvatarUser.js +0 -78
- package/src/getBotInitialData.js +0 -42
- package/src/getRegion.js +0 -7
- package/src/getThreadHistoryDeprecated.js +0 -93
- package/src/getThreadInfoDeprecated.js +0 -80
- package/src/getThreadListDeprecated.js +0 -75
- package/src/listenNotification.js +0 -85
- package/src/pinMessage.js +0 -59
- package/src/refreshFb_dtsg.js +0 -89
- package/src/searchStickers.js +0 -53
- package/src/sendMessageMqtt.js +0 -316
- package/src/setMessageReactionMqtt.js +0 -62
- package/src/setProfileGuard.js +0 -44
- package/src/setStoryReaction.js +0 -64
- package/src/stopListenMqtt.js +0 -23
- package/src/uploadAttachment.js +0 -94
package/src/getThreadList.js
CHANGED
@@ -1,250 +1,197 @@
|
|
1
1
|
"use strict";
|
2
2
|
|
3
3
|
const utils = require("../utils");
|
4
|
-
|
4
|
+
const log = require("npmlog");
|
5
5
|
|
6
|
-
function
|
7
|
-
return
|
8
|
-
|
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
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
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
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
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
|
-
|
117
|
-
|
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
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
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
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
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
|
-
|
156
|
-
|
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
|
-
|
171
|
-
|
172
|
-
|
173
|
-
)
|
174
|
-
|
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
|
-
|
197
|
-
|
198
|
-
|
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.
|
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
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
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
|
-
|
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
|
-
|
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
|
+
};
|