fca-smart-shankar 10.9.1 → 13.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (169) hide show
  1. package/.replit +12 -3
  2. package/.upm/store.json +1 -1
  3. package/CHANGELOG.md +2 -0
  4. package/LICENSE-MIT +21 -0
  5. package/README.md +175 -30
  6. package/generated-icon.png +0 -0
  7. package/index.js +511 -414
  8. package/package.json +370 -85
  9. package/replit.nix +5 -0
  10. package/shankar-fca.json +4 -0
  11. package/src/addExternalModule.js +14 -5
  12. package/src/addUserToGroup.js +56 -20
  13. package/src/changeAdminStatus.js +44 -20
  14. package/src/changeArchivedStatus.js +25 -11
  15. package/src/changeAvatar.js +136 -0
  16. package/src/{changeAvt.js → changeAvatarV2.js} +3 -2
  17. package/src/changeBio.js +26 -15
  18. package/src/changeBlockedStatus.js +21 -8
  19. package/src/changeBlockedStatusMqtt.js +80 -0
  20. package/src/changeCover.js +73 -0
  21. package/src/changeGroupImage.js +53 -24
  22. package/src/changeName.js +79 -0
  23. package/src/changeNickname.js +27 -13
  24. package/src/changeThreadColor.js +22 -19
  25. package/src/changeThreadEmoji.js +24 -11
  26. package/src/changeUsername.js +59 -0
  27. package/src/createCommentPost.js +230 -0
  28. package/src/createNewGroup.js +38 -20
  29. package/src/createPoll.js +27 -16
  30. package/src/createPost.js +277 -0
  31. package/src/data/getThreadInfo.json +1 -0
  32. package/src/deleteMessage.js +24 -13
  33. package/src/deleteThread.js +25 -12
  34. package/src/editMessage.js +71 -53
  35. package/src/editMessageOld.js +67 -0
  36. package/src/follow.js +74 -0
  37. package/src/forwardAttachment.js +27 -15
  38. package/src/getAccess.js +112 -0
  39. package/src/getAvatarUser.js +78 -0
  40. package/src/getCurrentUserID.js +1 -1
  41. package/src/getEmojiUrl.js +10 -8
  42. package/src/getFriendsList.js +25 -15
  43. package/src/getMessage.js +813 -81
  44. package/src/getRegion.js +7 -0
  45. package/src/getThreadHistory.js +241 -98
  46. package/src/getThreadHistoryDeprecated.js +93 -0
  47. package/src/getThreadInfo.js +90 -287
  48. package/src/getThreadInfoDeprecated.js +80 -0
  49. package/src/getThreadList.js +214 -157
  50. package/src/getThreadListDeprecated.js +75 -0
  51. package/src/getThreadPictures.js +39 -19
  52. package/src/getUID.js +113 -50
  53. package/src/getUserID.js +18 -14
  54. package/src/getUserInfo.js +65 -71
  55. package/src/handleFriendRequest.js +24 -13
  56. package/src/handleMessageRequest.js +36 -20
  57. package/src/httpGet.js +34 -18
  58. package/src/httpPost.js +35 -18
  59. package/src/httpPostFormData.js +53 -24
  60. package/src/listenMqtt.js +680 -944
  61. package/src/listenNotification.js +85 -0
  62. package/src/logout.js +22 -15
  63. package/src/markAsDelivered.js +25 -15
  64. package/src/markAsRead.js +45 -27
  65. package/src/markAsReadAll.js +21 -14
  66. package/src/markAsSeen.js +28 -18
  67. package/src/muteThread.js +17 -12
  68. package/src/pinMessage.js +59 -0
  69. package/src/refreshFb_dtsg.js +89 -0
  70. package/src/removeUserFromGroup.js +47 -17
  71. package/src/resolvePhotoUrl.js +21 -13
  72. package/src/searchForThread.js +23 -13
  73. package/src/searchStickers.js +53 -0
  74. package/src/sendMessage.js +178 -117
  75. package/src/sendMessageMqtt.js +322 -0
  76. package/src/sendTypingIndicator.js +46 -16
  77. package/src/sendTypingIndicatorV2.js +28 -0
  78. package/src/setMessageReaction.js +33 -20
  79. package/src/setMessageReactionMqtt.js +62 -0
  80. package/src/setPostReaction.js +105 -95
  81. package/src/setProfileGuard.js +45 -0
  82. package/src/setStoryReaction.js +64 -0
  83. package/src/setTitle.js +34 -18
  84. package/src/shareContact.js +92 -37
  85. package/src/shareLink.js +5 -4
  86. package/src/stopListenMqtt.js +26 -0
  87. package/src/threadColors.js +110 -18
  88. package/src/unfriend.js +18 -9
  89. package/src/unsendMessage.js +31 -34
  90. package/src/uploadAttachment.js +94 -0
  91. package/test/data/shareAttach.js +1 -1
  92. package/test/test.js +1 -1
  93. package/utils.js +1393 -2918
  94. package/.cache/replit/env/latest +0 -56
  95. package/.cache/replit/env/latest.json +0 -1
  96. package/.cache/replit/modules/nodejs-14.res +0 -1
  97. package/.cache/replit/modules/replit.res +0 -1
  98. package/.cache/replit/modules/web.res +0 -1
  99. package/.cache/replit/modules.stamp +0 -0
  100. package/.cache/typescript/5.5/package.json +0 -1
  101. package/.config/configstore/update-notifier-npm.json +0 -4
  102. package/.gitattributes +0 -2
  103. package/Extra/Balancer.js +0 -49
  104. package/Extra/Bypass/956/index.js +0 -234
  105. package/Extra/Bypass/test/aaaa.json +0 -170
  106. package/Extra/Bypass/test/index.js +0 -188
  107. package/Extra/Database/index.js +0 -469
  108. package/Extra/ExtraAddons.js +0 -82
  109. package/Extra/ExtraFindUID.js +0 -62
  110. package/Extra/ExtraGetThread.js +0 -365
  111. package/Extra/ExtraScreenShot.js +0 -430
  112. package/Extra/ExtraUptimeRobot.js +0 -38
  113. package/Extra/Html/Classic/script.js +0 -119
  114. package/Extra/Html/Classic/style.css +0 -8
  115. package/Extra/Security/AES_256_GCM/index.js +0 -0
  116. package/Extra/Security/Base/Step_1.js +0 -6
  117. package/Extra/Security/Base/Step_2.js +0 -22
  118. package/Extra/Security/Base/Step_3.js +0 -22
  119. package/Extra/Security/Base/index.js +0 -191
  120. package/Extra/Security/Index.js +0 -5
  121. package/Extra/Security/Step_1.js +0 -6
  122. package/Extra/Security/Step_2.js +0 -22
  123. package/Extra/Security/Step_3.js +0 -22
  124. package/Extra/Src/Change_Environment.js +0 -24
  125. package/Extra/Src/Check_Update.js +0 -67
  126. package/Extra/Src/History.js +0 -115
  127. package/Extra/Src/Instant_Update.js +0 -65
  128. package/Extra/Src/Last-Run.js +0 -65
  129. package/Extra/Src/Premium.js +0 -81
  130. package/Extra/Src/Release_Memory.js +0 -160
  131. package/Extra/Src/Websocket.js +0 -213
  132. package/Extra/Src/image/checkmate.jpg +0 -0
  133. package/Extra/Src/test.js +0 -28
  134. package/Extra/Src/uuid.js +0 -137
  135. package/Func/AcceptAgreement.js +0 -31
  136. package/Func/ClearCache.js +0 -64
  137. package/Func/ReportV1.js +0 -54
  138. package/LICENSE +0 -678
  139. package/Language/index.json +0 -228
  140. package/Main.js +0 -1444
  141. package/SECURITY.md +0 -18
  142. package/broadcast.js +0 -44
  143. package/logger.js +0 -66
  144. package/src/Dev_Horizon_Data.js +0 -125
  145. package/src/Dev_getThreadInfoOLD.js +0 -422
  146. package/src/Dev_shareTest2.js +0 -68
  147. package/src/Dev_shareTest3.js +0 -71
  148. package/src/Premium.js +0 -25
  149. package/src/Screenshot.js +0 -83
  150. package/src/getAccessToken.js +0 -28
  151. package/src/getThreadInfoOLD.js +0 -422
  152. package/src/getThreadMain.js +0 -220
  153. package/src/getUserInfoMain.js +0 -65
  154. package/src/getUserInfoV2.js +0 -32
  155. package/src/getUserInfoV3.js +0 -63
  156. package/src/getUserInfoV4.js +0 -55
  157. package/src/getUserInfoV5.js +0 -61
  158. package/src/listenMqttV1.js +0 -846
  159. package/src/sendMqttMessage.js +0 -71
  160. package/src/unsendMqttMessage.js +0 -66
  161. package/test/Database_Test.js +0 -4
  162. package/test/Db2.js +0 -530
  163. package/test/Shankar_Database/A_README.md +0 -1
  164. package/test/Shankar_Database/Database.db +0 -0
  165. package/test/env/.env +0 -0
  166. package/test/example-db.db +0 -0
  167. package/test/memoryleak.js +0 -18
  168. package/test/testname.js +0 -1342
  169. package/test/testv2.js +0 -3
