@dongdev/fca-unofficial 3.0.31 → 4.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (128) hide show
  1. package/LICENSE +191 -0
  2. package/README.md +238 -398
  3. package/dist/cjs.cjs +9 -0
  4. package/dist/index.d.mts +1250 -0
  5. package/dist/index.d.ts +1250 -0
  6. package/dist/index.js +27772 -0
  7. package/dist/index.mjs +27735 -0
  8. package/docs/ARCHITECTURE.md +467 -0
  9. package/docs/DOCS.md +709 -0
  10. package/fca-config.example.json +33 -0
  11. package/package.json +32 -22
  12. package/test/fca.test.cjs +540 -0
  13. package/CHANGELOG.md +0 -296
  14. package/DOCS.md +0 -2712
  15. package/func/checkUpdate.js +0 -222
  16. package/func/logAdapter.js +0 -33
  17. package/func/logger.js +0 -48
  18. package/index.d.ts +0 -751
  19. package/index.js +0 -8
  20. package/module/config.js +0 -40
  21. package/module/login.js +0 -133
  22. package/module/loginHelper.js +0 -1296
  23. package/module/options.js +0 -44
  24. package/src/api/action/addExternalModule.js +0 -25
  25. package/src/api/action/changeAvatar.js +0 -137
  26. package/src/api/action/changeBio.js +0 -75
  27. package/src/api/action/enableAutoSaveAppState.js +0 -73
  28. package/src/api/action/getCurrentUserID.js +0 -7
  29. package/src/api/action/handleFriendRequest.js +0 -57
  30. package/src/api/action/logout.js +0 -76
  31. package/src/api/action/refreshFb_dtsg.js +0 -48
  32. package/src/api/action/setPostReaction.js +0 -106
  33. package/src/api/action/unfriend.js +0 -54
  34. package/src/api/http/httpGet.js +0 -46
  35. package/src/api/http/httpPost.js +0 -52
  36. package/src/api/http/postFormData.js +0 -47
  37. package/src/api/messaging/addUserToGroup.js +0 -68
  38. package/src/api/messaging/changeAdminStatus.js +0 -126
  39. package/src/api/messaging/changeArchivedStatus.js +0 -55
  40. package/src/api/messaging/changeBlockedStatus.js +0 -48
  41. package/src/api/messaging/changeGroupImage.js +0 -91
  42. package/src/api/messaging/changeNickname.js +0 -70
  43. package/src/api/messaging/changeThreadColor.js +0 -79
  44. package/src/api/messaging/changeThreadEmoji.js +0 -111
  45. package/src/api/messaging/createNewGroup.js +0 -88
  46. package/src/api/messaging/createPoll.js +0 -46
  47. package/src/api/messaging/createThemeAI.js +0 -98
  48. package/src/api/messaging/deleteMessage.js +0 -136
  49. package/src/api/messaging/deleteThread.js +0 -56
  50. package/src/api/messaging/editMessage.js +0 -68
  51. package/src/api/messaging/forwardAttachment.js +0 -57
  52. package/src/api/messaging/getEmojiUrl.js +0 -29
  53. package/src/api/messaging/getFriendsList.js +0 -82
  54. package/src/api/messaging/getMessage.js +0 -829
  55. package/src/api/messaging/getThemePictures.js +0 -62
  56. package/src/api/messaging/handleMessageRequest.js +0 -65
  57. package/src/api/messaging/markAsDelivered.js +0 -57
  58. package/src/api/messaging/markAsRead.js +0 -88
  59. package/src/api/messaging/markAsReadAll.js +0 -49
  60. package/src/api/messaging/markAsSeen.js +0 -61
  61. package/src/api/messaging/muteThread.js +0 -50
  62. package/src/api/messaging/removeUserFromGroup.js +0 -62
  63. package/src/api/messaging/resolvePhotoUrl.js +0 -43
  64. package/src/api/messaging/scheduler.js +0 -264
  65. package/src/api/messaging/searchForThread.js +0 -53
  66. package/src/api/messaging/sendMessage.js +0 -270
  67. package/src/api/messaging/sendTypingIndicator.js +0 -74
  68. package/src/api/messaging/setMessageReaction.js +0 -90
  69. package/src/api/messaging/setTitle.js +0 -124
  70. package/src/api/messaging/shareContact.js +0 -49
  71. package/src/api/messaging/threadColors.js +0 -128
  72. package/src/api/messaging/unsendMessage.js +0 -81
  73. package/src/api/messaging/uploadAttachment.js +0 -492
  74. package/src/api/socket/core/connectMqtt.js +0 -258
  75. package/src/api/socket/core/emitAuth.js +0 -103
  76. package/src/api/socket/core/getSeqID.js +0 -320
  77. package/src/api/socket/core/getTaskResponseData.js +0 -25
  78. package/src/api/socket/core/parseDelta.js +0 -377
  79. package/src/api/socket/detail/buildStream.js +0 -215
  80. package/src/api/socket/detail/constants.js +0 -28
  81. package/src/api/socket/listenMqtt.js +0 -377
  82. package/src/api/socket/middleware/index.js +0 -216
  83. package/src/api/threads/getThreadHistory.js +0 -664
  84. package/src/api/threads/getThreadInfo.js +0 -296
  85. package/src/api/threads/getThreadList.js +0 -293
  86. package/src/api/threads/getThreadPictures.js +0 -78
  87. package/src/api/users/getUserID.js +0 -65
  88. package/src/api/users/getUserInfo.js +0 -402
  89. package/src/api/users/getUserInfoV2.js +0 -134
  90. package/src/core/sendReqMqtt.js +0 -96
  91. package/src/database/helpers.js +0 -53
  92. package/src/database/models/index.js +0 -88
  93. package/src/database/models/thread.js +0 -50
  94. package/src/database/models/user.js +0 -46
  95. package/src/database/threadData.js +0 -94
  96. package/src/database/userData.js +0 -98
  97. package/src/remote/remoteClient.js +0 -123
  98. package/src/utils/broadcast.js +0 -51
  99. package/src/utils/client.js +0 -10
  100. package/src/utils/constants.js +0 -23
  101. package/src/utils/cookies.js +0 -68
  102. package/src/utils/format/attachment.js +0 -357
  103. package/src/utils/format/cookie.js +0 -9
  104. package/src/utils/format/date.js +0 -50
  105. package/src/utils/format/decode.js +0 -44
  106. package/src/utils/format/delta.js +0 -194
  107. package/src/utils/format/ids.js +0 -64
  108. package/src/utils/format/index.js +0 -64
  109. package/src/utils/format/message.js +0 -88
  110. package/src/utils/format/presence.js +0 -132
  111. package/src/utils/format/readTyp.js +0 -44
  112. package/src/utils/format/thread.js +0 -42
  113. package/src/utils/format/utils.js +0 -141
  114. package/src/utils/headers.js +0 -115
  115. package/src/utils/loginParser/autoLogin.js +0 -125
  116. package/src/utils/loginParser/helpers.js +0 -43
  117. package/src/utils/loginParser/index.js +0 -10
  118. package/src/utils/loginParser/parseAndCheckLogin.js +0 -220
  119. package/src/utils/loginParser/textUtils.js +0 -28
  120. package/src/utils/request/client.js +0 -26
  121. package/src/utils/request/config.js +0 -23
  122. package/src/utils/request/defaults.js +0 -46
  123. package/src/utils/request/helpers.js +0 -46
  124. package/src/utils/request/index.js +0 -17
  125. package/src/utils/request/methods.js +0 -163
  126. package/src/utils/request/proxy.js +0 -21
  127. package/src/utils/request/retry.js +0 -77
  128. package/src/utils/request/sanitize.js +0 -49
