alicezetion 1.5.6 → 1.5.8

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 +489 -526
  4. package/leiamnash/addExternalModule.js +15 -19
  5. package/leiamnash/addUserToGroup.js +77 -113
  6. package/leiamnash/changeAdminStatus.js +47 -79
  7. package/leiamnash/changeArchivedStatus.js +41 -55
  8. package/leiamnash/changeBio.js +64 -77
  9. package/leiamnash/changeBlockedStatus.js +36 -47
  10. package/leiamnash/changeGroupImage.js +105 -129
  11. package/leiamnash/changeNickname.js +43 -59
  12. package/leiamnash/changeThreadColor.js +61 -71
  13. package/leiamnash/changeThreadEmoji.js +41 -55
  14. package/leiamnash/chat.js +324 -459
  15. package/leiamnash/createNewGroup.js +70 -86
  16. package/leiamnash/createPoll.js +59 -71
  17. package/leiamnash/deleteMessage.js +44 -56
  18. package/leiamnash/deleteThread.js +42 -56
  19. package/leiamnash/forwardAttachment.js +47 -60
  20. package/leiamnash/forwardMessage.js +0 -0
  21. package/leiamnash/getCurrentUserID.js +7 -7
  22. package/leiamnash/getEmojiUrl.js +27 -29
  23. package/leiamnash/getFriendsList.js +73 -84
  24. package/leiamnash/getThreadHistory.js +537 -645
  25. package/leiamnash/getThreadHistoryDeprecated.js +71 -93
  26. package/leiamnash/getThreadInfo.js +171 -206
  27. package/leiamnash/getThreadInfoDeprecated.js +56 -80
  28. package/leiamnash/getThreadList.js +213 -238
  29. package/leiamnash/getThreadListDeprecated.js +46 -75
  30. package/leiamnash/getThreadPictures.js +59 -79
  31. package/leiamnash/getUserID.js +61 -66
  32. package/leiamnash/getUserInfo.js +66 -72
  33. package/leiamnash/handleFriendRequest.js +46 -61
  34. package/leiamnash/handleMessageRequest.js +47 -65
  35. package/leiamnash/httpGet.js +47 -52
  36. package/leiamnash/httpPost.js +47 -52
  37. package/leiamnash/listen.js +553 -0
  38. package/leiamnash/listenMqtt-Test.js +687 -0
  39. package/leiamnash/listenMqtt.js +685 -789
  40. package/leiamnash/logout.js +68 -75
  41. package/leiamnash/markAsDelivered.js +47 -58
  42. package/leiamnash/markAsRead.js +70 -80
  43. package/leiamnash/markAsReadAll.js +39 -49
  44. package/leiamnash/markAsSeen.js +48 -59
  45. package/leiamnash/muteThread.js +45 -52
  46. package/leiamnash/removeUserFromGroup.js +45 -79
  47. package/leiamnash/resolvePhotoUrl.js +36 -45
  48. package/leiamnash/searchForThread.js +42 -53
  49. package/leiamnash/sendTypingIndicator.js +70 -103
  50. package/leiamnash/setMessageReaction.js +103 -117
  51. package/leiamnash/setPostReaction.js +63 -76
  52. package/leiamnash/setTitle.js +70 -86
  53. package/leiamnash/threadColors.js +41 -57
  54. package/leiamnash/unfriend.js +42 -52
  55. package/leiamnash/unsendMessage.js +39 -49
  56. package/package.json +73 -71
  57. package/utils.js +1198 -1356
@@ -1,789 +1,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
- };
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
+ };