package/src/listenMqtt.js CHANGED
@@ -1,956 +1,692 @@
1
- /* eslint-disable linebreak-style */
2
- 'use strict';
3
-
4
- const utils = require('../utils');
5
- const log = require('npmlog');
6
- const mqtt = require('mqtt');
7
- const WebSocket = require('ws');
8
- const HttpsProxyAgent = require('https-proxy-agent');
9
- const EventEmitter = require('events');
10
- const Duplexify = require('duplexify');
11
- const {
12
- Transform
13
- } = require('stream');
14
- var identity = function() {};
15
- var form = {};
16
- var getSeqID = function() {};
17
- global.Fca.Data.MsgCount = new Map();
18
- global.Fca.Data.event = new Map();
19
-
20
- const topics = ['/ls_req', '/ls_resp', '/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'];
21
-
22
- let WebSocket_Global;
23
-
24
- function buildProxy() {
25
- const Proxy = new Transform({
26
- objectMode: false,
27
- transform(chunk, enc, next) {
28
- if (WebSocket_Global.readyState !== WebSocket_Global.OPEN) {
29
- return next();
30
- }
31
-
32
- let data;
33
- if (typeof chunk === 'string') {
34
- data = Buffer.from(chunk, 'utf8');
35
- } else {
36
- data = chunk;
37
- }
38
-
39
- WebSocket_Global.send(data);
40
- next();
41
- },
42
- flush(done) {
43
- WebSocket_Global.close();
44
- done();
45
- },
46
- writev(chunks, cb) {
47
- const buffers = chunks.map(({ chunk }) => {
48
- if (typeof chunk === 'string') {
49
- return Buffer.from(chunk, 'utf8');
50
- }
51
- return chunk;
52
- });
53
- this._write(Buffer.concat(buffers), 'binary', cb);
54
- },
55
- });
56
- return Proxy;
57
- }
58
-
59
- function buildStream(options, WebSocket, Proxy) {
60
- const Stream = Duplexify(undefined, undefined, options);
61
- Stream.socket = WebSocket;
62
-
63
- WebSocket.onclose = () => {
64
- Stream.end();
65
- Stream.destroy();
66
- };
67
1
 
68
- WebSocket.onerror = (err) => {
69
- Stream.destroy(err);
70
- };
71
2
 
72
- WebSocket.onmessage = (event) => {
73
- const data = event.data instanceof ArrayBuffer ? Buffer.from(event.data) : Buffer.from(event.data, 'utf8');
74
- Stream.push(data);
75
- };
76
-
77
- WebSocket.onopen = () => {
78
- Stream.setReadable(Proxy);
79
- Stream.setWritable(Proxy);
80
- Stream.emit('connect');
81
- };
82
-
83
- WebSocket_Global = WebSocket;
84
- Proxy.on('close', () => WebSocket.close());
3
+ "use strict";
4
+ var utils = require("../utils");
5
+ var log = require("npmlog");
6
+ var mqtt = require('mqtt');
7
+ var websocket = require('websocket-stream');
8
+ var HttpsProxyAgent = require('https-proxy-agent');
9
+ const EventEmitter = require('events');
85
10
 
86
- return Stream;
87
- }
11
+ var identity = function () { };
12
+ var form = {};
13
+ var getSeqID = function () { };
14
+
15
+ var topics = [
16
+ "/legacy_web",
17
+ "/webrtc",
18
+ "/rtc_multi",
19
+ "/onevc",
20
+ "/br_sr", //Notification
21
+ //Need to publish /br_sr right after this
22
+ "/sr_res",
23
+ "/t_ms",
24
+ "/thread_typing",
25
+ "/orca_typing_notifications",
26
+ "/notify_disconnect",
27
+ //Need to publish /messenger_sync_create_queue right after this
28
+ "/orca_presence",
29
+ //Will receive /sr_res right here.
30
+
31
+ "/inbox",
32
+ "/mercury",
33
+ "/messaging_events",
34
+ "/orca_message_notifications",
35
+ "/pp",
36
+ "/webrtc_response",
37
+ ];
88
38
 
89
39
  function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
