@lazyneoaz/nkxchat 1.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/LICENSE +3 -0
- package/README.md +199 -0
- package/examples/login-with-cookies.js +102 -0
- package/examples/verify.js +70 -0
- package/index.js +2 -0
- package/package.json +84 -0
- package/src/apis/addExternalModule.js +24 -0
- package/src/apis/addUserToGroup.js +108 -0
- package/src/apis/changeAdminStatus.js +148 -0
- package/src/apis/changeArchivedStatus.js +61 -0
- package/src/apis/changeAvatar.js +103 -0
- package/src/apis/changeBio.js +69 -0
- package/src/apis/changeBlockedStatus.js +54 -0
- package/src/apis/changeGroupImage.js +136 -0
- package/src/apis/changeThreadColor.js +116 -0
- package/src/apis/changeThreadEmoji.js +53 -0
- package/src/apis/comment.js +207 -0
- package/src/apis/createAITheme.js +129 -0
- package/src/apis/createNewGroup.js +79 -0
- package/src/apis/createPoll.js +73 -0
- package/src/apis/deleteMessage.js +44 -0
- package/src/apis/deleteThread.js +52 -0
- package/src/apis/editMessage.js +70 -0
- package/src/apis/emoji.js +124 -0
- package/src/apis/enableAutoSaveAppState.js +69 -0
- package/src/apis/fetchThemeData.js +113 -0
- package/src/apis/follow.js +81 -0
- package/src/apis/forwardAttachment.js +178 -0
- package/src/apis/forwardMessage.js +52 -0
- package/src/apis/friend.js +243 -0
- package/src/apis/gcmember.js +122 -0
- package/src/apis/gcname.js +123 -0
- package/src/apis/gcrule.js +119 -0
- package/src/apis/getAccess.js +111 -0
- package/src/apis/getBotInfo.js +88 -0
- package/src/apis/getBotInitialData.js +43 -0
- package/src/apis/getEmojiUrl.js +40 -0
- package/src/apis/getFriendsList.js +79 -0
- package/src/apis/getMessage.js +423 -0
- package/src/apis/getTheme.js +123 -0
- package/src/apis/getThemeInfo.js +116 -0
- package/src/apis/getThemePictures.js +87 -0
- package/src/apis/getThreadColors.js +119 -0
- package/src/apis/getThreadHistory.js +239 -0
- package/src/apis/getThreadInfo.js +267 -0
- package/src/apis/getThreadList.js +232 -0
- package/src/apis/getThreadPictures.js +58 -0
- package/src/apis/getUserID.js +117 -0
- package/src/apis/getUserInfo.js +513 -0
- package/src/apis/getUserInfoV2.js +146 -0
- package/src/apis/handleFriendRequest.js +66 -0
- package/src/apis/handleMessageRequest.js +50 -0
- package/src/apis/httpGet.js +63 -0
- package/src/apis/httpPost.js +89 -0
- package/src/apis/httpPostFormData.js +69 -0
- package/src/apis/listenMqtt.js +924 -0
- package/src/apis/listenSpeed.js +178 -0
- package/src/apis/logout.js +63 -0
- package/src/apis/markAsDelivered.js +47 -0
- package/src/apis/markAsRead.js +95 -0
- package/src/apis/markAsReadAll.js +40 -0
- package/src/apis/markAsSeen.js +70 -0
- package/src/apis/mqttDeltaValue.js +252 -0
- package/src/apis/muteThread.js +45 -0
- package/src/apis/nickname.js +132 -0
- package/src/apis/notes.js +163 -0
- package/src/apis/pinMessage.js +150 -0
- package/src/apis/produceMetaTheme.js +160 -0
- package/src/apis/realtime.js +182 -0
- package/src/apis/refreshFb_dtsg.js +94 -0
- package/src/apis/removeUserFromGroup.js +117 -0
- package/src/apis/resolvePhotoUrl.js +58 -0
- package/src/apis/searchForThread.js +154 -0
- package/src/apis/sendEffect.js +306 -0
- package/src/apis/sendMessage.js +353 -0
- package/src/apis/sendMessageMqtt.js +255 -0
- package/src/apis/sendTypingIndicator.js +40 -0
- package/src/apis/setMessageReaction.js +27 -0
- package/src/apis/setMessageReactionMqtt.js +61 -0
- package/src/apis/setPostReaction.js +118 -0
- package/src/apis/setThreadTheme.js +210 -0
- package/src/apis/setThreadThemeMqtt.js +94 -0
- package/src/apis/setTitle.js +26 -0
- package/src/apis/share.js +106 -0
- package/src/apis/shareContact.js +66 -0
- package/src/apis/stickers.js +257 -0
- package/src/apis/story.js +181 -0
- package/src/apis/theme.js +233 -0
- package/src/apis/unfriend.js +47 -0
- package/src/apis/unsendMessage.js +17 -0
- package/src/apis/uploadAttachment.js +87 -0
- package/src/database/appStateBackup.js +189 -0
- package/src/database/models/index.js +56 -0
- package/src/database/models/thread.js +31 -0
- package/src/database/models/user.js +32 -0
- package/src/database/threadData.js +101 -0
- package/src/database/userData.js +90 -0
- package/src/engine/client.js +92 -0
- package/src/engine/models/buildAPI.js +118 -0
- package/src/engine/models/loginHelper.js +492 -0
- package/src/engine/models/setOptions.js +88 -0
- package/src/types/index.d.ts +498 -0
- package/src/utils/antiSuspension.js +516 -0
- package/src/utils/auth-helpers.js +149 -0
- package/src/utils/autoReLogin.js +237 -0
- package/src/utils/axios.js +368 -0
- package/src/utils/cache.js +54 -0
- package/src/utils/clients.js +279 -0
- package/src/utils/constants.js +525 -0
- package/src/utils/formatters/data/formatAttachment.js +370 -0
- package/src/utils/formatters/data/formatDelta.js +109 -0
- package/src/utils/formatters/index.js +159 -0
- package/src/utils/formatters/value/formatCookie.js +91 -0
- package/src/utils/formatters/value/formatDate.js +36 -0
- package/src/utils/formatters/value/formatID.js +16 -0
- package/src/utils/formatters.js +1369 -0
- package/src/utils/headers.js +235 -0
- package/src/utils/index.js +152 -0
- package/src/utils/monitoring.js +333 -0
- package/src/utils/rateLimiter.js +251 -0
- package/src/utils/tokenRefresh.js +285 -0
- package/src/utils/user-agents.js +238 -0
- package/src/utils/validation.js +157 -0
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
|
|
2
|
+
"use strict";
|
|
3
|
+
|
|
4
|
+
const utils = require('../utils');
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Formats an event reminder object from a GraphQL response.
|
|
8
|
+
* @param {Object} reminder The raw event reminder object.
|
|
9
|
+
* @returns {Object} A formatted event reminder object.
|
|
10
|
+
*/
|
|
11
|
+
function formatEventReminders(reminder) {
|
|
12
|
+
return {
|
|
13
|
+
reminderID: reminder.id,
|
|
14
|
+
eventCreatorID: reminder.lightweight_event_creator.id,
|
|
15
|
+
time: reminder.time,
|
|
16
|
+
eventType: reminder.lightweight_event_type.toLowerCase(),
|
|
17
|
+
locationName: reminder.location_name,
|
|
18
|
+
locationCoordinates: reminder.location_coordinates,
|
|
19
|
+
locationPage: reminder.location_page,
|
|
20
|
+
eventStatus: reminder.lightweight_event_status.toLowerCase(),
|
|
21
|
+
note: reminder.note,
|
|
22
|
+
repeatMode: reminder.repeat_mode.toLowerCase(),
|
|
23
|
+
eventTitle: reminder.event_title,
|
|
24
|
+
triggerMessage: reminder.trigger_message,
|
|
25
|
+
secondsToNotifyBefore: reminder.seconds_to_notify_before,
|
|
26
|
+
allowsRsvp: reminder.allows_rsvp,
|
|
27
|
+
relatedEvent: reminder.related_event,
|
|
28
|
+
members: reminder.event_reminder_members.edges.map(function (member) {
|
|
29
|
+
return {
|
|
30
|
+
memberID: member.node.id,
|
|
31
|
+
state: member.guest_list_state.toLowerCase(),
|
|
32
|
+
};
|
|
33
|
+
}),
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Formats a thread object from a GraphQL response.
|
|
39
|
+
* @param {Object} data The raw GraphQL data for a thread.
|
|
40
|
+
* @returns {Object | null} A formatted thread object or null if data is invalid.
|
|
41
|
+
* @throws {Error} If Facebook returns a GraphQL error
|
|
42
|
+
*/
|
|
43
|
+
function formatThreadGraphQLResponse(data) {
|
|
44
|
+
// Check for GraphQL errors and throw with details instead of silently returning null
|
|
45
|
+
if (data.errors) {
|
|
46
|
+
const details = data.errors.map(e => e.message || e).join(', ');
|
|
47
|
+
const error = new Error(`GraphQL error in getThreadInfo: ${details}`);
|
|
48
|
+
Object.assign(error, {
|
|
49
|
+
details: details,
|
|
50
|
+
fullErrors: data.errors
|
|
51
|
+
});
|
|
52
|
+
utils.error("formatThreadGraphQLResponse", error);
|
|
53
|
+
throw error;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
const messageThread = data.message_thread;
|
|
57
|
+
if (!messageThread) {
|
|
58
|
+
const error = new Error("No message_thread in GraphQL response - thread may not exist or access may be restricted");
|
|
59
|
+
Object.assign(error, {
|
|
60
|
+
details: "The GraphQL query returned successfully but contained no message_thread data"
|
|
61
|
+
});
|
|
62
|
+
utils.error("formatThreadGraphQLResponse", error);
|
|
63
|
+
throw error;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
const threadID = messageThread.thread_key.thread_fbid
|
|
67
|
+
? messageThread.thread_key.thread_fbid
|
|
68
|
+
: messageThread.thread_key.other_user_id;
|
|
69
|
+
|
|
70
|
+
const lastM = messageThread.last_message;
|
|
71
|
+
const snippetID =
|
|
72
|
+
lastM?.nodes?.[0]?.message_sender?.messaging_actor?.id || null;
|
|
73
|
+
const snippetText = lastM?.nodes?.[0]?.snippet || null;
|
|
74
|
+
const lastR = messageThread.last_read_receipt;
|
|
75
|
+
const lastReadTimestamp = lastR?.nodes?.[0]?.timestamp_precise || null;
|
|
76
|
+
|
|
77
|
+
return {
|
|
78
|
+
threadID: threadID,
|
|
79
|
+
threadName: messageThread.name,
|
|
80
|
+
participantIDs: messageThread.all_participants.edges.map(
|
|
81
|
+
(d) => d.node.messaging_actor.id,
|
|
82
|
+
),
|
|
83
|
+
userInfo: messageThread.all_participants.edges.map((d) => ({
|
|
84
|
+
id: d.node.messaging_actor.id,
|
|
85
|
+
name: d.node.messaging_actor.name,
|
|
86
|
+
firstName: d.node.messaging_actor.short_name,
|
|
87
|
+
vanity: d.node.messaging_actor.username,
|
|
88
|
+
url: d.node.messaging_actor.url,
|
|
89
|
+
thumbSrc: d.node.messaging_actor.big_image_src.uri,
|
|
90
|
+
profileUrl: d.node.messaging_actor.big_image_src.uri,
|
|
91
|
+
gender: d.node.messaging_actor.gender,
|
|
92
|
+
type: d.node.messaging_actor.__typename,
|
|
93
|
+
isFriend: d.node.messaging_actor.is_viewer_friend,
|
|
94
|
+
isBirthday: !!d.node.messaging_actor.is_birthday,
|
|
95
|
+
})),
|
|
96
|
+
unreadCount: messageThread.unread_count,
|
|
97
|
+
messageCount: messageThread.messages_count,
|
|
98
|
+
timestamp: messageThread.updated_time_precise,
|
|
99
|
+
muteUntil: messageThread.mute_until,
|
|
100
|
+
isGroup: messageThread.thread_type == "GROUP",
|
|
101
|
+
isSubscribed: messageThread.is_viewer_subscribed,
|
|
102
|
+
isArchived: messageThread.has_viewer_archived,
|
|
103
|
+
folder: messageThread.folder,
|
|
104
|
+
cannotReplyReason: messageThread.cannot_reply_reason,
|
|
105
|
+
eventReminders: messageThread.event_reminders
|
|
106
|
+
? messageThread.event_reminders.nodes.map(formatEventReminders)
|
|
107
|
+
: null,
|
|
108
|
+
emoji: messageThread.customization_info
|
|
109
|
+
? messageThread.customization_info.emoji
|
|
110
|
+
: null,
|
|
111
|
+
color:
|
|
112
|
+
messageThread.customization_info &&
|
|
113
|
+
messageThread.customization_info.outgoing_bubble_color
|
|
114
|
+
? messageThread.customization_info.outgoing_bubble_color.slice(2)
|
|
115
|
+
: null,
|
|
116
|
+
threadTheme: messageThread.thread_theme,
|
|
117
|
+
nicknames:
|
|
118
|
+
messageThread.customization_info &&
|
|
119
|
+
messageThread.customization_info.participant_customizations
|
|
120
|
+
? messageThread.customization_info.participant_customizations.reduce(
|
|
121
|
+
function (res, val) {
|
|
122
|
+
if (val.nickname) res[val.participant_id] = val.nickname;
|
|
123
|
+
return res;
|
|
124
|
+
},
|
|
125
|
+
{},
|
|
126
|
+
)
|
|
127
|
+
: {},
|
|
128
|
+
adminIDs: messageThread.thread_admins,
|
|
129
|
+
approvalMode: Boolean(messageThread.approval_mode),
|
|
130
|
+
approvalQueue: messageThread.group_approval_queue.nodes.map((a) => ({
|
|
131
|
+
inviterID: a.inviter.id,
|
|
132
|
+
requesterID: a.requester.id,
|
|
133
|
+
timestamp: a.request_timestamp,
|
|
134
|
+
request_source: a.request_source,
|
|
135
|
+
})),
|
|
136
|
+
reactionsMuteMode: messageThread.reactions_mute_mode.toLowerCase(),
|
|
137
|
+
mentionsMuteMode: messageThread.mentions_mute_mode.toLowerCase(),
|
|
138
|
+
isPinProtected: messageThread.is_pin_protected,
|
|
139
|
+
relatedPageThread: messageThread.related_page_thread,
|
|
140
|
+
name: messageThread.name,
|
|
141
|
+
snippet: snippetText,
|
|
142
|
+
snippetSender: snippetID,
|
|
143
|
+
snippetAttachments: [],
|
|
144
|
+
serverTimestamp: messageThread.updated_time_precise,
|
|
145
|
+
imageSrc: messageThread.image ? messageThread.image.uri : null,
|
|
146
|
+
isCanonicalUser: messageThread.is_canonical_neo_user,
|
|
147
|
+
isCanonical: messageThread.thread_type != "GROUP",
|
|
148
|
+
recipientsLoadable: true,
|
|
149
|
+
hasEmailParticipant: false,
|
|
150
|
+
readOnly: false,
|
|
151
|
+
canReply: messageThread.cannot_reply_reason == null,
|
|
152
|
+
lastMessageTimestamp: messageThread.last_message
|
|
153
|
+
? messageThread.last_message.timestamp_precise
|
|
154
|
+
: null,
|
|
155
|
+
lastMessageType: "message",
|
|
156
|
+
lastReadTimestamp: lastReadTimestamp,
|
|
157
|
+
threadType: messageThread.thread_type == "GROUP" ? 2 : 1,
|
|
158
|
+
inviteLink: {
|
|
159
|
+
enable: messageThread.joinable_mode
|
|
160
|
+
? messageThread.joinable_mode.mode == 1
|
|
161
|
+
: false,
|
|
162
|
+
link: messageThread.joinable_mode
|
|
163
|
+
? messageThread.joinable_mode.link
|
|
164
|
+
: null,
|
|
165
|
+
},
|
|
166
|
+
};
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* @param {Object} defaultFuncs
|
|
171
|
+
* @param {Object} api
|
|
172
|
+
* @param {Object} ctx
|
|
173
|
+
* @returns {function(threadID: string | string[]): Promise<Object>}
|
|
174
|
+
*/
|
|
175
|
+
module.exports = function (defaultFuncs, api, ctx) {
|
|
176
|
+
/**
|
|
177
|
+
* Retrieves information about one or more threads.
|
|
178
|
+
* @param {string|string[]} threadID A single thread ID or an array of thread IDs.
|
|
179
|
+
* @returns {Promise<Object>} A promise that resolves with an object of thread info, or a single thread object if one ID was passed.
|
|
180
|
+
*/
|
|
181
|
+
return async function getThreadInfo(threadID) {
|
|
182
|
+
const threadIDs = Array.isArray(threadID) ? threadID : [threadID];
|
|
183
|
+
|
|
184
|
+
// Validate thread IDs
|
|
185
|
+
if (!ctx.validator.validateIDArray(threadIDs, ctx.validator.isValidThreadID)) {
|
|
186
|
+
throw new Error("Invalid thread ID(s)");
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
let form = {};
|
|
190
|
+
threadIDs.forEach((t, i) => {
|
|
191
|
+
form["o" + i] = {
|
|
192
|
+
doc_id: "3449967031715030",
|
|
193
|
+
query_params: {
|
|
194
|
+
id: t,
|
|
195
|
+
message_limit: 0,
|
|
196
|
+
load_messages: false,
|
|
197
|
+
load_read_receipts: false,
|
|
198
|
+
before: null,
|
|
199
|
+
},
|
|
200
|
+
};
|
|
201
|
+
});
|
|
202
|
+
|
|
203
|
+
form = {
|
|
204
|
+
queries: JSON.stringify(form),
|
|
205
|
+
batch_name: "MessengerGraphQLThreadFetcher",
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
try {
|
|
209
|
+
const resData = await defaultFuncs
|
|
210
|
+
.post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
|
|
211
|
+
.then(utils.parseAndCheckLogin(ctx, defaultFuncs));
|
|
212
|
+
|
|
213
|
+
if (resData.error) {
|
|
214
|
+
throw resData;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
const threadInfos = {};
|
|
218
|
+
for (let i = resData.length - 2; i >= 0; i--) {
|
|
219
|
+
const res = resData[i];
|
|
220
|
+
|
|
221
|
+
// Check for error_results and throw instead of silently continuing
|
|
222
|
+
if (res.error_results) {
|
|
223
|
+
const error = new Error(`Facebook returned error_results for thread query: ${res.error_results} errors`);
|
|
224
|
+
Object.assign(error, {
|
|
225
|
+
error_count: res.error_results,
|
|
226
|
+
thread_index: i
|
|
227
|
+
});
|
|
228
|
+
utils.error("getThreadInfo", error);
|
|
229
|
+
throw error;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
const oKey = Object.keys(res)[0];
|
|
233
|
+
const responseData = res[oKey];
|
|
234
|
+
|
|
235
|
+
// Check for errors in the response object
|
|
236
|
+
if (responseData.errors || responseData.error_results) {
|
|
237
|
+
const details = responseData.errors
|
|
238
|
+
? JSON.stringify(responseData.errors)
|
|
239
|
+
: `error_results: ${responseData.error_results}`;
|
|
240
|
+
const error = new Error(`GraphQL error in thread response: ${details}`);
|
|
241
|
+
Object.assign(error, {
|
|
242
|
+
details: details,
|
|
243
|
+
thread_index: i,
|
|
244
|
+
fullErrors: responseData.errors
|
|
245
|
+
});
|
|
246
|
+
utils.error("getThreadInfo", error);
|
|
247
|
+
throw error;
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
const threadInfo = formatThreadGraphQLResponse(responseData.data);
|
|
251
|
+
if (threadInfo) {
|
|
252
|
+
threadInfos[threadInfo.threadID || threadID[threadID.length - 1 - i]] = threadInfo;
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
// Cache the thread infos
|
|
257
|
+
for (const id in threadInfos) {
|
|
258
|
+
ctx.cache.set(`thread_${id}`, threadInfos[id]);
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
return Array.isArray(threadID) ? threadInfos : Object.values(threadInfos)[0] || null;
|
|
262
|
+
} catch (err) {
|
|
263
|
+
utils.error("getThreadInfo", err);
|
|
264
|
+
throw err;
|
|
265
|
+
}
|
|
266
|
+
};
|
|
267
|
+
};
|
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
|
|
2
|
+
"use strict";
|
|
3
|
+
|
|
4
|
+
const utils = require('../utils');
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Formats an event reminder object from a GraphQL response.
|
|
8
|
+
* @param {Object} reminder The raw event reminder object.
|
|
9
|
+
* @returns {Object} A formatted event reminder object.
|
|
10
|
+
*/
|
|
11
|
+
function formatEventReminders(reminder) {
|
|
12
|
+
return {
|
|
13
|
+
reminderID: reminder.id,
|
|
14
|
+
eventCreatorID: reminder.lightweight_event_creator.id,
|
|
15
|
+
time: reminder.time,
|
|
16
|
+
eventType: reminder.lightweight_event_type.toLowerCase(),
|
|
17
|
+
locationName: reminder.location_name,
|
|
18
|
+
locationCoordinates: reminder.location_coordinates,
|
|
19
|
+
locationPage: reminder.location_page,
|
|
20
|
+
eventStatus: reminder.lightweight_event_status.toLowerCase(),
|
|
21
|
+
note: reminder.note,
|
|
22
|
+
repeatMode: reminder.repeat_mode.toLowerCase(),
|
|
23
|
+
eventTitle: reminder.event_title,
|
|
24
|
+
triggerMessage: reminder.trigger_message,
|
|
25
|
+
secondsToNotifyBefore: reminder.seconds_to_notify_before,
|
|
26
|
+
allowsRsvp: reminder.allows_rsvp,
|
|
27
|
+
relatedEvent: reminder.related_event,
|
|
28
|
+
members: reminder.event_reminder_members.edges.map(function (member) {
|
|
29
|
+
return {
|
|
30
|
+
memberID: member.node.id,
|
|
31
|
+
state: member.guest_list_state.toLowerCase(),
|
|
32
|
+
};
|
|
33
|
+
}),
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Formats a thread object from a GraphQL response.
|
|
39
|
+
* @param {Object} messageThread The raw message_thread object from GraphQL.
|
|
40
|
+
* @returns {Object | null} A formatted thread object or null if data is invalid.
|
|
41
|
+
*/
|
|
42
|
+
function formatThreadGraphQLResponse(messageThread) {
|
|
43
|
+
if (!messageThread || !messageThread.thread_key) return null;
|
|
44
|
+
|
|
45
|
+
const threadID = messageThread.thread_key.thread_fbid
|
|
46
|
+
? messageThread.thread_key.thread_fbid
|
|
47
|
+
: messageThread.thread_key.other_user_id;
|
|
48
|
+
|
|
49
|
+
const lastM = messageThread.last_message;
|
|
50
|
+
const snippetID =
|
|
51
|
+
lastM?.nodes?.[0]?.message_sender?.messaging_actor?.id || null;
|
|
52
|
+
const snippetText = lastM?.nodes?.[0]?.snippet || null;
|
|
53
|
+
const lastR = messageThread.last_read_receipt;
|
|
54
|
+
const lastReadTimestamp = lastR?.nodes?.[0]?.timestamp_precise || null;
|
|
55
|
+
|
|
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
|
+
const p = d.node.messaging_actor;
|
|
64
|
+
return {
|
|
65
|
+
id: p.id,
|
|
66
|
+
name: p.name,
|
|
67
|
+
firstName: p.short_name,
|
|
68
|
+
vanity: p.username,
|
|
69
|
+
url: p.url,
|
|
70
|
+
thumbSrc: p.big_image_src?.uri,
|
|
71
|
+
profileUrl: p.big_image_src?.uri,
|
|
72
|
+
gender: p.gender,
|
|
73
|
+
type: p.__typename,
|
|
74
|
+
isFriend: p.is_viewer_friend,
|
|
75
|
+
isBirthday: !!p.is_birthday,
|
|
76
|
+
isEmployee: p.is_employee,
|
|
77
|
+
isMessengerUser: p.is_messenger_user,
|
|
78
|
+
isVerified: p.is_verified,
|
|
79
|
+
isMessageBlockedByViewer: p.is_message_blocked_by_viewer,
|
|
80
|
+
isViewerCoworker: p.is_viewer_coworker,
|
|
81
|
+
acceptsMessengerUserFeedback: p.accepts_messenger_user_feedback,
|
|
82
|
+
isMessengerPlatformBot: p.is_messenger_platform_bot,
|
|
83
|
+
};
|
|
84
|
+
}),
|
|
85
|
+
unreadCount: messageThread.unread_count,
|
|
86
|
+
messageCount: messageThread.messages_count,
|
|
87
|
+
timestamp: messageThread.updated_time_precise,
|
|
88
|
+
muteUntil: messageThread.mute_until,
|
|
89
|
+
isGroup: messageThread.thread_type == "GROUP",
|
|
90
|
+
isSubscribed: messageThread.is_viewer_subscribed,
|
|
91
|
+
isArchived: messageThread.has_viewer_archived,
|
|
92
|
+
folder: messageThread.folder,
|
|
93
|
+
cannotReplyReason: messageThread.cannot_reply_reason,
|
|
94
|
+
eventReminders: messageThread.event_reminders
|
|
95
|
+
? messageThread.event_reminders.nodes.map(formatEventReminders)
|
|
96
|
+
: null,
|
|
97
|
+
emoji: messageThread.customization_info
|
|
98
|
+
? messageThread.customization_info.emoji
|
|
99
|
+
: null,
|
|
100
|
+
color:
|
|
101
|
+
messageThread.customization_info &&
|
|
102
|
+
messageThread.customization_info.outgoing_bubble_color
|
|
103
|
+
? messageThread.customization_info.outgoing_bubble_color.slice(2)
|
|
104
|
+
: null,
|
|
105
|
+
threadTheme: messageThread.thread_theme,
|
|
106
|
+
nicknames:
|
|
107
|
+
messageThread.customization_info &&
|
|
108
|
+
messageThread.customization_info.participant_customizations
|
|
109
|
+
? messageThread.customization_info.participant_customizations.reduce(
|
|
110
|
+
(res, val) => {
|
|
111
|
+
if (val.nickname) res[val.participant_id] = val.nickname;
|
|
112
|
+
return res;
|
|
113
|
+
},
|
|
114
|
+
{},
|
|
115
|
+
)
|
|
116
|
+
: {},
|
|
117
|
+
adminIDs: (messageThread.thread_admins || []).map(a => a.id),
|
|
118
|
+
approvalMode: Boolean(messageThread.approval_mode),
|
|
119
|
+
approvalQueue: (messageThread.group_approval_queue?.nodes || []).map((a) => ({
|
|
120
|
+
inviterID: a.inviter.id,
|
|
121
|
+
requesterID: a.requester.id,
|
|
122
|
+
timestamp: a.request_timestamp,
|
|
123
|
+
request_source: a.request_source,
|
|
124
|
+
})),
|
|
125
|
+
reactionsMuteMode: messageThread.reactions_mute_mode?.toLowerCase() || 'all_reactions',
|
|
126
|
+
mentionsMuteMode: messageThread.mentions_mute_mode?.toLowerCase() || 'all_mentions',
|
|
127
|
+
isPinProtected: messageThread.is_pin_protected,
|
|
128
|
+
relatedPageThread: messageThread.related_page_thread,
|
|
129
|
+
name: messageThread.name,
|
|
130
|
+
snippet: snippetText,
|
|
131
|
+
snippetSender: snippetID,
|
|
132
|
+
snippetAttachments: [],
|
|
133
|
+
serverTimestamp: messageThread.updated_time_precise,
|
|
134
|
+
imageSrc: messageThread.image ? messageThread.image.uri : null,
|
|
135
|
+
isCanonicalUser: messageThread.is_canonical_neo_user,
|
|
136
|
+
isCanonical: messageThread.thread_type != "GROUP",
|
|
137
|
+
recipientsLoadable: true,
|
|
138
|
+
hasEmailParticipant: false,
|
|
139
|
+
readOnly: false,
|
|
140
|
+
canReply: messageThread.cannot_reply_reason == null,
|
|
141
|
+
lastMessageTimestamp: messageThread.last_message
|
|
142
|
+
? messageThread.last_message.timestamp_precise
|
|
143
|
+
: null,
|
|
144
|
+
lastMessageType: "message",
|
|
145
|
+
lastReadTimestamp: lastReadTimestamp,
|
|
146
|
+
threadType: messageThread.thread_type == "GROUP" ? 2 : 1,
|
|
147
|
+
inviteLink: {
|
|
148
|
+
enable: messageThread.joinable_mode?.mode == 1,
|
|
149
|
+
link: messageThread.joinable_mode?.link || null,
|
|
150
|
+
},
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
/**
|
|
155
|
+
* @param {Object} defaultFuncs
|
|
156
|
+
* @param {Object} api
|
|
157
|
+
* @param {Object} ctx
|
|
158
|
+
* @returns {function(limit: number, timestamp: number | null, tags: string[]): Promise<Array<Object>>}
|
|
159
|
+
*/
|
|
160
|
+
module.exports = function (defaultFuncs, api, ctx) {
|
|
161
|
+
/**
|
|
162
|
+
* Retrieves a list of threads.
|
|
163
|
+
* @param {number} limit - The number of threads to retrieve.
|
|
164
|
+
* @param {number|null} timestamp - A timestamp to start fetching threads before. Use null for the most recent.
|
|
165
|
+
* @param {string[]} tags - An array of tags to filter threads by (e.g., ["INBOX", "ARCHIVED"]).
|
|
166
|
+
* @returns {Promise<Object[]>} A promise that resolves with an array of formatted thread objects.
|
|
167
|
+
*/
|
|
168
|
+
return async function getThreadList(limit, timestamp = null, tags = ["INBOX"]) {
|
|
169
|
+
if (utils.getType(limit) !== "Number" || !Number.isInteger(limit) || limit <= 0) {
|
|
170
|
+
throw new Error("getThreadList: limit must be a positive integer.");
|
|
171
|
+
}
|
|
172
|
+
if (utils.getType(timestamp) !== "Null" && (utils.getType(timestamp) !== "Number" || !Number.isInteger(timestamp))) {
|
|
173
|
+
throw new Error("getThreadList: timestamp must be an integer or null.");
|
|
174
|
+
}
|
|
175
|
+
if (utils.getType(tags) === "String") {
|
|
176
|
+
tags = [tags];
|
|
177
|
+
}
|
|
178
|
+
if (utils.getType(tags) !== "Array") {
|
|
179
|
+
throw new Error("getThreadList: tags must be an array.");
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
const form = {
|
|
183
|
+
av: ctx.i_userID || ctx.userID,
|
|
184
|
+
queries: JSON.stringify({
|
|
185
|
+
o0: {
|
|
186
|
+
doc_id: "3336396659757871",
|
|
187
|
+
query_params: {
|
|
188
|
+
limit: limit + (timestamp ? 1 : 0),
|
|
189
|
+
before: timestamp,
|
|
190
|
+
tags: tags,
|
|
191
|
+
includeDeliveryReceipts: true,
|
|
192
|
+
includeSeqID: false,
|
|
193
|
+
},
|
|
194
|
+
},
|
|
195
|
+
}),
|
|
196
|
+
batch_name: "MessengerGraphQLThreadlistFetcher",
|
|
197
|
+
};
|
|
198
|
+
|
|
199
|
+
try {
|
|
200
|
+
const resData = await defaultFuncs
|
|
201
|
+
.post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
|
|
202
|
+
.then(utils.parseAndCheckLogin(ctx, defaultFuncs));
|
|
203
|
+
|
|
204
|
+
if (!resData || !Array.isArray(resData) || resData.length === 0) {
|
|
205
|
+
throw new Error("getThreadList: Invalid response from server");
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const lastResult = resData[resData.length - 1];
|
|
209
|
+
if (lastResult && lastResult.error_results && lastResult.error_results > 0) {
|
|
210
|
+
throw new Error(JSON.stringify(resData[0]?.o0?.errors || "Unknown error"));
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
if (lastResult && lastResult.successful_results === 0) {
|
|
214
|
+
throw new Error("getThreadList: there was no successful_results");
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
if (!resData[0] || !resData[0].o0 || !resData[0].o0.data) {
|
|
218
|
+
throw new Error("getThreadList: Invalid data structure in response");
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
let nodes = resData[0].o0.data.viewer.message_threads.nodes;
|
|
222
|
+
if (timestamp) {
|
|
223
|
+
nodes.shift();
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
return nodes.map(formatThreadGraphQLResponse);
|
|
227
|
+
} catch (err) {
|
|
228
|
+
utils.error("getThreadList", err);
|
|
229
|
+
throw err;
|
|
230
|
+
}
|
|
231
|
+
};
|
|
232
|
+
};
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
|
|
5
|
+
module.exports = (defaultFuncs, api, ctx) => {
|
|
6
|
+
return async function getThreadPictures(threadID, offset, limit, callback) {
|
|
7
|
+
let resolveFunc = () => {};
|
|
8
|
+
let rejectFunc = () => {};
|
|
9
|
+
const returnPromise = new Promise((resolve, reject) => {
|
|
10
|
+
resolveFunc = resolve;
|
|
11
|
+
rejectFunc = reject;
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
if (!callback) {
|
|
15
|
+
if (utils.getType(limit) === "Function") {
|
|
16
|
+
callback = limit;
|
|
17
|
+
limit = 50;
|
|
18
|
+
} else if (utils.getType(offset) === "Function") {
|
|
19
|
+
callback = offset;
|
|
20
|
+
offset = 0;
|
|
21
|
+
limit = 50;
|
|
22
|
+
} else {
|
|
23
|
+
callback = (err, result) => {
|
|
24
|
+
if (err) return rejectFunc(err);
|
|
25
|
+
resolveFunc(result);
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
offset = offset || 0;
|
|
31
|
+
limit = limit || 50;
|
|
32
|
+
|
|
33
|
+
try {
|
|
34
|
+
const form = {
|
|
35
|
+
thread_id: threadID,
|
|
36
|
+
offset: offset,
|
|
37
|
+
limit: limit
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
const res = await defaultFuncs.post(
|
|
41
|
+
"https://www.facebook.com/ajax/mercury/thread_images.php",
|
|
42
|
+
ctx.jar,
|
|
43
|
+
form
|
|
44
|
+
).then(utils.parseAndCheckLogin(ctx, defaultFuncs));
|
|
45
|
+
|
|
46
|
+
if (res && res.error) {
|
|
47
|
+
throw res;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
return callback(null, res?.payload || res);
|
|
51
|
+
} catch (err) {
|
|
52
|
+
utils.error("getThreadPictures", err);
|
|
53
|
+
callback(err);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
return returnPromise;
|
|
57
|
+
};
|
|
58
|
+
};
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
|
|
5
|
+
function formatData(data) {
|
|
6
|
+
return {
|
|
7
|
+
userID: utils.formatID(data.uid.toString()),
|
|
8
|
+
photoUrl: data.photo,
|
|
9
|
+
indexRank: data.index_rank,
|
|
10
|
+
name: data.text,
|
|
11
|
+
isVerified: data.is_verified,
|
|
12
|
+
profileUrl: data.path,
|
|
13
|
+
category: data.category,
|
|
14
|
+
score: data.score,
|
|
15
|
+
type: data.type
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
module.exports = (defaultFuncs, api, ctx) => {
|
|
20
|
+
return async function getUID(link, callback) {
|
|
21
|
+
let resolveFunc = () => {};
|
|
22
|
+
let rejectFunc = () => {};
|
|
23
|
+
const returnPromise = new Promise((resolve, reject) => {
|
|
24
|
+
resolveFunc = resolve;
|
|
25
|
+
rejectFunc = reject;
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
if (!callback) {
|
|
29
|
+
callback = (err, result) => {
|
|
30
|
+
if (err) return rejectFunc(err);
|
|
31
|
+
resolveFunc(result);
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if (!link || typeof link !== 'string') {
|
|
36
|
+
const error = { error: "getUID: link parameter must be a non-empty string" };
|
|
37
|
+
utils.error("getUID", error);
|
|
38
|
+
return callback(error);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Check if it's a profile URL
|
|
42
|
+
const isProfileUrl = link.match(/\.com/);
|
|
43
|
+
if (!isProfileUrl) {
|
|
44
|
+
// Treat as username/name, use search
|
|
45
|
+
try {
|
|
46
|
+
const form = {
|
|
47
|
+
value: link.toLowerCase(),
|
|
48
|
+
viewer: ctx.userID,
|
|
49
|
+
rsp: "search",
|
|
50
|
+
context: "search",
|
|
51
|
+
path: "/home.php",
|
|
52
|
+
request_id: ctx.clientID || utils.getGUID()
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
const res = await defaultFuncs.get("https://www.facebook.com/ajax/typeahead/search.php", ctx.jar, form)
|
|
56
|
+
.then(utils.parseAndCheckLogin(ctx, defaultFuncs));
|
|
57
|
+
|
|
58
|
+
if (res.error) {
|
|
59
|
+
throw res;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
if (!res.payload || !res.payload.entries) {
|
|
63
|
+
const error = {
|
|
64
|
+
error: "getUID: No results found. This may be due to Facebook security restrictions or account checkpoint.",
|
|
65
|
+
details: "Your account may require verification. Please visit facebook.com to verify."
|
|
66
|
+
};
|
|
67
|
+
throw error;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
const data = res.payload.entries;
|
|
71
|
+
|
|
72
|
+
if (data.length === 0) {
|
|
73
|
+
utils.warn(`getUID: No user found with name "${link}"`);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
callback(null, data.map(formatData));
|
|
77
|
+
} catch (err) {
|
|
78
|
+
if (err.error && typeof err.error === 'string' && err.error.includes('checkpoint')) {
|
|
79
|
+
err.friendlyMessage = "Account checkpoint required - Please verify your account on facebook.com";
|
|
80
|
+
}
|
|
81
|
+
utils.error("getUID", err);
|
|
82
|
+
callback(err);
|
|
83
|
+
}
|
|
84
|
+
return returnPromise;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
// Handle profile URL
|
|
88
|
+
try {
|
|
89
|
+
let uid;
|
|
90
|
+
if (link.includes('profile.php?id=')) {
|
|
91
|
+
uid = link.split('profile.php?id=')[1].split('&')[0];
|
|
92
|
+
} else {
|
|
93
|
+
// For username URLs, fetch the page to get userID
|
|
94
|
+
const res = await defaultFuncs.get(link, ctx.jar)
|
|
95
|
+
.then(utils.parseAndCheckLogin(ctx, defaultFuncs));
|
|
96
|
+
|
|
97
|
+
const userIDMatch = res.match(/"userID":"(\d+)"/) || res.match(/"id":"(\d+)"/) || res.match(/entity_id["\s:]+(\d+)/);
|
|
98
|
+
if (userIDMatch) {
|
|
99
|
+
uid = userIDMatch[1];
|
|
100
|
+
} else {
|
|
101
|
+
throw new Error("Could not extract user ID from profile URL");
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
if (!uid || !/^\d+$/.test(uid)) {
|
|
106
|
+
throw new Error("Invalid user ID extracted");
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
callback(null, uid);
|
|
110
|
+
} catch (err) {
|
|
111
|
+
utils.error("getUID", err);
|
|
112
|
+
callback(err);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
return returnPromise;
|
|
116
|
+
};
|
|
117
|
+
};
|