alicezetion 1.7.8 → 1.7.9

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