90
- const chatOn = ctx.globalOptions.online;
91
- const foreground = false;
92
-
93
- const sessionID = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER) + 1;
94
- const GUID = utils.getGUID()
95
- const username = {
96
- u: ctx.userID,
97
- s: sessionID,
98
- chat_on: chatOn,
99
- fg: foreground,
100
- d: GUID,
101
- ct: 'websocket',
102
- aid: '219994525426954',
103
- aids: null,
104
- mqtt_sid: '',
105
- cp: 3,
106
- ecp: 10,
107
- st: [],
108
- pm: [],
109
- dc: '',
110
- no_auto_fg: true,
111
- gas: null,
112
- pack: [],
113
- p: null,
114
- php_override: ""
115
- };
116
-
117
- const cookies = ctx.jar.getCookies('https://www.facebook.com').join('; ');
118
-
119
- let host;
120
- if (ctx.mqttEndpoint) {
121
- host = `${ctx.mqttEndpoint}&sid=${sessionID}&cid=${GUID}`;
122
- } else if (ctx.region) {
123
- host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLowerCase()}&sid=${sessionID}&cid=${GUID}`;
124
- } else {
125
- host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}&cid=${GUID}`;
126
- }
127
-
128
- const options = {
129
- clientId: 'mqttwsclient',
130
- protocolId: 'MQIsdp',
131
- protocolVersion: 3,
132
- username: JSON.stringify(username),
133
- clean: true,
134
- wsOptions: {
135
- headers: {
136
- Cookie: cookies,
137
- Origin: 'https://www.facebook.com',
138
- 'User-Agent': ctx.globalOptions.userAgent,
139
- Referer: 'https://www.facebook.com/',
140
- Host: new URL(host).hostname,
141
- },
142
- origin: 'https://www.facebook.com',
143
- protocolVersion: 13,
144
- binaryType: 'arraybuffer',
145
- },
146
- keepalive: 60,
147
- reschedulePings: true,
148
- reconnectPeriod: 3,
149
- };
150
-
151
- if (ctx.globalOptions.proxy !== undefined) {
152
- const agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
153
- options.wsOptions.agent = agent;
154
- }
155
-
156
- ctx.mqttClient = new mqtt.Client(() => buildStream(options, new WebSocket(host, options.wsOptions), buildProxy()), options);
157
- global.mqttClient = ctx.mqttClient;
158
-
159
- global.mqttClient.on('error', (err) => {
160
- log.error('listenMqtt', err);
161
- global.mqttClient.end();
162
-
163
- if (ctx.globalOptions.autoReconnect) {
164
- getSeqID();
165
- } else {
166
- globalCallback({
167
- type: 'stop_listen',
168
- error: 'Server Đã Sập - Auto Restart'
169
- }, null);
170
- return process.exit(1);
171
- }
172
- });
173
-
174
- global.mqttClient.on('connect', () => {
175
- if (!global.Fca.Data.Setup || global.Fca.Data.Setup === undefined) {
176
- if (global.Fca.Require.FastConfig.RestartMQTT_Minutes !== 0 && global.Fca.Data.StopListening !== true) {
177
- global.Fca.Data.Setup = true;
178
- setTimeout(() => {
179
- global.Fca.Require.logger.Warning('Closing MQTT Client...');
180
- ctx.mqttClient.end();
181
- global.Fca.Require.logger.Warning('Reconnecting MQTT Client...');
182
- global.Fca.Data.Setup = false;
183
- getSeqID();
184
- }, Number(global.Fca.Require.FastConfig.RestartMQTT_Minutes) * 60 * 1000);
185
- }
186
- }
187
-
188
- if (process.env.OnStatus === undefined) {
189
- global.Fca.Require.logger.Normal('Bạn Đang Sài Phiên Bản: Premium Access');
190
-
191
- if (Number(global.Fca.Require.FastConfig.AutoRestartMinutes) === 0) {
192
- // something
193
- } else if (Number(global.Fca.Require.FastConfig.AutoRestartMinutes) < 10) {
194
- log.warn('AutoRestartMinutes', 'The number of minutes to automatically restart must be more than 10 minutes');
195
- } else if (Number(global.Fca.Require.FastConfig.AutoRestartMinutes) < 0) {
196
- log.warn('AutoRestartMinutes', 'Invalid auto-restart minutes!');
197
- } else {
198
- global.Fca.Require.logger.Normal(global.Fca.getText(global.Fca.Require.Language.Src.AutoRestart, global.Fca.Require.FastConfig.AutoRestartMinutes));
199
- global.Fca.Require.logger.Normal(`Auto Restart MQTT Client After: ${global.Fca.Require.FastConfig.RestartMQTT_Minutes} Minutes`);
200
- setTimeout(() => {
201
- global.Fca.Require.logger.Normal(global.Fca.Require.Language.Src.OnRestart);
202
- process.exit(1);
203
- }, Number(global.Fca.Require.FastConfig.AutoRestartMinutes) * 60000);
204
- }
205
- require('../broadcast').startBroadcasting();
206
- const MemoryManager = require('../Extra/Src/Release_Memory');
207
- const path = require('path');
208
-
209
- const SettingMemoryManager = {
210
- warningThreshold: 0.7,
211
- releaseThreshold: 0.8,
212
- maxThreshold: 0.9,
213
- interval: 300 * 1000,
214
- logLevel: 'warn',
215
- logFile: path.join(process.cwd(), 'Horizon_Database' ,'memory.log'),
216
- smartReleaseEnabled: true,
217
- allowLog: (global.Fca.Require.FastConfig.AntiStuckAndMemoryLeak.LogFile.Use || false)
218
- };
219
-
220
- const memoryManager = new MemoryManager(SettingMemoryManager);
221
-
222
- memoryManager.autoStart(60 * 60 * 1000);
223
-
224
- if (global.Fca.Require.FastConfig.AntiStuckAndMemoryLeak.AutoRestart.Use) {
225
- memoryManager.onMaxMemory(function() {
226
- global.Fca.Require.logger.Warning('Memory Usage >= 90% - Auto Restart Avoid Crash');
227
- process.exit(1);
228
- });
229
- }
230
- process.env.OnStatus = true;
231
- }
232
-
233
- topics.forEach((topicsub) => global.mqttClient.subscribe(topicsub));
234
-
235
-
236
- let topic;
237
- const queue = {
238
- sync_api_version: 11,
239
- max_deltas_able_to_process: 100,
240
- delta_batch_size: 500,
241
- encoding: 'JSON',
242
- entity_fbid: ctx.userID,
243
- };
244
-
245
- topic = "/messenger_sync_create_queue";
246
- queue.initial_titan_sequence_id = ctx.lastSeqId;
247
- queue.device_params = null;
248
-
249
- global.mqttClient.publish(topic, JSON.stringify(queue), {
250
- qos: 1,
251
- retain: false
252
- });
253
-
254
- var rTimeout = setTimeout(function() {
255
- global.mqttClient.end();
256
- getSeqID();
257
- }, 3000);
258
-
259
- ctx.tmsWait = function() {
260
- clearTimeout(rTimeout);
261
- ctx.globalOptions.emitReady ? globalCallback({
262
- type: "ready",
263
- error: null
264
- }) : '';
265
- delete ctx.tmsWait;
266
- };
267
- });
268
-
269
- const HandleMessage = function(topic, message, _packet) {
270
- const jsonMessage = JSON.parse(message.toString());
271
- if (topic === "/t_ms") {
272
- if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
273
-
274
- if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
275
- ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
276
- ctx.syncToken = jsonMessage.syncToken;
277
- }
278
-
279
- if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
280
- //If it contains more than 1 delta
281
- for (var i in jsonMessage.deltas) {
282
- var delta = jsonMessage.deltas[i];
283
- parseDelta(defaultFuncs, api, ctx, globalCallback, {
284
- "delta": delta
285
- });
286
- }
287
- } else if (topic === "/ls_resp") {
288
- const payload = JSON.parse(jsonMessage.payload); //'{"name":null,"step":[1,[1,[4,0,1,[5,"taskExists",[19,"415"]]],[23,[2,0],[1,[5,"replaceOptimsiticMessage","7192532113093667880","mid.$gABfX5li9LA6VdUymnWPRAdlkiawo"]]]],[1,[4,0,1,[5,"taskExists",[19,"415"]]],[23,[2,0],[1,[5,"mailboxTaskCompletionApiOnTaskCompletion",[19,"415"],true]]]],[1,[4,0,1,[5,"taskExists",[19,"415"]]],[23,[2,0],[1,[5,"removeTask",[19,"415"],[9]]]]]]}'
289
- const request_ID = jsonMessage.request_id;
290
-
291
- if (ctx.callback_Task[request_ID] != undefined && ctx.callback_Task[request_ID].type != undefined) {
292
- const {
293
- callback,
294
- type
295
- } = ctx.callback_Task[request_ID];
296
- const Data = new getRespData(type, payload);
297
- if (!callback) {
298
- return;
299
- }
300
- else if (!Data) {
301
- callback("Something went wrong 🐳", null);
302
- } else {
303
- callback(null, Data);
304
- }
305
- }
306
- } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
307
- var typ = {
308
- type: "typ",
309
- isTyping: !!jsonMessage.state,
310
- from: jsonMessage.sender_fbid.toString(),
311
- threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
312
- };
313
- (function() {
314
- globalCallback(null, typ);
315
- })();
316
- } else if (topic === "/orca_presence") {
317
- if (!ctx.globalOptions.updatePresence) {
318
- for (var i in jsonMessage.list) {
319
- var data = jsonMessage.list[i];
320
- var userID = data["u"];
321
-
322
- var presence = {
323
- type: "presence",
324
- userID: userID.toString(),
325
- //Convert to ms
326
- timestamp: data["l"] * 1000,
327
- statuses: data["p"]
328
- };
329
- (function() {
330
- globalCallback(null, presence);
331
- })();
332
- }
333
- }
334
- }
335
-
336
- };
337
-
338
- global.mqttClient.on('message', HandleMessage);
339
-
340
- process.on('SIGINT', () => {
341
- LogUptime();
342
- process.kill(process.pid);
343
- });
344
-
345
- process.on('exit', LogUptime);
346
-
347
-
348
- }
349
-
350
- function getRespData(Type, payload) {
351
- try {
352
- switch (Type) {
353
- case "sendMqttMessage": {
354
- return {
355
- type: Type,
356
- threadID: payload.step[1][2][2][1][2], //this is sick bro
357
- messageID: payload.step[1][2][2][1][3],
358
- payload: payload.step[1][2]
359
- };
360
- }
361
- default: { //!very LAZY :> cook yourself
362
- return {
363
- Data: payload.step[1][2][2][1],
364
- type: Type,
365
- payload: payload.step[1][2]
366
- };
367
- }
368
- }
369
- } catch (e) {
370
- return null;
371
- }
372
- }
373
-
374
- function LogUptime() {
375
- const uptime = process.uptime();
376
- const {
377
- join
378
- } = require('path');
379
- const filePath = join(__dirname, '../CountTime.json');
380
-
381
- let time1;
382
- if (global.Fca.Require.fs.existsSync(filePath)) {
383
- time1 = Number(global.Fca.Require.fs.readFileSync(filePath, 'utf8')) || 0;
384
- } else {
385
- time1 = 0;
386
- }
387
-
388
- global.Fca.Require.fs.writeFileSync(filePath, String(Number(uptime) + time1), 'utf8');
389
- }
390
-
391
- if (global.Fca.Require.FastConfig.AntiGetInfo.AntiGetThreadInfo) {
392
- setInterval(() => {
393
- try {
394
- const { updateMessageCount, getData, hasData } = require('../Extra/ExtraGetThread');
395
- const Data = global.Fca.Data.MsgCount;
396
- const Arr = Array.from(Data.keys());
397
- for (let i of Arr) {
398
- const Count = parseInt(Data.get(i));
399
- if (hasData(i)) {
400
- let x = getData(i);
401
- x.messageCount += Count;
402
- updateMessageCount(i, x);
403
- Data.delete(i);
404
- }
405
- }
406
-
407
- } catch (e) {
408
- console.log(e);
409
- }
410
- }, 30 * 1000);
40
+ //Don't really know what this does but I think it's for the active state?
41
+ //TODO: Move to ctx when implemented
42
+ var chatOn = ctx.globalOptions.online;
43
+ var foreground = false;
44
+
45
+ const sessionID = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER) + 1;
46
+ const GUID = utils.getGUID()
47
+ const username = {
48
+ u: ctx.userID,
49
+ s: sessionID,
50
+ chat_on: chatOn,
51
+ fg: foreground,
52
+ d: GUID,
53
+ ct: 'websocket',
54
+ aid: '219994525426954',
55
+ aids: null,
56
+ mqtt_sid: '',
57
+ cp: 3,
58
+ ecp: 10,
59
+ st: [],
60
+ pm: [],
61
+ dc: '',
62
+ no_auto_fg: true,
63
+ gas: null,
64
+ pack: [],
65
+ a: ctx.globalOptions.userAgent,
66
+ p: null,
67
+ aids: null,
68
+ php_override: ""
69
+ };
70
+
71
+ const cookies = ctx.jar.getCookies('https://www.facebook.com').join('; ');
72
+
73
+ let host;
74
+ if (ctx.mqttEndpoint) {
75
+ host = `${ctx.mqttEndpoint}&sid=${sessionID}`;
76
+ } else if (ctx.region) {
77
+ host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLowerCase()}&sid=${sessionID}`;
78
+ } else {
79
+ host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}`;
80
+ }
81
+
82
+ const options = {
83
+ clientId: 'mqttwsclient',
84
+ protocolId: 'MQIsdp',
85
+ protocolVersion: 3,
86
+ username: JSON.stringify(username),
87
+ clean: true,
88
+ wsOptions: {
89
+ headers: {
90
+ Cookie: cookies,
91
+ Origin: 'https://www.facebook.com',
92
+ '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',
93
+ Referer: 'https://www.facebook.com/',
94
+ Host: new URL(host).hostname,
95
+ },
96
+ origin: 'https://www.facebook.com',
97
+ protocolVersion: 13,
98
+ binaryType: 'arraybuffer',
99
+ },
100
+ keepalive: 60,
101
+ reschedulePings: true,
102
+ reconnectPeriod: 3,
103
+ };
104
+
105
+ if (typeof ctx.globalOptions.proxy != "undefined") {
106
+ var agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
107
+ options.wsOptions.agent = agent;
108
+ }
109
+
110
+ ctx.mqttClient = new mqtt.Client(_ => websocket(host, options.wsOptions), options);
111
+
112
+ var mqttClient = ctx.mqttClient;
113
+
114
+ mqttClient.on('error', function (err) {
115
+ log.error("listenMqtt", err);
116
+ mqttClient.end();
117
+ if (ctx.globalOptions.autoReconnect) getSeqID();
118
+ else globalCallback({ type: "stop_listen", error: "Connection refused: Server unavailable" }, null);
119
+ });
120
+
121
+ mqttClient.on('connect', function () {
122
+ topics.forEach(topicsub => mqttClient.subscribe(topicsub));
123
+
124
+ var topic;
125
+ var queue = {
126
+ sync_api_version: 10,
127
+ max_deltas_able_to_process: 1000,
128
+ delta_batch_size: 500,
129
+ encoding: "JSON",
130
+ entity_fbid: ctx.userID,
131
+ };
132
+
133
+ if (ctx.syncToken) {
134
+ topic = "/messenger_sync_get_diffs";
135
+ queue.last_seq_id = ctx.lastSeqId;
136
+ queue.sync_token = ctx.syncToken;
137
+ } else {
138
+ topic = "/messenger_sync_create_queue";
139
+ queue.initial_titan_sequence_id = ctx.lastSeqId;
140
+ queue.device_params = null;
141
+ }
142
+
143
+ mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
144
+ mqttClient.publish("/foreground_state", JSON.stringify({ "foreground": chatOn }), { qos: 1 });
145
+ var rTimeout = setTimeout(function () {
146
+ mqttClient.end();
147
+ getSeqID();
148
+ }, 3000);
149
+
150
+ ctx.tmsWait = function () {
151
+ clearTimeout(rTimeout);
152
+ ctx.globalOptions.emitReady ? globalCallback({
153
+ type: "ready",
154
+ error: null
155
+ }) : "";
156
+ delete ctx.tmsWait;
157
+ };
158
+ });
159
+
160
+ mqttClient.on('message', function (topic, message, _packet) {
161
+ try {
162
+ var jsonMessage = JSON.parse(message);
163
+ } catch (ex) {
164
+ return log.error("listenMqtt", ex);
165
+ }
166
+ if (topic === "/t_ms") {
167
+ if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
168
+
169
+ if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
170
+ ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
171
+ ctx.syncToken = jsonMessage.syncToken;
172
+ }
173
+
174
+ if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
175
+
176
+ //If it contains more than 1 delta
177
+ for (var i in jsonMessage.deltas) {
178
+ var delta = jsonMessage.deltas[i];
179
+ parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
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
+ } else if (topic === "/orca_presence") {
190
+ if (!ctx.globalOptions.updatePresence) {
191
+ for (var i in jsonMessage.list) {
192
+ var data = jsonMessage.list[i];
193
+ var userID = data["u"];
194
+
195
+ var presence = {
196
+ type: "presence",
197
+ userID: userID.toString(),
198
+ //Convert to ms
199
+ timestamp: data["l"] * 1000,
200
+ statuses: data["p"]
201
+ };
202
+ (function () { globalCallback(null, presence); })();
203
+ }
204
+ }
205
+ }
206
+
207
+ });
208
+
209
+ mqttClient.on('close', function () {
210
+ //(function () { globalCallback("Connection closed."); })();
211
+ // client.end();
212
+ });
411
213
  }
