fca-mod 1.3.4

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.

Potentially problematic release.


This version of fca-mod might be problematic. Click here for more details.

Files changed (117) hide show
  1. package/.cache/replit/__replit_disk_meta.json +1 -0
  2. package/.cache/replit/nix/env.json +1 -0
  3. package/.config/configstore/update-notifier-npm.json +4 -0
  4. package/.config/configstore/update-notifier-npm.json.1735545094 +4 -0
  5. package/.gitattributes +2 -0
  6. package/.replit +89 -0
  7. package/.upm/store.json +1 -0
  8. package/Extra/Database/index.js +470 -0
  9. package/Extra/ExtraAddons.js +80 -0
  10. package/Extra/ExtraFindUID.js +62 -0
  11. package/Extra/ExtraGetThread.js +118 -0
  12. package/Extra/ExtraScreenShot.js +430 -0
  13. package/Extra/ExtraUptimeRobot.js +38 -0
  14. package/Extra/Html/Classic/script.js +119 -0
  15. package/Extra/Html/Classic/style.css +8 -0
  16. package/Extra/Security/Index.js +146 -0
  17. package/Extra/Security/Step_1.js +6 -0
  18. package/Extra/Security/Step_2.js +22 -0
  19. package/Extra/Security/Step_3.js +22 -0
  20. package/Extra/Src/Change_Environment.js +24 -0
  21. package/Extra/Src/Check_Update.js +66 -0
  22. package/Extra/Src/History.js +115 -0
  23. package/Extra/Src/Instant_Update.js +65 -0
  24. package/Extra/Src/Last-Run.js +65 -0
  25. package/Extra/Src/Premium.js +83 -0
  26. package/Extra/Src/Release_Memory.js +41 -0
  27. package/Extra/Src/uuid.js +137 -0
  28. package/FastConfigFca.json +23 -0
  29. package/Func/AcceptAgreement.js +32 -0
  30. package/Func/ClearCache.js +64 -0
  31. package/Func/ReportV1.js +54 -0
  32. package/LICENSE +24 -0
  33. package/Language/index.json +206 -0
  34. package/Main.js +1098 -0
  35. package/README.md +136 -0
  36. package/SECURITY.md +14 -0
  37. package/broadcast.js +40 -0
  38. package/index.js +313 -0
  39. package/logger.js +66 -0
  40. package/package.json +102 -0
  41. package/replit.nix +8 -0
  42. package/src/Dev_Horizon_Data.js +125 -0
  43. package/src/Premium.js +30 -0
  44. package/src/Screenshot.js +85 -0
  45. package/src/addExternalModule.js +16 -0
  46. package/src/addUserToGroup.js +79 -0
  47. package/src/changeAdminStatus.js +79 -0
  48. package/src/changeArchivedStatus.js +41 -0
  49. package/src/changeAvt.js +85 -0
  50. package/src/changeBio.js +65 -0
  51. package/src/changeBlockedStatus.js +36 -0
  52. package/src/changeGroupImage.js +106 -0
  53. package/src/changeNickname.js +45 -0
  54. package/src/changeThreadColor.js +62 -0
  55. package/src/changeThreadEmoji.js +42 -0
  56. package/src/createNewGroup.js +70 -0
  57. package/src/createPoll.js +60 -0
  58. package/src/deleteMessage.js +45 -0
  59. package/src/deleteThread.js +43 -0
  60. package/src/forwardAttachment.js +48 -0
  61. package/src/getAccessToken.js +32 -0
  62. package/src/getCurrentUserID.js +7 -0
  63. package/src/getEmojiUrl.js +27 -0
  64. package/src/getFriendsList.js +73 -0
  65. package/src/getMessage.js +80 -0
  66. package/src/getThreadHistory.js +537 -0
  67. package/src/getThreadInfo.js +355 -0
  68. package/src/getThreadList.js +213 -0
  69. package/src/getThreadMain.js +220 -0
  70. package/src/getThreadPictures.js +59 -0
  71. package/src/getUID.js +59 -0
  72. package/src/getUserID.js +62 -0
  73. package/src/getUserInfo.js +141 -0
  74. package/src/getUserInfoMain.js +65 -0
  75. package/src/getUserInfoV2.js +36 -0
  76. package/src/getUserInfoV3.js +63 -0
  77. package/src/getUserInfoV4.js +55 -0
  78. package/src/getUserInfoV5.js +61 -0
  79. package/src/handleFriendRequest.js +46 -0
  80. package/src/handleMessageRequest.js +49 -0
  81. package/src/httpGet.js +49 -0
  82. package/src/httpPost.js +48 -0
  83. package/src/httpPostFormData.js +41 -0
  84. package/src/listenMqtt.js +796 -0
  85. package/src/logout.js +68 -0
  86. package/src/markAsDelivered.js +48 -0
  87. package/src/markAsRead.js +70 -0
  88. package/src/markAsReadAll.js +43 -0
  89. package/src/markAsSeen.js +51 -0
  90. package/src/muteThread.js +47 -0
  91. package/src/removeUserFromGroup.js +49 -0
  92. package/src/resolvePhotoUrl.js +37 -0
  93. package/src/searchForThread.js +43 -0
  94. package/src/sendMessage.js +334 -0
  95. package/src/sendTypingIndicator.js +80 -0
  96. package/src/setMessageReaction.js +109 -0
  97. package/src/setPostReaction.js +102 -0
  98. package/src/setTitle.js +74 -0
  99. package/src/threadColors.js +39 -0
  100. package/src/unfriend.js +43 -0
  101. package/src/unsendMessage.js +40 -0
  102. package/test/Database_Test.js +3 -0
  103. package/test/Db2.js +362 -0
  104. package/test/data/shareAttach.js +146 -0
  105. package/test/data/something.mov +0 -0
  106. package/test/data/test.png +0 -0
  107. package/test/data/test.txt +7 -0
  108. package/test/env/.env +0 -0
  109. package/test/example-config.json +18 -0
  110. package/test/example-db.db +0 -0
  111. package/test/kteam_Database/A_README.md +1 -0
  112. package/test/kteam_Database/Database.db +0 -0
  113. package/test/memoryleak.js +18 -0
  114. package/test/test-page.js +140 -0
  115. package/test/test.js +385 -0
  116. package/test/testv2.js +18 -0
  117. package/utils.js +1644 -0