@@ -1,296 +0,0 @@
1
- 'use strict';
2
-
3
- const fs = require('fs');
4
- const path = require('path');
5
- const { parseAndCheckLogin } = require('../../utils/client');
6
- const log = require('../../../func/logAdapter');
7
-
8
- function formatEventReminders(reminder) {
9
- return {
10
- reminderID: reminder.id,
11
- eventCreatorID: reminder.lightweight_event_creator.id,
12
- time: reminder.time,
13
- eventType: reminder.lightweight_event_type.toLowerCase(),
14
- locationName: reminder.location_name,
15
- locationCoordinates: reminder.location_coordinates,
16
- locationPage: reminder.location_page,
17
- eventStatus: reminder.lightweight_event_status.toLowerCase(),
18
- note: reminder.note,
19
- repeatMode: reminder.repeat_mode.toLowerCase(),
20
- eventTitle: reminder.event_title,
21
- triggerMessage: reminder.trigger_message,
22
- secondsToNotifyBefore: reminder.seconds_to_notify_before,
23
- allowsRsvp: reminder.allows_rsvp,
24
- relatedEvent: reminder.related_event,
25
- members: reminder.event_reminder_members.edges.map((member) => ({
26
- memberID: member.node.id,
27
- state: member.guest_list_state.toLowerCase(),
28
- })),
29
- };
30
- }
31
-
32
- function formatThreadGraphQLResponse(data) {
33
- if (data.errors) {
34
- const details = data.errors.map(e => e.message || e).join(", ");
35
- const error = new Error(`GraphQL error in getThreadInfo: ${details}`);
36
- throw error;
37
- }
38
-
39
- const messageThread = data.message_thread;
40
- if (!messageThread) {
41
- throw new Error("No message_thread in GraphQL response");
42
- }
43
-
44
- const threadID =
45
- messageThread.thread_key.thread_fbid ||
46
- messageThread.thread_key.other_user_id;
47
-
48
- const lastM = messageThread.last_message;
49
- const snippetID =
50
- lastM?.nodes?.[0]?.message_sender?.messaging_actor?.id || null;
51
- const snippetText = lastM?.nodes?.[0]?.snippet || null;
52
- const lastReadTimestamp =
53
- messageThread.last_read_receipt?.nodes?.[0]?.timestamp_precise || null;
54
-
55
- return {
56
- threadID,
57
- threadName: messageThread.name,
58
- participantIDs: messageThread.all_participants.edges.map(
59
- d => d.node.messaging_actor.id
60
- ),
61
- userInfo: messageThread.all_participants.edges.map(d => ({
62
- id: d.node.messaging_actor.id,
63
- name: d.node.messaging_actor.name,
64
- firstName: d.node.messaging_actor.short_name,
65
- vanity: d.node.messaging_actor.username,
66
- url: d.node.messaging_actor.url,
67
- thumbSrc: d.node.messaging_actor.big_image_src.uri,
68
- profileUrl: d.node.messaging_actor.big_image_src.uri,
69
- gender: d.node.messaging_actor.gender,
70
- type: d.node.messaging_actor.__typename,
71
- isFriend: d.node.messaging_actor.is_viewer_friend,
72
- isBirthday: !!d.node.messaging_actor.is_birthday,
73
- })),
74
- unreadCount: messageThread.unread_count,
75
- messageCount: messageThread.messages_count,
76
- timestamp: messageThread.updated_time_precise,
77
- muteUntil: messageThread.mute_until,
78
- isGroup: messageThread.thread_type === "GROUP",
79
- isSubscribed: messageThread.is_viewer_subscribed,
80
- isArchived: messageThread.has_viewer_archived,
81
- folder: messageThread.folder,
82
- cannotReplyReason: messageThread.cannot_reply_reason,
83
- eventReminders: messageThread.event_reminders
84
- ? messageThread.event_reminders.nodes.map(formatEventReminders)
85
- : null,
86
- emoji: messageThread.customization_info?.emoji || null,
87
- color: messageThread.customization_info?.outgoing_bubble_color
88
- ? messageThread.customization_info.outgoing_bubble_color.slice(2)
89
- : null,
90
- threadTheme: messageThread.thread_theme,
91
- nicknames:
92
- messageThread.customization_info?.participant_customizations?.reduce(
93
- (res, val) => {
94
- if (val.nickname) res[val.participant_id] = val.nickname;
95
- return res;
96
- },
97
- {}
98
- ) || {},
99
- adminIDs: messageThread.thread_admins,
100
- approvalMode: Boolean(messageThread.approval_mode),
101
- approvalQueue:
102
- messageThread.group_approval_queue?.nodes?.map(a => ({
103
- inviterID: a.inviter.id,
104
- requesterID: a.requester.id,
105
- timestamp: a.request_timestamp,
106
- request_source: a.request_source,
107
- })) || [],
108
- reactionsMuteMode: messageThread.reactions_mute_mode?.toLowerCase(),
109
- mentionsMuteMode: messageThread.mentions_mute_mode?.toLowerCase(),
110
- isPinProtected: messageThread.is_pin_protected,
111
- relatedPageThread: messageThread.related_page_thread,
112
- name: messageThread.name,
113
- snippet: snippetText,
114
- snippetSender: snippetID,
115
- snippetAttachments: [],
116
- serverTimestamp: messageThread.updated_time_precise,
117
- imageSrc: messageThread.image?.uri || null,
118
- isCanonicalUser: messageThread.is_canonical_neo_user,
119
- isCanonical: messageThread.thread_type !== "GROUP",
120
- recipientsLoadable: true,
121
- hasEmailParticipant: false,
122
- readOnly: false,
123
- canReply: messageThread.cannot_reply_reason == null,
124
- lastMessageTimestamp:
125
- messageThread.last_message?.timestamp_precise || null,
126
- lastMessageType: "message",
127
- lastReadTimestamp,
128
- threadType: messageThread.thread_type === "GROUP" ? 2 : 1,
129
- inviteLink: {
130
- enable: messageThread.joinable_mode?.mode === 1,
131
- link: messageThread.joinable_mode?.link || null,
132
- },
133
- };
134
- }
135
-
136
- module.exports = function (defaultFuncs, api, ctx) {
137
- const dbFiles = fs.readdirSync(path.join(__dirname, "../../database"))
138
- .filter(f => path.extname(f) === ".js")
139
- .reduce((acc, file) => {
140
- const mod = require(path.join(__dirname, "../../database", file));
141
- acc[path.basename(file, ".js")] = typeof mod === "function" ? mod(api) : mod;
142
- return acc;
143
- }, {});
144
-
145
- const { threadData } = dbFiles;
146
- const { create, get, update } = threadData || {};
147
- const FRESH_MS = 10 * 60 * 1000;
148
- return function getThreadInfo(threadID, callback) {
149
- let resolveFunc;
150
- let rejectFunc;
151
-
152
- const returnPromise = new Promise((resolve, reject) => {
153
- resolveFunc = resolve;
154
- rejectFunc = reject;
155
- });
156
-
157
- if (typeof callback !== "function") {
158
- callback = (err, data) => {
159
- if (err) {
160
- return rejectFunc(err);
161
- }
162
- return resolveFunc(data);
163
- };
164
- }
165
-
166
- const threadIDs = Array.isArray(threadID) ? threadID.map(String) : [String(threadID)];
167
-
168
- const now = Date.now();
169
-
170
- const loadFromDb = async ids => {
171
- if (!threadData || typeof get !== "function") return { fresh: {}, stale: ids };
172
- const fresh = {};
173
- const stale = [];
174
- const rows = await Promise.all(ids.map(id => get(id).catch(() => null)));
175
- for (let i = 0; i < ids.length; i++) {
176
- const id = ids[i];
177
- const row = rows[i];
178
- if (row && row.data) {
179
- const updatedAt = row.updatedAt ? new Date(row.updatedAt).getTime() : 0;
180
- if (updatedAt && now - updatedAt <= FRESH_MS) {
181
- fresh[id] = row.data;
182
- } else {
183
- stale.push(id);
184
- }
185
- } else {
186
- stale.push(id);
187
- }
188
- }
189
- return { fresh, stale };
190
- };
191
-
192
- const fetchFromGraphQL = async ids => {
193
- if (!ids.length) return {};
194
- const queries = {};
195
- ids.forEach((t, i) => {
196
- queries["o" + i] = {
197
- doc_id: "3449967031715030",
198
- query_params: {
199
- id: t,
200
- message_limit: 0,
201
- load_messages: false,
202
- load_read_receipts: false,
203
- before: null
204
- }
205
- };
206
- });
207
-
208
- const form = {
209
- queries: JSON.stringify(queries),
210
- batch_name: "MessengerGraphQLThreadFetcher"
211
- };
212
-
213
- const resData = await defaultFuncs
214
- .post(
215
- "https://www.facebook.com/api/graphqlbatch/",
216
- ctx.jar,
217
- form
218
- )
219
- .then(parseAndCheckLogin(ctx, defaultFuncs));
220
-
221
- if (resData.error) {
222
- throw resData;
223
- }
224
-
225
- const out = {};
226
- for (let i = resData.length - 2; i >= 0; i--) {
227
- const res = resData[i];
228
- const oKey = Object.keys(res)[0];
229
- const responseData = res[oKey];
230
- try {
231
- const info = formatThreadGraphQLResponse(responseData.data);
232
- if (info && info.threadID) {
233
- out[info.threadID] = info;
234
- }
235
- } catch (e) {
236
- // Skip malformed entries but continue processing others
237
- log.error("getThreadInfoGraphQL", e && e.message ? e.message : String(e));
238
- }
239
- }
240
- return out;
241
- };
242
-
243
- (async () => {
244
- try {
245
- const { fresh, stale } = await loadFromDb(threadIDs);
246
- let fetched = {};
247
-
248
- if (stale.length) {
249
- fetched = await fetchFromGraphQL(stale);
250
-
251
- // Persist fetched data back to DB
252
- if (threadData && (typeof create === "function" || typeof update === "function")) {
253
- const tasks = [];
254
- for (const id of stale) {
255
- const info = fetched[id];
256
- if (!info) continue;
257
- const payload = { data: info };
258
- if (typeof update === "function") {
259
- tasks.push(update(id, payload).catch(() => null));
260
- } else if (typeof create === "function") {
261
- tasks.push(create(id, payload).catch(() => null));
262
- }
263
- }
264
- if (tasks.length) {
265
- try {
266
- await Promise.all(tasks);
267
- } catch {
268
- // Swallow DB errors – not critical for API behavior
269
- }
270
- }
271
- }
272
- }
273
-
274
- const resultMap = {};
275
- for (const id of threadIDs) {
276
- resultMap[id] = fresh[id] || fetched[id] || null;
277
- }
278
-
279
- const result = Array.isArray(threadID)
280
- ? resultMap
281
- : resultMap[threadIDs[0]] || null;
282
-
283
- return callback(null, result);
284
- } catch (err) {
285
- // Horizon-style anti-get-info message to hint possible spam/limit
286
- log.error(
287
- "getThreadInfoGraphQL",
288
- "Lỗi: getThreadInfoGraphQL Có Thể Do Bạn Spam Quá Nhiều, Hãy Thử Lại !"
289
- );
290
- return callback(err);
291
- }
292
- })();
293
-
294
- return returnPromise;
295
- };
296
- };
@@ -1,293 +0,0 @@
1
- "use strict";
2
-
3
- const log = require("../../../func/logAdapter");
4
- const { parseAndCheckLogin } = require("../../utils/client");
5
- const { formatID, getType } = require("../../utils/format");
6
- function createProfileUrl(url, username, id) {
7
- if (url) return url;
8
- return (
9
- "https://www.facebook.com/" + (username || formatID(id.toString()))
10
- );
11
- }
12
-
13
- function formatParticipants(participants) {
14
- return participants.edges.map(p => {
15
- p = p.node.messaging_actor;
16
- switch (p["__typename"]) {
17
- case "User":
18
- return {
19
- accountType: p["__typename"],
20
- userID: formatID(p.id.toString()), // do we need .toString()? when it is not a string?
21
- name: p.name,
22
- shortName: p.short_name,
23
- gender: p.gender,
24
- url: p.url, // how about making it profileURL
25
- profilePicture: p.big_image_src.uri,
26
- username: p.username || null,
27
- // TODO: maybe better names for these?
28
- isViewerFriend: p.is_viewer_friend, // true/false
29
- isMessengerUser: p.is_messenger_user, // true/false
30
- isVerified: p.is_verified, // true/false
31
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer, // true/false
32
- isViewerCoworker: p.is_viewer_coworker, // true/false
33
- isEmployee: p.is_employee // null? when it is something other? can someone check?
34
- };
35
- case "Page":
36
- return {
37
- accountType: p["__typename"],
38
- userID: formatID(p.id.toString()), // or maybe... pageID?
39
- name: p.name,
40
- url: p.url,
41
- profilePicture: p.big_image_src.uri,
42
- username: p.username || null,
43
- // uhm... better names maybe?
44
- acceptsMessengerUserFeedback: p.accepts_messenger_user_feedback, // true/false
45
- isMessengerUser: p.is_messenger_user, // true/false
46
- isVerified: p.is_verified, // true/false
47
- isMessengerPlatformBot: p.is_messenger_platform_bot, // true/false
48
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer // true/false
49
- };
50
- case "ReducedMessagingActor":
51
- case "UnavailableMessagingActor":
52
- return {
53
- accountType: p["__typename"],
54
- userID: formatID(p.id.toString()),
55
- name: p.name,
56
- url: createProfileUrl(p.url, p.username, p.id), // in this case p.url is null all the time
57
- profilePicture: p.big_image_src.uri, // in this case it is default facebook photo, we could determine gender using it
58
- username: p.username || null, // maybe we could use it to generate profile URL?
59
- isMessageBlockedByViewer: p.is_message_blocked_by_viewer // true/false
60
- };
61
- default:
62
- log.warn(
63
- "getThreadList",
64
- "Found participant with unsupported typename. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues\n" +
65
- JSON.stringify(p, null, 2)
66
- );
67
- return {
68
- accountType: p["__typename"],
69
- userID: formatID(p.id.toString()),
70
- name: p.name || `[unknown ${p["__typename"]}]` // probably it will always be something... but fallback to [unknown], just in case
71
- };
72
- }
73
- });
74
- }
75
-
76
- // "FF8C0077" -> "8C0077"
77
- function formatColor(color) {
78
- if (color && color.match(/^(?:[0-9a-fA-F]{8})$/g)) return color.slice(2);
79
- return color;
80
- }
81
-
82
- function getThreadName(t) {
83
- if (t.name || t.thread_key.thread_fbid) return t.name;
84
-
85
- for (let po of t.all_participants.edges) {
86
- let p = po.node;
87
- if (p.messaging_actor.id === t.thread_key.other_user_id)
88
- return p.messaging_actor.name;
89
- }
90
- }
91
-
92
- function mapNicknames(customizationInfo) {
93
- return customizationInfo && customizationInfo.participant_customizations
94
- ? customizationInfo.participant_customizations.map(u => {
95
- return {
96
- userID: u.participant_id,
97
- nickname: u.nickname
98
- };
99
- })
100
- : [];
101
- }
102
-
103
- function formatThreadList(data) {
104
- return data.map(t => {
105
- let lastMessageNode =
106
- t.last_message && t.last_message.nodes && t.last_message.nodes.length > 0
107
- ? t.last_message.nodes[0]
108
- : null;
109
- return {
110
- threadID: t.thread_key
111
- ? formatID(t.thread_key.thread_fbid || t.thread_key.other_user_id)
112
- : null, // shall never be null
113
- name: getThreadName(t),
114
- unreadCount: t.unread_count,
115
- messageCount: t.messages_count,
116
- imageSrc: t.image ? t.image.uri : null,
117
- emoji: t.customization_info ? t.customization_info.emoji : null,
118
- color: formatColor(
119
- t.customization_info ? t.customization_info.outgoing_bubble_color : null
120
- ),
121
- threadTheme: t.thread_theme,
122
- nicknames: mapNicknames(t.customization_info),
123
- muteUntil: t.mute_until,
124
- participants: formatParticipants(t.all_participants),
125
- adminIDs: t.thread_admins.map(a => a.id),
126
- folder: t.folder,
127
- isGroup: t.thread_type === "GROUP",
128
- customizationEnabled: t.customization_enabled, // false for ONE_TO_ONE with Page or ReducedMessagingActor
129
- participantAddMode: t.participant_add_mode_as_string, // "ADD" if "GROUP" and null if "ONE_TO_ONE"
130
- montageThread: t.montage_thread
131
- ? Buffer.from(t.montage_thread.id, "base64").toString()
132
- : null, // base64 encoded string "message_thread:0000000000000000"
133
- reactionsMuteMode: t.reactions_mute_mode,
134
- mentionsMuteMode: t.mentions_mute_mode,
135
- isArchived: t.has_viewer_archived,
136
- isSubscribed: t.is_viewer_subscribed,
137
- timestamp: t.updated_time_precise, // in miliseconds
138
- snippet: lastMessageNode ? lastMessageNode.snippet : null,
139
- snippetAttachments: lastMessageNode
140
- ? lastMessageNode.extensible_attachment
141
- : null, // TODO: not sure if it works
142
- snippetSender: lastMessageNode
143
- ? formatID(
144
- (lastMessageNode.message_sender.messaging_actor.id || "").toString()
145
- )
146
- : null,
147
- lastMessageTimestamp: lastMessageNode
148
- ? lastMessageNode.timestamp_precise
149
- : null, // timestamp in miliseconds
150
- lastReadTimestamp:
151
- t.last_read_receipt && t.last_read_receipt.nodes.length > 0
152
- ? t.last_read_receipt.nodes[0]
153
- ? t.last_read_receipt.nodes[0].timestamp_precise
154
- : null
155
- : null,
156
- cannotReplyReason: t.cannot_reply_reason,
157
- approvalMode: Boolean(t.approval_mode),
158
- participantIDs: formatParticipants(t.all_participants).map(
159
- participant => participant.userID
160
- ),
161
- threadType: t.thread_type === "GROUP" ? 2 : 1, // "GROUP" or "ONE_TO_ONE"
162
- inviteLink: {
163
- enable: t.joinable_mode ? t.joinable_mode.mode == 1 : false,
164
- link: t.joinable_mode ? t.joinable_mode.link : null
165
- }
166
- };
167
- });
168
- }
169
-
170
- module.exports = function(defaultFuncs, api, ctx) {
171
- return function getThreadList(limit, timestamp, tags, callback) {
172
- if (
173
- !callback &&
174
- (getType(tags) === "Function" ||
175
- getType(tags) === "AsyncFunction")
176
- ) {
177
- callback = tags;
178
- tags = [""];
179
- }
180
- if (
181
- getType(limit) !== "Number" ||
182
- !Number.isInteger(limit) ||
183
- limit <= 0
184
- )
185
- throw { error: "getThreadList: limit must be a positive integer" };
186
- if (
187
- getType(timestamp) !== "Null" &&
188
- (getType(timestamp) !== "Number" || !Number.isInteger(timestamp))
189
- )
190
- throw { error: "getThreadList: timestamp must be an integer or null" };
191
- if (getType(tags) === "String") tags = [tags];
192
- if (getType(tags) !== "Array")
193
- throw { error: "getThreadList: tags must be an array" };
194
- var resolveFunc = function() {};
195
- var rejectFunc = function() {};
196
- var returnPromise = new Promise(function(resolve, reject) {
197
- resolveFunc = resolve;
198
- rejectFunc = reject;
199
- });
200
- if (
201
- getType(callback) !== "Function" &&
202
- getType(callback) !== "AsyncFunction"
203
- ) {
204
- callback = function(err, data) {
205
- if (err) return rejectFunc(err);
206
- resolveFunc(data);
207
- };
208
- }
209
- const form = {
210
- av: ctx.userID,
211
- queries: JSON.stringify({
212
- o0: {
213
- doc_id: "3336396659757871",
214
- query_params: {
215
- limit: limit + (timestamp ? 1 : 0),
216
- before: timestamp,
217
- tags: tags,
218
- includeDeliveryReceipts: true,
219
- includeSeqID: false
220
- }
221
- }
222
- }),
223
- batch_name: "MessengerGraphQLThreadlistFetcher"
224
- };
225
- defaultFuncs
226
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
227
- .then(parseAndCheckLogin(ctx, defaultFuncs))
228
- .then(resData => {
229
- // Validate resData is an array and has elements
230
- if (!resData || !Array.isArray(resData) || resData.length === 0) {
231
- throw {
232
- error: "getThreadList: Invalid response data - resData is not a valid array",
233
- res: resData
234
- };
235
- }
236
-
237
- // Validate last element exists and has required properties
238
- const lastElement = resData[resData.length - 1];
239
- if (!lastElement || typeof lastElement !== "object") {
240
- throw {
241
- error: "getThreadList: Invalid response data - last element is missing or invalid",
242
- res: resData
243
- };
244
- }
245
-
246
- if (lastElement.error_results > 0) {
247
- // Check if first element and o0 exist before accessing errors
248
- if (resData[0] && resData[0].o0 && resData[0].o0.errors) {
249
- throw resData[0].o0.errors;
250
- } else {
251
- throw {
252
- error: "getThreadList: Error results > 0 but error details not available",
253
- res: resData
254
- };
255
- }
256
- }
257
-
258
- if (lastElement.successful_results === 0) {
259
- throw {
260
- error: "getThreadList: there was no successful_results",
261
- res: resData
262
- };
263
- }
264
-
265
- // Validate first element and nested data structure
266
- if (!resData[0] || !resData[0].o0 || !resData[0].o0.data ||
267
- !resData[0].o0.data.viewer || !resData[0].o0.data.viewer.message_threads ||
268
- !Array.isArray(resData[0].o0.data.viewer.message_threads.nodes)) {
269
- throw {
270
- error: "getThreadList: Invalid response data structure - missing required fields",
271
- res: resData
272
- };
273
- }
274
-
275
- if (timestamp) {
276
- const nodes = resData[0].o0.data.viewer.message_threads.nodes;
277
- if (Array.isArray(nodes) && nodes.length > 0) {
278
- nodes.shift();
279
- }
280
- }
281
-
282
- callback(
283
- null,
284
- formatThreadList(resData[0].o0.data.viewer.message_threads.nodes)
285
- );
286
- })
287
- .catch(err => {
288
- log.error("getThreadList", err);
289
- return callback(err);
290
- });
291
- return returnPromise;
292
- };
293
- };
@@ -1,78 +0,0 @@
1
- "use strict";
2
-
3
- const log = require("../../../func/logAdapter");
4
- const { parseAndCheckLogin } = require("../../utils/client");
5
- module.exports = function(defaultFuncs, api, ctx) {
6
- return function getThreadPictures(threadID, offset, limit, callback) {
7
- let resolveFunc = function() {};
8
- let rejectFunc = function() {};
9
- const returnPromise = new Promise(function(resolve, reject) {
10
- resolveFunc = resolve;
11
- rejectFunc = reject;
12
- });
13
-
14
- if (!callback) {
15
- callback = function(err, friendList) {
16
- if (err) {
17
- return rejectFunc(err);
18
- }
19
- resolveFunc(friendList);
20
- };
21
- }
22
-
23
- let form = {
24
- thread_id: threadID,
25
- offset: offset,
26
- limit: limit
27
- };
28
-
29
- defaultFuncs
30
- .post(
31
- "https://www.facebook.com/ajax/messaging/attachments/sharedphotos.php",
32
- ctx.jar,
33
- form
34
- )
35
- .then(parseAndCheckLogin(ctx, defaultFuncs))
36
- .then(function(resData) {
37
- if (resData.error) {
38
- throw resData;
39
- }
40
- return Promise.all(
41
- resData.payload.imagesData.map(function(image) {
42
- form = {
43
- thread_id: threadID,
44
- image_id: image.fbid
45
- };
46
- return defaultFuncs
47
- .post(
48
- "https://www.facebook.com/ajax/messaging/attachments/sharedphotos.php",
49
- ctx.jar,
50
- form
51
- )
52
- .then(parseAndCheckLogin(ctx, defaultFuncs))
53
- .then(function(resData) {
54
- if (resData.error) {
55
- throw resData;
56
- }
57
- // the response is pretty messy
58
- const queryThreadID =
59
- resData.jsmods.require[0][3][1].query_metadata.query_path[0]
60
- .message_thread;
61
- const imageData =
62
- resData.jsmods.require[0][3][1].query_results[queryThreadID]
63
- .message_images.edges[0].node.image2;
64
- return imageData;
65
- });
66
- })
67
- );
68
- })
69
- .then(function(resData) {
70
- callback(null, resData);
71
- })
72
- .catch(function(err) {
73
- log.error("Error in getThreadPictures", err);
74
- callback(err);
75
- });
76
- return returnPromise;
77
- };
78
- };