412
214
 
413
- function parseDelta(defaultFuncs, api, ctx, globalCallback, {
414
- delta
415
- }) {
416
- if (delta.class === 'NewMessage') {
417
- if (ctx.globalOptions.pageID && ctx.globalOptions.pageID !== delta.queue) return;
418
-
419
- const resolveAttachmentUrl = (i) => {
420
- if (!delta.attachments || i === delta.attachments.length || utils.getType(delta.attachments) !== 'Array') {
421
- let fmtMsg;
422
- try {
423
- fmtMsg = utils.formatDeltaMessage(delta);
424
- } catch (err) {
425
- return log.error('Lỗi Nhẹ', err);
426
- }
427
-
428
- if (fmtMsg) {
429
- const isGroup = fmtMsg.isGroup;
430
- const threadID = fmtMsg.threadID;
431
- const messageID = fmtMsg.messageID;
432
-
433
- global.Fca.Data.event.set("Data", {
434
- isGroup,
435
- threadID,
436
- messageID
437
- });
438
-
439
- if (global.Fca.Require.FastConfig.AntiGetInfo.AntiGetThreadInfo) {
440
- global.Fca.Data.MsgCount.set(fmtMsg.threadID, ((global.Fca.Data.MsgCount.get(fmtMsg.threadID)) + 1 || 1));
441
- }
442
-
443
- if (ctx.globalOptions.autoMarkDelivery) {
444
- markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
445
- }
446
-
447
- if (!ctx.globalOptions.selfListen && fmtMsg.senderID === ctx.userID) return;
448
- globalCallback(null, fmtMsg);
449
- }
450
- } else {
451
- const attachment = delta.attachments[i];
452
- if (attachment.mercury.attach_type === 'photo') {
453
- api.resolvePhotoUrl(attachment.fbid, (err, url) => {
454
- if (!err) attachment.mercury.metadata.url = url;
455
- resolveAttachmentUrl(i + 1);
456
- });
457
- } else {
458
- resolveAttachmentUrl(i + 1);
459
- }
460
- }
461
- };
462
-
463
- resolveAttachmentUrl(0);
464
- } else if (delta.class === 'ClientPayload') {
465
- const clientPayload = utils.decodeClientPayload(delta.payload);
466
- if (clientPayload && clientPayload.deltas) {
467
- for (const delta of clientPayload.deltas) {
468
- if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
469
- const messageReaction = {
470
- type: 'message_reaction',
471
- threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
472
- messageID: delta.deltaMessageReaction.messageId,
473
- reaction: delta.deltaMessageReaction.reaction,
474
- senderID: delta.deltaMessageReaction.senderId.toString(),
475
- userID: delta.deltaMessageReaction.userId.toString(),
476
- };
477
- globalCallback(null, messageReaction);
478
- } else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
479
- const messageUnsend = {
480
- type: 'message_unsend',
481
- threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
482
- messageID: delta.deltaRecallMessageData.messageID,
483
- senderID: delta.deltaRecallMessageData.senderID.toString(),
484
- deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
485
- timestamp: delta.deltaRecallMessageData.timestamp,
486
- };
487
- globalCallback(null, messageUnsend);
488
- } else if (delta.deltaMessageReply) {
489
- const mdata =
490
- delta.deltaMessageReply.message === undefined ?
491
- [] :
492
- delta.deltaMessageReply.message.data === undefined ?
493
- [] :
494
- delta.deltaMessageReply.message.data.prng === undefined ?
495
- [] :
496
- JSON.parse(delta.deltaMessageReply.message.data.prng);
497
-
498
- const m_id = mdata.map((u) => u.i);
499
- const m_offset = mdata.map((u) => u.o);
500
- const m_length = mdata.map((u) => u.l);
501
-
502
- const mentions = {};
503
- for (let i = 0; i < m_id.length; i++) {
504
- mentions[m_id[i]] = (delta.deltaMessageReply.message.body || '').substring(m_offset[i], m_offset[i] + m_length[i]);
505
- }
506
-
507
- const callbackToReturn = {
508
- type: 'message_reply',
509
- threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
510
- messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
511
- senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
512
- attachments: ( delta.deltaMessageReply.message.attachments || [] )
513
- .map((att) => {
514
- const mercury = JSON.parse(att.mercuryJSON);
515
- Object.assign(att, mercury);
516
- return att;
517
- })
518
- .map((att) => {
519
- let x;
520
- try {
521
- x = utils._formatAttachment(att);
522
- } catch (ex) {
523
- x = att;
524
- x.error = ex;
525
- x.type = 'unknown';
526
- }
527
- return x;
528
- }),
529
- args: (delta.deltaMessageReply.message.body || '').trim().split(/\s+/),
530
- body: delta.deltaMessageReply.message.body || '',
531
- isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
532
- mentions,
533
- timestamp: parseInt(delta.deltaMessageReply.message.messageMetadata.timestamp),
534
- participantIDs: (delta.deltaMessageReply.message.participants || []).map((e) => e.toString()),
535
- };
536
-
537
- if (delta.deltaMessageReply.repliedToMessage) {
538
- const mdata =
539
- delta.deltaMessageReply.repliedToMessage === undefined ?
540
- [] :
541
- delta.deltaMessageReply.repliedToMessage.data === undefined ?
542
- [] :
543
- delta.deltaMessageReply.repliedToMessage.data.prng === undefined ?
544
- [] :
545
- JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
546
-
547
- const m_id = mdata.map((u) => u.i);
548
- const m_offset = mdata.map((u) => u.o);
549
- const m_length = mdata.map((u) => u.l);
550
-
551
- const rmentions = {};
552
- for (let i = 0; i < m_id.length; i++) {
553
- rmentions[m_id[i]] = (delta.deltaMessageReply.repliedToMessage.body || '').substring(m_offset[i], m_offset[i] + m_length[i]);
554
- }
555
-
556
- callbackToReturn.messageReply = {
557
- threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
558
- messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
559
- senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
560
- attachments: delta.deltaMessageReply.repliedToMessage.attachments
561
- .map((att) => {
562
- let mercury;
563
- try {
564
- mercury = JSON.parse(att.mercuryJSON);
565
- Object.assign(att, mercury);
566
- } catch (ex) {
567
- mercury = {};
568
- }
569
- return att;
570
- })
571
- .map((att) => {
572
- let x;
573
- try {
574
- x = utils._formatAttachment(att);
575
- } catch (ex) {
576
- x = att;
577
- x.error = ex;
578
- x.type = 'unknown';
579
- }
580
- return x;
581
- }),
582
- args: (delta.deltaMessageReply.repliedToMessage.body || '').trim().split(/\s+/),
583
- body: delta.deltaMessageReply.repliedToMessage.body || '',
584
- isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
585
- mentions: rmentions,
586
- timestamp: parseInt(delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp),
587
- participantIDs: (delta.deltaMessageReply.repliedToMessage.participants || []).map((e) => e.toString()),
588
- };
589
- } else if (delta.deltaMessageReply.replyToMessageId) {
590
- return defaultFuncs
591
- .post('https://www.facebook.com/api/graphqlbatch/', ctx.jar, {
592
- av: ctx.globalOptions.pageID,
593
- queries: JSON.stringify({
594
- o0: {
595
- doc_id: '2848441488556444',
596
- query_params: {
597
- thread_and_message_id: {
598
- thread_id: callbackToReturn.threadID,
599
- message_id: delta.deltaMessageReply.replyToMessageId.id,
600
- },
601
- },
602
- },
603
- }),
604
- })
605
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
606
- .then((resData) => {
607
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
608
- if (resData[resData.length - 1].successful_results === 0) throw {
609
- error: 'forcedFetch: there was no successful_results',
610
- res: resData
611
- };
612
- const fetchData = resData[0].o0.data.message;
613
- const mobj = {};
614
-
615
- for (const n in fetchData.message.ranges) {
616
- mobj[fetchData.message.ranges[n].entity.id] = (fetchData.message.text || '').substr(fetchData.message.ranges[n].offset, fetchData.message.ranges[n].length);
617
- }
618
- callbackToReturn.messageReply = {
619
- type: 'Message',
620
- threadID: callbackToReturn.threadID,
621
- messageID: fetchData.message_id,
622
- senderID: fetchData.message_sender.id.toString(),
623
- attachments: fetchData.message.blob_attachment.map((att) => utils._formatAttachment({
624
- blob_attachment: att
625
- })),
626
- args: (fetchData.message.text || '').trim().split(/\s+/) || [],
627
- body: fetchData.message.text || '',
628
- isGroup: callbackToReturn.isGroup,
629
- mentions: mobj,
630
- timestamp: parseInt(fetchData.timestamp_precise),
631
- };
632
- })
633
-
634
- .catch((err) => log.error('forcedFetch', err))
635
- .finally(() => {
636
- if (ctx.globalOptions.autoMarkDelivery) {
637
- markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
638
- }
639
-
640
- if (!ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID) return;
641
- globalCallback(null, callbackToReturn);
642
- });
643
- } else {
644
- callbackToReturn.delta = delta;
645
- }
646
- if (ctx.globalOptions.autoMarkDelivery) {
647
- markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
648
- }
649
-
650
- if (!ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID) return;
651
- globalCallback(null, callbackToReturn);
652
- }
653
- }
654
-
655
- return;
656
- }
657
- }
658
- switch (delta.class) {
659
- case 'ReadReceipt': {
660
- let fmtMsg;
661
- try {
662
- fmtMsg = utils.formatDeltaReadReceipt(delta);
663
- } catch (err) {
664
- return log.error('Lỗi Nhẹ', err);
665
- }
666
- globalCallback(null, fmtMsg);
667
- break;
668
- }
669
- case 'AdminTextMessage': {
670
- switch (delta.type) {
671
- case 'joinable_group_link_mode_change':
672
- case 'magic_words':
673
- case 'pin_messages_v2':
674
- case 'change_thread_theme':
675
- case 'change_thread_icon':
676
- case 'change_thread_nickname':
677
- case 'change_thread_admins':
678
- case 'change_thread_approval_mode':
679
- case 'group_poll':
680
- case 'messenger_call_log':
681
- case 'participant_joined_group_call': {
682
- let fmtMsg;
683
- try {
684
- fmtMsg = utils.formatDeltaEvent(delta);
685
- } catch (err) {
686
- console.log(delta);
687
- return log.error('Lỗi Nhẹ', err);
688
- }
689
- globalCallback(null, fmtMsg);
690
- break;
691
- }
692
- }
693
- break;
694
- }
695
-
696
- //For group images
697
- case 'ForcedFetch': {
698
- if (!delta.threadKey) return;
699
- const mid = delta.messageId;
700
- const tid = delta.threadKey.threadFbId;
701
-
702
- if (mid && tid) {
703
- const form = {
704
- av: ctx.globalOptions.pageID,
705
- queries: JSON.stringify({
706
- o0: {
707
- doc_id: '2848441488556444',
708
- query_params: {
709
- thread_and_message_id: {
710
- thread_id: tid.toString(),
711
- message_id: mid,
712
- },
713
- },
714
- },
715
- }),
716
- };
717
- defaultFuncs
718
- .post('https://www.facebook.com/api/graphqlbatch/', ctx.jar, form)
719
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
720
- .then((resData) => {
721
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
722
-
723
- if (resData[resData.length - 1].successful_results === 0) throw {
724
- error: 'forcedFetch: there was no successful_results',
725
- res: resData
726
- };
727
-
728
- const fetchData = resData[0].o0.data.message;
729
-
730
- if (utils.getType(fetchData) === 'Object') {
731
- log.info('forcedFetch', fetchData);
732
- switch (fetchData.__typename) {
733
- case 'ThreadImageMessage':
734
- if (!ctx.globalOptions.selfListen && fetchData.message_sender.id.toString() === ctx.userID) return;
735
- if (!ctx.loggedIn) return;
736
-
737
- globalCallback(null, {
738
- type: 'change_thread_image',
739
- threadID: utils.formatID(tid.toString()),
740
- snippet: fetchData.snippet,
741
- timestamp: fetchData.timestamp_precise,
742
- author: fetchData.message_sender.id,
743
- image: {
744
- attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
745
- width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
746
- height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
747
- url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri,
748
- },
749
- });
750
- break;
751
- case 'UserMessage': {
752
- const event = {
753
- type: 'message',
754
- senderID: utils.formatID(fetchData.message_sender.id),
755
- body: fetchData.message.text || '',
756
- threadID: utils.formatID(tid.toString()),
757
- messageID: fetchData.message_id,
758
- attachments: [{
759
- type: 'share',
760
- ID: fetchData.extensible_attachment.legacy_attachment_id,
761
- url: fetchData.extensible_attachment.story_attachment.url,
762
- title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
763
- description: fetchData.extensible_attachment.story_attachment.description.text,
764
- source: fetchData.extensible_attachment.story_attachment.source,
765
- image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
766
- width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
767
- height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
768
- playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
769
- duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
770
- subattachments: fetchData.extensible_attachment.subattachments,
771
- properties: fetchData.extensible_attachment.story_attachment.properties,
772
- }],
773
- mentions: {},
774
- timestamp: parseInt(fetchData.timestamp_precise),
775
- isGroup: (fetchData.message_sender.id !== tid.toString()),
776
- };
777
-
778
- log.info('ff-Return', event);
779
- globalCallback(null, event);
780
- break;
781
- }
782
- default:
783
- log.error('forcedFetch', fetchData);
784
- }
785
- } else {
786
- log.error('forcedFetch', fetchData);
787
- }
788
- })
789
- .catch((err) => log.error('forcedFetch', err));
790
- }
791
- break;
792
- }
793
- case 'ThreadName':
794
- case 'ParticipantsAddedToGroupThread':
795
- case 'ParticipantLeftGroupThread': {
796
- let formattedEvent;
797
- try {
798
- formattedEvent = utils.formatDeltaEvent(delta);
799
- } catch (err) {
800
- console.log(err);
801
- return log.error('Lỗi Nhẹ', err);
802
- }
803
-
804
- if (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) return;
805
- if (!ctx.loggedIn) return;
806
- globalCallback(null, formattedEvent);
807
- break;
808
- }
809
- case 'NewMessage': {
810
- const hasLiveLocation = delta => {
811
- const attachment = delta.attachments?.[0]?.mercury?.extensible_attachment;
812
- const storyAttachment = attachment?.story_attachment;
813
- return storyAttachment?.style_list?.includes('message_live_location');
814
- };
815
-
816
- if (delta.attachments?.length === 1 && hasLiveLocation(delta)) {
817
- delta.class = 'UserLocation';
818
-
819
- try {
820
- const fmtMsg = utils.formatDeltaEvent(delta);
821
- globalCallback(null, fmtMsg);
822
- } catch (err) {
823
- console.log(delta);
824
- log.error('Lỗi Nhẹ', err);
825
- }
826
- }
827
- break;
828
- }
829
- }
215
+ function parseDelta(defaultFuncs, api, ctx, globalCallback, v) {
216
+ if (v.delta.class == "NewMessage") {
217
+ //Not tested for pages
218
+ if (ctx.globalOptions.pageID && ctx.globalOptions.pageID != v.queue) return;
219
+
220
+ (function resolveAttachmentUrl(i) {
221
+ if (v.delta.attachments && (i == v.delta.attachments.length)) {
222
+ var fmtMsg;
223
+ try {
224
+ fmtMsg = utils.formatDeltaMessage(v);
225
+ } catch (err) {
226
+ return globalCallback({
227
+ error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
228
+ detail: err,
229
+ res: v,
230
+ type: "parse_error"
231
+ });
232
+ }
233
+ if (fmtMsg)
234
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
235
+
236
+ return !ctx.globalOptions.selfListen && fmtMsg.senderID === ctx.userID ? undefined : (function () { globalCallback(null, fmtMsg); })();
237
+ } else {
238
+ if (v.delta.attachments && (v.delta.attachments[i].mercury.attach_type == "photo")) {
239
+ api.resolvePhotoUrl(v.delta.attachments[i].fbid, (err, url) => {
240
+ if (!err) v.delta.attachments[i].mercury.metadata.url = url;
241
+ return resolveAttachmentUrl(i + 1);
242
+ });
243
+ } else return resolveAttachmentUrl(i + 1);
244
+ }
245
+ })(0);
246
+ }
247
+
248
+ if (v.delta.class == "ClientPayload") {
249
+ var clientPayload = utils.decodeClientPayload(v.delta.payload);
250
+ if (clientPayload && clientPayload.deltas) {
251
+ for (var i in clientPayload.deltas) {
252
+ var delta = clientPayload.deltas[i];
253
+ if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
254
+ (function () {
255
+ globalCallback(null, {
256
+ type: "message_reaction",
257
+ threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
258
+ messageID: delta.deltaMessageReaction.messageId,
259
+ reaction: delta.deltaMessageReaction.reaction,
260
+ senderID: delta.deltaMessageReaction.senderId.toString(),
261
+ userID: delta.deltaMessageReaction.userId.toString()
262
+ });
263
+ })();
264
+ } else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
265
+ (function () {
266
+ globalCallback(null, {
267
+ type: "message_unsend",
268
+ threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
269
+ messageID: delta.deltaRecallMessageData.messageID,
270
+ senderID: delta.deltaRecallMessageData.senderID.toString(),
271
+ deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
272
+ timestamp: delta.deltaRecallMessageData.timestamp
273
+ });
274
+ })();
275
+ } else if (delta.deltaMessageReply) {
276
+ //Mention block - #1
277
+ var mdata =
278
+ delta.deltaMessageReply.message === undefined ? [] :
279
+ delta.deltaMessageReply.message.data === undefined ? [] :
280
+ delta.deltaMessageReply.message.data.prng === undefined ? [] :
281
+ JSON.parse(delta.deltaMessageReply.message.data.prng);
282
+ var m_id = mdata.map(u => u.i);
283
+ var m_offset = mdata.map(u => u.o);
284
+ var m_length = mdata.map(u => u.l);
285
+
286
+ var mentions = {};
287
+
288
+ 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]);
289
+ //Mention block - 1#
290
+ var callbackToReturn = {
291
+ type: "message_reply",
292
+ threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
293
+ messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
294
+ senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
295
+ attachments: delta.deltaMessageReply.message.attachments.map(function (att) {
296
+ var mercury = JSON.parse(att.mercuryJSON);
297
+ Object.assign(att, mercury);
298
+ return att;
299
+ }).map(att => {
300
+ var x;
301
+ try {
302
+ x = utils._formatAttachment(att);
303
+ } catch (ex) {
304
+ x = att;
305
+ x.error = ex;
306
+ x.type = "unknown";
307
+ }
308
+ return x;
309
+ }),
310
+ args: (delta.deltaMessageReply.message.body || "").trim().split(/\s+/),
311
+ body: (delta.deltaMessageReply.message.body || ""),
312
+ isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
313
+ mentions: mentions,
314
+ timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
315
+ participantIDs: (delta.deltaMessageReply.message.participants || []).map(e => e.toString())
316
+ };
317
+
318
+ if (delta.deltaMessageReply.repliedToMessage) {
319
+ //Mention block - #2
320
+ mdata =
321
+ delta.deltaMessageReply.repliedToMessage === undefined ? [] :
322
+ delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
323
+ delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
324
+ JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
325
+ m_id = mdata.map(u => u.i);
326
+ m_offset = mdata.map(u => u.o);
327
+ m_length = mdata.map(u => u.l);
328
+
329
+ var rmentions = {};
330
+
331
+ 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]);
332
+ //Mention block - 2#
333
+ callbackToReturn.messageReply = {
334
+ threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
335
+ messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
336
+ senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
337
+ attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function (att) {
338
+ var mercury = JSON.parse(att.mercuryJSON);
339
+ Object.assign(att, mercury);
340
+ return att;
341
+ }).map(att => {
342
+ var x;
343
+ try {
344
+ x = utils._formatAttachment(att);
345
+ } catch (ex) {
346
+ x = att;
347
+ x.error = ex;
348
+ x.type = "unknown";
349
+ }
350
+ return x;
351
+ }),
352
+ args: (delta.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
353
+ body: delta.deltaMessageReply.repliedToMessage.body || "",
354
+ isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
355
+ mentions: rmentions,
356
+ timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp,
357
+ participantIDs: (delta.deltaMessageReply.repliedToMessage.participants || []).map(e => e.toString())
358
+ };
359
+ } else if (delta.deltaMessageReply.replyToMessageId) {
360
+ return defaultFuncs
361
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
362
+ "av": ctx.globalOptions.pageID,
363
+ "queries": JSON.stringify({
364
+ "o0": {
365
+ //Using the same doc_id as forcedFetch
366
+ "doc_id": "2848441488556444",
367
+ "query_params": {
368
+ "thread_and_message_id": {
369
+ "thread_id": callbackToReturn.threadID,
370
+ "message_id": delta.deltaMessageReply.replyToMessageId.id,
371
+ }
372
+ }
373
+ }
374
+ })
375
+ })
376
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
377
+ .then((resData) => {
378
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
379
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
380
+ var fetchData = resData[0].o0.data.message;
381
+ var mobj = {};
382
+ 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);
383
+
384
+ callbackToReturn.messageReply = {
385
+ threadID: callbackToReturn.threadID,
386
+ messageID: fetchData.message_id,
387
+ senderID: fetchData.message_sender.id.toString(),
388
+ attachments: fetchData.message.blob_attachment.map(att => {
389
+ var x;
390
+ try {
391
+ x = utils._formatAttachment({ blob_attachment: att });
392
+ } catch (ex) {
393
+ x = att;
394
+ x.error = ex;
395
+ x.type = "unknown";
396
+ }
397
+ return x;
398
+ }),
399
+ args: (fetchData.message.text || "").trim().split(/\s+/) || [],
400
+ body: fetchData.message.text || "",
401
+ isGroup: callbackToReturn.isGroup,
402
+ mentions: mobj,
403
+ timestamp: parseInt(fetchData.timestamp_precise)
404
+ };
405
+ })
406
+ .catch(err => log.error("forcedFetch", err))
407
+ .finally(function () {
408
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
409
+ !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
410
+ });
411
+ } else callbackToReturn.delta = delta;
412
+
413
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
414
+
415
+ return !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
416
+ }
417
+ }
418
+ return;
419
+ }
420
+ }
421
+
422
+ if (v.delta.class !== "NewMessage" && !ctx.globalOptions.listenEvents) return;
423
+ switch (v.delta.class) {
424
+ case "ReadReceipt":
425
+ var fmtMsg;
426
+ try {
427
+ fmtMsg = utils.formatDeltaReadReceipt(v.delta);
428
+ } catch (err) {
429
+ return globalCallback({
430
+ error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
431
+ detail: err,
432
+ res: v.delta,
433
+ type: "parse_error"
434
+ });
435
+ }
436
+ return (function () { globalCallback(null, fmtMsg); })();
437
+ case "AdminTextMessage":
438
+ switch (v.delta.type) {
439
+ case "change_thread_theme":
440
+ case "change_thread_nickname":
441
+ case "change_thread_admins":
442
+ case "change_thread_approval_mode":
443
+ case "joinable_group_link_mode_change":
444
+ case "rtc_call_log":
445
+ case "group_poll":
446
+ case "update_vote":
447
+ case "magic_words":
448
+ case "messenger_call_log":
449
+ case "participant_joined_group_call":
450
+ var fmtMsg;
451
+ try {
452
+ fmtMsg = utils.formatDeltaEvent(v.delta);
453
+ } catch (err) {
454
+ return globalCallback({
455
+ error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
456
+ detail: err,
457
+ res: v.delta,
458
+ type: "parse_error"
459
+ });
460
+ }
461
+ return (function () { globalCallback(null, fmtMsg); })();
462
+ default:
463
+ return;
464
+ }
465
+ break;
466
+ //For group images
467
+ case "ForcedFetch":
468
+ if (!v.delta.threadKey) return;
469
+ var mid = v.delta.messageId;
470
+ var tid = v.delta.threadKey.threadFbId;
471
+ if (mid && tid) {
472
+ const form = {
473
+ "av": ctx.globalOptions.pageID,
474
+ "queries": JSON.stringify({
475
+ "o0": {
476
+ //This doc_id is valid as of March 25, 2020
477
+ "doc_id": "2848441488556444",
478
+ "query_params": {
479
+ "thread_and_message_id": {
480
+ "thread_id": tid.toString(),
481
+ "message_id": mid,
482
+ }
483
+ }
484
+ }
485
+ })
486
+ };
487
+
488
+ defaultFuncs
489
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
490
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
491
+ .then((resData) => {
492
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
493
+
494
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
495
+
496
+ var fetchData = resData[0].o0.data.message;
497
+
498
+ if (utils.getType(fetchData) == "Object") {
499
+ log.info("forcedFetch", fetchData);
500
+ switch (fetchData.__typename) {
501
+ case "ThreadImageMessage":
502
+ (!ctx.globalOptions.selfListen &&
503
+ fetchData.message_sender.id.toString() === ctx.userID) ||
504
+ !ctx.loggedIn ?
505
+ undefined :
506
+ (function () {
507
+ globalCallback(null, {
508
+ type: "change_thread_image",
509
+ threadID: utils.formatID(tid.toString()),
510
+ snippet: fetchData.snippet,
511
+ timestamp: fetchData.timestamp_precise,
512
+ author: fetchData.message_sender.id,
513
+ image: {
514
+ attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
515
+ width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
516
+ height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
517
+ url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
518
+ }
519
+ });
520
+ })();
521
+ break;
522
+ case "UserMessage":
523
+ log.info("ff-Return", {
524
+ type: "message",
525
+ senderID: utils.formatID(fetchData.message_sender.id),
526
+ body: fetchData.message.text || "",
527
+ threadID: utils.formatID(tid.toString()),
528
+ messageID: fetchData.message_id,
529
+ attachments: [{
530
+ type: "share",
531
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
532
+ url: fetchData.extensible_attachment.story_attachment.url,
533
+
534
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
535
+ description: fetchData.extensible_attachment.story_attachment.description.text,
536
+ source: fetchData.extensible_attachment.story_attachment.source,
537
+
538
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
539
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
540
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
541
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
542
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
543
+
544
+ subattachments: fetchData.extensible_attachment.subattachments,
545
+ properties: fetchData.extensible_attachment.story_attachment.properties,
546
+ }],
547
+ mentions: {},
548
+ timestamp: parseInt(fetchData.timestamp_precise),
549
+ isGroup: (fetchData.message_sender.id != tid.toString())
550
+ });
551
+ globalCallback(null, {
552
+ type: "message",
553
+ senderID: utils.formatID(fetchData.message_sender.id),
554
+ body: fetchData.message.text || "",
555
+ threadID: utils.formatID(tid.toString()),
556
+ messageID: fetchData.message_id,
557
+ attachments: [{
558
+ type: "share",
559
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
560
+ url: fetchData.extensible_attachment.story_attachment.url,
561
+
562
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
563
+ description: fetchData.extensible_attachment.story_attachment.description.text,
564
+ source: fetchData.extensible_attachment.story_attachment.source,
565
+
566
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
567
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
568
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
569
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
570
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
571
+
572
+ subattachments: fetchData.extensible_attachment.subattachments,
573
+ properties: fetchData.extensible_attachment.story_attachment.properties,
574
+ }],
575
+ mentions: {},
576
+ timestamp: parseInt(fetchData.timestamp_precise),
577
+ isGroup: (fetchData.message_sender.id != tid.toString())
578
+ });
579
+ }
580
+ } else log.error("forcedFetch", fetchData);
581
+ })
582
+ .catch((err) => log.error("forcedFetch", err));
583
+ }
584
+ break;
585
+ case "ThreadName":
586
+ case "ParticipantsAddedToGroupThread":
587
+ case "ParticipantLeftGroupThread":
588
+ var formattedEvent;
589
+ try {
590
+ formattedEvent = utils.formatDeltaEvent(v.delta);
591
+ } catch (err) {
592
+ return globalCallback({
593
+ error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
594
+ detail: err,
595
+ res: v.delta,
596
+ type: "parse_error"
597
+ });
598
+ }
599
+ return (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) || !ctx.loggedIn ? undefined : (function () { globalCallback(null, formattedEvent); })();
600
+ }
830
601
  }
