stfca 1.0.1 → 1.0.3

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 (141) hide show
  1. package/CHANGELOG.md +32 -0
  2. package/README.md +45 -2
  3. package/checkUpdate.js +109 -0
  4. package/index.js +460 -1
  5. package/package.json +22 -46
  6. package/src/{api/action/addExternalModule.js → addExternalModule.js} +25 -25
  7. package/src/addUserToGroup.js +115 -0
  8. package/src/changeAdminStatus.js +103 -0
  9. package/src/changeArchivedStatus.js +55 -0
  10. package/src/{api/action/changeAvatar.js → changeAvatar.js} +136 -137
  11. package/src/changeAvatarV2.js +86 -0
  12. package/src/changeAvt.js +85 -0
  13. package/src/{api/action/changeBio.js → changeBio.js} +76 -75
  14. package/src/{api/messaging/changeBlockedStatus.js → changeBlockedStatus.js} +49 -48
  15. package/src/changeBlockedStatusMqtt.js +80 -0
  16. package/src/changeCover.js +72 -0
  17. package/src/changeGroupImage.js +135 -0
  18. package/src/changeName.js +79 -0
  19. package/src/changeNickname.js +59 -0
  20. package/src/changeThreadColor.js +65 -0
  21. package/src/changeThreadEmoji.js +55 -0
  22. package/src/changeUsername.js +59 -0
  23. package/src/createCommentPost.js +230 -0
  24. package/src/{api/messaging/createNewGroup.js → createNewGroup.js} +88 -88
  25. package/src/createPoll.js +71 -0
  26. package/src/createPost.js +276 -0
  27. package/src/{api/messaging/deleteMessage.js → deleteMessage.js} +56 -56
  28. package/src/{api/messaging/deleteThread.js → deleteThread.js} +56 -56
  29. package/src/editMessage.js +68 -0
  30. package/src/editMessageOld.js +67 -0
  31. package/src/follow.js +74 -0
  32. package/src/forwardAttachment.js +60 -0
  33. package/src/getAccess.js +112 -0
  34. package/src/getAvatarUser.js +78 -0
  35. package/src/{api/action/getCurrentUserID.js → getCurrentUserID.js} +7 -7
  36. package/src/{api/messaging/getEmojiUrl.js → getEmojiUrl.js} +2 -2
  37. package/src/{api/messaging/getFriendsList.js → getFriendsList.js} +83 -82
  38. package/src/{api/messaging/getMessage.js → getMessage.js} +847 -829
  39. package/src/getRegion.js +7 -0
  40. package/src/{api/threads/getThreadHistory.js → getThreadHistory.js} +680 -664
  41. package/src/getThreadHistoryDeprecated.js +71 -0
  42. package/src/getThreadInfo.js +232 -0
  43. package/src/getThreadInfoDeprecated.js +56 -0
  44. package/src/getThreadList.js +213 -0
  45. package/src/getThreadListDeprecated.js +46 -0
  46. package/src/getThreadPictures.js +59 -0
  47. package/src/getUID.js +119 -0
  48. package/src/{api/users/getUserID.js → getUserID.js} +61 -65
  49. package/src/getUserInfo.js +66 -0
  50. package/src/handleFriendRequest.js +46 -0
  51. package/src/handleMessageRequest.js +47 -0
  52. package/src/httpGet.js +49 -0
  53. package/src/httpPost.js +48 -0
  54. package/src/listenMqtt.js +833 -0
  55. package/src/logout.js +75 -0
  56. package/src/markAsDelivered.js +47 -0
  57. package/src/markAsRead.js +70 -0
  58. package/src/markAsReadAll.js +40 -0
  59. package/src/markAsSeen.js +48 -0
  60. package/src/muteThread.js +45 -0
  61. package/src/refreshFb_dtsg.js +89 -0
  62. package/src/removeUserFromGroup.js +79 -0
  63. package/src/{api/messaging/resolvePhotoUrl.js → resolvePhotoUrl.js} +45 -43
  64. package/src/{api/messaging/searchForThread.js → searchForThread.js} +53 -52
  65. package/src/searchStickers.js +53 -0
  66. package/src/sendMessage.js +379 -0
  67. package/src/{api/messaging/sendMessageMqtt.js → sendMessageMqtt.js} +322 -323
  68. package/src/sendTypingIndicator.js +101 -0
  69. package/src/sendTypingIndicatorV2.js +28 -0
  70. package/src/setMessageReaction.js +122 -0
  71. package/src/setMessageReactionMqtt.js +62 -0
  72. package/src/{api/action/setPostReaction.js → setPostReaction.js} +112 -106
  73. package/src/setStoryReaction.js +64 -0
  74. package/src/setTitle.js +90 -0
  75. package/src/shareContact.js +110 -0
  76. package/src/shareLink.js +59 -0
  77. package/src/stopListenMqtt.js +23 -0
  78. package/src/{api/messaging/threadColors.js → threadColors.js} +131 -128
  79. package/src/{api/action/unfriend.js → unfriend.js} +52 -54
  80. package/src/unsendMessage.js +45 -0
  81. package/src/{api/messaging/uploadAttachment.js → uploadAttachment.js} +93 -95
  82. package/utils.js +2876 -0
  83. package/LICENSE-MIT +0 -4
  84. package/index.d.ts +0 -615
  85. package/module/config.js +0 -33
  86. package/module/login.js +0 -48
  87. package/module/loginHelper.js +0 -722
  88. package/module/options.js +0 -44
  89. package/src/api/action/handleFriendRequest.js +0 -57
  90. package/src/api/action/logout.js +0 -76
  91. package/src/api/action/refreshFb_dtsg.js +0 -71
  92. package/src/api/http/httpGet.js +0 -46
  93. package/src/api/http/httpPost.js +0 -52
  94. package/src/api/http/postFormData.js +0 -47
  95. package/src/api/messaging/addUserToGroup.js +0 -68
  96. package/src/api/messaging/changeAdminStatus.js +0 -122
  97. package/src/api/messaging/changeArchivedStatus.js +0 -55
  98. package/src/api/messaging/changeGroupImage.js +0 -90
  99. package/src/api/messaging/changeNickname.js +0 -70
  100. package/src/api/messaging/changeThreadColor.js +0 -79
  101. package/src/api/messaging/changeThreadEmoji.js +0 -106
  102. package/src/api/messaging/createPoll.js +0 -43
  103. package/src/api/messaging/editMessage.js +0 -68
  104. package/src/api/messaging/forwardAttachment.js +0 -51
  105. package/src/api/messaging/handleMessageRequest.js +0 -65
  106. package/src/api/messaging/markAsDelivered.js +0 -57
  107. package/src/api/messaging/markAsRead.js +0 -88
  108. package/src/api/messaging/markAsReadAll.js +0 -49
  109. package/src/api/messaging/markAsSeen.js +0 -61
  110. package/src/api/messaging/muteThread.js +0 -50
  111. package/src/api/messaging/removeUserFromGroup.js +0 -105
  112. package/src/api/messaging/sendMessage.js +0 -379
  113. package/src/api/messaging/sendTypingIndicator.js +0 -67
  114. package/src/api/messaging/setMessageReaction.js +0 -75
  115. package/src/api/messaging/setTitle.js +0 -119
  116. package/src/api/messaging/shareContact.js +0 -49
  117. package/src/api/messaging/unsendMessage.js +0 -81
  118. package/src/api/socket/core/connectMqtt.js +0 -179
  119. package/src/api/socket/core/getSeqID.js +0 -25
  120. package/src/api/socket/core/getTaskResponseData.js +0 -22
  121. package/src/api/socket/core/markDelivery.js +0 -12
  122. package/src/api/socket/core/parseDelta.js +0 -351
  123. package/src/api/socket/detail/buildStream.js +0 -208
  124. package/src/api/socket/detail/constants.js +0 -24
  125. package/src/api/socket/listenMqtt.js +0 -133
  126. package/src/api/threads/getThreadInfo.js +0 -358
  127. package/src/api/threads/getThreadList.js +0 -248
  128. package/src/api/threads/getThreadPictures.js +0 -78
  129. package/src/api/users/getUserInfo.js +0 -319
  130. package/src/api/users/getUserInfoV2.js +0 -133
  131. package/src/core/sendReqMqtt.js +0 -63
  132. package/src/database/models/index.js +0 -49
  133. package/src/database/models/thread.js +0 -31
  134. package/src/database/models/user.js +0 -32
  135. package/src/database/threadData.js +0 -98
  136. package/src/database/userData.js +0 -89
  137. package/src/utils/client.js +0 -214
  138. package/src/utils/constants.js +0 -23
  139. package/src/utils/format.js +0 -1111
  140. package/src/utils/headers.js +0 -41
  141. package/src/utils/request.js +0 -215
