fca-smart-shankar 13.0.0 → 16.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/.cache/replit/env/latest +56 -0
  2. package/.cache/replit/env/latest.json +1 -0
  3. package/.cache/replit/modules/nodejs-14.res +1 -0
  4. package/.cache/replit/modules/replit.res +1 -0
  5. package/.cache/replit/modules/web.res +1 -0
  6. package/.cache/replit/modules.stamp +0 -0
  7. package/.cache/typescript/5.5/package.json +1 -0
  8. package/.config/configstore/update-notifier-npm.json +4 -0
  9. package/.gitattributes +2 -0
  10. package/.replit +3 -12
  11. package/.upm/store.json +1 -1
  12. package/Extra/Balancer.js +49 -0
  13. package/Extra/Bypass/956/index.js +234 -0
  14. package/Extra/Bypass/test/aaaa.json +170 -0
  15. package/Extra/Bypass/test/index.js +188 -0
  16. package/Extra/Database/index.js +469 -0
  17. package/Extra/ExtraAddons.js +82 -0
  18. package/Extra/ExtraFindUID.js +62 -0
  19. package/Extra/ExtraGetThread.js +365 -0
  20. package/Extra/ExtraScreenShot.js +430 -0
  21. package/Extra/ExtraUptimeRobot.js +38 -0
  22. package/Extra/Html/Classic/script.js +119 -0
  23. package/Extra/Html/Classic/style.css +8 -0
  24. package/Extra/Security/AES_256_GCM/index.js +0 -0
  25. package/Extra/Security/Base/Step_1.js +6 -0
  26. package/Extra/Security/Base/Step_2.js +22 -0
  27. package/Extra/Security/Base/Step_3.js +22 -0
  28. package/Extra/Security/Base/index.js +191 -0
  29. package/Extra/Security/Index.js +5 -0
  30. package/Extra/Security/Step_1.js +6 -0
  31. package/Extra/Security/Step_2.js +22 -0
  32. package/Extra/Security/Step_3.js +22 -0
  33. package/Extra/Src/Change_Environment.js +24 -0
  34. package/Extra/Src/Check_Update.js +67 -0
  35. package/Extra/Src/History.js +115 -0
  36. package/Extra/Src/Instant_Update.js +65 -0
  37. package/Extra/Src/Last-Run.js +65 -0
  38. package/Extra/Src/Premium.js +81 -0
  39. package/Extra/Src/Release_Memory.js +160 -0
  40. package/Extra/Src/Websocket.js +213 -0
  41. package/Extra/Src/image/checkmate.jpg +0 -0
  42. package/Extra/Src/test.js +28 -0
  43. package/Extra/Src/uuid.js +137 -0
  44. package/Func/AcceptAgreement.js +31 -0
  45. package/Func/ClearCache.js +64 -0
  46. package/Func/ReportV1.js +54 -0
  47. package/LICENSE +678 -0
  48. package/Language/index.json +228 -0
  49. package/Main.js +1444 -0
  50. package/README.md +30 -175
  51. package/SECURITY.md +18 -0
  52. package/broadcast.js +44 -0
  53. package/index.js +414 -511
  54. package/logger.js +66 -0
  55. package/package.json +85 -370
  56. package/src/Dev_Horizon_Data.js +125 -0
  57. package/src/Dev_getThreadInfoOLD.js +422 -0
  58. package/src/Dev_shareTest2.js +68 -0
  59. package/src/Dev_shareTest3.js +71 -0
  60. package/src/Premium.js +25 -0
  61. package/src/Screenshot.js +83 -0
  62. package/src/addExternalModule.js +5 -14
  63. package/src/addUserToGroup.js +20 -56
  64. package/src/changeAdminStatus.js +20 -44
  65. package/src/changeArchivedStatus.js +11 -25
  66. package/src/{changeAvatarV2.js → changeAvt.js} +2 -3
  67. package/src/changeBio.js +15 -26
  68. package/src/changeBlockedStatus.js +8 -21
  69. package/src/changeGroupImage.js +24 -53
  70. package/src/changeNickname.js +13 -27
  71. package/src/changeThreadColor.js +19 -22
  72. package/src/changeThreadEmoji.js +11 -24
  73. package/src/createNewGroup.js +20 -38
  74. package/src/createPoll.js +16 -27
  75. package/src/deleteMessage.js +13 -24
  76. package/src/deleteThread.js +12 -25
  77. package/src/editMessage.js +53 -71
  78. package/src/forwardAttachment.js +15 -27
  79. package/src/getAccessToken.js +28 -0
  80. package/src/getCurrentUserID.js +1 -1
  81. package/src/getEmojiUrl.js +8 -10
  82. package/src/getFriendsList.js +15 -25
  83. package/src/getMessage.js +81 -813
  84. package/src/getThreadHistory.js +98 -241
  85. package/src/getThreadInfo.js +287 -90
  86. package/src/getThreadInfoOLD.js +422 -0
  87. package/src/getThreadList.js +157 -214
  88. package/src/getThreadMain.js +220 -0
  89. package/src/getThreadPictures.js +19 -39
  90. package/src/getUID.js +50 -113
  91. package/src/getUserID.js +14 -18
  92. package/src/getUserInfo.js +71 -65
  93. package/src/getUserInfoMain.js +65 -0
  94. package/src/getUserInfoV2.js +32 -0
  95. package/src/getUserInfoV3.js +63 -0
  96. package/src/getUserInfoV4.js +55 -0
  97. package/src/getUserInfoV5.js +61 -0
  98. package/src/handleFriendRequest.js +13 -24
  99. package/src/handleMessageRequest.js +20 -36
  100. package/src/httpGet.js +18 -34
  101. package/src/httpPost.js +18 -35
  102. package/src/httpPostFormData.js +24 -53
  103. package/src/listenMqtt.js +944 -680
  104. package/src/listenMqttV1.js +846 -0
  105. package/src/logout.js +15 -22
  106. package/src/markAsDelivered.js +15 -25
  107. package/src/markAsRead.js +27 -45
  108. package/src/markAsReadAll.js +14 -21
  109. package/src/markAsSeen.js +18 -28
  110. package/src/muteThread.js +12 -17
  111. package/src/removeUserFromGroup.js +17 -47
  112. package/src/resolvePhotoUrl.js +13 -21
  113. package/src/searchForThread.js +13 -23
  114. package/src/sendMessage.js +117 -178
  115. package/src/sendMqttMessage.js +71 -0
  116. package/src/sendTypingIndicator.js +16 -46
  117. package/src/setMessageReaction.js +20 -33
  118. package/src/setPostReaction.js +95 -105
  119. package/src/setTitle.js +18 -34
  120. package/src/shareContact.js +37 -92
  121. package/src/shareLink.js +4 -5
  122. package/src/threadColors.js +18 -110
  123. package/src/unfriend.js +9 -18
  124. package/src/unsendMessage.js +34 -31
  125. package/src/unsendMqttMessage.js +66 -0
  126. package/test/Database_Test.js +4 -0
  127. package/test/Db2.js +530 -0
  128. package/test/Shankar_Database/A_README.md +1 -0
  129. package/test/Shankar_Database/Database.db +0 -0
  130. package/test/data/shareAttach.js +1 -1
  131. package/test/env/.env +0 -0
  132. package/test/example-db.db +0 -0
  133. package/test/memoryleak.js +18 -0
  134. package/test/test.js +1 -1
  135. package/test/testname.js +1342 -0
  136. package/test/testv2.js +3 -0
  137. package/utils.js +2918 -1393
  138. package/CHANGELOG.md +0 -2
  139. package/LICENSE-MIT +0 -21
  140. package/generated-icon.png +0 -0
  141. package/replit.nix +0 -5
  142. package/shankar-fca.json +0 -4
  143. package/src/changeAvatar.js +0 -136
  144. package/src/changeBlockedStatusMqtt.js +0 -80
  145. package/src/changeCover.js +0 -73
  146. package/src/changeName.js +0 -79
  147. package/src/changeUsername.js +0 -59
  148. package/src/createCommentPost.js +0 -230
  149. package/src/createPost.js +0 -277
  150. package/src/data/getThreadInfo.json +0 -1
  151. package/src/editMessageOld.js +0 -67
  152. package/src/follow.js +0 -74
  153. package/src/getAccess.js +0 -112
  154. package/src/getAvatarUser.js +0 -78
  155. package/src/getRegion.js +0 -7
  156. package/src/getThreadHistoryDeprecated.js +0 -93
  157. package/src/getThreadInfoDeprecated.js +0 -80
  158. package/src/getThreadListDeprecated.js +0 -75
  159. package/src/listenNotification.js +0 -85
  160. package/src/pinMessage.js +0 -59
  161. package/src/refreshFb_dtsg.js +0 -89
  162. package/src/searchStickers.js +0 -53
  163. package/src/sendMessageMqtt.js +0 -322
  164. package/src/sendTypingIndicatorV2.js +0 -28
  165. package/src/setMessageReactionMqtt.js +0 -62
  166. package/src/setProfileGuard.js +0 -45
  167. package/src/setStoryReaction.js +0 -64
  168. package/src/stopListenMqtt.js +0 -26
  169. package/src/uploadAttachment.js +0 -94
package/src/listenMqtt.js CHANGED
@@ -1,692 +1,956 @@
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
+ }
1
58
 
59
+ function buildStream(options, WebSocket, Proxy) {
60
+ const Stream = Duplexify(undefined, undefined, options);
61
+ Stream.socket = WebSocket;
2
62
 
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');
63
+ WebSocket.onclose = () => {
64
+ Stream.end();
65
+ Stream.destroy();
66
+ };
10
67
 
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
- ];
68
+ WebSocket.onerror = (err) => {
69
+ Stream.destroy(err);
70
+ };
71
+
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());
85
+
86
+ return Stream;
87
+ }
38
88
 
39
89
  function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
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
- });
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);
213
411
  }
214
412
 
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
- }
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
+ }
601
830
  }
602
831
 
603
832
  function markDelivery(ctx, api, threadID, messageID) {
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
- }
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
+ }
616
845
  }
617
846
 
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
- };
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
+ };