alicezetion 1.1.5 → 1.1.6

Sign up to get free protection for your applications and to get access to all the features.
@@ -6,672 +6,682 @@ var mqtt = require('mqtt');
6
6
  var websocket = require('websocket-stream');
7
7
  var HttpsProxyAgent = require('https-proxy-agent');
8
8
  const EventEmitter = require('events');
9
-
10
- var identity = function() {};
9
+ var identity = function () { };
11
10
  var form = {};
12
- var getSeqID = function() {};
13
-
14
- var topics = [
15
- "/legacy_web",
16
- "/webrtc",
17
- "/rtc_multi",
18
- "/onevc",
19
- "/br_sr", //Notification
20
- //Need to publish /br_sr right after this
21
- "/sr_res",
22
- "/t_ms",
23
- "/thread_typing",
24
- "/orca_typing_notifications",
25
- "/notify_disconnect",
26
- //Need to publish /messenger_sync_create_queue right after this
27
- "/orca_presence",
28
- //Will receive /sr_res right here.
29
-
30
- "/inbox",
31
- "/mercury",
32
- "/messaging_events",
33
- "/orca_message_notifications",
34
- "/pp",
35
- "/webrtc_response",
11
+ var getSeqID = function () { };
12
+
13
+ var topics = ["/legacy_web",
14
+ "/webrtc",
15
+ "/rtc_multi",
16
+ "/onevc",
17
+ "/br_sr",
18
+ "/sr_res",
19
+ "/t_ms",
20
+ "/thread_typing",
21
+ "/orca_typing_notifications",
22
+ "/notify_disconnect",
23
+ "/orca_presence",
24
+ "/inbox",
25
+ "/mercury",
26
+ "/messaging_events",
27
+ "/orca_message_notifications",
28
+ "/pp",
29
+ "/webrtc_response"
36
30
  ];
37
31
 
32
+ /* [ Noti ? ]
33
+ ! "/br_sr", //Notification
34
+ * => Need to publish /br_sr right after this
35
+
36
+ ! "/notify_disconnect",
37
+ * => Need to publish /messenger_sync_create_queue right after this
38
+
39
+ ! "/orca_presence",
40
+ * => Will receive /sr_res right here.
41
+ */
42
+
38
43
  function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
39
- //Don't really know what this does but I think it's for the active state?
40
- //TODO: Move to ctx when implemented
41
- var chatOn = ctx.globalOptions.online;
42
- var foreground = false;
43
-
44
- var sessionID = Math.floor(Math.random() * 9007199254740991) + 1;
45
- var username = {
46
- u: ctx.userID,
47
- s: sessionID,
48
- chat_on: chatOn,
49
- fg: foreground,
50
- d: utils.getGUID(),
51
- ct: "websocket",
52
- //App id from facebook
53
- aid: "219994525426954",
54
- mqtt_sid: "",
55
- cp: 3,
56
- ecp: 10,
57
- st: [],
58
- pm: [],
59
- dc: "",
60
- no_auto_fg: true,
61
- gas: null,
62
- pack: []
63
- };
64
- var cookies = ctx.jar.getCookies("https://www.facebook.com").join("; ");
65
-
66
- var host;
67
- if (ctx.mqttEndpoint) host = `${ctx.mqttEndpoint}&sid=${sessionID}`;
68
- else if (ctx.region) host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLocaleLowerCase()}&sid=${sessionID}`;
69
- else host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}`;
70
-
71
- var options = {
72
- clientId: "mqttwsclient",
73
- protocolId: 'MQIsdp',
74
- protocolVersion: 3,
75
- username: JSON.stringify(username),
76
- clean: true,
77
- wsOptions: {
78
- headers: {
79
- 'Cookie': cookies,
80
- 'Origin': 'https://www.facebook.com',
81
- 'User-Agent': ctx.globalOptions.userAgent,
82
- 'Referer': 'https://www.facebook.com/',
83
- 'Host': new URL(host).hostname //'edge-chat.facebook.com'
84
- },
85
- origin: 'https://www.facebook.com',
86
- protocolVersion: 13
87
- },
88
- keepalive: 10,
89
- reschedulePings: false
90
- };
91
-
92
- if (typeof ctx.globalOptions.proxy != "undefined") {
93
- var agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
94
- options.wsOptions.agent = agent;
95
- }
96
-
97
- ctx.mqttClient = new mqtt.Client(_ => websocket(host, options.wsOptions), options);
98
-
99
- var mqttClient = ctx.mqttClient;
100
-
101
- mqttClient.on('error', function(err) {
102
- log.error("listenMqtt", "Connection refused: Server unavailable. Exiting...");
103
- mqttClient.end();
104
- process.exit();
105
- if (ctx.globalOptions.autoReconnect) getSeqID();
106
- else {
107
- globalCallback({ type: "stop_listen", error: "Connection refused: Server unavailable" }, null);
108
- }
109
- });
110
-
111
- mqttClient.on('connect', function() {
112
- topics.forEach(topicsub => mqttClient.subscribe(topicsub));
113
-
114
- var topic;
115
- var queue = {
116
- sync_api_version: 10,
117
- max_deltas_able_to_process: 1000,
118
- delta_batch_size: 500,
119
- encoding: "JSON",
120
- entity_fbid: ctx.userID,
121
- };
122
-
123
- if (ctx.syncToken) {
124
- topic = "/messenger_sync_get_diffs";
125
- queue.last_seq_id = ctx.lastSeqId;
126
- queue.sync_token = ctx.syncToken;
127
- } else {
128
- topic = "/messenger_sync_create_queue";
129
- queue.initial_titan_sequence_id = ctx.lastSeqId;
130
- queue.device_params = null;
131
- }
132
-
133
- mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
134
- //onbot
135
- mqttClient.publish("/foreground_state", JSON.stringify({"foreground": chatOn}), {qos: 1});
136
-
137
- var rTimeout = setTimeout(function() {
138
- mqttClient.end();
139
- getSeqID();
140
- }, 5000);
141
-
142
- ctx.tmsWait = function() {
143
- clearTimeout(rTimeout);
144
- ctx.globalOptions.emitReady ? globalCallback({
145
- type: "ready",
146
- error: null
147
- }) : "";
148
- delete ctx.tmsWait;
149
- };
150
- });
151
-
152
- mqttClient.on('message', function(topic, message, _packet) {
153
- try {
154
- var jsonMessage = JSON.parse(message);
155
- } catch (ex) {
156
- return log.error("listenMqtt", "SyntaxError: Unexpected token in JSON at position 0");
157
- }
158
- if (topic === "/t_ms") {
159
- if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
160
-
161
- if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
162
- ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
163
- ctx.syncToken = jsonMessage.syncToken;
164
- }
165
-
166
- if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
167
-
168
- //If it contains more than 1 delta
169
- for (var i in jsonMessage.deltas) {
170
- var delta = jsonMessage.deltas[i];
171
- parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
172
- }
173
- } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
174
- var typ = {
175
- type: "typ",
176
- isTyping: !!jsonMessage.state,
177
- from: jsonMessage.sender_fbid.toString(),
178
- threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
179
- };
180
- (function() { globalCallback(null, typ); })();
181
- } else if (topic === "/orca_presence") {
182
- if (!ctx.globalOptions.updatePresence) {
183
- for (var i in jsonMessage.list) {
184
- var data = jsonMessage.list[i];
185
- var userID = data["u"];
186
-
187
- var presence = {
188
- type: "presence",
189
- userID: userID.toString(),
190
- //Convert to ms
191
- timestamp: data["l"] * 1000,
192
- statuses: data["p"]
193
- };
194
- (function() { globalCallback(null, presence); })();
195
- }
196
- }
197
- }
198
-
199
- });
200
-
201
- mqttClient.on('close', function() {
202
- //(function () { globalCallback("Connection closed."); })();
203
- // client.end();
204
- });
44
+ //Don't really know what this does but I think it's for the active state?
45
+ //TODO: Move to ctx when implemented
46
+ var chatOn = ctx.globalOptions.online;
47
+ var foreground = false;
48
+
49
+ var sessionID = Math.floor(Math.random() * 9007199254740991) + 1;
50
+ var username = {u: ctx.userID,s: sessionID,chat_on: chatOn,fg: foreground,d: utils.getGUID(),ct: "websocket",aid: "219994525426954", mqtt_sid: "",cp: 3,ecp: 10,st: [],pm: [],dc: "",no_auto_fg: true,gas: null,pack: []};
51
+ var cookies = ctx.jar.getCookies('https://www.facebook.com').join("; ");
52
+
53
+ var host;
54
+ if (ctx.mqttEndpoint) host = `${ctx.mqttEndpoint}&sid=${sessionID}`;
55
+ else if (ctx.region) host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLocaleLowerCase()}&sid=${sessionID}`;
56
+ else host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}`;
57
+
58
+ var options = {
59
+ clientId: "mqttwsclient",
60
+ protocolId: 'MQIsdp',
61
+ protocolVersion: 3,
62
+ username: JSON.stringify(username),
63
+ clean: true,
64
+ wsOptions: {
65
+ headers: {
66
+ 'Cookie': cookies,
67
+ 'Origin': 'https://www.facebook.com',
68
+ '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'),
69
+ 'Referer': 'https://www.facebook.com/',
70
+ 'Host': new URL(host).hostname //'edge-chat.facebook.com'
71
+ },
72
+ origin: 'https://www.facebook.com',
73
+ protocolVersion: 13
74
+ },
75
+ keepalive: 10,
76
+ reschedulePings: true,
77
+ connectTimeout: 10000,
78
+ reconnectPeriod: 1000
79
+ };
80
+
81
+ if (typeof ctx.globalOptions.proxy != "undefined") {
82
+ var agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
83
+ options.wsOptions.agent = agent;
84
+ }
85
+
86
+ ctx.mqttClient = new mqtt.Client(_ => websocket(host, options.wsOptions), options);
87
+
88
+ var mqttClient = ctx.mqttClient;
89
+ mqttClient.on('error', function (err) {
90
+ log.error("listenMqtt", err);
91
+ mqttClient.end();
92
+ if (ctx.globalOptions.autoReconnect) getSeqID();
93
+ else {
94
+ globalCallback({
95
+ type: "stop_listen",
96
+ error: "Connection refused: Server unavailable"
97
+ }, null);
98
+ }
99
+ });
100
+
101
+ mqttClient.on('connect', function () {
102
+ topics.forEach(topicsub => mqttClient.subscribe(topicsub));
103
+
104
+ var topic;
105
+ var queue = {
106
+ sync_api_version: 11,
107
+ max_deltas_able_to_process: 100,
108
+ delta_batch_size: 500,
109
+ encoding: "JSON",
110
+ entity_fbid: ctx.userID,
111
+ };
112
+
113
+ if (ctx.syncToken) {
114
+ topic = "/messenger_sync_get_diffs";
115
+ queue.last_seq_id = ctx.lastSeqId;
116
+ queue.sync_token = ctx.syncToken;
117
+ } else {
118
+ topic = "/messenger_sync_create_queue";
119
+ queue.initial_titan_sequence_id = ctx.lastSeqId;
120
+ queue.device_params = null;
121
+ }
122
+ mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
123
+
124
+ // set status online
125
+ // fix by NTKhang
126
+ mqttClient.publish("/foreground_state", JSON.stringify({"foreground": chatOn}), {qos: 1});
127
+
128
+ var rTimeout = setTimeout(function () {
129
+ mqttClient.end();
130
+ getSeqID();
131
+ }, 3000);
132
+
133
+ ctx.tmsWait = function () {
134
+ clearTimeout(rTimeout);
135
+ ctx.globalOptions.emitReady ? globalCallback({type: "ready",error: null}) : '';
136
+ delete ctx.tmsWait;
137
+ };
138
+ });
139
+
140
+ mqttClient.on('message', function (topic, message, _packet) {
141
+ const jsonMessage = JSON.parse(message.toString());
142
+ if (topic === "/t_ms") {
143
+ if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
144
+
145
+ if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
146
+ ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
147
+ ctx.syncToken = jsonMessage.syncToken;
148
+ }
149
+
150
+ if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
151
+ //If it contains more than 1 delta
152
+ for (var i in jsonMessage.deltas) {
153
+ var delta = jsonMessage.deltas[i];
154
+ parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
155
+ }
156
+ } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
157
+ var typ = {
158
+ type: "typ",
159
+ isTyping: !!jsonMessage.state,
160
+ from: jsonMessage.sender_fbid.toString(),
161
+ threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
162
+ };
163
+ (function () { globalCallback(null, typ); })();
164
+ } else if (topic === "/orca_presence") {
165
+ if (!ctx.globalOptions.updatePresence) {
166
+ for (var i in jsonMessage.list) {
167
+ var data = jsonMessage.list[i];
168
+ var userID = data["u"];
169
+
170
+ var presence = {
171
+ type: "presence",
172
+ userID: userID.toString(),
173
+ //Convert to ms
174
+ timestamp: data["l"] * 1000,
175
+ statuses: data["p"]
176
+ };
177
+ (function () { globalCallback(null, presence); })();
178
+ }
179
+ }
180
+ }
181
+
182
+ });
183
+
184
+ process.on('SIGINT', function () {
185
+ process.kill(process.pid);
186
+ });
187
+
188
+ process.on('exit', (code) => {
189
+
190
+ });
191
+
192
+ mqttClient.on('close', function () {
193
+
194
+ });
195
+
196
+ mqttClient.on('disconnect',function () {
197
+ process.exit(7378278);
198
+ });
205
199
  }
