fca-neokex-fix 1.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 (114) hide show
  1. package/CHANGELOG.md +220 -0
  2. package/LICENSE +26 -0
  3. package/README.md +346 -0
  4. package/THEME_FEATURES.md +137 -0
  5. package/examples/README.md +131 -0
  6. package/examples/apply-ai-theme.js +127 -0
  7. package/examples/check-current-theme.js +74 -0
  8. package/examples/simple-bot.js +114 -0
  9. package/examples/test-bot.js +752 -0
  10. package/examples/test-logging.js +85 -0
  11. package/examples/theme-usage-example.js +53 -0
  12. package/index.js +2 -0
  13. package/package.json +105 -0
  14. package/src/apis/addExternalModule.js +24 -0
  15. package/src/apis/addUserToGroup.js +108 -0
  16. package/src/apis/changeAdminStatus.js +148 -0
  17. package/src/apis/changeArchivedStatus.js +61 -0
  18. package/src/apis/changeAvatar.js +103 -0
  19. package/src/apis/changeBio.js +69 -0
  20. package/src/apis/changeBlockedStatus.js +54 -0
  21. package/src/apis/changeGroupImage.js +136 -0
  22. package/src/apis/changeThreadColor.js +116 -0
  23. package/src/apis/comment.js +207 -0
  24. package/src/apis/createAITheme.js +129 -0
  25. package/src/apis/createNewGroup.js +79 -0
  26. package/src/apis/createPoll.js +73 -0
  27. package/src/apis/deleteMessage.js +44 -0
  28. package/src/apis/deleteThread.js +52 -0
  29. package/src/apis/editMessage.js +70 -0
  30. package/src/apis/emoji.js +124 -0
  31. package/src/apis/fetchThemeData.js +65 -0
  32. package/src/apis/follow.js +81 -0
  33. package/src/apis/forwardMessage.js +52 -0
  34. package/src/apis/friend.js +243 -0
  35. package/src/apis/gcmember.js +122 -0
  36. package/src/apis/gcname.js +123 -0
  37. package/src/apis/gcrule.js +119 -0
  38. package/src/apis/getAccess.js +111 -0
  39. package/src/apis/getBotInfo.js +88 -0
  40. package/src/apis/getBotInitialData.js +43 -0
  41. package/src/apis/getFriendsList.js +79 -0
  42. package/src/apis/getMessage.js +423 -0
  43. package/src/apis/getTheme.js +104 -0
  44. package/src/apis/getThemeInfo.js +96 -0
  45. package/src/apis/getThreadHistory.js +239 -0
  46. package/src/apis/getThreadInfo.js +257 -0
  47. package/src/apis/getThreadList.js +222 -0
  48. package/src/apis/getThreadPictures.js +58 -0
  49. package/src/apis/getUserID.js +83 -0
  50. package/src/apis/getUserInfo.js +495 -0
  51. package/src/apis/getUserInfoV2.js +146 -0
  52. package/src/apis/handleMessageRequest.js +50 -0
  53. package/src/apis/httpGet.js +63 -0
  54. package/src/apis/httpPost.js +89 -0
  55. package/src/apis/httpPostFormData.js +69 -0
  56. package/src/apis/listenMqtt.js +796 -0
  57. package/src/apis/listenSpeed.js +170 -0
  58. package/src/apis/logout.js +63 -0
  59. package/src/apis/markAsDelivered.js +47 -0
  60. package/src/apis/markAsRead.js +95 -0
  61. package/src/apis/markAsReadAll.js +41 -0
  62. package/src/apis/markAsSeen.js +70 -0
  63. package/src/apis/mqttDeltaValue.js +330 -0
  64. package/src/apis/muteThread.js +45 -0
  65. package/src/apis/nickname.js +132 -0
  66. package/src/apis/notes.js +163 -0
  67. package/src/apis/pinMessage.js +141 -0
  68. package/src/apis/produceMetaTheme.js +180 -0
  69. package/src/apis/realtime.js +161 -0
  70. package/src/apis/removeUserFromGroup.js +117 -0
  71. package/src/apis/resolvePhotoUrl.js +58 -0
  72. package/src/apis/searchForThread.js +154 -0
  73. package/src/apis/sendMessage.js +281 -0
  74. package/src/apis/sendMessageMqtt.js +188 -0
  75. package/src/apis/sendTypingIndicator.js +41 -0
  76. package/src/apis/setMessageReaction.js +27 -0
  77. package/src/apis/setMessageReactionMqtt.js +61 -0
  78. package/src/apis/setThreadTheme.js +260 -0
  79. package/src/apis/setThreadThemeMqtt.js +94 -0
  80. package/src/apis/share.js +107 -0
  81. package/src/apis/shareContact.js +66 -0
  82. package/src/apis/stickers.js +257 -0
  83. package/src/apis/story.js +181 -0
  84. package/src/apis/theme.js +233 -0
  85. package/src/apis/unfriend.js +47 -0
  86. package/src/apis/unsendMessage.js +17 -0
  87. package/src/database/appStateBackup.js +189 -0
  88. package/src/database/models/index.js +56 -0
  89. package/src/database/models/thread.js +31 -0
  90. package/src/database/models/user.js +32 -0
  91. package/src/database/threadData.js +101 -0
  92. package/src/database/userData.js +90 -0
  93. package/src/engine/client.js +91 -0
  94. package/src/engine/models/buildAPI.js +109 -0
  95. package/src/engine/models/loginHelper.js +326 -0
  96. package/src/engine/models/setOptions.js +53 -0
  97. package/src/utils/auth-helpers.js +149 -0
  98. package/src/utils/autoReLogin.js +169 -0
  99. package/src/utils/axios.js +290 -0
  100. package/src/utils/clients.js +270 -0
  101. package/src/utils/constants.js +396 -0
  102. package/src/utils/formatters/data/formatAttachment.js +370 -0
  103. package/src/utils/formatters/data/formatDelta.js +153 -0
  104. package/src/utils/formatters/index.js +159 -0
  105. package/src/utils/formatters/value/formatCookie.js +91 -0
  106. package/src/utils/formatters/value/formatDate.js +36 -0
  107. package/src/utils/formatters/value/formatID.js +16 -0
  108. package/src/utils/formatters.js +1067 -0
  109. package/src/utils/headers.js +199 -0
  110. package/src/utils/index.js +151 -0
  111. package/src/utils/monitoring.js +358 -0
  112. package/src/utils/rateLimiter.js +380 -0
  113. package/src/utils/tokenRefresh.js +311 -0
  114. package/src/utils/user-agents.js +238 -0
