shadowx-fca 2.4.0 → 2.5.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/checkUpdate.js +1 -1
- package/index.js +664 -418
- package/package.json +1 -1
- package/src/GetBotInfo.js +57 -0
- package/src/OldMessage.js +38 -10
- package/src/comment.js +213 -0
- package/src/emoji.js +124 -0
- package/src/friend.js +243 -0
- package/src/gcmember.js +122 -0
- package/src/getUserInfo.js +222 -43
- package/src/listenMqtt.js +9 -116
- package/src/nickname.js +132 -0
- package/src/sendMessage.js +224 -235
- package/src/sendTypingIndicator.js +45 -101
- package/src/share.js +62 -0
- package/src/shareContact.js +17 -61
- package/src/stickers.js +117 -0
- package/src/story.js +181 -0
- package/src/theme.js +233 -0
- package/utils.js +24 -1311
package/src/friend.js
ADDED
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* S4hiilAns4ri
|
|
7
|
+
* @description A module for managing friend-related actions like listing friends, handling requests, and getting suggestions.
|
|
8
|
+
* @param {Object} defaultFuncs The default functions provided by the API wrapper.
|
|
9
|
+
* @param {Object} api The full API object.
|
|
10
|
+
* @param {Object} ctx The context object containing the user's session state (e.g., userID, jar, fb_dtsg).
|
|
11
|
+
* @returns {Object} A `friendModule` object with methods for friend interactions.
|
|
12
|
+
*/
|
|
13
|
+
module.exports = function (defaultFuncs, api, ctx) {
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* A private helper function to standardize friend data from various GraphQL endpoints.
|
|
17
|
+
* @private
|
|
18
|
+
* @param {Object} data The raw data object from a GraphQL response.
|
|
19
|
+
* @param {('requests'|'suggestions'|'list')} type The type of data to format, which determines the path to the user list.
|
|
20
|
+
* @returns {Array<Object>} An array of formatted friend objects, each containing `userID`, `name`, `profilePicture`, `socialContext`, and `url`.
|
|
21
|
+
*/
|
|
22
|
+
function formatFriends(data, type) {
|
|
23
|
+
const viewer = data?.data?.viewer;
|
|
24
|
+
let edges;
|
|
25
|
+
if (type === 'requests' && viewer?.friend_requests?.edges) {
|
|
26
|
+
edges = viewer.friend_requests.edges;
|
|
27
|
+
} else if (type === 'suggestions' && viewer?.people_you_may_know?.edges) {
|
|
28
|
+
edges = viewer.people_you_may_know.edges;
|
|
29
|
+
} else if (type === 'list' && data?.data?.node?.all_collections?.nodes[0]?.style_renderer?.collection?.pageItems?.edges) {
|
|
30
|
+
edges = data.data.node.all_collections.nodes[0].style_renderer.collection.pageItems.edges;
|
|
31
|
+
} else {
|
|
32
|
+
return [];
|
|
33
|
+
}
|
|
34
|
+
return edges.map(edge => {
|
|
35
|
+
const node = edge.node;
|
|
36
|
+
return {
|
|
37
|
+
userID: node.id || node.node?.id,
|
|
38
|
+
name: node.name || node.title?.text,
|
|
39
|
+
profilePicture: node.profile_picture?.uri || node.image?.uri,
|
|
40
|
+
socialContext: node.social_context?.text || node.subtitle_text?.text,
|
|
41
|
+
url: node.url
|
|
42
|
+
};
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
const friendModule = {
|
|
47
|
+
/**
|
|
48
|
+
* @namespace api.friend
|
|
49
|
+
* @description A collection of functions for interacting with friends.
|
|
50
|
+
* @license Ex-it
|
|
51
|
+
* @author S4hiilAns4ri
|
|
52
|
+
*/
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Fetches the list of incoming friend requests.
|
|
56
|
+
* @async
|
|
57
|
+
* @returns {Promise<Array<Object>>} A promise that resolves to an array of friend request objects.
|
|
58
|
+
* @throws {Error} If the API request fails or returns an error.
|
|
59
|
+
*/
|
|
60
|
+
requests: async function() {
|
|
61
|
+
try {
|
|
62
|
+
const form = {
|
|
63
|
+
av: ctx.userID,
|
|
64
|
+
__user: ctx.userID,
|
|
65
|
+
__a: "1",
|
|
66
|
+
fb_dtsg: ctx.fb_dtsg,
|
|
67
|
+
jazoest: ctx.jazoest,
|
|
68
|
+
lsd: ctx.lsd,
|
|
69
|
+
fb_api_caller_class: "RelayModern",
|
|
70
|
+
fb_api_req_friendly_name: "FriendingCometRootContentQuery",
|
|
71
|
+
variables: JSON.stringify({ scale: 3 }),
|
|
72
|
+
doc_id: "9103543533085580"
|
|
73
|
+
};
|
|
74
|
+
const res = await defaultFuncs.post("https://www.facebook.com/api/graphql/", ctx.jar, form, {});
|
|
75
|
+
if (res.data.errors) throw new Error(JSON.stringify(res.data.errors));
|
|
76
|
+
return formatFriends(res.data, 'requests');
|
|
77
|
+
} catch (err) {
|
|
78
|
+
throw err;
|
|
79
|
+
}
|
|
80
|
+
},
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Accepts a friend request.
|
|
84
|
+
* @async
|
|
85
|
+
* @param {string} identifier The user ID or name of the person whose friend request is to be accepted. If a name is provided, the function will search through pending requests.
|
|
86
|
+
* @returns {Promise<Object>} A promise that resolves to the API response data upon successful acceptance.
|
|
87
|
+
* @throws {Error} If the identifier is missing, the user is not found in requests, or the API call fails.
|
|
88
|
+
*/
|
|
89
|
+
accept: async function(identifier) {
|
|
90
|
+
try {
|
|
91
|
+
if (!identifier) throw new Error("A name or user ID is required.");
|
|
92
|
+
let targetUserID = identifier;
|
|
93
|
+
if (isNaN(identifier)) {
|
|
94
|
+
const requests = await friendModule.requests();
|
|
95
|
+
const found = requests.find(req => req.name.toLowerCase().includes(identifier.toLowerCase()));
|
|
96
|
+
if (!found) throw new Error(`Could not find any friend request matching "${identifier}".`);
|
|
97
|
+
targetUserID = found.userID;
|
|
98
|
+
}
|
|
99
|
+
const variables = {
|
|
100
|
+
input: {
|
|
101
|
+
friend_requester_id: targetUserID,
|
|
102
|
+
friending_channel: "FRIENDS_HOME_MAIN",
|
|
103
|
+
actor_id: ctx.userID,
|
|
104
|
+
client_mutation_id: Math.floor(Math.random() * 10 + 1).toString()
|
|
105
|
+
},
|
|
106
|
+
scale: 3
|
|
107
|
+
};
|
|
108
|
+
const form = {
|
|
109
|
+
av: ctx.userID,
|
|
110
|
+
__user: ctx.userID,
|
|
111
|
+
__a: "1",
|
|
112
|
+
fb_dtsg: ctx.fb_dtsg,
|
|
113
|
+
jazoest: ctx.jazoest,
|
|
114
|
+
lsd: ctx.lsd,
|
|
115
|
+
fb_api_caller_class: "RelayModern",
|
|
116
|
+
fb_api_req_friendly_name: "FriendingCometFriendRequestConfirmMutation",
|
|
117
|
+
variables: JSON.stringify(variables),
|
|
118
|
+
doc_id: "24630768433181357"
|
|
119
|
+
};
|
|
120
|
+
const res = await defaultFuncs.post("https://www.facebook.com/api/graphql/", ctx.jar, form, {});
|
|
121
|
+
if (res.data.errors) throw new Error(JSON.stringify(res.data.errors));
|
|
122
|
+
return res.data.data;
|
|
123
|
+
} catch (err) {
|
|
124
|
+
if (err.message?.includes("1431004")) {
|
|
125
|
+
throw new Error("I cannot accept this friend request right now. There might be a problem with the account or you need to wait.");
|
|
126
|
+
}
|
|
127
|
+
throw err;
|
|
128
|
+
}
|
|
129
|
+
},
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Fetches the friend list for a given user ID.
|
|
133
|
+
* @async
|
|
134
|
+
* @param {string} [userID=ctx.userID] The ID of the user whose friend list to fetch. Defaults to the logged-in user.
|
|
135
|
+
* @returns {Promise<Array<Object>>} A promise that resolves to an array of formatted friend objects.
|
|
136
|
+
* @throws {Error} If the API request fails.
|
|
137
|
+
*/
|
|
138
|
+
list: async function(userID = ctx.userID) {
|
|
139
|
+
try {
|
|
140
|
+
const sectionToken = Buffer.from(`app_section:${userID}:2356318349`).toString('base64');
|
|
141
|
+
const variables = {
|
|
142
|
+
collectionToken: null,
|
|
143
|
+
scale: 2,
|
|
144
|
+
sectionToken: sectionToken,
|
|
145
|
+
useDefaultActor: false,
|
|
146
|
+
userID: userID
|
|
147
|
+
};
|
|
148
|
+
const form = {
|
|
149
|
+
av: ctx.userID,
|
|
150
|
+
__user: ctx.userID,
|
|
151
|
+
__a: "1",
|
|
152
|
+
fb_dtsg: ctx.fb_dtsg,
|
|
153
|
+
jazoest: ctx.jazoest,
|
|
154
|
+
lsd: ctx.lsd,
|
|
155
|
+
fb_api_caller_class: "RelayModern",
|
|
156
|
+
fb_api_req_friendly_name: "ProfileCometTopAppSectionQuery",
|
|
157
|
+
variables: JSON.stringify(variables),
|
|
158
|
+
doc_id: "24492266383698794"
|
|
159
|
+
};
|
|
160
|
+
const res = await defaultFuncs.post("https://www.facebook.com/api/graphql/", ctx.jar, form, {});
|
|
161
|
+
if (res.data.errors) throw new Error(JSON.stringify(res.data.errors));
|
|
162
|
+
return formatFriends(res.data, 'list');
|
|
163
|
+
} catch(err) {
|
|
164
|
+
throw err;
|
|
165
|
+
}
|
|
166
|
+
},
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* @namespace api.friend.suggest
|
|
170
|
+
* @description Functions for managing friend suggestions.
|
|
171
|
+
*/
|
|
172
|
+
suggest: {
|
|
173
|
+
/**
|
|
174
|
+
* Fetches a list of suggested friends (People You May Know).
|
|
175
|
+
* @async
|
|
176
|
+
* @param {number} [limit=30] The maximum number of suggestions to fetch.
|
|
177
|
+
* @returns {Promise<Array<Object>>} A promise that resolves to an array of suggested friend objects.
|
|
178
|
+
* @throws {Error} If the API request fails.
|
|
179
|
+
*/
|
|
180
|
+
list: async function(limit = 30) {
|
|
181
|
+
try {
|
|
182
|
+
const form = {
|
|
183
|
+
av: ctx.userID,
|
|
184
|
+
__user: ctx.userID,
|
|
185
|
+
__a: "1",
|
|
186
|
+
fb_dtsg: ctx.fb_dtsg,
|
|
187
|
+
jazoest: ctx.jazoest,
|
|
188
|
+
lsd: ctx.lsd,
|
|
189
|
+
fb_api_caller_class: "RelayModern",
|
|
190
|
+
fb_api_req_friendly_name: "FriendingCometPYMKPanelPaginationQuery",
|
|
191
|
+
variables: JSON.stringify({ count: limit, cursor: null, scale: 3 }),
|
|
192
|
+
doc_id: "9917809191634193"
|
|
193
|
+
};
|
|
194
|
+
const res = await defaultFuncs.post("https://www.facebook.com/api/graphql/", ctx.jar, form, {});
|
|
195
|
+
if (res.data.errors) throw new Error(JSON.stringify(res.data.errors));
|
|
196
|
+
return formatFriends(res.data, 'suggestions');
|
|
197
|
+
} catch(err) {
|
|
198
|
+
throw err;
|
|
199
|
+
}
|
|
200
|
+
},
|
|
201
|
+
/**
|
|
202
|
+
* Sends a friend request to a user.
|
|
203
|
+
* @async
|
|
204
|
+
* @param {string} userID The ID of the user to send the friend request to.
|
|
205
|
+
* @returns {Promise<Object>} A promise that resolves to the API response data on success.
|
|
206
|
+
* @throws {Error} If the userID is missing or the API request fails.
|
|
207
|
+
*/
|
|
208
|
+
request: async function(userID) {
|
|
209
|
+
try {
|
|
210
|
+
if (!userID) throw new Error("userID is required.");
|
|
211
|
+
const variables = {
|
|
212
|
+
input: {
|
|
213
|
+
friend_requestee_ids: [userID],
|
|
214
|
+
friending_channel: "FRIENDS_HOME_MAIN",
|
|
215
|
+
actor_id: ctx.userID,
|
|
216
|
+
client_mutation_id: Math.floor(Math.random() * 10 + 1).toString()
|
|
217
|
+
},
|
|
218
|
+
scale: 3
|
|
219
|
+
};
|
|
220
|
+
const form = {
|
|
221
|
+
av: ctx.userID,
|
|
222
|
+
__user: ctx.userID,
|
|
223
|
+
__a: "1",
|
|
224
|
+
fb_dtsg: ctx.fb_dtsg,
|
|
225
|
+
jazoest: ctx.jazoest,
|
|
226
|
+
lsd: ctx.lsd,
|
|
227
|
+
fb_api_caller_class: "RelayModern",
|
|
228
|
+
fb_api_req_friendly_name: "FriendingCometFriendRequestSendMutation",
|
|
229
|
+
variables: JSON.stringify(variables),
|
|
230
|
+
doc_id: "23982103144788355"
|
|
231
|
+
};
|
|
232
|
+
const res = await defaultFuncs.post("https://www.facebook.com/api/graphql/", ctx.jar, form, {});
|
|
233
|
+
if (res.data.errors) throw new Error(JSON.stringify(res.data.errors));
|
|
234
|
+
return res.data.data;
|
|
235
|
+
} catch(err) {
|
|
236
|
+
throw err;
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
};
|
|
241
|
+
|
|
242
|
+
return friendModule;
|
|
243
|
+
};
|
package/src/gcmember.js
ADDED
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
|
|
5
|
+
module.exports = function (defaultFuncs, api, ctx) {
|
|
6
|
+
/**
|
|
7
|
+
* Author: S4hiilAns4ri (github.com/S4hiilAns4ri)
|
|
8
|
+
* Mqtt
|
|
9
|
+
* Adds or removes members from a group chat with pre-checking.
|
|
10
|
+
*
|
|
11
|
+
* @param {"add" | "remove"} action The action to perform.
|
|
12
|
+
* @param {string|string[]} userIDs The user ID or array of user IDs.
|
|
13
|
+
* @param {string} threadID The ID of the group chat.
|
|
14
|
+
* @param {Function} [callback] Optional callback function.
|
|
15
|
+
* @returns {Promise<object>} A promise that resolves with information about the action.
|
|
16
|
+
*/
|
|
17
|
+
return async function gcmember(action, userIDs, threadID, callback) {
|
|
18
|
+
let _callback;
|
|
19
|
+
if (typeof threadID === 'function') {
|
|
20
|
+
_callback = threadID;
|
|
21
|
+
threadID = null;
|
|
22
|
+
} else if (typeof callback === 'function') {
|
|
23
|
+
_callback = callback;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
let resolvePromise, rejectPromise;
|
|
27
|
+
const returnPromise = new Promise((resolve, reject) => {
|
|
28
|
+
resolvePromise = resolve;
|
|
29
|
+
rejectPromise = reject;
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
if (typeof _callback != "function") {
|
|
33
|
+
_callback = (err, data) => {
|
|
34
|
+
// Note: We will now rarely use the 'err' parameter for validation errors
|
|
35
|
+
if (err) return rejectPromise(err);
|
|
36
|
+
resolvePromise(data);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
try {
|
|
41
|
+
const validActions = ["add", "remove"];
|
|
42
|
+
action = action ? action.toLowerCase() : "";
|
|
43
|
+
|
|
44
|
+
// --- ERROR CHECKS NOW RETURN AN OBJECT INSTEAD OF THROWING ---
|
|
45
|
+
if (!validActions.includes(action)) {
|
|
46
|
+
return _callback(null, { type: "error_gc", error: `Invalid action. Must be one of: ${validActions.join(", ")}` });
|
|
47
|
+
}
|
|
48
|
+
if (!userIDs || userIDs.length === 0) {
|
|
49
|
+
return _callback(null, { type: "error_gc", error: "userIDs is required." });
|
|
50
|
+
}
|
|
51
|
+
if (!threadID) {
|
|
52
|
+
return _callback(null, { type: "error_gc", error: "threadID is required." });
|
|
53
|
+
}
|
|
54
|
+
if (!ctx.mqttClient) {
|
|
55
|
+
return _callback(null, { type: "error_gc", error: "Not connected to MQTT" });
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const threadInfo = await api.getThreadInfo(threadID);
|
|
59
|
+
if (!threadInfo) {
|
|
60
|
+
return _callback(null, { type: "error_gc", error: "Could not retrieve thread information." });
|
|
61
|
+
}
|
|
62
|
+
if (threadInfo.isGroup === false) {
|
|
63
|
+
return _callback(null, { type: "error_gc", error: "This feature is only for group chats, not private messages." });
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
const currentMembers = threadInfo.participantIDs;
|
|
67
|
+
const usersToModify = Array.isArray(userIDs) ? userIDs : [userIDs];
|
|
68
|
+
let queryPayload, query;
|
|
69
|
+
let finalUsers = usersToModify;
|
|
70
|
+
|
|
71
|
+
ctx.wsReqNumber = (ctx.wsReqNumber || 0) + 1;
|
|
72
|
+
ctx.wsTaskNumber = (ctx.wsTaskNumber || 0) + 1;
|
|
73
|
+
|
|
74
|
+
if (action === 'add') {
|
|
75
|
+
const usersToAdd = usersToModify.filter(id => !currentMembers.includes(id));
|
|
76
|
+
if (usersToAdd.length === 0) {
|
|
77
|
+
return _callback(null, { type: "error_gc", error: "All specified users are already in the group." });
|
|
78
|
+
}
|
|
79
|
+
finalUsers = usersToAdd;
|
|
80
|
+
queryPayload = { thread_key: parseInt(threadID), contact_ids: finalUsers.map(id => parseInt(id)), sync_group: 1 };
|
|
81
|
+
query = { label: "23", payload: JSON.stringify(queryPayload), queue_name: threadID, task_id: ctx.wsTaskNumber };
|
|
82
|
+
|
|
83
|
+
} else { // action is 'remove'
|
|
84
|
+
const userToRemove = usersToModify[0];
|
|
85
|
+
if (!currentMembers.includes(userToRemove)) {
|
|
86
|
+
return _callback(null, { type: "error_gc", error: `User with ID ${userToRemove} is not in this group chat.` });
|
|
87
|
+
}
|
|
88
|
+
finalUsers = [userToRemove];
|
|
89
|
+
queryPayload = { thread_id: threadID, contact_id: userToRemove, sync_group: 1 };
|
|
90
|
+
query = { label: "140", payload: JSON.stringify(queryPayload), queue_name: "remove_participant_v2", task_id: ctx.wsTaskNumber };
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const context = {
|
|
94
|
+
app_id: ctx.appID,
|
|
95
|
+
payload: { epoch_id: parseInt(utils.generateOfflineThreadingID()), tasks: [query], version_id: "24631415369801570" },
|
|
96
|
+
request_id: ctx.wsReqNumber,
|
|
97
|
+
type: 3
|
|
98
|
+
};
|
|
99
|
+
context.payload = JSON.stringify(context.payload);
|
|
100
|
+
|
|
101
|
+
ctx.mqttClient.publish('/ls_req', JSON.stringify(context), { qos: 1, retain: false }, (err) => {
|
|
102
|
+
if (err) return _callback(err); // For network errors, we still reject
|
|
103
|
+
|
|
104
|
+
const gcmemberInfo = {
|
|
105
|
+
type: "gc_member_update",
|
|
106
|
+
threadID: threadID,
|
|
107
|
+
userIDs: finalUsers,
|
|
108
|
+
action: action,
|
|
109
|
+
senderID: ctx.userID,
|
|
110
|
+
BotID: ctx.userID,
|
|
111
|
+
timestamp: Date.now(),
|
|
112
|
+
};
|
|
113
|
+
return _callback(null, gcmemberInfo);
|
|
114
|
+
});
|
|
115
|
+
} catch (err) {
|
|
116
|
+
|
|
117
|
+
return _callback(null, { type: "error_gc", error: err.message || "An unknown error occurred." });
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
return returnPromise;
|
|
121
|
+
};
|
|
122
|
+
};
|
package/src/getUserInfo.js
CHANGED
|
@@ -1,66 +1,245 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
// sahilchat-fca — Author: S4hiilAns4ri (github.com/S4hiilAns4ri)
|
|
3
|
+
const utils = require('../utils');
|
|
4
|
+
const _ = require('lodash');
|
|
5
|
+
const deepdash = require('deepdash');
|
|
6
|
+
deepdash(_);
|
|
2
7
|
|
|
3
|
-
|
|
4
|
-
|
|
8
|
+
/**
|
|
9
|
+
* @param {object} data
|
|
10
|
+
* @param {string} userID
|
|
11
|
+
* @returns {object|null}
|
|
12
|
+
*/
|
|
13
|
+
function findMainUserObject(data, userID) {
|
|
14
|
+
let mainUserObject = null;
|
|
15
|
+
if (!Array.isArray(data)) return null;
|
|
16
|
+
function deepFind(obj) {
|
|
17
|
+
if (mainUserObject || typeof obj !== 'object' || obj === null) return;
|
|
18
|
+
if (obj.id === userID && obj.__typename === 'User' && obj.profile_tabs) {
|
|
19
|
+
mainUserObject = obj;
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
for (const k in obj) {
|
|
23
|
+
if (obj.hasOwnProperty(k)) {
|
|
24
|
+
deepFind(obj[k]);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
deepFind({ all: data });
|
|
29
|
+
return mainUserObject;
|
|
30
|
+
}
|
|
5
31
|
|
|
6
|
-
|
|
7
|
-
|
|
32
|
+
/**
|
|
33
|
+
* @param {object} socialContext
|
|
34
|
+
* @param {string} keyword
|
|
35
|
+
* @returns {string|null}
|
|
36
|
+
*/
|
|
37
|
+
function findSocialContextText(socialContext, keyword) {
|
|
38
|
+
if (socialContext && Array.isArray(socialContext.content)) {
|
|
39
|
+
for (const item of socialContext.content) {
|
|
40
|
+
const text = item?.text?.text;
|
|
41
|
+
if (text && text.toLowerCase().includes(keyword.toLowerCase())) {
|
|
42
|
+
return text;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
8
48
|
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
49
|
+
/**
|
|
50
|
+
* @param {Array<Object>} dataArray
|
|
51
|
+
* @param {string} key
|
|
52
|
+
* @returns {any}
|
|
53
|
+
*/
|
|
54
|
+
function findFirstValueByKey(dataArray, key) {
|
|
55
|
+
if (!Array.isArray(dataArray)) return null;
|
|
56
|
+
let found = null;
|
|
57
|
+
function deepSearch(obj) {
|
|
58
|
+
if (found !== null || typeof obj !== 'object' || obj === null) return;
|
|
59
|
+
if (obj.hasOwnProperty(key)) {
|
|
60
|
+
found = obj[key];
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
for (const k in obj) {
|
|
64
|
+
if (obj.hasOwnProperty(k)) {
|
|
65
|
+
deepSearch(obj[k]);
|
|
66
|
+
}
|
|
24
67
|
}
|
|
25
68
|
}
|
|
69
|
+
for (const obj of dataArray) {
|
|
70
|
+
deepSearch(obj);
|
|
71
|
+
}
|
|
72
|
+
return found;
|
|
73
|
+
}
|
|
26
74
|
|
|
27
|
-
|
|
75
|
+
/**
|
|
76
|
+
* @param {Array<Object>} allJsonData
|
|
77
|
+
* @returns {string|null}
|
|
78
|
+
*/
|
|
79
|
+
function findBioFromProfileTiles(allJsonData) {
|
|
80
|
+
try {
|
|
81
|
+
const bio = findFirstValueByKey(allJsonData, 'profile_status_text');
|
|
82
|
+
return bio?.text || null;
|
|
83
|
+
} catch {
|
|
84
|
+
return null;
|
|
85
|
+
}
|
|
28
86
|
}
|
|
29
87
|
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
88
|
+
/**
|
|
89
|
+
* @param {Array<Object>} allJsonData
|
|
90
|
+
* @returns {string|null}
|
|
91
|
+
*/
|
|
92
|
+
function findLiveCityFromProfileTiles(allJsonData) {
|
|
93
|
+
try {
|
|
94
|
+
const result = _.findDeep(allJsonData, (value, key, parent) => {
|
|
95
|
+
return key === 'text' &&
|
|
96
|
+
typeof value === 'string' &&
|
|
97
|
+
value.includes('Lives in') &&
|
|
98
|
+
parent?.ranges?.[0]?.entity?.category_type === "CITY_WITH_ID";
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
if (result) {
|
|
102
|
+
return result.value;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
return null;
|
|
106
|
+
} catch (err) {
|
|
107
|
+
return null;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
module.exports = (defaultFuncs, api, ctx) => {
|
|
112
|
+
function createDefaultUser(id) {
|
|
113
|
+
return {
|
|
114
|
+
id,
|
|
115
|
+
name: "Facebook User",
|
|
116
|
+
firstName: "Facebook",
|
|
117
|
+
lastName: null,
|
|
118
|
+
vanity: id,
|
|
119
|
+
profilePicUrl: `https://graph.facebook.com/${id}/picture?width=720&height=720&access_token=6628568379%7Cc1e620fa708a1d5696fb991c1bde5662`,
|
|
120
|
+
profileUrl: `https://www.facebook.com/profile.php?id=${id}`,
|
|
121
|
+
gender: "no specific gender",
|
|
122
|
+
type: "user",
|
|
123
|
+
isFriend: false,
|
|
124
|
+
isBirthday: false
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
return function getUserInfo(id, usePayload, callback, groupFields = []) {
|
|
129
|
+
let resolveFunc = () => {};
|
|
130
|
+
let rejectFunc = () => {};
|
|
131
|
+
const returnPromise = new Promise((resolve, reject) => {
|
|
35
132
|
resolveFunc = resolve;
|
|
36
133
|
rejectFunc = reject;
|
|
37
134
|
});
|
|
38
135
|
|
|
136
|
+
if (typeof usePayload === 'function') {
|
|
137
|
+
callback = usePayload;
|
|
138
|
+
usePayload = true;
|
|
139
|
+
}
|
|
140
|
+
if (usePayload === undefined) usePayload = true;
|
|
39
141
|
if (!callback) {
|
|
40
|
-
callback =
|
|
142
|
+
callback = (err, data) => {
|
|
41
143
|
if (err) return rejectFunc(err);
|
|
42
|
-
resolveFunc(
|
|
144
|
+
resolveFunc(data);
|
|
43
145
|
};
|
|
44
146
|
}
|
|
45
147
|
|
|
46
|
-
|
|
148
|
+
const originalIdIsArray = Array.isArray(id);
|
|
149
|
+
const ids = originalIdIsArray ? id : [id];
|
|
47
150
|
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
form[
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
151
|
+
if (usePayload) {
|
|
152
|
+
const form = {};
|
|
153
|
+
ids.forEach((v, i) => { form[`ids[${i}]`] = v; });
|
|
154
|
+
const getGenderString = (code) => code === 1 ? "male" : code === 2 ? "female" : "no specific gender";
|
|
155
|
+
defaultFuncs.post("https://www.facebook.com/chat/user_info/", ctx.jar, form)
|
|
156
|
+
.then(resData => utils.parseAndCheckLogin(ctx, defaultFuncs)(resData))
|
|
157
|
+
.then(resData => {
|
|
158
|
+
if (resData?.error && resData?.error !== 3252001) throw resData;
|
|
159
|
+
const retObj = {};
|
|
160
|
+
const profiles = resData?.payload?.profiles;
|
|
161
|
+
if (profiles) {
|
|
162
|
+
for (const prop in profiles) {
|
|
163
|
+
if (profiles.hasOwnProperty(prop)) {
|
|
164
|
+
const inner = profiles[prop];
|
|
165
|
+
const nameParts = inner.name ? inner.name.split(' ') : [];
|
|
166
|
+
retObj[prop] = {
|
|
167
|
+
id: prop,
|
|
168
|
+
name: inner.name,
|
|
169
|
+
firstName: inner.firstName,
|
|
170
|
+
lastName: nameParts.length > 1 ? nameParts[nameParts.length - 1] : null,
|
|
171
|
+
vanity: inner.vanity,
|
|
172
|
+
profilePicUrl: `https://graph.facebook.com/${prop}/picture?width=720&height=720&access_token=6628568379%7Cc1e620fa708a1d5696fb991c1bde5662`,
|
|
173
|
+
profileUrl: inner.uri,
|
|
174
|
+
gender: getGenderString(inner.gender),
|
|
175
|
+
type: inner.type,
|
|
176
|
+
isFriend: inner.is_friend,
|
|
177
|
+
isBirthday: !!inner.is_birthday,
|
|
178
|
+
searchTokens: inner.searchTokens,
|
|
179
|
+
alternateName: inner.alternateName
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
} else {
|
|
184
|
+
for (const prop of ids) {
|
|
185
|
+
retObj[prop] = createDefaultUser(prop);
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
return originalIdIsArray ? callback(null, Object.values(retObj)) : callback(null, retObj[ids[0]]);
|
|
189
|
+
}).catch(err => {
|
|
190
|
+
utils.error("getUserInfo (payload)", err);
|
|
191
|
+
return callback(err);
|
|
192
|
+
});
|
|
193
|
+
} else {
|
|
194
|
+
const fetchProfile = async (userID) => {
|
|
195
|
+
try {
|
|
196
|
+
const url = `https://www.facebook.com/${userID}`;
|
|
197
|
+
const allJsonData = await utils.json(url, ctx.jar, null, ctx.globalOptions, ctx);
|
|
198
|
+
if (!allJsonData || allJsonData.length === 0) throw new Error(`Could not find JSON data for ID: ${userID}`);
|
|
199
|
+
const mainUserObject = findMainUserObject(allJsonData, userID);
|
|
200
|
+
if (!mainUserObject) throw new Error(`Could not isolate main user object for ID: ${userID}`);
|
|
201
|
+
const get = (obj, path) => {
|
|
202
|
+
if (!obj || !path) return null;
|
|
203
|
+
return path.split('.').reduce((prev, curr) => (prev ? prev[curr] : undefined), obj);
|
|
204
|
+
};
|
|
205
|
+
const name = mainUserObject.name;
|
|
206
|
+
const nameParts = name ? name.split(' ') : [];
|
|
207
|
+
const result = {
|
|
208
|
+
id: mainUserObject.id,
|
|
209
|
+
name: name,
|
|
210
|
+
firstName: nameParts[0] || get(mainUserObject, 'short_name') || get(findFirstValueByKey(allJsonData, 'profile_owner'), 'short_name'),
|
|
211
|
+
lastName: nameParts.length > 1 ? nameParts[nameParts.length - 1] : null,
|
|
212
|
+
vanity: get(mainUserObject, 'vanity') || get(findFirstValueByKey(allJsonData, 'props'), 'userVanity') || null,
|
|
213
|
+
profileUrl: mainUserObject.url,
|
|
214
|
+
profilePicUrl: `https://graph.facebook.com/${userID}/picture?width=720&height=720&access_token=6628568379%7Cc1e620fa708a1d5696fb991c1bde5662`,
|
|
215
|
+
gender: mainUserObject.gender,
|
|
216
|
+
type: mainUserObject.__typename,
|
|
217
|
+
isFriend: mainUserObject.is_viewer_friend,
|
|
218
|
+
isBirthday: !!mainUserObject.is_birthday,
|
|
219
|
+
isVerified: !!mainUserObject.show_verified_badge_on_profile,
|
|
220
|
+
bio: findBioFromProfileTiles(allJsonData) || get(findFirstValueByKey(allJsonData, 'delegate_page'), 'best_description.text'),
|
|
221
|
+
live_city: findLiveCityFromProfileTiles(allJsonData),
|
|
222
|
+
headline: get(mainUserObject, 'contextual_headline.text') || get(findFirstValueByKey(allJsonData, 'meta_verified_section'), 'headline'),
|
|
223
|
+
followers: findSocialContextText(mainUserObject.profile_social_context, "followers"),
|
|
224
|
+
following: findSocialContextText(mainUserObject.profile_social_context, "following"),
|
|
225
|
+
coverPhoto: get(mainUserObject, 'cover_photo.photo.image.uri')
|
|
226
|
+
};
|
|
227
|
+
return result;
|
|
228
|
+
} catch (err) {
|
|
229
|
+
utils.error(`Failed to fetch profile for ${userID}: ${err.message}`, err);
|
|
230
|
+
return createDefaultUser(userID);
|
|
231
|
+
}
|
|
232
|
+
};
|
|
63
233
|
|
|
234
|
+
Promise.all(ids.map(fetchProfile))
|
|
235
|
+
.then(results => {
|
|
236
|
+
return originalIdIsArray ? callback(null, results) : callback(null, results[0] || null);
|
|
237
|
+
})
|
|
238
|
+
.catch(err => {
|
|
239
|
+
utils.error("getUserInfo (fetch)", err);
|
|
240
|
+
callback(err);
|
|
241
|
+
});
|
|
242
|
+
}
|
|
64
243
|
return returnPromise;
|
|
65
244
|
};
|
|
66
245
|
};
|