831
602
 
832
603
  function markDelivery(ctx, api, threadID, messageID) {
833
- if (threadID && messageID) {
834
- api.markAsDelivered(threadID, messageID, (err) => {
835
- if (err) log.error('markAsDelivered', err);
836
- else {
837
- if (ctx.globalOptions.autoMarkRead) {
838
- api.markAsRead(threadID, (err) => {
839
- if (err) log.error('markAsDelivered', err);
840
- });
841
- }
842
- }
843
- });
844
- }
604
+ if (threadID && messageID) {
605
+ api.markAsDelivered(threadID, messageID, (err) => {
606
+ if (err) log.error("markAsDelivered", err);
607
+ else {
608
+ if (ctx.globalOptions.autoMarkRead) {
609
+ api.markAsRead(threadID, (err) => {
610
+ if (err) log.error("markAsDelivered", err);
611
+ });
612
+ }
613
+ }
614
+ });
615
+ }
845
616
  }
846
617
 
847
-
848
-
849
- module.exports = function(defaultFuncs, api, ctx) {
850
- var globalCallback = identity;
851
- var okeoke;
852
- getSeqID = function getSeqID() {
853
- ctx.t_mqttCalled = false;
854
- defaultFuncs
855
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
856
- .then(res => {
857
- okeoke = res;
858
- return res;
859
- })
860
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
861
- .then((resData) => {
862
- if (utils.getType(resData) != "Array") {
863
- if (okeoke.request.uri && okeoke.request.uri.href.includes("https://www.facebook.com/checkpoint/")) {
864
- if (okeoke.request.uri.href.includes('601051028565049')) {
865
- return global.Fca.BypassAutomationNotification(undefined, ctx.jar, ctx.globalOptions, undefined ,process.env.UID)
866
- }
867
- }
868
- if (global.Fca.Require.FastConfig.AutoLogin) {
869
- return global.Fca.Require.logger.Warning(global.Fca.Require.Language.Index.AutoLogin, function() {
870
- return global.Fca.Action('AutoLogin');
871
- });
872
- } else if (!global.Fca.Require.FastConfig.AutoLogin) {
873
- return global.Fca.Require.logger.Error(global.Fca.Require.Language.Index.ErrAppState);
874
- }
875
- return;
876
- } else {
877
- if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
878
- if (resData[resData.length - 1].successful_results === 0) throw {
879
- error: "getSeqId: there was no successful_results",
880
- res: resData
881
- };
882
- if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
883
- ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
884
- listenMqtt(defaultFuncs, api, ctx, globalCallback);
885
- } else throw {
886
- error: "getSeqId: no sync_sequence_id found.",
887
- res: resData
888
- };
889
- }
890
- })
891
- .catch((err) => {
892
- log.error("getSeqId", err);
893
- if (okeoke.request.uri && okeoke.request.uri.href.includes("https://www.facebook.com/checkpoint/")) {
894
- if (okeoke.request.uri.href.includes('601051028565049')) {
895
- return global.Fca.BypassAutomationNotification(undefined, ctx.jar, ctx.globalOptions, undefined ,process.env.UID)
896
- }
897
- }
898
- if (utils.getType(err) == "Object" && err.error === global.Fca.Require.Language.Index.ErrAppState) ctx.loggedIn = false;
899
- return globalCallback(err);
900
- });
901
- };
902
-
903
- return function(callback) {
904
- class MessageEmitter extends EventEmitter {
905
- stopListening(callback) {
906
- callback = callback || (() => {});
907
- globalCallback = identity;
908
- if (ctx.mqttClient) {
909
- ctx.mqttClient.unsubscribe("/webrtc");
910
- ctx.mqttClient.unsubscribe("/rtc_multi");
911
- ctx.mqttClient.unsubscribe("/onevc");
912
- ctx.mqttClient.publish("/browser_close", "{}");
913
- ctx.mqttClient.end(false, function(...data) {
914
- ctx.mqttClient = undefined;
915
- });
916
- }
917
- global.Fca.Data.StopListening = true;
918
- }
919
- }
920
-
921
- var msgEmitter = new MessageEmitter();
922
- globalCallback = (callback || function(error, message) {
923
- if (error) return msgEmitter.emit("error", error);
924
- msgEmitter.emit("message", message);
925
- });
926
-
927
- //Reset some stuff
928
- if (!ctx.firstListen) ctx.lastSeqId = null;
929
- ctx.syncToken = undefined;
930
- ctx.t_mqttCalled = false;
931
-
932
- //Same request as getThreadList
933
- form = {
934
- av: ctx.globalOptions.pageID,
935
- queries: JSON.stringify({
936
- o0: {
937
- doc_id: '3336396659757871',
938
- query_params: {
939
- limit: 1,
940
- before: null,
941
- tags: ['INBOX'],
942
- includeDeliveryReceipts: false,
943
- includeSeqID: true,
944
- },
945
- },
946
- }),
947
- };
948
-
949
-
950
- if (!ctx.firstListen || !ctx.lastSeqId) getSeqID();
951
- else listenMqtt(defaultFuncs, api, ctx, globalCallback);
952
- ctx.firstListen = false;
953
-
954
- return msgEmitter;
955
- };
956
- };
618
+ module.exports = function (defaultFuncs, api, ctx) {
619
+ var globalCallback = identity;
620
+ getSeqID = function getSeqID() {
621
+ ctx.t_mqttCalled = false;
622
+ defaultFuncs
623
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
624
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
625
+ .then((resData) => {
626
+ if (utils.getType(resData) != "Array") throw { error: "Not logged in", res: resData };
627
+ if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
628
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "getSeqId: there was no successful_results", res: resData };
629
+ if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
630
+ ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
631
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
632
+ } else throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
633
+ })
634
+ .catch((err) => {
635
+ log.error("getSeqId", err);
636
+ if (utils.getType(err) == "Object" && err.error === "Not logged in") ctx.loggedIn = false;
637
+ return globalCallback(err);
638
+ });
639
+ };
640
+
641
+ return async function (callback) {
642
+ class MessageEmitter extends EventEmitter {
643
+ stopListening(callback) {
644
+ callback = callback || (() => { });
645
+ globalCallback = identity;
646
+ if (ctx.mqttClient) {
647
+ ctx.mqttClient.unsubscribe("/webrtc");
648
+ ctx.mqttClient.unsubscribe("/rtc_multi");
649
+ ctx.mqttClient.unsubscribe("/onevc");
650
+ ctx.mqttClient.publish("/browser_close", "{}");
651
+ ctx.mqttClient.end(false, function (...data) {
652
+ callback(data);
653
+ ctx.mqttClient = undefined;
654
+ });
655
+ }
656
+ }
657
+ }
658
+
659
+ var msgEmitter = new MessageEmitter();
660
+ globalCallback = (callback || function (error, message) {
661
+ if (error) return msgEmitter.emit("error", error);
662
+ msgEmitter.emit("message", message);
663
+ });
664
+
665
+ //Reset some stuff
666
+ if (!ctx.firstListen) ctx.lastSeqId = null;
667
+ ctx.syncToken = undefined;
668
+ ctx.t_mqttCalled = false;
669
+
670
+ //Same request as getThreadList
671
+ form = {
672
+ "av": ctx.globalOptions.pageID,
673
+ "queries": JSON.stringify({
674
+ "o0": {
675
+ "doc_id": "3336396659757871",
676
+ "query_params": {
677
+ "limit": 1,
678
+ "before": null,
679
+ "tags": ["INBOX"],
680
+ "includeDeliveryReceipts": false,
681
+ "includeSeqID": true
682
+ }
683
+ }
684
+ })
685
+ };
686
+
687
+ if (!ctx.firstListen || !ctx.lastSeqId) getSeqID();
688
+ else listenMqtt(defaultFuncs, api, ctx, globalCallback);
689
+ ctx.firstListen = false;
690
+ return msgEmitter;
691
+ };
692
+ };