@@ -0,0 +1,796 @@
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 { json } = require("body-parser");
10
+ var identity = function () { };
11
+ var form = {};
12
+ var getSeqID = function () { };
13
+
14
+ var topics = ["/legacy_web","/webrtc","/rtc_multi","/onevc","/br_sr","/sr_res","/t_ms","/thread_typing","/orca_typing_notifications","/notify_disconnect","/orca_presence","/inbox","/mercury", "/messaging_events", "/orca_message_notifications", "/pp","/webrtc_response"];
15
+
16
+ /* [ Noti ? ]
17
+ ! "/br_sr", //Notification
18
+ * => Need to publish /br_sr right after this
19
+
20
+ ! "/notify_disconnect",
21
+ * => Need to publish /messenger_sync_create_queue right after this
22
+
23
+ ! "/orca_presence",
24
+ * => Will receive /sr_res right here.
25
+ */
26
+
27
+ function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
28
+ //Don't really know what this does but I think it's for the active state?
29
+ //TODO: Move to ctx when implemented
30
+ var chatOn = ctx.globalOptions.online;
31
+ var foreground = false;
32
+
33
+ var sessionID = Math.floor(Math.random() * 9007199254740991) + 1;
34
+ var username = {u: ctx.userID,s: sessionID,chat_on: chatOn,fg: foreground,d: utils.getGUID(),ct: "websocket",aid: "219994525426954", mqtt_sid: "",cp: 3,ecp: 10,st: [],pm: [],dc: "",no_auto_fg: true,gas: null,pack: []};
35
+ var cookies = ctx.jar.getCookies('https://www.facebook.com').join("; ");
36
+
37
+ var host;
38
+ if (ctx.mqttEndpoint) host = `${ctx.mqttEndpoint}&sid=${sessionID}`;
39
+ else if (ctx.region) host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLocaleLowerCase()}&sid=${sessionID}`;
40
+ else host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}`;
41
+
42
+ var options = {
43
+ clientId: "mqttwsclient",
44
+ protocolId: 'MQIsdp',
45
+ protocolVersion: 3,
46
+ username: JSON.stringify(username),
47
+ clean: true,
48
+ wsOptions: {
49
+ headers: {
50
+ 'Cookie': cookies,
51
+ 'Origin': 'https://www.facebook.com',
52
+ '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'),
53
+ 'Referer': 'https://www.facebook.com/',
54
+ 'Host': new URL(host).hostname //'edge-chat.facebook.com'
55
+ },
56
+ origin: 'https://www.facebook.com',
57
+ protocolVersion: 13
58
+ },
59
+ keepalive: 10,
60
+ reschedulePings: true,
61
+ connectTimeout: 10000,
62
+ reconnectPeriod: 1000
63
+ };
64
+
65
+ if (typeof ctx.globalOptions.proxy != "undefined") {
66
+ var agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
67
+ options.wsOptions.agent = agent;
68
+ }
69
+ ctx.mqttClient = new mqtt.Client(_ => websocket(host, options.wsOptions), options);
70
+ var mqttClient = ctx.mqttClient;
71
+
72
+ if (process.env.OnStatus == undefined && global.Fca.Require.FastConfig.RestartMQTT_Minutes != 0) {
73
+ setInterval(() => {
74
+ mqttClient.end();
75
+ mqttClient = null;
76
+ ctx.mqttClient = null;
77
+ global.Fca.Require.logger.Normal('Restarting MQTT Client...')
78
+ restartMQTT();
79
+ }, global.Fca.Require.FastConfig.RestartMQTT_Minutes * 60000);
80
+ }
81
+
82
+ function restartMQTT() {
83
+ global.Fca.Require.logger.Normal('Restarting MQTT Client Sucessfully');
84
+
85
+ const newClient = new mqtt.Client(_ => websocket(host, options.wsOptions), options);
86
+
87
+ mqttClient = newClient;
88
+ ctx.mqttClient = newClient;
89
+
90
+ mqttClient.on('connect', function () {
91
+
92
+ topics.forEach(topicsub => mqttClient.subscribe(topicsub));
93
+
94
+ var topic;
95
+ var queue = {
96
+ sync_api_version: 11,
97
+ max_deltas_able_to_process: 100,
98
+ delta_batch_size: 500,
99
+ encoding: "JSON",
100
+ entity_fbid: ctx.userID,
101
+ };
102
+
103
+ if (ctx.syncToken) {
104
+ topic = "/messenger_sync_get_diffs";
105
+ queue.last_seq_id = ctx.lastSeqId;
106
+ queue.sync_token = ctx.syncToken;
107
+ } else {
108
+ topic = "/messenger_sync_create_queue";
109
+ queue.initial_titan_sequence_id = ctx.lastSeqId;
110
+ queue.device_params = null;
111
+ }
112
+ mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
113
+
114
+ var rTimeout = setTimeout(function () {
115
+ mqttClient.end();
116
+ getSeqID();
117
+ }, 3000);
118
+
119
+ ctx.tmsWait = function () {
120
+ clearTimeout(rTimeout);
121
+ ctx.globalOptions.emitReady ? globalCallback({type: "ready",error: null}) : '';
122
+ delete ctx.tmsWait;
123
+ };
124
+ });
125
+
126
+ mqttClient.on('message', function (topic, message, _packet) {
127
+ const jsonMessage = JSON.parse(message.toString());
128
+ if (topic === "/t_ms") {
129
+ if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
130
+
131
+ if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
132
+ ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
133
+ ctx.syncToken = jsonMessage.syncToken;
134
+ }
135
+
136
+ if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
137
+ //If it contains more than 1 delta
138
+ for (var i in jsonMessage.deltas) {
139
+ var delta = jsonMessage.deltas[i];
140
+ parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
141
+ }
142
+ } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
143
+ var typ = {
144
+ type: "typ",
145
+ isTyping: !!jsonMessage.state,
146
+ from: jsonMessage.sender_fbid.toString(),
147
+ threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
148
+ };
149
+ (function () { globalCallback(null, typ); })();
150
+ } else if (topic === "/orca_presence") {
151
+ if (!ctx.globalOptions.updatePresence) {
152
+ for (var i in jsonMessage.list) {
153
+ var data = jsonMessage.list[i];
154
+ var userID = data["u"];
155
+
156
+ var presence = {
157
+ type: "presence",
158
+ userID: userID.toString(),
159
+ //Convert to ms
160
+ timestamp: data["l"] * 1000,
161
+ statuses: data["p"]
162
+ };
163
+ (function () { globalCallback(null, presence); })();
164
+ }
165
+ }
166
+ }
167
+
168
+ });
169
+ newClient.on('error', (error) => {
170
+ console.log(`MQTT client error: ${error}`);
171
+ newClient.end(); // Close the new client connection
172
+ setTimeout(restartMQTT, 5000); // Wait for 5 seconds before retrying
173
+ });
174
+ mqttClient = newClient;
175
+ return ctx.mqttClient = newClient;
176
+ }
177
+
178
+ mqttClient.on('error', function (err) {
179
+ log.error("listenMqtt", err);
180
+ mqttClient.end();
181
+ if (ctx.globalOptions.autoReconnect) getSeqID();
182
+ else {
183
+ globalCallback({ type: "stop_listen", error: "Server Đã Sập - Auto Restart" }, null);
184
+ return process.exit(1);
185
+ }
186
+ });
187
+
188
+ mqttClient.on('connect', function () {
189
+ if (process.env.OnStatus == undefined) {
190
+ global.Fca.Require.logger.Normal(global.Fca.Data.PremText || "Hiện Status Lỗi :s")
191
+ if (Number(global.Fca.Require.FastConfig.AutoRestartMinutes) == 0) {
192
+ // something
193
+ }
194
+ else if (Number(global.Fca.Require.FastConfig.AutoRestartMinutes < 10)) {
195
+ log.warn("AutoRestartMinutes","The number of minutes to automatically restart must be more than 10 minutes");
196
+ }
197
+ else if (Number(global.Fca.Require.FastConfig.AutoRestartMinutes) < 0) {
198
+ log.warn("AutoRestartMinutes","Invalid auto-restart minutes!");
199
+ }
200
+ else {
201
+ global.Fca.Require.logger.Normal(global.Fca.getText(global.Fca.Require.Language.Src.AutoRestart,global.Fca.Require.FastConfig.AutoRestartMinutes));
202
+ global.Fca.Require.logger.Normal("Auto Restart MQTT Client After: " + global.Fca.Require.FastConfig.RestartMQTT_Minutes + " Minutes");
203
+ setInterval(() => {
204
+ global.Fca.Require.logger.Normal(global.Fca.Require.Language.Src.OnRestart);
205
+ process.exit(1);
206
+ }, Number(global.Fca.Require.FastConfig.AutoRestartMinutes) * 60000);
207
+ }
208
+ require('../broadcast');
209
+ process.env.OnStatus = true;
210
+ }
211
+
212
+ topics.forEach(topicsub => mqttClient.subscribe(topicsub));
213
+
214
+ var topic;
215
+ var queue = {
216
+ sync_api_version: 11,
217
+ max_deltas_able_to_process: 100,
218
+ delta_batch_size: 500,
219
+ encoding: "JSON",
220
+ entity_fbid: ctx.userID,
221
+ };
222
+
223
+ if (ctx.syncToken) {
224
+ topic = "/messenger_sync_get_diffs";
225
+ queue.last_seq_id = ctx.lastSeqId;
226
+ queue.sync_token = ctx.syncToken;
227
+ } else {
228
+ topic = "/messenger_sync_create_queue";
229
+ queue.initial_titan_sequence_id = ctx.lastSeqId;
230
+ queue.device_params = null;
231
+ }
232
+ mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
233
+
234
+ var rTimeout = setTimeout(function () {
235
+ mqttClient.end();
236
+ getSeqID();
237
+ }, 3000);
238
+
239
+ ctx.tmsWait = function () {
240
+ clearTimeout(rTimeout);
241
+ ctx.globalOptions.emitReady ? globalCallback({type: "ready",error: null}) : '';
242
+ delete ctx.tmsWait;
243
+ };
244
+ });
245
+
246
+ mqttClient.on('message', function (topic, message, _packet) {
247
+ const jsonMessage = JSON.parse(message.toString());
248
+ if (topic === "/t_ms") {
249
+ if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
250
+
251
+ if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
252
+ ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
253
+ ctx.syncToken = jsonMessage.syncToken;
254
+ }
255
+
256
+ if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
257
+ //If it contains more than 1 delta
258
+ for (var i in jsonMessage.deltas) {
259
+ var delta = jsonMessage.deltas[i];
260
+ parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
261
+ }
262
+ } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
263
+ var typ = {
264
+ type: "typ",
265
+ isTyping: !!jsonMessage.state,
266
+ from: jsonMessage.sender_fbid.toString(),
267
+ threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
268
+ };
269
+ (function () { globalCallback(null, typ); })();
270
+ } else if (topic === "/orca_presence") {
271
+ if (!ctx.globalOptions.updatePresence) {
272
+ for (var i in jsonMessage.list) {
273
+ var data = jsonMessage.list[i];
274
+ var userID = data["u"];
275
+
276
+ var presence = {
277
+ type: "presence",
278
+ userID: userID.toString(),
279
+ //Convert to ms
280
+ timestamp: data["l"] * 1000,
281
+ statuses: data["p"]
282
+ };
283
+ (function () { globalCallback(null, presence); })();
284
+ }
285
+ }
286
+ }
287
+
288
+ });
289
+
290
+ process.on('SIGINT', function () {
291
+ LogUptime();process.kill(process.pid);
292
+ });
293
+
294
+ process.on('exit', (code) => {
295
+ LogUptime();
296
+ });
297
+
298
+ }
299
+
300
+ function LogUptime() {
301
+ var uptime = process.uptime();
302
+ var { join } = require('path');
303
+ if (global.Fca.Require.fs.existsSync(join(__dirname, '../CountTime.json'))) {
304
+ var Time1 = (Number(global.Fca.Require.fs.readFileSync(join(__dirname, '../CountTime.json'), 'utf8')) || 0);
305
+ global.Fca.Require.fs.writeFileSync(join(__dirname, '../CountTime.json'), String(Number(uptime) + Time1), 'utf8');
306
+ }
307
+ else {
308
+ var Time1 = 0;
309
+ global.Fca.Require.fs.writeFileSync(join(__dirname, '../CountTime.json'), String(Number(uptime) + Time1), 'utf8');
310
+ }
311
+ }
312
+ function parseDelta(defaultFuncs, api, ctx, globalCallback, v) {
313
+ if (v.delta.class == "NewMessage") {
314
+ //Not tested for pages
315
+ if (ctx.globalOptions.pageID && ctx.globalOptions.pageID != v.queue) return;
316
+
317
+ (function resolveAttachmentUrl(i) {
318
+ if (v.delta.attachments && (i == v.delta.attachments.length)) {
319
+ var fmtMsg;
320
+ try {
321
+ fmtMsg = utils.formatDeltaMessage(v);
322
+ } catch (err) {
323
+ return log.error("Lỗi Nhẹ", err);
324
+ }
325
+ global.Fca.Data.event = fmtMsg;
326
+ try {
327
+ if (process.env.HalzionVersion == 1973) {
328
+ var { updateMessageCount,getData,hasData } = require('../Extra/ExtraGetThread');
329
+ if (hasData(fmtMsg.threadID)) {
330
+ var x = getData(fmtMsg.threadID);
331
+ x.messageCount+=1;
332
+ updateMessageCount(fmtMsg.threadID,x);
333
+ }
334
+ }
335
+ }
336
+ catch (e) {
337
+ //temp
338
+ }
339
+ if (fmtMsg)
340
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
341
+
342
+ return !ctx.globalOptions.selfListen && fmtMsg.senderID === ctx.userID ? undefined : (function () { globalCallback(null, fmtMsg); })();
343
+ } else {
344
+ if (v.delta.attachments && (v.delta.attachments[i].mercury.attach_type == "photo")) {
345
+ api.resolvePhotoUrl(v.delta.attachments[i].fbid, (err, url) => {
346
+ if (!err) v.delta.attachments[i].mercury.metadata.url = url;
347
+ return resolveAttachmentUrl(i + 1);
348
+ });
349
+ } else return resolveAttachmentUrl(i + 1);
350
+ }
351
+ })(0);
352
+ }
353
+
354
+ if (v.delta.class == "ClientPayload") {
355
+ var clientPayload = utils.decodeClientPayload(v.delta.payload);
356
+ if (clientPayload && clientPayload.deltas) {
357
+ for (var i in clientPayload.deltas) {
358
+ var delta = clientPayload.deltas[i];
359
+ if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
360
+ (function () {
361
+ globalCallback(null, {
362
+ type: "message_reaction",
363
+ threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
364
+ messageID: delta.deltaMessageReaction.messageId,
365
+ reaction: delta.deltaMessageReaction.reaction,
366
+ senderID: delta.deltaMessageReaction.senderId.toString(),
367
+ userID: delta.deltaMessageReaction.userId.toString()
368
+ });
369
+ })();
370
+ } else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
371
+ (function () {
372
+ globalCallback(null, {
373
+ type: "message_unsend",
374
+ threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
375
+ messageID: delta.deltaRecallMessageData.messageID,
376
+ senderID: delta.deltaRecallMessageData.senderID.toString(),
377
+ deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
378
+ timestamp: delta.deltaRecallMessageData.timestamp
379
+ });
380
+ })();
381
+ } else if (delta.deltaMessageReply) {
382
+ //Mention block - #1
383
+ var mdata =
384
+ delta.deltaMessageReply.message === undefined ? [] :
385
+ delta.deltaMessageReply.message.data === undefined ? [] :
386
+ delta.deltaMessageReply.message.data.prng === undefined ? [] :
387
+ JSON.parse(delta.deltaMessageReply.message.data.prng);
388
+ var m_id = mdata.map(u => u.i);
389
+ var m_offset = mdata.map(u => u.o);
390
+ var m_length = mdata.map(u => u.l);
391
+
392
+ var mentions = {};
393
+
394
+ 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]);
395
+ //Mention block - 1#
396
+ var callbackToReturn = {
397
+ type: "message_reply",
398
+ threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
399
+ messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
400
+ senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
401
+ attachments: delta.deltaMessageReply.message.attachments.map(function (att) {
402
+ var mercury = JSON.parse(att.mercuryJSON);
403
+ Object.assign(att, mercury);
404
+ return att;
405
+ }).map(att => {
406
+ var x;
407
+ try {
408
+ x = utils._formatAttachment(att);
409
+ } catch (ex) {
410
+ x = att;
411
+ x.error = ex;
412
+ x.type = "unknown";
413
+ }
414
+ return x;
415
+ }),
416
+ args: (delta.deltaMessageReply.message.body || "").trim().split(/\s+/),
417
+ body: (delta.deltaMessageReply.message.body || ""),
418
+ isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
419
+ mentions: mentions,
420
+ timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
421
+ participantIDs: (delta.deltaMessageReply.message.participants || []).map(e => e.toString())
422
+ };
423
+
424
+ if (delta.deltaMessageReply.repliedToMessage) {
425
+ //Mention block - #2
426
+ mdata =
427
+ delta.deltaMessageReply.repliedToMessage === undefined ? [] :
428
+ delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
429
+ delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
430
+ JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
431
+ m_id = mdata.map(u => u.i);
432
+ m_offset = mdata.map(u => u.o);
433
+ m_length = mdata.map(u => u.l);
434
+
435
+ var rmentions = {};
436
+
437
+ 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]);
438
+ //Mention block - 2#
439
+ callbackToReturn.messageReply = {
440
+ threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
441
+ messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
442
+ senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
443
+ attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function (att) {
444
+ var mercury = JSON.parse(att.mercuryJSON);
445
+ Object.assign(att, mercury);
446
+ return att;
447
+ }).map(att => {
448
+ var x;
449
+ try {
450
+ x = utils._formatAttachment(att);
451
+ } catch (ex) {
452
+ x = att;
453
+ x.error = ex;
454
+ x.type = "unknown";
455
+ }
456
+ return x;
457
+ }),
458
+ args: (delta.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
459
+ body: delta.deltaMessageReply.repliedToMessage.body || "",
460
+ isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
461
+ mentions: rmentions,
462
+ timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp,
463
+ participantIDs: (delta.deltaMessageReply.repliedToMessage.participants || []).map(e => e.toString())
464
+ };
465
+ } else if (delta.deltaMessageReply.replyToMessageId) {
466
+ return defaultFuncs
467
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
468
+ "av": ctx.globalOptions.pageID,
469
+ "queries": JSON.stringify({
470
+ "o0": {
471
+ //Using the same doc_id as forcedFetch
472
+ "doc_id": "2848441488556444",
473
+ "query_params": {
474
+ "thread_and_message_id": {
475
+ "thread_id": callbackToReturn.threadID,
476
+ "message_id": delta.deltaMessageReply.replyToMessageId.id,
477
+ }
478
+ }
479
+ }
480
+ })
481
+ })
482
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
483
+ .then((resData) => {
484
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
485
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
486
+ var fetchData = resData[0].o0.data.message;
487
+ var mobj = {};
488
+ 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);
489
+
490
+ callbackToReturn.messageReply = {
491
+ type: "Message",
492
+ threadID: callbackToReturn.threadID,
493
+ messageID: fetchData.message_id,
494
+ senderID: fetchData.message_sender.id.toString(),
495
+ attachments: fetchData.message.blob_attachment.map(att => {
496
+ var x;
497
+ try {
498
+ x = utils._formatAttachment({ blob_attachment: att });
499
+ } catch (ex) {
500
+ x = att;
501
+ x.error = ex;
502
+ x.type = "unknown";
503
+ }
504
+ return x;
505
+ }),
506
+ args: (fetchData.message.text || "").trim().split(/\s+/) || [],
507
+ body: fetchData.message.text || "",
508
+ isGroup: callbackToReturn.isGroup,
509
+ mentions: mobj,
510
+ timestamp: parseInt(fetchData.timestamp_precise)
511
+ };
512
+ })
513
+ .catch(err => log.error("forcedFetch", err))
514
+ .finally(function () {
515
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
516
+ !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
517
+ });
518
+ } else callbackToReturn.delta = delta;
519
+
520
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
521
+
522
+ return !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
523
+ }
524
+ }
525
+ return;
526
+ }
527
+ }
528
+
529
+ if (v.delta.class !== "NewMessage" && !ctx.globalOptions.listenEvents) return;
530
+ switch (v.delta.class) {
531
+ case "ReadReceipt":
532
+ var fmtMsg;
533
+ try {
534
+ fmtMsg = utils.formatDeltaReadReceipt(v.delta);
535
+ } catch (err) {
536
+ return log.error("Lỗi Nhẹ", err);
537
+ }
538
+ return (function () { globalCallback(null, fmtMsg); })();
539
+ case "AdminTextMessage":
540
+ switch (v.delta.type) {
541
+ case "joinable_group_link_mode_change":
542
+ case "magic_words":
543
+ case "change_thread_theme":
544
+ case "change_thread_icon":
545
+ case "change_thread_nickname":
546
+ case "change_thread_admins":
547
+ case "change_thread_approval_mode":
548
+ case "group_poll":
549
+ case "messenger_call_log":
550
+ case "participant_joined_group_call":
551
+ var fmtMsg;
552
+ try {
553
+ fmtMsg = utils.formatDeltaEvent(v.delta);
554
+ } catch (err) {
555
+ return log.error("Lỗi Nhẹ", err);
556
+ }
557
+ return (function () { globalCallback(null, fmtMsg); })();
558
+ default:
559
+ return;
560
+ }
561
+ //For group images
562
+ case "ForcedFetch":
563
+ if (!v.delta.threadKey) return;
564
+ var mid = v.delta.messageId;
565
+ var tid = v.delta.threadKey.threadFbId;
566
+ if (mid && tid) {
567
+ const form = {
568
+ "av": ctx.globalOptions.pageID,
569
+ "queries": JSON.stringify({
570
+ "o0": {
571
+ //This doc_id is valid as of March 25, 2020
572
+ "doc_id": "2848441488556444",
573
+ "query_params": {
574
+ "thread_and_message_id": {
575
+ "thread_id": tid.toString(),
576
+ "message_id": mid,
577
+ }
578
+ }
579
+ }
580
+ })
581
+ };
582
+
583
+ defaultFuncs
584
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
585
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
586
+ .then((resData) => {
587
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
588
+
589
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
590
+
591
+ var fetchData = resData[0].o0.data.message;
592
+
593
+ if (utils.getType(fetchData) == "Object") {
594
+ log.info("forcedFetch", fetchData);
595
+ switch (fetchData.__typename) {
596
+ case "ThreadImageMessage":
597
+ (!ctx.globalOptions.selfListen &&
598
+ fetchData.message_sender.id.toString() === ctx.userID) ||
599
+ !ctx.loggedIn ?
600
+ undefined :
601
+ (function () {
602
+ globalCallback(null, {
603
+ type: "change_thread_image",
604
+ threadID: utils.formatID(tid.toString()),
605
+ snippet: fetchData.snippet,
606
+ timestamp: fetchData.timestamp_precise,
607
+ author: fetchData.message_sender.id,
608
+ image: {
609
+ attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
610
+ width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
611
+ height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
612
+ url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
613
+ }
614
+ });
615
+ })();
616
+ break;
617
+ case "UserMessage":
618
+ log.info("ff-Return", {
619
+ type: "message",
620
+ senderID: utils.formatID(fetchData.message_sender.id),
621
+ body: fetchData.message.text || "",
622
+ threadID: utils.formatID(tid.toString()),
623
+ messageID: fetchData.message_id,
624
+ attachments: [{
625
+ type: "share",
626
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
627
+ url: fetchData.extensible_attachment.story_attachment.url,
628
+
629
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
630
+ description: fetchData.extensible_attachment.story_attachment.description.text,
631
+ source: fetchData.extensible_attachment.story_attachment.source,
632
+
633
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
634
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
635
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
636
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
637
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
638
+
639
+ subattachments: fetchData.extensible_attachment.subattachments,
640
+ properties: fetchData.extensible_attachment.story_attachment.properties,
641
+ }],
642
+ mentions: {},
643
+ timestamp: parseInt(fetchData.timestamp_precise),
644
+ isGroup: (fetchData.message_sender.id != tid.toString())
645
+ });
646
+ globalCallback(null, {
647
+ type: "message",
648
+ senderID: utils.formatID(fetchData.message_sender.id),
649
+ body: fetchData.message.text || "",
650
+ threadID: utils.formatID(tid.toString()),
651
+ messageID: fetchData.message_id,
652
+ attachments: [{
653
+ type: "share",
654
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
655
+ url: fetchData.extensible_attachment.story_attachment.url,
656
+
657
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
658
+ description: fetchData.extensible_attachment.story_attachment.description.text,
659
+ source: fetchData.extensible_attachment.story_attachment.source,
660
+
661
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
662
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
663
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
664
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
665
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
666
+
667
+ subattachments: fetchData.extensible_attachment.subattachments,
668
+ properties: fetchData.extensible_attachment.story_attachment.properties,
669
+ }],
670
+ mentions: {},
671
+ timestamp: parseInt(fetchData.timestamp_precise),
672
+ isGroup: (fetchData.message_sender.id != tid.toString())
673
+ });
674
+ }
675
+ } else log.error("forcedFetch", fetchData);
676
+ })
677
+ .catch((err) => log.error("forcedFetch", err));
678
+ }
679
+ break;
680
+ case "ThreadName":
681
+ case "ParticipantsAddedToGroupThread":
682
+ case "ParticipantLeftGroupThread":
683
+ var formattedEvent;
684
+ try {
685
+ formattedEvent = utils.formatDeltaEvent(v.delta);
686
+ } catch (err) {
687
+ return log.error("Lỗi Nhẹ", err);
688
+ }
689
+ return (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) || !ctx.loggedIn ? undefined : (function () { globalCallback(null, formattedEvent); })();
690
+ }
691
+ }
692
+
693
+
694
+ function markDelivery(ctx, api, threadID, messageID) {
695
+ if (threadID && messageID) {
696
+ api.markAsDelivered(threadID, messageID, (err) => {
697
+ if (err) log.error("markAsDelivered", err);
698
+ else {
699
+ if (ctx.globalOptions.autoMarkRead) {
700
+ api.markAsRead(threadID, (err) => {
701
+ if (err) log.error("markAsDelivered", err);
702
+ });
703
+ }
704
+ }
705
+ });
706
+ }
707
+ }
708
+
709
+ module.exports = function (defaultFuncs, api, ctx) {
710
+ var globalCallback = identity;
711
+ getSeqID = function getSeqID() {
712
+ ctx.t_mqttCalled = false;
713
+ defaultFuncs
714
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
715
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
716
+ .then((resData) => {
717
+ if (utils.getType(resData) != "Array") {
718
+ if (global.Fca.Require.FastConfig.AutoLogin) {
719
+ return global.Fca.Require.logger.Warning(global.Fca.Require.Language.Index.AutoLogin, function() {
720
+ return global.Fca.Action('AutoLogin');
721
+ });
722
+ }
723
+ else if (!global.Fca.Require.FastConfig.AutoLogin) {
724
+ return global.Fca.Require.logger.Error(global.Fca.Require.Language.Index.ErrAppState);
725
+ }
726
+ return;
727
+ }
728
+ else {
729
+ if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
730
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "getSeqId: there was no successful_results", res: resData };
731
+ if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
732
+ ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
733
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
734
+ }
735
+ else throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
736
+ }
737
+ })
738
+ .catch((err) => {
739
+ log.error("getSeqId", err);
740
+ if (utils.getType(err) == "Object" && err.error === global.Fca.Require.Language.Index.ErrAppState) ctx.loggedIn = false;
741
+ return globalCallback(err);
742
+ });
743
+ };
744
+
745
+ return function (callback) {
746
+ class MessageEmitter extends EventEmitter {
747
+ stopListening(callback) {
748
+ callback = callback || (() => { });
749
+ globalCallback = identity;
750
+ if (ctx.mqttClient) {
751
+ ctx.mqttClient.unsubscribe("/webrtc");
752
+ ctx.mqttClient.unsubscribe("/rtc_multi");
753
+ ctx.mqttClient.unsubscribe("/onevc");
754
+ ctx.mqttClient.publish("/browser_close", "{}");
755
+ ctx.mqttClient.end(false, function (...data) {
756
+ callback(data);
757
+ ctx.mqttClient = undefined;
758
+ });
759
+ }
760
+ }
761
+ }
762
+
763
+ var msgEmitter = new MessageEmitter();
764
+ globalCallback = (callback || function (error, message) {
765
+ if (error) return msgEmitter.emit("error", error);
766
+ msgEmitter.emit("message", message);
767
+ });
768
+
769
+ //Reset some stuff
770
+ if (!ctx.firstListen) ctx.lastSeqId = null;
771
+ ctx.syncToken = undefined;
772
+ ctx.t_mqttCalled = false;
773
+
774
+ //Same request as getThreadList
775
+ form = {
776
+ "av": ctx.globalOptions.pageID,
777
+ "queries": JSON.stringify({
778
+ "o0": {
779
+ "doc_id": "3336396659757871",
780
+ "query_params": {
781
+ "limit": 1,
782
+ "before": null,
783
+ "tags": ["INBOX"],
784
+ "includeDeliveryReceipts": false,
785
+ "includeSeqID": true
786
+ }
787
+ }
788
+ })
789
+ };
790
+
791
+ if (!ctx.firstListen || !ctx.lastSeqId) getSeqID();
792
+ else listenMqtt(defaultFuncs, api, ctx, globalCallback);
793
+ ctx.firstListen = false;
794
+ return msgEmitter;
795
+ };
796
+ };