206
200
 
207
201
  function parseDelta(defaultFuncs, api, ctx, globalCallback, v) {
208
- if (v.delta.class == "NewMessage") {
209
- //Not tested for pages
210
- if (ctx.globalOptions.pageID && ctx.globalOptions.pageID != v.queue) return;
211
-
212
- (function resolveAttachmentUrl(i) {
213
- if (v.delta.attachments && (i == v.delta.attachments.length)) {
214
- var fmtMsg;
215
- try {
216
- fmtMsg = utils.formatDeltaMessage(v);
217
- } catch (err) {
218
- return globalCallback({
219
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
220
- detail: err,
221
- res: v,
222
- type: "parse_error"
223
- });
224
- }
225
- if (fmtMsg)
226
- if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
227
-
228
- return !ctx.globalOptions.selfListen && fmtMsg.senderID === ctx.userID ? undefined : (function() { globalCallback(null, fmtMsg); })();
229
- } else {
230
- if (v.delta.attachments && (v.delta.attachments[i].mercury.attach_type == "photo")) {
231
- api.resolvePhotoUrl(v.delta.attachments[i].fbid, (err, url) => {
232
- if (!err) v.delta.attachments[i].mercury.metadata.url = url;
233
- return resolveAttachmentUrl(i + 1);
234
- });
235
- } else return resolveAttachmentUrl(i + 1);
236
- }
237
- })(0);
238
- }
239
-
240
- if (v.delta.class == "ClientPayload") {
241
- var clientPayload = utils.decodeClientPayload(v.delta.payload);
242
- if (clientPayload && clientPayload.deltas) {
243
- for (var i in clientPayload.deltas) {
244
- var delta = clientPayload.deltas[i];
245
- if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
246
- (function() {
247
- globalCallback(null, {
248
- type: "message_reaction",
249
- threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
250
- messageID: delta.deltaMessageReaction.messageId,
251
- reaction: delta.deltaMessageReaction.reaction,
252
- senderID: delta.deltaMessageReaction.senderId.toString(),
253
- userID: delta.deltaMessageReaction.userId.toString()
254
- });
255
- })();
256
- } else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
257
- (function() {
258
- globalCallback(null, {
259
- type: "message_unsend",
260
- threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
261
- messageID: delta.deltaRecallMessageData.messageID,
262
- senderID: delta.deltaRecallMessageData.senderID.toString(),
263
- deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
264
- timestamp: delta.deltaRecallMessageData.timestamp
265
- });
266
- })();
267
- } else if (delta.deltaMessageReply) {
268
- //Mention block - #1
269
- var mdata = delta.deltaMessageReply.message === undefined ? [] :
270
- delta.deltaMessageReply.message.data === undefined ? [] :
271
- delta.deltaMessageReply.message.data.prng === undefined ? [] :
272
- JSON.parse(delta.deltaMessageReply.message.data.prng);
273
- var m_id = mdata.map(u => u.i);
274
- var m_offset = mdata.map(u => u.o);
275
- var m_length = mdata.map(u => u.l);
276
-
277
- var mentions = {};
278
-
279
- 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]);
280
- //Mention block - 1#
281
- var callbackToReturn = {
282
- type: "message_reply",
283
- threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
284
- messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
285
- senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
286
- attachments: delta.deltaMessageReply.message.attachments.map(function(att) {
287
- var mercury = JSON.parse(att.mercuryJSON);
288
- Object.assign(att, mercury);
289
- return att;
290
- }).map(att => {
291
- var x;
292
- try {
293
- x = utils._formatAttachment(att);
294
- } catch (ex) {
295
- x = att;
296
- x.error = ex;
297
- x.type = "unknown";
298
- }
299
- return x;
300
- }),
301
- args: (delta.deltaMessageReply.message.body || "").trim().split(/\s+/),
302
- body: (delta.deltaMessageReply.message.body || ""),
303
- isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
304
- mentions: mentions,
305
- timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
306
- participantIDs: (delta.deltaMessageReply.message.participants || []).map(e => e.toString())
307
- };
308
-
309
- if (delta.deltaMessageReply.repliedToMessage) {
310
- //Mention block - #2
311
- mdata = delta.deltaMessageReply.repliedToMessage === undefined ? [] :
312
- delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
313
- delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
314
- JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
315
- m_id = mdata.map(u => u.i);
316
- m_offset = mdata.map(u => u.o);
317
- m_length = mdata.map(u => u.l);
318
-
319
- var rmentions = {};
320
-
321
- 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]);
322
- //Mention block - 2#
323
- callbackToReturn.messageReply = {
324
- threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
325
- messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
326
- senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
327
- attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function(att) {
328
- var mercury = JSON.parse(att.mercuryJSON);
329
- Object.assign(att, mercury);
330
- return att;
331
- }).map(att => {
332
- var x;
333
- try {
334
- x = utils._formatAttachment(att);
335
- } catch (ex) {
336
- x = att;
337
- x.error = ex;
338
- x.type = "unknown";
339
- }
340
- return x;
341
- }),
342
- args: (delta.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
343
- body: delta.deltaMessageReply.repliedToMessage.body || "",
344
- isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
345
- mentions: rmentions,
346
- timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp,
347
- participantIDs: (delta.deltaMessageReply.repliedToMessage.participants || []).map(e => e.toString())
348
- };
349
- } else if (delta.deltaMessageReply.replyToMessageId) {
350
- return defaultFuncs
351
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
352
- "av": ctx.globalOptions.pageID,
353
- "queries": JSON.stringify({
354
- "o0": {
355
- //Using the same doc_id as forcedFetch
356
- "doc_id": "2848441488556444",
357
- "query_params": {
358
- "thread_and_message_id": {
359
- "thread_id": callbackToReturn.threadID,
360
- "message_id": delta.deltaMessageReply.replyToMessageId.id,
361
- }
362
- }
363
- }
364
- })
365
- })
366
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
367
- .then((resData) => {
368
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
369
- if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
370
- var fetchData = resData[0].o0.data.message;
371
- var mobj = {};
372
- 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);
373
-
374
- callbackToReturn.messageReply = {
375
- threadID: callbackToReturn.threadID,
376
- messageID: fetchData.message_id,
377
- senderID: fetchData.message_sender.id.toString(),
378
- attachments: fetchData.message.blob_attachment.map(att => {
379
- var x;
380
- try {
381
- x = utils._formatAttachment({ blob_attachment: att });
382
- } catch (ex) {
383
- x = att;
384
- x.error = ex;
385
- x.type = "unknown";
386
- }
387
- return x;
388
- }),
389
- args: (fetchData.message.text || "").trim().split(/\s+/) || [],
390
- body: fetchData.message.text || "",
391
- isGroup: callbackToReturn.isGroup,
392
- mentions: mobj,
393
- timestamp: parseInt(fetchData.timestamp_precise)
394
- };
395
- })
396
- .catch(err => log.error("forcedFetch", err))
397
- .finally(function() {
398
- if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
399
- !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function() { globalCallback(null, callbackToReturn); })();
400
- });
401
- } else callbackToReturn.delta = delta;
402
-
403
- if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
404
-
405
- return !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function() { globalCallback(null, callbackToReturn); })();
406
- }
407
- }
408
- return;
409
- }
410
- }
411
-
412
- if (v.delta.class !== "NewMessage" && !ctx.globalOptions.listenEvents) return;
413
- switch (v.delta.class) {
414
- case "ReadReceipt":
415
- var fmtMsg;
416
- try {
417
- fmtMsg = utils.formatDeltaReadReceipt(v.delta);
418
- } catch (err) {
419
- return globalCallback({
420
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
421
- detail: err,
422
- res: v.delta,
423
- type: "parse_error"
424
- });
425
- }
426
- return (function() { globalCallback(null, fmtMsg); })();
427
- case "AdminTextMessage":
428
- switch (v.delta.type) {
429
- case "change_thread_theme":
430
- case "change_thread_icon":
431
- case "change_thread_nickname":
432
- case "change_thread_admins":
433
- case "change_thread_approval_mode":
434
- case "group_poll":
435
- case "messenger_call_log":
436
- case "participant_joined_group_call":
437
- var fmtMsg;
438
- try {
439
- fmtMsg = utils.formatDeltaEvent(v.delta);
440
- } catch (err) {
441
- return globalCallback({
442
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
443
- detail: err,
444
- res: v.delta,
445
- type: "parse_error"
446
- });
447
- }
448
- return (function() { globalCallback(null, fmtMsg); })();
449
- default:
450
- return;
451
- }
452
- break;
453
- //For group images
454
- case "ForcedFetch":
455
- if (!v.delta.threadKey) return;
456
- var mid = v.delta.messageId;
457
- var tid = v.delta.threadKey.threadFbId;
458
- if (mid && tid) {
459
- const form = {
460
- "av": ctx.globalOptions.pageID,
461
- "queries": JSON.stringify({
462
- "o0": {
463
- //This doc_id is valid as of March 25, 2020
464
- "doc_id": "2848441488556444",
465
- "query_params": {
466
- "thread_and_message_id": {
467
- "thread_id": tid.toString(),
468
- "message_id": mid,
469
- }
470
- }
471
- }
472
- })
473
- };
474
-
475
- defaultFuncs
476
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
477
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
478
- .then((resData) => {
479
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
480
-
481
- if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
482
-
483
- var fetchData = resData[0].o0.data.message;
484
-
485
- if (utils.getType(fetchData) == "Object") {
486
- log.info("forcedFetch", fetchData);
487
- switch (fetchData.__typename) {
488
- case "ThreadImageMessage":
489
- (!ctx.globalOptions.selfListen && fetchData.message_sender.id.toString() === ctx.userID) ||
490
- !ctx.loggedIn ? undefined : (function() {
491
- globalCallback(null, {
492
- type: "change_thread_image",
493
- threadID: utils.formatID(tid.toString()),
494
- snippet: fetchData.snippet,
495
- timestamp: fetchData.timestamp_precise,
496
- author: fetchData.message_sender.id,
497
- image: {
498
- attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
499
- width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
500
- height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
501
- url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
502
- }
503
- });
504
- })();
505
- break;
506
- case "UserMessage":
507
- log.info("ff-Return", {
508
- type: "message",
509
- senderID: utils.formatID(fetchData.message_sender.id),
510
- body: fetchData.message.text || "",
511
- threadID: utils.formatID(tid.toString()),
512
- messageID: fetchData.message_id,
513
- attachments: [{
514
- type: "share",
515
- ID: fetchData.extensible_attachment.legacy_attachment_id,
516
- url: fetchData.extensible_attachment.story_attachment.url,
517
-
518
- title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
519
- description: fetchData.extensible_attachment.story_attachment.description.text,
520
- source: fetchData.extensible_attachment.story_attachment.source,
521
-
522
- image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
523
- width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
524
- height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
525
- playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
526
- duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
527
-
528
- subattachments: fetchData.extensible_attachment.subattachments,
529
- properties: fetchData.extensible_attachment.story_attachment.properties,
530
- }],
531
- mentions: {},
532
- timestamp: parseInt(fetchData.timestamp_precise),
533
- isGroup: (fetchData.message_sender.id != tid.toString())
534
- });
535
- globalCallback(null, {
536
- type: "message",
537
- senderID: utils.formatID(fetchData.message_sender.id),
538
- body: fetchData.message.text || "",
539
- threadID: utils.formatID(tid.toString()),
540
- messageID: fetchData.message_id,
541
- attachments: [{
542
- type: "share",
543
- ID: fetchData.extensible_attachment.legacy_attachment_id,
544
- url: fetchData.extensible_attachment.story_attachment.url,
545
-
546
- title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
547
- description: fetchData.extensible_attachment.story_attachment.description.text,
548
- source: fetchData.extensible_attachment.story_attachment.source,
549
-
550
- image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
551
- width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
552
- height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
553
- playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
554
- duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
555
-
556
- subattachments: fetchData.extensible_attachment.subattachments,
557
- properties: fetchData.extensible_attachment.story_attachment.properties,
558
- }],
559
- mentions: {},
560
- timestamp: parseInt(fetchData.timestamp_precise),
561
- isGroup: (fetchData.message_sender.id != tid.toString())
562
- });
563
- }
564
- } else log.error("forcedFetch", fetchData);
565
- })
566
- .catch((err) => log.error("forcedFetch", err));
567
- }
568
- break;
569
- case "ThreadName":
570
- case "ParticipantsAddedToGroupThread":
571
- case "ParticipantLeftGroupThread":
572
- var formattedEvent;
573
- try {
574
- formattedEvent = utils.formatDeltaEvent(v.delta);
575
- } catch (err) {
576
- return globalCallback({
577
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
578
- detail: err,
579
- res: v.delta,
580
- type: "parse_error"
581
- });
582
- }
583
- return (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) || !ctx.loggedIn ? undefined : (function() { globalCallback(null, formattedEvent); })();
584
- }
202
+ if (v.delta.class == "NewMessage") {
203
+ //Not tested for pages
204
+ if (ctx.globalOptions.pageID && ctx.globalOptions.pageID != v.queue) return;
205
+
206
+ (function resolveAttachmentUrl(i) {
207
+ if (v.delta.attachments && (i == v.delta.attachments.length)) {
208
+ var fmtMsg;
209
+ try {
210
+ fmtMsg = utils.formatDeltaMessage(v);
211
+ } catch (err) {
212
+ return globalCallback({
213
+ error: "Problem parsing message object. Please open an issue at https://github.com/VangBanLaNhat/fca-unofficial/issues.",
214
+ detail: err,
215
+ res: v,
216
+ type: "parse_error"
217
+ });
218
+ }
219
+
220
+ if (fmtMsg)
221
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
222
+
223
+ return !ctx.globalOptions.selfListen && fmtMsg.senderID === ctx.userID ? undefined : (function () { globalCallback(null, fmtMsg); })();
224
+ } else {
225
+ if (v.delta.attachments && (v.delta.attachments[i].mercury.attach_type == "photo")) {
226
+ api.resolvePhotoUrl(v.delta.attachments[i].fbid, (err, url) => {
227
+ if (!err) v.delta.attachments[i].mercury.metadata.url = url;
228
+ return resolveAttachmentUrl(i + 1);
229
+ });
230
+ } else return resolveAttachmentUrl(i + 1);
231
+ }
232
+ })(0);
233
+ }
234
+
235
+ if (v.delta.class == "ClientPayload") {
236
+ var clientPayload = utils.decodeClientPayload(v.delta.payload);
237
+ if (clientPayload && clientPayload.deltas) {
238
+ for (var i in clientPayload.deltas) {
239
+ var delta = clientPayload.deltas[i];
240
+ if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
241
+ (function () {
242
+ globalCallback(null, {
243
+ type: "message_reaction",
244
+ threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
245
+ messageID: delta.deltaMessageReaction.messageId,
246
+ reaction: delta.deltaMessageReaction.reaction,
247
+ senderID: delta.deltaMessageReaction.senderId.toString(),
248
+ userID: delta.deltaMessageReaction.userId.toString()
249
+ });
250
+ })();
251
+ } else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
252
+ (function () {
253
+ globalCallback(null, {
254
+ type: "message_unsend",
255
+ threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
256
+ messageID: delta.deltaRecallMessageData.messageID,
257
+ senderID: delta.deltaRecallMessageData.senderID.toString(),
258
+ deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
259
+ timestamp: delta.deltaRecallMessageData.timestamp
260
+ });
261
+ })();
262
+ } else if (delta.deltaMessageReply) {
263
+ //Mention block - #1
264
+ var mdata =
265
+ delta.deltaMessageReply.message === undefined ? [] :
266
+ delta.deltaMessageReply.message.data === undefined ? [] :
267
+ delta.deltaMessageReply.message.data.prng === undefined ? [] :
268
+ JSON.parse(delta.deltaMessageReply.message.data.prng);
269
+ var m_id = mdata.map(u => u.i);
270
+ var m_offset = mdata.map(u => u.o);
271
+ var m_length = mdata.map(u => u.l);
272
+
273
+ var mentions = {};
274
+
275
+ 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]);
276
+ //Mention block - 1#
277
+ var callbackToReturn = {
278
+ type: "message_reply",
279
+ threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
280
+ messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
281
+ senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
282
+ attachments: delta.deltaMessageReply.message.attachments.map(function (att) {
283
+ var mercury = JSON.parse(att.mercuryJSON);
284
+ Object.assign(att, mercury);
285
+ return att;
286
+ }).map(att => {
287
+ var x;
288
+ try {
289
+ x = utils._formatAttachment(att);
290
+ } catch (ex) {
291
+ x = att;
292
+ x.error = ex;
293
+ x.type = "unknown";
294
+ }
295
+ return x;
296
+ }),
297
+ args: (delta.deltaMessageReply.message.body || "").trim().split(/\s+/),
298
+ body: (delta.deltaMessageReply.message.body || ""),
299
+ isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
300
+ mentions: mentions,
301
+ timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
302
+ participantIDs: (delta.deltaMessageReply.message.participants || []).map(e => e.toString())
303
+ };
304
+
305
+ if (delta.deltaMessageReply.repliedToMessage) {
306
+ //Mention block - #2
307
+ mdata =
308
+ delta.deltaMessageReply.repliedToMessage === undefined ? [] :
309
+ delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
310
+ delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
311
+ JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
312
+ m_id = mdata.map(u => u.i);
313
+ m_offset = mdata.map(u => u.o);
314
+ m_length = mdata.map(u => u.l);
315
+
316
+ var rmentions = {};
317
+
318
+ 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]);
319
+ //Mention block - 2#
320
+ callbackToReturn.messageReply = {
321
+ threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
322
+ messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
323
+ senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
324
+ attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function (att) {
325
+ var mercury = JSON.parse(att.mercuryJSON);
326
+ Object.assign(att, mercury);
327
+ return att;
328
+ }).map(att => {
329
+ var x;
330
+ try {
331
+ x = utils._formatAttachment(att);
332
+ } catch (ex) {
333
+ x = att;
334
+ x.error = ex;
335
+ x.type = "unknown";
336
+ }
337
+ return x;
338
+ }),
339
+ args: (delta.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
340
+ body: delta.deltaMessageReply.repliedToMessage.body || "",
341
+ isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
342
+ mentions: rmentions,
343
+ timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp,
344
+ participantIDs: (delta.deltaMessageReply.repliedToMessage.participants || []).map(e => e.toString())
345
+ };
346
+ } else if (delta.deltaMessageReply.replyToMessageId) {
347
+ return defaultFuncs
348
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
349
+ "av": ctx.globalOptions.pageID,
350
+ "queries": JSON.stringify({
351
+ "o0": {
352
+ //Using the same doc_id as forcedFetch
353
+ "doc_id": "2848441488556444",
354
+ "query_params": {
355
+ "thread_and_message_id": {
356
+ "thread_id": callbackToReturn.threadID,
357
+ "message_id": delta.deltaMessageReply.replyToMessageId.id,
358
+ }
359
+ }
360
+ }
361
+ })
362
+ })
363
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
364
+ .then((resData) => {
365
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
366
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
367
+ var fetchData = resData[0].o0.data.message;
368
+ var mobj = {};
369
+ 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);
370
+
371
+ callbackToReturn.messageReply = {
372
+ type: "Message",
373
+ threadID: callbackToReturn.threadID,
374
+ messageID: fetchData.message_id,
375
+ senderID: fetchData.message_sender.id.toString(),
376
+ attachments: fetchData.message.blob_attachment.map(att => {
377
+ var x;
378
+ try {
379
+ x = utils._formatAttachment({ blob_attachment: att });
380
+ } catch (ex) {
381
+ x = att;
382
+ x.error = ex;
383
+ x.type = "unknown";
384
+ }
385
+ return x;
386
+ }),
387
+ args: (fetchData.message.text || "").trim().split(/\s+/) || [],
388
+ body: fetchData.message.text || "",
389
+ isGroup: callbackToReturn.isGroup,
390
+ mentions: mobj,
391
+ timestamp: parseInt(fetchData.timestamp_precise)
392
+ };
393
+ })
394
+ .catch(err => log.error("forcedFetch", err))
395
+ .finally(function () {
396
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
397
+ !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
398
+ });
399
+ } else callbackToReturn.delta = delta;
400
+
401
+ if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
402
+
403
+ return !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
404
+ }
405
+ }
406
+ return;
407
+ }
408
+ }
409
+
410
+ if (v.delta.class !== "NewMessage" && !ctx.globalOptions.listenEvents) return;
411
+ switch (v.delta.class) {
412
+ case "ReadReceipt":
413
+ var fmtMsg;
414
+ try {
415
+ fmtMsg = utils.formatDeltaReadReceipt(v.delta);
416
+ } catch (err) {
417
+ return globalCallback({
418
+ error: "Problem parsing message object. Please open an issue at https://github.com/VangBanLaNhat/fca-unofficial/issues.",
419
+ detail: err,
420
+ res: v.delta,
421
+ type: "parse_error"
422
+ });
423
+ }
424
+ return (function () { globalCallback(null, fmtMsg); })();
425
+ case "AdminTextMessage":
426
+ switch (v.delta.type) {
427
+ case "joinable_group_link_mode_change":
428
+ case "magic_words":
429
+ case "change_thread_theme":
430
+ case "change_thread_icon":
431
+ case "change_thread_nickname":
432
+ case "change_thread_admins":
433
+ case "change_thread_approval_mode":
434
+ case "group_poll":
435
+ case "messenger_call_log":
436
+ case "participant_joined_group_call":
437
+ var fmtMsg;
438
+ try {
439
+ fmtMsg = utils.formatDeltaEvent(v.delta);
440
+ } catch (err) {
441
+ return globalCallback({
442
+ error: "Problem parsing message object. Please open an issue at https://github.com/VangBanLaNhat/fca-unofficial/issues.",
443
+ detail: err,
444
+ res: v.delta,
445
+ type: "parse_error"
446
+ });
447
+ }
448
+ return (function () { globalCallback(null, fmtMsg); })();
449
+ default:
450
+ return;
451
+ }
452
+ //For group images
453
+ case "ForcedFetch":
454
+ if (!v.delta.threadKey) return;
455
+ var mid = v.delta.messageId;
456
+ var tid = v.delta.threadKey.threadFbId;
457
+ if (mid && tid) {
458
+ const form = {
459
+ "av": ctx.globalOptions.pageID,
460
+ "queries": JSON.stringify({
461
+ "o0": {
462
+ //This doc_id is valid as of March 25, 2020
463
+ "doc_id": "2848441488556444",
464
+ "query_params": {
465
+ "thread_and_message_id": {
466
+ "thread_id": tid.toString(),
467
+ "message_id": mid,
468
+ }
469
+ }
470
+ }
471
+ })
472
+ };
473
+
474
+ defaultFuncs
475
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
476
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
477
+ .then((resData) => {
478
+ if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
479
+
480
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
481
+
482
+ var fetchData = resData[0].o0.data.message;
483
+
484
+ if (utils.getType(fetchData) == "Object") {
485
+ log.info("forcedFetch", fetchData);
486
+ switch (fetchData.__typename) {
487
+ case "ThreadImageMessage":
488
+ (!ctx.globalOptions.selfListen &&
489
+ fetchData.message_sender.id.toString() === ctx.userID) ||
490
+ !ctx.loggedIn ?
491
+ undefined :
492
+ (function () {
493
+ globalCallback(null, {
494
+ type: "change_thread_image",
495
+ threadID: utils.formatID(tid.toString()),
496
+ snippet: fetchData.snippet,
497
+ timestamp: fetchData.timestamp_precise,
498
+ author: fetchData.message_sender.id,
499
+ image: {
500
+ attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
501
+ width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
502
+ height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
503
+ url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
504
+ }
505
+ });
506
+ })();
507
+ break;
508
+ case "UserMessage":
509
+ log.info("ff-Return", {
510
+ type: "message",
511
+ senderID: utils.formatID(fetchData.message_sender.id),
512
+ body: fetchData.message.text || "",
513
+ threadID: utils.formatID(tid.toString()),
514
+ messageID: fetchData.message_id,
515
+ attachments: [{
516
+ type: "share",
517
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
518
+ url: fetchData.extensible_attachment.story_attachment.url,
519
+
520
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
521
+ description: fetchData.extensible_attachment.story_attachment.description.text,
522
+ source: fetchData.extensible_attachment.story_attachment.source,
523
+
524
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
525
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
526
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
527
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
528
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
529
+
530
+ subattachments: fetchData.extensible_attachment.subattachments,
531
+ properties: fetchData.extensible_attachment.story_attachment.properties,
532
+ }],
533
+ mentions: {},
534
+ timestamp: parseInt(fetchData.timestamp_precise),
535
+ isGroup: (fetchData.message_sender.id != tid.toString())
536
+ });
537
+ globalCallback(null, {
538
+ type: "message",
539
+ senderID: utils.formatID(fetchData.message_sender.id),
540
+ body: fetchData.message.text || "",
541
+ threadID: utils.formatID(tid.toString()),
542
+ messageID: fetchData.message_id,
543
+ attachments: [{
544
+ type: "share",
545
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
546
+ url: fetchData.extensible_attachment.story_attachment.url,
547
+
548
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
549
+ description: fetchData.extensible_attachment.story_attachment.description.text,
550
+ source: fetchData.extensible_attachment.story_attachment.source,
551
+
552
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
553
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
554
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
555
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
556
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
557
+
558
+ subattachments: fetchData.extensible_attachment.subattachments,
559
+ properties: fetchData.extensible_attachment.story_attachment.properties,
560
+ }],
561
+ mentions: {},
562
+ timestamp: parseInt(fetchData.timestamp_precise),
563
+ isGroup: (fetchData.message_sender.id != tid.toString())
564
+ });
565
+ }
566
+ } else log.error("forcedFetch", fetchData);
567
+ })
568
+ .catch((err) => log.error("forcedFetch", err));
569
+ }
570
+ break;
571
+ case "ThreadName":
572
+ case "ParticipantsAddedToGroupThread":
573
+ case "ParticipantLeftGroupThread":
574
+ var formattedEvent;
575
+ try {
576
+ formattedEvent = utils.formatDeltaEvent(v.delta);
577
+ } catch (err) {
578
+ return log.error("Lỗi Nhẹ", err);
579
+ }
580
+ return (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) || !ctx.loggedIn ? undefined : (function () { globalCallback(null, formattedEvent); })();
581
+ }
585
582
  }