@@ -0,0 +1,239 @@
1
+ "use strict";
2
+
3
+ const utils = require('../utils');
4
+
5
+
6
+ /**
7
+ * Formats a single attachment object from a GraphQL response.
8
+ * @param {Object} attachment The raw attachment object.
9
+ * @returns {Object} A formatted attachment object.
10
+ */
11
+ function formatAttachmentsGraphQLResponse(attachment) {
12
+ switch (attachment.__typename) {
13
+ case "MessageImage":
14
+ return {
15
+ type: "photo",
16
+ ID: attachment.legacy_attachment_id,
17
+ filename: attachment.filename,
18
+ thumbnailUrl: attachment.thumbnail.uri,
19
+ previewUrl: attachment.preview.uri,
20
+ previewWidth: attachment.preview.width,
21
+ previewHeight: attachment.preview.height,
22
+ largePreviewUrl: attachment.large_preview.uri,
23
+ largePreviewHeight: attachment.large_preview.height,
24
+ largePreviewWidth: attachment.large_preview.width,
25
+ url: attachment.large_preview.uri,
26
+ width: attachment.original_dimensions.x,
27
+ height: attachment.original_dimensions.y,
28
+ name: attachment.filename
29
+ };
30
+ case "MessageAnimatedImage":
31
+ return {
32
+ type: "animated_image",
33
+ ID: attachment.legacy_attachment_id,
34
+ filename: attachment.filename,
35
+ previewUrl: attachment.preview_image.uri,
36
+ previewWidth: attachment.preview_image.width,
37
+ previewHeight: attachment.preview_image.height,
38
+ url: attachment.animated_image.uri,
39
+ width: attachment.animated_image.width,
40
+ height: attachment.animated_image.height,
41
+ name: attachment.filename,
42
+ facebookUrl: attachment.animated_image.uri,
43
+ };
44
+ case "MessageVideo":
45
+ return {
46
+ type: "video",
47
+ ID: attachment.legacy_attachment_id,
48
+ filename: attachment.filename,
49
+ duration: attachment.playable_duration_in_ms,
50
+ thumbnailUrl: attachment.large_image.uri,
51
+ previewUrl: attachment.large_image.uri,
52
+ previewWidth: attachment.large_image.width,
53
+ previewHeight: attachment.large_image.height,
54
+ url: attachment.playable_url,
55
+ width: attachment.original_dimensions.x,
56
+ height: attachment.original_dimensions.y,
57
+ videoType: attachment.video_type.toLowerCase(),
58
+ };
59
+ case "MessageFile":
60
+ return {
61
+ type: "file",
62
+ ID: attachment.message_file_fbid,
63
+ filename: attachment.filename,
64
+ url: attachment.url,
65
+ isMalicious: attachment.is_malicious,
66
+ contentType: attachment.content_type,
67
+ name: attachment.filename,
68
+ };
69
+ case "MessageAudio":
70
+ return {
71
+ type: "audio",
72
+ ID: attachment.url_shimhash,
73
+ filename: attachment.filename,
74
+ duration: attachment.playable_duration_in_ms,
75
+ audioType: attachment.audio_type,
76
+ url: attachment.playable_url,
77
+ isVoiceMail: attachment.is_voicemail,
78
+ };
79
+ default:
80
+ return {
81
+ type: "unknown",
82
+ error: "Don't know about attachment type " + attachment.__typename,
83
+ };
84
+ }
85
+ }
86
+
87
+ /**
88
+ * Formats a share (extensible) attachment from a GraphQL response.
89
+ * @param {Object} attachment The raw extensible attachment object.
90
+ * @returns {Object} A formatted share attachment object.
91
+ */
92
+ function formatExtensibleAttachment(attachment) {
93
+ if (attachment.story_attachment) {
94
+ return {
95
+ type: "share",
96
+ ID: attachment.legacy_attachment_id,
97
+ url: attachment.story_attachment.url,
98
+ title: attachment.story_attachment.title_with_entities.text,
99
+ description: attachment.story_attachment.description && attachment.story_attachment.description.text,
100
+ source: attachment.story_attachment.source == null ? null : attachment.story_attachment.source.text,
101
+ image: attachment.story_attachment.media?.image?.uri,
102
+ width: attachment.story_attachment.media?.image?.width,
103
+ height: attachment.story_attachment.media?.image?.height,
104
+ playable: attachment.story_attachment.media?.is_playable,
105
+ duration: attachment.story_attachment.media?.playable_duration_in_ms,
106
+ playableUrl: attachment.story_attachment.media?.playable_url,
107
+ subattachments: attachment.story_attachment.subattachments,
108
+ properties: attachment.story_attachment.properties.reduce((obj, cur) => {
109
+ obj[cur.key] = cur.value.text;
110
+ return obj;
111
+ }, {}),
112
+ };
113
+ }
114
+ return { type: "unknown", error: "Don't know what to do with extensible_attachment." };
115
+ }
116
+
117
+ /**
118
+ * Formats the response from a GraphQL message history query.
119
+ * @param {Object} data The raw GraphQL response data.
120
+ * @returns {Array<Object>} An array of formatted message objects.
121
+ */
122
+ function formatMessagesGraphQLResponse(data) {
123
+ const messageThread = data.o0.data.message_thread;
124
+ if (!messageThread) return [];
125
+
126
+ const threadID = messageThread.thread_key.thread_fbid ? messageThread.thread_key.thread_fbid : messageThread.thread_key.other_user_id;
127
+
128
+ return messageThread.messages.nodes.map(d => {
129
+ switch (d.__typename) {
130
+ case "UserMessage":
131
+ const mentions = {};
132
+ if (d.message?.ranges) {
133
+ d.message.ranges.forEach(e => {
134
+ mentions[e.entity.id] = d.message.text.substring(e.offset, e.offset + e.length);
135
+ });
136
+ }
137
+ return {
138
+ type: "message",
139
+ attachments: d.sticker ? [{
140
+ type: "sticker",
141
+ ID: d.sticker.id,
142
+ url: d.sticker.url,
143
+ packID: d.sticker.pack?.id,
144
+ frameCount: d.sticker.frame_count,
145
+ frameRate: d.sticker.frame_rate,
146
+ framesPerRow: d.sticker.frames_per_row,
147
+ framesPerCol: d.sticker.frames_per_col,
148
+ stickerID: d.sticker.id,
149
+ }] : (d.blob_attachments || []).map(formatAttachmentsGraphQLResponse).concat(d.extensible_attachment ? [formatExtensibleAttachment(d.extensible_attachment)] : []),
150
+ body: d.message?.text || "",
151
+ isGroup: messageThread.thread_type === "GROUP",
152
+ messageID: d.message_id,
153
+ senderID: d.message_sender.id,
154
+ threadID: threadID,
155
+ timestamp: d.timestamp_precise,
156
+ mentions: mentions,
157
+ isUnread: d.unread,
158
+ messageReactions: d.message_reactions?.map(r => ({ reaction: r.reaction, userID: r.user.id })) || [],
159
+ };
160
+ case "ThreadNameMessage":
161
+ case "ThreadImageMessage":
162
+ case "ParticipantLeftMessage":
163
+ case "ParticipantsAddedMessage":
164
+ case "GenericAdminTextMessage": {
165
+
166
+
167
+ return {
168
+ type: "event",
169
+ messageID: d.message_id,
170
+ threadID: threadID,
171
+ isGroup: messageThread.thread_type === "GROUP",
172
+ senderID: d.message_sender.id,
173
+ author: d.message_sender.id,
174
+ timestamp: d.timestamp_precise,
175
+ snippet: d.snippet,
176
+ logMessageType: utils.getAdminTextMessageType(d.extensible_message_admin_text_type || d.__typename),
177
+ logMessageData: d.extensible_message_admin_text || d,
178
+ };
179
+ }
180
+ default:
181
+ return { type: "unknown", error: "Unknown message type " + d.__typename, raw: d };
182
+ }
183
+ });
184
+ }
185
+
186
+ /**
187
+ * @param {Object} defaultFuncs
188
+ * @param {Object} api
189
+ * @param {Object} ctx
190
+ * @returns {function(threadID: string, amount: number, timestamp: number | null): Promise<Array<Object>>}
191
+ */
192
+ module.exports = function (defaultFuncs, api, ctx) {
193
+ /**
194
+ * Retrieves the message history for a specific thread.
195
+ * @param {string} threadID The ID of the thread to fetch history from.
196
+ * @param {number} amount The number of messages to retrieve.
197
+ * @param {number | null} timestamp The timestamp to start fetching messages before.
198
+ * @returns {Promise<Array<Object>>} A promise that resolves with an array of formatted message objects.
199
+ */
200
+ return async function getThreadHistory(threadID, amount, timestamp) {
201
+ if (!threadID || !amount) {
202
+ throw new Error("getThreadHistory: threadID and amount are required.");
203
+ }
204
+
205
+ const form = {
206
+ av: ctx.globalOptions.pageID,
207
+ queries: JSON.stringify({
208
+ o0: {
209
+ doc_id: "1498317363570230",
210
+ query_params: {
211
+ id: threadID,
212
+ message_limit: amount,
213
+ load_messages: 1,
214
+ load_read_receipts: false,
215
+ before: timestamp || null,
216
+ },
217
+ },
218
+ }),
219
+ };
220
+
221
+ try {
222
+ const resData = await defaultFuncs.post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form);
223
+ const parsedData = await utils.parseAndCheckLogin(ctx, defaultFuncs)(resData);
224
+
225
+ if (parsedData.error || (Array.isArray(parsedData) && parsedData[parsedData.length - 1].error_results !== 0)) {
226
+ throw parsedData;
227
+ }
228
+
229
+ if (!Array.isArray(parsedData) || !parsedData[0] || !parsedData[0].o0 || !parsedData[0].o0.data) {
230
+ throw { error: "getThreadHistory: Malformed response from GraphQL.", res: parsedData };
231
+ }
232
+
233
+ return formatMessagesGraphQLResponse(parsedData[0]);
234
+ } catch (err) {
235
+ utils.error("getThreadHistory", err);
236
+ throw err;
237
+ }
238
+ };
239
+ };
@@ -0,0 +1,257 @@
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
+ let form = {};
185
+ threadIDs.forEach((t, i) => {
186
+ form["o" + i] = {
187
+ doc_id: "3449967031715030",
188
+ query_params: {
189
+ id: t,
190
+ message_limit: 0,
191
+ load_messages: false,
192
+ load_read_receipts: false,
193
+ before: null,
194
+ },
195
+ };
196
+ });
197
+
198
+ form = {
199
+ queries: JSON.stringify(form),
200
+ batch_name: "MessengerGraphQLThreadFetcher",
201
+ };
202
+
203
+ try {
204
+ const resData = await defaultFuncs
205
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
206
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs));
207
+
208
+ if (resData.error) {
209
+ throw resData;
210
+ }
211
+
212
+ const threadInfos = {};
213
+ for (let i = resData.length - 2; i >= 0; i--) {
214
+ const res = resData[i];
215
+
216
+ // Check for error_results and throw instead of silently continuing
217
+ if (res.error_results) {
218
+ const error = new Error(`Facebook returned error_results for thread query: ${res.error_results} errors`);
219
+ Object.assign(error, {
220
+ error_count: res.error_results,
221
+ thread_index: i
222
+ });
223
+ utils.error("getThreadInfo", error);
224
+ throw error;
225
+ }
226
+
227
+ const oKey = Object.keys(res)[0];
228
+ const responseData = res[oKey];
229
+
230
+ // Check for errors in the response object
231
+ if (responseData.errors || responseData.error_results) {
232
+ const details = responseData.errors
233
+ ? JSON.stringify(responseData.errors)
234
+ : `error_results: ${responseData.error_results}`;
235
+ const error = new Error(`GraphQL error in thread response: ${details}`);
236
+ Object.assign(error, {
237
+ details: details,
238
+ thread_index: i,
239
+ fullErrors: responseData.errors
240
+ });
241
+ utils.error("getThreadInfo", error);
242
+ throw error;
243
+ }
244
+
245
+ const threadInfo = formatThreadGraphQLResponse(responseData.data);
246
+ if (threadInfo) {
247
+ threadInfos[threadInfo.threadID || threadID[threadID.length - 1 - i]] = threadInfo;
248
+ }
249
+ }
250
+
251
+ return Array.isArray(threadID) ? threadInfos : Object.values(threadInfos)[0] || null;
252
+ } catch (err) {
253
+ utils.error("getThreadInfo", err);
254
+ throw err;
255
+ }
256
+ };
257
+ };