alicezetion 1.6.1 → 1.6.3

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