586
583
 
584
+
587
585
  function markDelivery(ctx, api, threadID, messageID) {
588
- if (threadID && messageID) {
589
- api.markAsDelivered(threadID, messageID, (err) => {
590
- if (err) {
591
- log.error("markAsDelivered", err);
592
- } else {
593
- if (ctx.globalOptions.autoMarkRead) {
594
- api.markAsRead(threadID, (err) => {
595
- if (err) log.error("markAsDelivered", err);
596
- });
597
- }
598
- }
599
- });
600
- }
586
+ if (threadID && messageID) {
587
+ api.markAsDelivered(threadID, messageID, (err) => {
588
+ if (err) log.error("markAsDelivered", err);
589
+ else {
590
+ if (ctx.globalOptions.autoMarkRead) {
591
+ api.markAsRead(threadID, (err) => {
592
+ if (err) log.error("markAsDelivered", err);
593
+ });
594
+ }
595
+ }
596
+ });
597
+ }
601
598
  }
602
599
 
603
- module.exports = function(defaultFuncs, api, ctx) {
604
- var globalCallback = identity;
605
- getSeqID = function getSeqID() {
606
- ctx.t_mqttCalled = false;
607
- defaultFuncs
608
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
609
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
610
- .then((resData) => {
611
- if (utils.getType(resData) != "Array") throw { error: "Not logged in", res: resData };
612
- if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
613
- if (resData[resData.length - 1].successful_results === 0) throw { error: "getSeqId: there was no successful_results", res: resData };
614
- if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
615
- ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
616
- listenMqtt(defaultFuncs, api, ctx, globalCallback);
617
- } else throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
618
- })
619
- .catch((err) => {
620
- log.error("getSeqId", err);
621
- if (utils.getType(err) == "Object" && err.error === "Not logged in") ctx.loggedIn = false;
622
- return globalCallback(err);
623
- });
624
- };
625
-
626
- return function(callback) {
627
- class MessageEmitter extends EventEmitter {
628
- stopListening(callback) {
629
- callback = callback || (() => {});
630
- globalCallback = identity;
631
- if (ctx.mqttClient) {
632
- ctx.mqttClient.unsubscribe("/webrtc");
633
- ctx.mqttClient.unsubscribe("/rtc_multi");
634
- ctx.mqttClient.unsubscribe("/onevc");
635
- ctx.mqttClient.publish("/browser_close", "{}");
636
- ctx.mqttClient.end(false, function(...data) {
637
- callback(data);
638
- ctx.mqttClient = undefined;
639
- });
640
- }
641
- }
642
- }
643
-
644
- var msgEmitter = new MessageEmitter();
645
- globalCallback = (callback || function(error, message) {
646
- if (error) return msgEmitter.emit("error", error);
647
- msgEmitter.emit("message", message);
648
- });
649
-
650
- //Reset some stuff
651
- if (!ctx.firstListen) ctx.lastSeqId = null;
652
- ctx.syncToken = undefined;
653
- ctx.t_mqttCalled = false;
654
-
655
- //Same request as getThreadList
656
- form = {
657
- "av": ctx.globalOptions.pageID,
658
- "queries": JSON.stringify({
659
- "o0": {
660
- "doc_id": "3336396659757871",
661
- "query_params": {
662
- "limit": 1,
663
- "before": null,
664
- "tags": ["INBOX"],
665
- "includeDeliveryReceipts": false,
666
- "includeSeqID": true
667
- }
668
- }
669
- })
670
- };
671
-
672
- if (!ctx.firstListen || !ctx.lastSeqId) getSeqID();
673
- else listenMqtt(defaultFuncs, api, ctx, globalCallback);
674
- ctx.firstListen = false;
675
- return msgEmitter;
676
- };
600
+ module.exports = function (defaultFuncs, api, ctx) {
601
+ var globalCallback = identity;
602
+ getSeqID = function getSeqID() {
603
+ ctx.t_mqttCalled = false;
604
+ defaultFuncs
605
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
606
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
607
+ .then((resData) => {
608
+ if (utils.getType(resData) != "Array") {
609
+ switch (global.Fca.Require.FastConfig.AutoLogin) {
610
+ case true: {
611
+ global.Fca.Require.logger.Warning(global.Fca.Require.Language.Index.AutoLogin, function() {
612
+ return global.Fca.AutoLogin();
613
+ });
614
+ break;
615
+ }
616
+ case false: {
617
+ throw { error: global.Fca.Require.Language.Index.ErrAppState };
618
+
619
+ }
620
+ }
621
+ }
622
+ if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
623
+ if (resData[resData.length - 1].successful_results === 0) throw { error: "getSeqId: there was no successful_results", res: resData };
624
+ if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
625
+ ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
626
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
627
+ } else throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
628
+ })
629
+ .catch((err) => {
630
+ log.error("getSeqId", err);
631
+ if (utils.getType(err) == "Object" && err.error === global.Fca.Require.Language.Index.ErrAppState) ctx.loggedIn = false;
632
+ return globalCallback(err);
633
+ });
634
+ };
635
+
636
+ return function (callback) {
637
+ class MessageEmitter extends EventEmitter {
638
+ stopListening(callback) {
639
+ callback = callback || (() => { });
640
+ globalCallback = identity;
641
+ if (ctx.mqttClient) {
642
+ ctx.mqttClient.unsubscribe("/webrtc");
643
+ ctx.mqttClient.unsubscribe("/rtc_multi");
644
+ ctx.mqttClient.unsubscribe("/onevc");
645
+ ctx.mqttClient.publish("/browser_close", "{}");
646
+ ctx.mqttClient.end(false, function (...data) {
647
+ callback(data);
648
+ ctx.mqttClient = undefined;
649
+ });
650
+ }
651
+ }
652
+ }
653
+
654
+ var msgEmitter = new MessageEmitter();
655
+ globalCallback = (callback || function (error, message) {
656
+ if (error) return msgEmitter.emit("error", error);
657
+ msgEmitter.emit("message", message);
658
+ });
659
+
660
+ //Reset some stuff
661
+ if (!ctx.firstListen) ctx.lastSeqId = null;
662
+ ctx.syncToken = undefined;
663
+ ctx.t_mqttCalled = false;
664
+
665
+ //Same request as getThreadList
666
+ form = {
667
+ "av": ctx.globalOptions.pageID,
668
+ "queries": JSON.stringify({
669
+ "o0": {
670
+ "doc_id": "3336396659757871",
671
+ "query_params": {
672
+ "limit": 1,
673
+ "before": null,
674
+ "tags": ["INBOX"],
675
+ "includeDeliveryReceipts": false,
676
+ "includeSeqID": true
677
+ }
678
+ }
679
+ })
680
+ };
681
+
682
+ if (!ctx.firstListen || !ctx.lastSeqId) getSeqID();
683
+ else listenMqtt(defaultFuncs, api, ctx, globalCallback);
684
+ ctx.firstListen = false;
685
+ return msgEmitter;
686
+ };
677
687
  };