@@ -0,0 +1,833 @@
1
+ /* eslint-disable no-redeclare */
2
+ "use strict";
3
+ var utils = require("../utils");
4
+ var log = require("npmlog");
5
+ var mqtt = require('mqtt');
6
+ var websocket = require('websocket-stream');
7
+ var HttpsProxyAgent = require('https-proxy-agent');
8
+ const EventEmitter = require('events');
9
+ const debugSeq = false;
10
+ var identity = function () { };
11
+ var form = {};
12
+ var getSeqID = function () { };
13
+ var topics = [
14
+ "/legacy_web",
15
+ "/webrtc",
16
+ "/rtc_multi",
17
+ "/onevc",
18
+ "/br_sr", //Notification
19
+ //Need to publish /br_sr right after this
20
+ "/sr_res",
21
+ "/t_ms",
22
+ "/thread_typing",
23
+ "/orca_typing_notifications",
24
+ "/notify_disconnect",
25
+ //Need to publish /messenger_sync_create_queue right after this
26
+ "/orca_presence",
27
+ //Will receive /sr_res right here.
28
+
29
+ "/inbox",
30
+ "/mercury",
31
+ "/messaging_events",
32
+ "/orca_message_notifications",
33
+ "/pp",
34
+ "/webrtc_response",
35
+ ];
36
+
37
+ function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
38
+ //Don't really know what this does but I think it's for the active state?
39
+ //TODO: Move to ctx when implemented
40
+ var chatOn = ctx.globalOptions.online;
41
+ var foreground = false;
42
+
43
+ var sessionID = Math.floor(Math.random() * 9007199254740991) + 1;
44
+ var GUID = utils.getGUID();
45
+ const username = {
46
+ u: ctx.userID,
47
+ s: sessionID,
48
+ chat_on: chatOn,
49
+ fg: foreground,
50
+ d: GUID,
51
+ ct: 'websocket',
52
+ aid: '219994525426954',
53
+ aids: null,
54
+ mqtt_sid: '',
55
+ cp: 3,
56
+ ecp: 10,
57
+ st: [],
58
+ pm: [],
59
+ dc: '',
60
+ no_auto_fg: true,
61
+ gas: null,
62
+ pack: [],
63
+ p: null,
64
+ php_override: ""
65
+ };
66
+ var cookies = ctx.jar.getCookies("https://www.facebook.com").join("; ");
67
+
68
+ var host;
69
+ if (ctx.mqttEndpoint) host = `${ctx.mqttEndpoint}&sid=${sessionID}&cid=${GUID}`;
70
+ else if (ctx.region) host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLocaleLowerCase()}&sid=${sessionID}&cid=${GUID}`;
71
+ else host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}&cid=${GUID}`;
72
+
73
+ const options = {
74
+ clientId: 'mqttwsclient',
75
+ protocolId: 'MQIsdp',
76
+ protocolVersion: 3,
77
+ username: JSON.stringify(username),
78
+ clean: true,
79
+ wsOptions: {
80
+ headers: {
81
+ Cookie: cookies,
82
+ Origin: 'https://www.facebook.com',
83
+ 'User-Agent': ctx.globalOptions.userAgent || 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.64 Safari/537.36',
84
+ Referer: 'https://www.facebook.com/',
85
+ Host: new URL(host).hostname,
86
+ },
87
+ origin: 'https://www.facebook.com',
88
+ protocolVersion: 13,
89
+ binaryType: 'arraybuffer',
90
+ },
91
+ keepalive: 60,
92
+ reschedulePings: true,
93
+ reconnectPeriod: 3,
94
+ };
95
+
96
+ if (typeof ctx.globalOptions.proxy != "undefined") {
97
+ var agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
98
+ options.wsOptions.agent = agent;
99
+ }
100
+
101
+ ctx.mqttClient = new mqtt.Client(_ => websocket(host, options.wsOptions), options);
102
+
103
+ global.mqttClient = ctx.mqttClient;
104
+
105
+ mqttClient.on('error', function (err) {
106
+ log.error("listenMqtt", err);
107
+ mqttClient.end();
108
+ if (ctx.globalOptions.autoReconnect) getSeqID();
109
+ else globalCallback({ type: "stop_listen", error: "Connection refused: Server unavailable" }, null);
110
+ });
111
+
112
+ mqttClient.on('connect', function () {
113
+ topics.forEach(topicsub => mqttClient.subscribe(topicsub));
114
+
115
+ // Display connection success message with branding
116
+ console.log('\n✅ ST-FCA MQTT Connected');
117
+ console.log(`📍 Region: ${ctx.region || 'PNB'}`);
118
+ console.log(`🔄 Auto-reconnect: ${ctx.globalOptions.autoReconnect ? 'Enabled' : 'Disabled'}${ctx.globalOptions.autoReconnect ? ' (reconnects every 3s on disconnect)' : ''}`);
119
+ console.log('🎨 Maintained & Enhanced by ST | Sheikh Tamim\n');
120
+
121
+ var topic;
122
+ var queue = {
123
+ sync_api_version: 10,
124
+ max_deltas_able_to_process: 1000,
125
+ delta_batch_size: 500,
126
+ encoding: "JSON",
127
+ entity_fbid: ctx.userID,
128
+ };
129
+
130
+ if (ctx.syncToken) {
131
+ topic = "/messenger_sync_get_diffs";
132
+ queue.last_seq_id = ctx.lastSeqId;
133
+ queue.sync_token = ctx.syncToken;
134
+ }
135
+ else {
136
+ topic = "/messenger_sync_create_queue";
137
+ queue.initial_titan_sequence_id = ctx.lastSeqId;
138
+ queue.device_params = null;
139
+ }
140
+
141
+ mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
142
+
143
+ var rTimeout = setTimeout(function () {
144
+ mqttClient.end();
145
+ getSeqID();
146
+ }, 5000);
147
+
148
+ ctx.tmsWait = function () {
149
+ clearTimeout(rTimeout);
150
+ ctx.globalOptions.emitReady ? globalCallback({
151
+ type: "ready",
152
+ error: null
153
+ }) : "";
154
+ delete ctx.tmsWait;
155
+ };
156
+ });
157
+
158
+ mqttClient.on('message', function (topic, message, _packet) {
159
+ try {
160
+ var jsonMessage = JSON.parse(message);
161
+ }
162
+ catch (ex) {
163
+ return log.error("listenMqtt", ex);
164
+ }
165
+ if (topic === "/t_ms") {
166
+ if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
167
+
168
+ if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
169
+ ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
170
+ ctx.syncToken = jsonMessage.syncToken;
171
+ }
172
+
173
+ if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
174
+
175
+ //If it contains more than 1 delta
176
+ for (var i in jsonMessage.deltas) {
177
+ var delta = jsonMessage.deltas[i];
178
+ parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
179
+ }
180
+ }
181
+ else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
182
+ var typ = {
183
+ type: "typ",
184
+ isTyping: !!jsonMessage.state,
185
+ from: jsonMessage.sender_fbid.toString(),
186
+ threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
187
+ };
188
+ (function () { globalCallback(null, typ); })();
189
+ }
190
+ else if (topic === "/orca_presence") {
191
+ if (!ctx.globalOptions.updatePresence) {
192
+ for (var i in jsonMessage.list) {
193
+ var data = jsonMessage.list[i];
194
+ var userID = data["u"];
195
+
196
+ var presence = {
197
+ type: "presence",
198
+ userID: userID.toString(),
199
+ //Convert to ms
200
+ timestamp: data["l"] * 1000,
201
+ statuses: data["p"]
202
+ };
203
+ (function () { globalCallback(null, presence); })();
204
+ }
205
+ }
206
+ }
207
+
208
+ });
209
+
210
+ mqttClient.on('close', function () { });
211
+ }
212
+
213
+ function parseDelta(defaultFuncs, api, ctx, globalCallback, v) {
214
+ if (v.delta.class == "NewMessage") {
215
+ //Not tested for pages
216
+ if (ctx.globalOptions.pageID && ctx.globalOptions.pageID != v.queue) return;
217
+
218
+ (function resolveAttachmentUrl(i) {
219
+ if (i == (v.delta.attachments || []).length) {
220
+ let fmtMsg;
221
+ try {
222
+ fmtMsg = utils.formatDeltaMessage(v);
223
+ } catch (err) {
224
+ return globalCallback({
225
+ error: "Problem parsing message object.",
226
+ detail: err,
227
+ res: v,
228
+ type: "parse_error"
229
+ });
230
+ }
231
+ if (fmtMsg) {
232
+ if (ctx.globalOptions.autoMarkDelivery) {
233
+ markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
234
+ }
235
+ }
236
+ return !ctx.globalOptions.selfListen &&
237
+ (fmtMsg.senderID === ctx.i_userID || fmtMsg.senderID === ctx.userID) ?
238
+ undefined :
239
+ (function () { globalCallback(null, fmtMsg); })();
240
+ } else {
241
+ if (v.delta.attachments[i].mercury.attach_type == "photo") {
242
+ api.resolvePhotoUrl(
243
+ v.delta.attachments[i].fbid,
244
+ (err, url) => {
245
+ if (!err)
246
+ v.delta.attachments[
247
+ i
248
+ ].mercury.metadata.url = url;
249
+ return resolveAttachmentUrl(i + 1);
250
+ }
251
+ );
252
+ } else {
253
+ return resolveAttachmentUrl(i + 1);
254
+ }
255
+ }
256
+ })(0);
257
+ }
258
+
259
+ if (v.delta.class == "ClientPayload") {
260
+ var clientPayload = utils.decodeClientPayload(v.delta.payload);
261
+ if (clientPayload && clientPayload.deltas) {
262
+ for (var i in clientPayload.deltas) {
263
+ var delta = clientPayload.deltas[i];
264
+ if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
265
+ (function () {
266
+ globalCallback(null, {
267
+ type: "message_reaction",
268
+ threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
269
+ messageID: delta.deltaMessageReaction.messageId,
270
+ reaction: delta.deltaMessageReaction.reaction,
271
+ senderID: delta.deltaMessageReaction.senderId.toString(),
272
+ userID: delta.deltaMessageReaction.userId.toString()
273
+ });
274
+ })();
275
+ }
276
+ else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
277
+ (function () {
278
+ globalCallback(null, {
279
+ type: "message_unsend",
280
+ threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
281
+ messageID: delta.deltaRecallMessageData.messageID,
282
+ senderID: delta.deltaRecallMessageData.senderID.toString(),
283
+ deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
284
+ timestamp: delta.deltaRecallMessageData.timestamp
285
+ });
286
+ })();
287
+ }
288
+ else if (delta.deltaMessageReply) {
289
+ //Mention block - #1
290
+ var mdata = delta.deltaMessageReply.message === undefined ? [] :
291
+ delta.deltaMessageReply.message.data === undefined ? [] :
292
+ delta.deltaMessageReply.message.data.prng === undefined ? [] :
293
+ JSON.parse(delta.deltaMessageReply.message.data.prng);
294
+ var m_id = mdata.map(u => u.i);
295
+ var m_offset = mdata.map(u => u.o);
296
+ var m_length = mdata.map(u => u.l);
297
+
298
+ var mentions = {};
299
+
300
+ for (var i = 0; i < m_id.length; i++) mentions[m_id[i]] = (delta.deltaMessageReply.message.body || "").substring(m_offset[i], m_offset[i] + m_length[i]);
301
+ //Mention block - 1#
302
+ var callbackToReturn = {
303
+ type: "message_reply",
304
+ threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
305
+ messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
306
+ senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
307
+ attachments: (delta.deltaMessageReply.message.attachments || []).map(function (att) {
308
+ var mercury = JSON.parse(att.mercuryJSON);
309
+ Object.assign(att, mercury);
310
+ return att;
311
+ }).map(att => {
312
+ var x;
313
+ try {
314
+ x = utils._formatAttachment(att);
315
+ }
316
+ catch (ex) {
317
+ x = att;
318
+ x.error = ex;
319
+ x.type = "unknown";
320
+ }
321
+ return x;
322
+ }),
323
+ args: (delta.deltaMessageReply.message.body || "").trim().split(/\s+/),
324
+ body: (delta.deltaMessageReply.message.body || ""),
325
+ isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
326
+ mentions: mentions,
327
+ timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
328
+ participantIDs: (delta.deltaMessageReply.message.messageMetadata.cid.canonicalParticipantFbids || delta.deltaMessageReply.message.participants || []).map(e => e.toString())
329
+ };
330
+
331
+ if (delta.deltaMessageReply.repliedToMessage) {
332
+ //Mention block - #2
333
+ mdata = delta.deltaMessageReply.repliedToMessage === undefined ? [] :
334
+ delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
335
+ delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
336
+ JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
337
+ m_id = mdata.map(u => u.i);
338
+ m_offset = mdata.map(u => u.o);
339
+ m_length = mdata.map(u => u.l);
340
+
341
+ var rmentions = {};
342
+
343
+ for (var i = 0; i < m_id.length; i++) rmentions[m_id[i]] = (delta.deltaMessageReply.repliedToMessage.body || "").substring(m_offset[i], m_offset[i] + m_length[i]);
344
+ //Mention block - 2#
345
+ callbackToReturn.messageReply = {
346
+ threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
347
+ messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
348
+ senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
349
+ attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function (att) {
350
+ var mercury = JSON.parse(att.mercuryJSON);
351
+ Object.assign(att, mercury);
352
+ return att;
353
+ }).map(att => {
354
+ var x;
355
+ try {
356
+ x = utils._formatAttachment(att);
357
+ }
358
+ catch (ex) {
359
+ x = att;
360
+ x.error = ex;
361
+ x.type = "unknown";
362
+ }
363
+ return x;
364
+ }),
365
+ args: (delta.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
366
+ body: delta.deltaMessageReply.repliedToMessage.body || "",
367
+ isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
368
+ mentions: rmentions,
369
+ timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp
370
+ };
371
+ }
372
+ else if (delta.deltaMessageReply.replyToMessageId) {
373
+ return defaultFuncs
374
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
375
+ "av": ctx.globalOptions.pageID,
376
+ "queries": JSON.stringify({
377
+ "o0": {
378
+ //Using the same doc_id as forcedFetch
379
+ "doc_id": "2848441488556444",
380
+ "query_params": {
381
+ "thread_and_message_id": {
382
+ "thread_id": callbackToReturn.threadID,
383
+ "message_id": delta.deltaMessageReply.replyToMessageId.id,
384
+ }
385
+ }
386
+ }
387
+ })
388
+ })
389
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
390
+ .then((resData) => {
391
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
392
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
393
+ var fetchData = resData[0].o0.data.message;
394
+ var mobj = {};
395
+ for (var n in fetchData.message.ranges) mobj[fetchData.message.ranges[n].entity.id] = (fetchData.message.text || "").substr(fetchData.message.ranges[n].offset, fetchData.message.ranges[n].length);
396
+
397
+ callbackToReturn.messageReply = {
398
+ threadID: callbackToReturn.threadID,
399
+ messageID: fetchData.message_id,
400
+ senderID: fetchData.message_sender.id.toString(),
401
+ attachments: fetchData.message.blob_attachment.map(att => {
402
+ var x;
403
+ try {
404
+ x = utils._formatAttachment({ blob_attachment: att });
405
+ }
406
+ catch (ex) {
407
+ x = att;
408
+ x.error = ex;
409
+ x.type = "unknown";
410
+ }
411
+ return x;
412
+ }),
413
+ args: (fetchData.message.text || "").trim().split(/\s+/) || [],
414
+ body: fetchData.message.text || "",
415
+ isGroup: callbackToReturn.isGroup,
416
+ mentions: mobj,
417
+ timestamp: parseInt(fetchData.timestamp_precise)
418
+ };
419
+ })
420
+ .catch(err => log.error("forcedFetch", err))
421
+ .finally(function () {
422
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
423
+ !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
424
+ });
425
+ }
426
+ else callbackToReturn.delta = delta;
427
+
428
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
429
+
430
+ return !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
431
+ }
432
+ }
433
+ return;
434
+ }
435
+ }
436
+
437
+ if (v.delta.class !== "NewMessage" && !ctx.globalOptions.listenEvents) return;
438
+ switch (v.delta.class) {
439
+ case "JoinableMode": {
440
+ let fmtMsg;
441
+ try {
442
+ fmtMsg = utils.formatDeltaEvent(v.delta);
443
+ } catch (err) {
444
+ return globalCallback({
445
+ error: "Lỗi gòi!!",
446
+ detail: err,
447
+ res: v.delta,
448
+ type: "parse_error"
449
+ });
450
+ }
451
+ return globalCallback(null, fmtMsg);
452
+ }
453
+ case "AdminTextMessage":
454
+ switch (v.delta.type) {
455
+ case 'confirm_friend_request':
456
+ case 'shared_album_delete':
457
+ case 'shared_album_addition':
458
+ case 'pin_messages_v2':
459
+ case 'unpin_messages_v2':
460
+ case "change_thread_theme":
461
+ case "change_thread_nickname":
462
+ case "change_thread_icon":
463
+ case "change_thread_quick_reaction":
464
+ case "change_thread_admins":
465
+ case "group_poll":
466
+ case "joinable_group_link_mode_change":
467
+ case "magic_words":
468
+ case "change_thread_approval_mode":
469
+ case "messenger_call_log":
470
+ case "participant_joined_group_call":
471
+ var fmtMsg;
472
+ try {
473
+ fmtMsg = utils.formatDeltaEvent(v.delta);
474
+ }
475
+ catch (err) {
476
+ return globalCallback({
477
+ error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
478
+ detail: err,
479
+ res: v.delta,
480
+ type: "parse_error"
481
+ });
482
+ }
483
+ return (function () { globalCallback(null, fmtMsg); })();
484
+ default:
485
+ // console.log(v.delta)
486
+ return;
487
+ }
488
+ //For group images
489
+ case "ForcedFetch":
490
+ if (!v.delta.threadKey) return;
491
+ var mid = v.delta.messageId;
492
+ var tid = v.delta.threadKey.threadFbId;
493
+ if (mid && tid) {
494
+ const form = {
495
+ "av": ctx.globalOptions.pageID,
496
+ "queries": JSON.stringify({
497
+ "o0": {
498
+ //This doc_id is valid as of March 25, 2020
499
+ "doc_id": "2848441488556444",
500
+ "query_params": {
501
+ "thread_and_message_id": {
502
+ "thread_id": tid.toString(),
503
+ "message_id": mid,
504
+ }
505
+ }
506
+ }
507
+ })
508
+ };
509
+
510
+ defaultFuncs
511
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
512
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
513
+ .then((resData) => {
514
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
515
+
516
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
517
+
518
+ var fetchData = resData[0].o0.data.message;
519
+
520
+ if (utils.getType(fetchData) == "Object") {
521
+ log.info("forcedFetch", fetchData);
522
+ switch (fetchData.__typename) {
523
+ case "ThreadImageMessage":
524
+ (!ctx.globalOptions.selfListen && fetchData.message_sender.id.toString() === ctx.userID) ||
525
+ !ctx.loggedIn ? undefined : (function () {
526
+ globalCallback(null, {
527
+ type: "change_thread_image",
528
+ threadID: utils.formatID(tid.toString()),
529
+ snippet: fetchData.snippet,
530
+ timestamp: fetchData.timestamp_precise,
531
+ author: fetchData.message_sender.id,
532
+ image: {
533
+ attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
534
+ width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
535
+ height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
536
+ url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
537
+ }
538
+ });
539
+ })();
540
+ break;
541
+ case "UserMessage":
542
+ log.info("ff-Return", {
543
+ type: "message",
544
+ senderID: utils.formatID(fetchData.message_sender.id),
545
+ body: fetchData.message.text || "",
546
+ threadID: utils.formatID(tid.toString()),
547
+ messageID: fetchData.message_id,
548
+ attachments: [{
549
+ type: "share",
550
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
551
+ url: fetchData.extensible_attachment.story_attachment.url,
552
+
553
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
554
+ description: fetchData.extensible_attachment.story_attachment.description.text,
555
+ source: fetchData.extensible_attachment.story_attachment.source,
556
+
557
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
558
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
559
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
560
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
561
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
562
+
563
+ subattachments: fetchData.extensible_attachment.subattachments,
564
+ properties: fetchData.extensible_attachment.story_attachment.properties,
565
+ }],
566
+ mentions: {},
567
+ timestamp: parseInt(fetchData.timestamp_precise),
568
+ participantIDs: (fetchData.participants || (fetchData.messageMetadata ? fetchData.messageMetadata.cid ? fetchData.messageMetadata.cid.canonicalParticipantFbids : fetchData.messageMetadata.participantIds : []) || []),
569
+ isGroup: (fetchData.message_sender.id != tid.toString())
570
+ });
571
+ globalCallback(null, {
572
+ type: "message",
573
+ senderID: utils.formatID(fetchData.message_sender.id),
574
+ body: fetchData.message.text || "",
575
+ threadID: utils.formatID(tid.toString()),
576
+ messageID: fetchData.message_id,
577
+ attachments: [{
578
+ type: "share",
579
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
580
+ url: fetchData.extensible_attachment.story_attachment.url,
581
+
582
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
583
+ description: fetchData.extensible_attachment.story_attachment.description.text,
584
+ source: fetchData.extensible_attachment.story_attachment.source,
585
+
586
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
587
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
588
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
589
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
590
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
591
+
592
+ subattachments: fetchData.extensible_attachment.subattachments,
593
+ properties: fetchData.extensible_attachment.story_attachment.properties,
594
+ }],
595
+ mentions: {},
596
+ timestamp: parseInt(fetchData.timestamp_precise),
597
+ participantIDs: (fetchData.participants || (fetchData.messageMetadata ? fetchData.messageMetadata.cid ? fetchData.messageMetadata.cid.canonicalParticipantFbids : fetchData.messageMetadata.participantIds : []) || []),
598
+ isGroup: (fetchData.message_sender.id != tid.toString())
599
+ });
600
+ }
601
+ }
602
+ else log.error("forcedFetch", fetchData);
603
+ })
604
+ .catch((err) => log.error("forcedFetch", err));
605
+ }
606
+ break;
607
+ case "ThreadName":
608
+ case "ParticipantsAddedToGroupThread":
609
+ case "ParticipantLeftGroupThread":
610
+ var formattedEvent;
611
+ try {
612
+ formattedEvent = utils.formatDeltaEvent(v.delta);
613
+ }
614
+ catch (err) {
615
+ return globalCallback({
616
+ error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
617
+ detail: err,
618
+ res: v.delta,
619
+ type: "parse_error"
620
+ });
621
+ }
622
+ return (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) || !ctx.loggedIn ? undefined : (function () { globalCallback(null, formattedEvent); })();
623
+ }
624
+ }
625
+
626
+ function markDelivery(ctx, api, threadID, messageID) {
627
+ if (threadID && messageID) {
628
+ api.markAsDelivered(threadID, messageID, (err) => {
629
+ if (err) log.error("markAsDelivered", err);
630
+ else {
631
+ if (ctx.globalOptions.autoMarkRead) {
632
+ api.markAsRead(threadID, (err) => {
633
+ if (err) log.error("markAsDelivered", err);
634
+ });
635
+ }
636
+ }
637
+ });
638
+ }
639
+ }
640
+
641
+ module.exports = function (defaultFuncs, api, ctx) {
642
+ let globalCallback = identity;
643
+ // function getSeqID() {
644
+ // ctx.t_mqttCalled = false;
645
+ // async function attemptRequest(retries = 3) {
646
+ // try {
647
+ // if (!ctx.fb_dtsg) {
648
+ // const dtsg = await api.getFreshDtsg();
649
+ // if (!dtsg) {
650
+ // if (retries > 0) {
651
+ // logger.Warning("Failed to get fb_dtsg, retrying...");
652
+ // await utils.sleep(2000); // Longer delay for token retry
653
+ // return attemptRequest(retries - 1);
654
+ // }
655
+ // throw { error: "Could not obtain fb_dtsg after multiple attempts" };
656
+ // }
657
+ // ctx.fb_dtsg = dtsg;
658
+ // }
659
+
660
+ // const form = {
661
+ // av: ctx.userID,
662
+ // fb_dtsg: ctx.fb_dtsg,
663
+ // queries: JSON.stringify({
664
+ // o0: {
665
+ // doc_id: '3336396659757871',
666
+ // query_params: {
667
+ // limit: 1,
668
+ // before: null,
669
+ // tags: ['INBOX'],
670
+ // includeDeliveryReceipts: false,
671
+ // includeSeqID: true
672
+ // }
673
+ // }
674
+ // }),
675
+ // __user: ctx.userID,
676
+ // __a: '1',
677
+ // __req: '8',
678
+ // __hs: '19577.HYP:comet_pkg.2.1..2.1',
679
+ // dpr: '1',
680
+ // fb_api_caller_class: 'RelayModern',
681
+ // fb_api_req_friendly_name: 'MessengerGraphQLThreadlistFetcher'
682
+ // };
683
+
684
+ // const headers = {
685
+ // 'Content-Type': 'application/x-www-form-urlencoded',
686
+ // 'Referer': 'https://www.facebook.com/',
687
+ // 'Origin': 'https://www.facebook.com',
688
+ // 'sec-fetch-site': 'same-origin',
689
+ // 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36',
690
+ // 'Cookie': ctx.jar.getCookieString('https://www.facebook.com'),
691
+ // 'accept': '*/*',
692
+ // 'accept-encoding': 'gzip, deflate, br'
693
+ // };
694
+
695
+ // const resData = await defaultFuncs
696
+ // .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form, { headers })
697
+ // .then(utils.parseAndCheckLogin(ctx, defaultFuncs));
698
+
699
+ // if (debugSeq) {
700
+ // console.log('GraphQL SeqID Response:', JSON.stringify(resData, null, 2));
701
+ // }
702
+
703
+ // if (resData.error === 1357004 || resData.error === 1357001) {
704
+ // if (retries > 0) {
705
+ // logger.Warning("Session error, refreshing token and retrying...");
706
+ // ctx.fb_dtsg = null; // Force new token
707
+ // await utils.sleep(2000);
708
+ // return attemptRequest(retries - 1);
709
+ // }
710
+ // throw { error: "Session refresh failed after retries" };
711
+ // }
712
+
713
+ // if (!Array.isArray(resData)) {
714
+ // throw { error: "Invalid response format", res: resData };
715
+ // }
716
+
717
+ // const seqID = resData[0]?.o0?.data?.viewer?.message_threads?.sync_sequence_id;
718
+ // if (!seqID) {
719
+ // throw { error: "Missing sync_sequence_id", res: resData };
720
+ // }
721
+
722
+ // ctx.lastSeqId = seqID;
723
+ // if (debugSeq) {
724
+ // console.log('Got SeqID:', ctx.lastSeqId);
725
+ // }
726
+
727
+ // return listenMqtt(defaultFuncs, api, ctx, globalCallback);
728
+
729
+ // } catch (err) {
730
+ // if (retries > 0) {
731
+ // console.log("Request failed, retrying...");
732
+
733
+ // return attemptRequest(retries - 1);
734
+ // }
735
+ // throw err;
736
+ // }
737
+ // }
738
+
739
+ // return attemptRequest()
740
+ // .catch((err) => {
741
+ // log.error("getSeqId", err);
742
+ // if (utils.getType(err) == "Object" && err.error === "Not logged in") ctx.loggedIn = false;
743
+ // return globalCallback(err);
744
+ // });
745
+ // }
746
+
747
+ getSeqID = function getSeqID() {
748
+ ctx.t_mqttCalled = false;
749
+ defaultFuncs
750
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
751
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
752
+ .then((resData) => {
753
+ if (utils.getType(resData) != "Array") throw { error: "Not logged in", res: resData };
754
+ if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
755
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "getSeqId: there was no successful_results", res: resData };
756
+ if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
757
+ ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
758
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
759
+ } else throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
760
+ })
761
+ .catch((err) => {
762
+ log.error("getSeqId", err);
763
+ if (utils.getType(err) == "Object" && err.error === "Not logged in") ctx.loggedIn = false;
764
+ return globalCallback(err);
765
+ });
766
+ };
767
+
768
+ return function (callback) {
769
+ class MessageEmitter extends EventEmitter {
770
+ stopListening(callback) {
771
+
772
+ callback = callback || (() => { });
773
+ globalCallback = identity;
774
+ if (ctx.mqttClient) {
775
+ ctx.mqttClient.unsubscribe("/webrtc");
776
+ ctx.mqttClient.unsubscribe("/rtc_multi");
777
+ ctx.mqttClient.unsubscribe("/onevc");
778
+ ctx.mqttClient.publish("/browser_close", "{}");
779
+ ctx.mqttClient.end(false, function (...data) {
780
+ callback(data);
781
+ ctx.mqttClient = undefined;
782
+ });
783
+ }
784
+ }
785
+
786
+ async stopListeningAsync() {
787
+ return new Promise((resolve) => {
788
+ this.stopListening(resolve);
789
+ });
790
+ }
791
+ }
792
+
793
+ const msgEmitter = new MessageEmitter();
794
+ globalCallback = (callback || function (error, message) {
795
+ if (error) {
796
+ return msgEmitter.emit("error", error);
797
+ }
798
+ msgEmitter.emit("message", message);
799
+ });
800
+
801
+ // Reset some stuff
802
+ if (!ctx.firstListen)
803
+ ctx.lastSeqId = null;
804
+ ctx.syncToken = undefined;
805
+ ctx.t_mqttCalled = false;
806
+
807
+ form = {
808
+ "av": ctx.globalOptions.pageID,
809
+ "queries": JSON.stringify({
810
+ "o0": {
811
+ "doc_id": "3336396659757871",
812
+ "query_params": {
813
+ "limit": 1,
814
+ "before": null,
815
+ "tags": ["INBOX"],
816
+ "includeDeliveryReceipts": false,
817
+ "includeSeqID": true
818
+ }
819
+ }
820
+ })
821
+ };
822
+
823
+ if (!ctx.firstListen || !ctx.lastSeqId) {
824
+ getSeqID(defaultFuncs, api, ctx, globalCallback);
825
+ } else {
826
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
827
+ }
828
+
829
+ api.stopListening = msgEmitter.stopListening;
830
+ api.stopListeningAsync = msgEmitter.stopListeningAsync;
831
+ return msgEmitter;
832
+ };
833
+ };