@dongdev/fca-unofficial 0.0.4 → 0.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (76) hide show
  1. package/.travis.yml +6 -6
  2. package/CHANGELOG.md +1 -1
  3. package/DOCS.md +1738 -1738
  4. package/LICENSE-MIT +21 -21
  5. package/README.md +219 -219
  6. package/index.js +307 -571
  7. package/lib/login.js +0 -0
  8. package/package.json +6 -5
  9. package/src/addExternalModule.js +19 -15
  10. package/src/addUserToGroup.js +113 -77
  11. package/src/changeAdminStatus.js +79 -47
  12. package/src/changeArchivedStatus.js +55 -41
  13. package/src/changeAvatar.js +126 -0
  14. package/src/changeBio.js +66 -54
  15. package/src/changeBlockedStatus.js +40 -29
  16. package/src/changeGroupImage.js +127 -101
  17. package/src/changeNickname.js +50 -36
  18. package/src/changeThreadColor.js +65 -61
  19. package/src/changeThreadEmoji.js +55 -41
  20. package/src/createNewGroup.js +86 -70
  21. package/src/createPoll.js +71 -59
  22. package/src/deleteMessage.js +56 -44
  23. package/src/deleteThread.js +56 -42
  24. package/src/forwardAttachment.js +60 -47
  25. package/src/getCurrentUserID.js +7 -7
  26. package/src/getEmojiUrl.js +29 -27
  27. package/src/getFriendsList.js +83 -73
  28. package/src/getMessage.js +796 -0
  29. package/src/getThreadHistory.js +666 -537
  30. package/src/getThreadInfo.js +232 -171
  31. package/src/getThreadList.js +192 -213
  32. package/src/getThreadPictures.js +79 -59
  33. package/src/getUserID.js +66 -61
  34. package/src/getUserInfo.js +74 -66
  35. package/src/handleFriendRequest.js +61 -46
  36. package/src/handleMessageRequest.js +65 -47
  37. package/src/httpGet.js +52 -44
  38. package/src/httpPost.js +52 -43
  39. package/src/httpPostFormData.js +63 -0
  40. package/src/listenMqtt.js +969 -709
  41. package/src/logout.js +62 -55
  42. package/src/markAsDelivered.js +58 -47
  43. package/src/markAsRead.js +80 -70
  44. package/src/markAsReadAll.js +49 -39
  45. package/src/markAsSeen.js +59 -48
  46. package/src/muteThread.js +52 -45
  47. package/src/postFormData.js +46 -0
  48. package/src/refreshFb_dtsg.js +81 -0
  49. package/src/removeUserFromGroup.js +79 -45
  50. package/src/resolvePhotoUrl.js +45 -36
  51. package/src/searchForThread.js +53 -42
  52. package/src/sendMessage.js +328 -328
  53. package/src/sendMessageMqtt.js +316 -0
  54. package/src/sendTypingIndicator.js +103 -70
  55. package/src/setMessageReaction.js +106 -98
  56. package/src/setPostReaction.js +102 -95
  57. package/src/setTitle.js +86 -70
  58. package/src/threadColors.js +131 -41
  59. package/src/unfriend.js +52 -42
  60. package/src/unsendMessage.js +49 -39
  61. package/src/uploadAttachment.js +95 -0
  62. package/utils.js +1501 -1196
  63. package/.gitattributes +0 -2
  64. package/src/Screenshot.js +0 -83
  65. package/src/changeAvt.js +0 -85
  66. package/src/getThreadHistoryDeprecated.js +0 -71
  67. package/src/getThreadInfoDeprecated.js +0 -56
  68. package/src/getThreadListDeprecated.js +0 -46
  69. package/src/shareContact.js +0 -46
  70. package/test/data/shareAttach.js +0 -146
  71. package/test/data/something.mov +0 -0
  72. package/test/data/test.png +0 -0
  73. package/test/data/test.txt +0 -7
  74. package/test/example-config.json +0 -18
  75. package/test/test-page.js +0 -140
  76. package/test/test.js +0 -385
package/src/listenMqtt.js CHANGED
@@ -1,710 +1,970 @@
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 GUID = utils.getGUID();
46
- const username = {
47
- u: ctx.userID,
48
- s: sessionID,
49
- chat_on: chatOn,
50
- fg: foreground,
51
- d: GUID,
52
- ct: 'websocket',
53
- aid: '219994525426954',
54
- aids: null,
55
- mqtt_sid: '',
56
- cp: 3,
57
- ecp: 10,
58
- st: [],
59
- pm: [],
60
- dc: '',
61
- no_auto_fg: true,
62
- gas: null,
63
- pack: [],
64
- p: null,
65
- php_override: ""
66
- };
67
- var cookies = ctx.jar.getCookies("https://www.facebook.com").join("; ");
68
-
69
- var host;
70
- if (ctx.mqttEndpoint) host = `${ctx.mqttEndpoint}&sid=${sessionID}&cid=${GUID}`;
71
- else if (ctx.region) host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLocaleLowerCase()}&sid=${sessionID}&cid=${GUID}`;
72
- else host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}&cid=${GUID}`;
73
-
74
- const options = {
75
- clientId: 'mqttwsclient',
76
- protocolId: 'MQIsdp',
77
- protocolVersion: 3,
78
- username: JSON.stringify(username),
79
- clean: true,
80
- wsOptions: {
81
- headers: {
82
- Cookie: cookies,
83
- Origin: 'https://www.facebook.com',
84
- 'User-Agent': ctx.globalOptions.userAgent || 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.64 Safari/537.36',
85
- Referer: 'https://www.facebook.com/',
86
- Host: new URL(host).hostname,
87
- },
88
- origin: 'https://www.facebook.com',
89
- protocolVersion: 13,
90
- binaryType: 'arraybuffer',
91
- },
92
- keepalive: 60,
93
- reschedulePings: true,
94
- reconnectPeriod: 3,
95
- };
96
-
97
- if (typeof ctx.globalOptions.proxy != "undefined") {
98
- var 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
- global.mqttClient = ctx.mqttClient;
105
-
106
- mqttClient.on('error', function (err) {
107
- log.error("listenMqtt", err);
108
- mqttClient.end();
109
- if (ctx.globalOptions.autoReconnect) getSeqID();
110
- else globalCallback({ type: "stop_listen", error: "Connection refused: Server unavailable" }, null);
111
- });
112
-
113
- mqttClient.on('connect', function () {
114
- topics.forEach(topicsub => mqttClient.subscribe(topicsub));
115
-
116
- var topic;
117
- var queue = {
118
- sync_api_version: 10,
119
- max_deltas_able_to_process: 1000,
120
- delta_batch_size: 500,
121
- encoding: "JSON",
122
- entity_fbid: ctx.userID,
123
- };
124
-
125
- if (ctx.syncToken) {
126
- topic = "/messenger_sync_get_diffs";
127
- queue.last_seq_id = ctx.lastSeqId;
128
- queue.sync_token = ctx.syncToken;
129
- }
130
- else {
131
- topic = "/messenger_sync_create_queue";
132
- queue.initial_titan_sequence_id = ctx.lastSeqId;
133
- queue.device_params = null;
134
- }
135
-
136
- mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
137
-
138
- var rTimeout = setTimeout(function () {
139
- mqttClient.end();
140
- getSeqID();
141
- }, 5000);
142
-
143
- ctx.tmsWait = function () {
144
- clearTimeout(rTimeout);
145
- ctx.globalOptions.emitReady ? globalCallback({
146
- type: "ready",
147
- error: null
148
- }) : "";
149
- delete ctx.tmsWait;
150
- };
151
- });
152
-
153
- mqttClient.on('message', function (topic, message, _packet) {
154
- try {
155
- var jsonMessage = JSON.parse(message);
156
- }
157
- catch (ex) {
158
- return log.error("listenMqtt", ex);
159
- }
160
- if (topic === "/t_ms") {
161
- if (ctx.tmsWait && typeof ctx.tmsWait == "function") ctx.tmsWait();
162
-
163
- if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
164
- ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
165
- ctx.syncToken = jsonMessage.syncToken;
166
- }
167
-
168
- if (jsonMessage.lastIssuedSeqId) ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
169
-
170
- //If it contains more than 1 delta
171
- for (var i in jsonMessage.deltas) {
172
- var delta = jsonMessage.deltas[i];
173
- parseDelta(defaultFuncs, api, ctx, globalCallback, { "delta": delta });
174
- }
175
- }
176
- else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
177
- var typ = {
178
- type: "typ",
179
- isTyping: !!jsonMessage.state,
180
- from: jsonMessage.sender_fbid.toString(),
181
- threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
182
- };
183
- (function () { globalCallback(null, typ); })();
184
- }
185
- else if (topic === "/orca_presence") {
186
- if (!ctx.globalOptions.updatePresence) {
187
- for (var i in jsonMessage.list) {
188
- var data = jsonMessage.list[i];
189
- var userID = data["u"];
190
-
191
- var presence = {
192
- type: "presence",
193
- userID: userID.toString(),
194
- //Convert to ms
195
- timestamp: data["l"] * 1000,
196
- statuses: data["p"]
197
- };
198
- (function () { globalCallback(null, presence); })();
199
- }
200
- }
201
- }
202
-
203
- });
204
-
205
- mqttClient.on('close', function () {});
206
- }
207
-
208
- function parseDelta(defaultFuncs, api, ctx, globalCallback, v) {
209
- if (v.delta.class == "NewMessage") {
210
- //Not tested for pages
211
- if (ctx.globalOptions.pageID && ctx.globalOptions.pageID != v.queue) return;
212
-
213
- (function resolveAttachmentUrl(i) {
214
- if (i == (v.delta.attachments || []).length) {
215
- let fmtMsg;
216
- try {
217
- fmtMsg = utils.formatDeltaMessage(v);
218
- } catch (err) {
219
- return globalCallback({
220
- error: "Problem parsing message object.",
221
- detail: err,
222
- res: v,
223
- type: "parse_error"
224
- });
225
- }
226
- if (fmtMsg) {
227
- if (ctx.globalOptions.autoMarkDelivery) {
228
- markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
229
- }
230
- }
231
- return !ctx.globalOptions.selfListen &&
232
- (fmtMsg.senderID === ctx.i_userID || fmtMsg.senderID === ctx.userID) ?
233
- undefined :
234
- (function () { globalCallback(null, fmtMsg); })();
235
- } else {
236
- if (v.delta.attachments[i].mercury.attach_type == "photo") {
237
- api.resolvePhotoUrl(
238
- v.delta.attachments[i].fbid,
239
- (err, url) => {
240
- if (!err)
241
- v.delta.attachments[
242
- i
243
- ].mercury.metadata.url = url;
244
- return resolveAttachmentUrl(i + 1);
245
- }
246
- );
247
- } else {
248
- return resolveAttachmentUrl(i + 1);
249
- }
250
- }
251
- })(0);
252
- }
253
-
254
- if (v.delta.class == "ClientPayload") {
255
- var clientPayload = utils.decodeClientPayload(v.delta.payload);
256
- if (clientPayload && clientPayload.deltas) {
257
- for (var i in clientPayload.deltas) {
258
- var delta = clientPayload.deltas[i];
259
- if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
260
- (function () {
261
- globalCallback(null, {
262
- type: "message_reaction",
263
- threadID: (delta.deltaMessageReaction.threadKey.threadFbId ? delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey.otherUserFbId).toString(),
264
- messageID: delta.deltaMessageReaction.messageId,
265
- reaction: delta.deltaMessageReaction.reaction,
266
- senderID: delta.deltaMessageReaction.senderId.toString(),
267
- userID: delta.deltaMessageReaction.userId.toString()
268
- });
269
- })();
270
- }
271
- else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
272
- (function () {
273
- globalCallback(null, {
274
- type: "message_unsend",
275
- threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ? delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
276
- messageID: delta.deltaRecallMessageData.messageID,
277
- senderID: delta.deltaRecallMessageData.senderID.toString(),
278
- deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
279
- timestamp: delta.deltaRecallMessageData.timestamp
280
- });
281
- })();
282
- }
283
- else if (delta.deltaMessageReply) {
284
- //Mention block - #1
285
- var mdata = delta.deltaMessageReply.message === undefined ? [] :
286
- delta.deltaMessageReply.message.data === undefined ? [] :
287
- delta.deltaMessageReply.message.data.prng === undefined ? [] :
288
- JSON.parse(delta.deltaMessageReply.message.data.prng);
289
- var m_id = mdata.map(u => u.i);
290
- var m_offset = mdata.map(u => u.o);
291
- var m_length = mdata.map(u => u.l);
292
-
293
- var mentions = {};
294
-
295
- 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]);
296
- //Mention block - 1#
297
- var callbackToReturn = {
298
- type: "message_reply",
299
- threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
300
- messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
301
- senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
302
- attachments: (delta.deltaMessageReply.message.attachments || []).map(function (att) {
303
- var mercury = JSON.parse(att.mercuryJSON);
304
- Object.assign(att, mercury);
305
- return att;
306
- }).map(att => {
307
- var x;
308
- try {
309
- x = utils._formatAttachment(att);
310
- }
311
- catch (ex) {
312
- x = att;
313
- x.error = ex;
314
- x.type = "unknown";
315
- }
316
- return x;
317
- }),
318
- args: (delta.deltaMessageReply.message.body || "").trim().split(/\s+/),
319
- body: (delta.deltaMessageReply.message.body || ""),
320
- isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
321
- mentions: mentions,
322
- timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
323
- participantIDs: (delta.deltaMessageReply.message.messageMetadata.cid.canonicalParticipantFbids || delta.deltaMessageReply.message.participants || []).map(e => e.toString())
324
- };
325
-
326
- if (delta.deltaMessageReply.repliedToMessage) {
327
- //Mention block - #2
328
- mdata = delta.deltaMessageReply.repliedToMessage === undefined ? [] :
329
- delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
330
- delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
331
- JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
332
- m_id = mdata.map(u => u.i);
333
- m_offset = mdata.map(u => u.o);
334
- m_length = mdata.map(u => u.l);
335
-
336
- var rmentions = {};
337
-
338
- 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]);
339
- //Mention block - 2#
340
- callbackToReturn.messageReply = {
341
- threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
342
- messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
343
- senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
344
- attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function (att) {
345
- var mercury = JSON.parse(att.mercuryJSON);
346
- Object.assign(att, mercury);
347
- return att;
348
- }).map(att => {
349
- var x;
350
- try {
351
- x = utils._formatAttachment(att);
352
- }
353
- catch (ex) {
354
- x = att;
355
- x.error = ex;
356
- x.type = "unknown";
357
- }
358
- return x;
359
- }),
360
- args: (delta.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
361
- body: delta.deltaMessageReply.repliedToMessage.body || "",
362
- isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
363
- mentions: rmentions,
364
- timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp
365
- };
366
- }
367
- else if (delta.deltaMessageReply.replyToMessageId) {
368
- return defaultFuncs
369
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
370
- "av": ctx.globalOptions.pageID,
371
- "queries": JSON.stringify({
372
- "o0": {
373
- //Using the same doc_id as forcedFetch
374
- "doc_id": "2848441488556444",
375
- "query_params": {
376
- "thread_and_message_id": {
377
- "thread_id": callbackToReturn.threadID,
378
- "message_id": delta.deltaMessageReply.replyToMessageId.id,
379
- }
380
- }
381
- }
382
- })
383
- })
384
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
385
- .then((resData) => {
386
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
387
- if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
388
- var fetchData = resData[0].o0.data.message;
389
- var mobj = {};
390
- 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);
391
-
392
- callbackToReturn.messageReply = {
393
- threadID: callbackToReturn.threadID,
394
- messageID: fetchData.message_id,
395
- senderID: fetchData.message_sender.id.toString(),
396
- attachments: fetchData.message.blob_attachment.map(att => {
397
- var x;
398
- try {
399
- x = utils._formatAttachment({ blob_attachment: att });
400
- }
401
- catch (ex) {
402
- x = att;
403
- x.error = ex;
404
- x.type = "unknown";
405
- }
406
- return x;
407
- }),
408
- args: (fetchData.message.text || "").trim().split(/\s+/) || [],
409
- body: fetchData.message.text || "",
410
- isGroup: callbackToReturn.isGroup,
411
- mentions: mobj,
412
- timestamp: parseInt(fetchData.timestamp_precise)
413
- };
414
- })
415
- .catch(err => log.error("forcedFetch", err))
416
- .finally(function () {
417
- if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
418
- !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
419
- });
420
- }
421
- else callbackToReturn.delta = delta;
422
-
423
- if (ctx.globalOptions.autoMarkDelivery) markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
424
-
425
- return !ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID ? undefined : (function () { globalCallback(null, callbackToReturn); })();
426
- }
427
- }
428
- return;
429
- }
430
- }
431
-
432
- if (v.delta.class !== "NewMessage" && !ctx.globalOptions.listenEvents) return;
433
- switch (v.delta.class) {
434
- case "ReadReceipt":
435
- var fmtMsg;
436
- try {
437
- fmtMsg = utils.formatDeltaReadReceipt(v.delta);
438
- }
439
- catch (err) {
440
- return globalCallback({
441
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
442
- detail: err,
443
- res: v.delta,
444
- type: "parse_error"
445
- });
446
- }
447
- return (function () { globalCallback(null, fmtMsg); })();
448
- case "AdminTextMessage":
449
- switch (v.delta.type) {
450
- case 'confirm_friend_request':
451
- case 'shared_album_delete':
452
- case 'shared_album_addition':
453
- case 'pin_messages_v2':
454
- case 'unpin_messages_v2':
455
- case "change_thread_theme":
456
- case "change_thread_nickname":
457
- case "change_thread_icon":
458
- case "change_thread_quick_reaction":
459
- case "change_thread_admins":
460
- case "group_poll":
461
- case "joinable_group_link_mode_change":
462
- case "magic_words":
463
- case "change_thread_approval_mode":
464
- case "messenger_call_log":
465
- case "participant_joined_group_call":
466
- var fmtMsg;
467
- try {
468
- fmtMsg = utils.formatDeltaEvent(v.delta);
469
- }
470
- catch (err) {
471
- return globalCallback({
472
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
473
- detail: err,
474
- res: v.delta,
475
- type: "parse_error"
476
- });
477
- }
478
- return (function () { globalCallback(null, fmtMsg); })();
479
- default:
480
- return;
481
- }
482
- //For group images
483
- case "ForcedFetch":
484
- if (!v.delta.threadKey) return;
485
- var mid = v.delta.messageId;
486
- var tid = v.delta.threadKey.threadFbId;
487
- if (mid && tid) {
488
- const form = {
489
- "av": ctx.globalOptions.pageID,
490
- "queries": JSON.stringify({
491
- "o0": {
492
- //This doc_id is valid as of March 25, 2020
493
- "doc_id": "2848441488556444",
494
- "query_params": {
495
- "thread_and_message_id": {
496
- "thread_id": tid.toString(),
497
- "message_id": mid,
498
- }
499
- }
500
- }
501
- })
502
- };
503
-
504
- defaultFuncs
505
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
506
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
507
- .then((resData) => {
508
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
509
-
510
- if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
511
-
512
- var fetchData = resData[0].o0.data.message;
513
-
514
- if (utils.getType(fetchData) == "Object") {
515
- log.info("forcedFetch", fetchData);
516
- switch (fetchData.__typename) {
517
- case "ThreadImageMessage":
518
- (!ctx.globalOptions.selfListen && fetchData.message_sender.id.toString() === ctx.userID) ||
519
- !ctx.loggedIn ? undefined : (function () {
520
- globalCallback(null, {
521
- type: "change_thread_image",
522
- threadID: utils.formatID(tid.toString()),
523
- snippet: fetchData.snippet,
524
- timestamp: fetchData.timestamp_precise,
525
- author: fetchData.message_sender.id,
526
- image: {
527
- attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
528
- width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
529
- height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
530
- url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
531
- }
532
- });
533
- })();
534
- break;
535
- case "UserMessage":
536
- log.info("ff-Return", {
537
- type: "message",
538
- senderID: utils.formatID(fetchData.message_sender.id),
539
- body: fetchData.message.text || "",
540
- threadID: utils.formatID(tid.toString()),
541
- messageID: fetchData.message_id,
542
- attachments: [{
543
- type: "share",
544
- ID: fetchData.extensible_attachment.legacy_attachment_id,
545
- url: fetchData.extensible_attachment.story_attachment.url,
546
-
547
- title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
548
- description: fetchData.extensible_attachment.story_attachment.description.text,
549
- source: fetchData.extensible_attachment.story_attachment.source,
550
-
551
- image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
552
- width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
553
- height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
554
- playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
555
- duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
556
-
557
- subattachments: fetchData.extensible_attachment.subattachments,
558
- properties: fetchData.extensible_attachment.story_attachment.properties,
559
- }],
560
- mentions: {},
561
- timestamp: parseInt(fetchData.timestamp_precise),
562
- participantIDs: (fetchData.participants || (fetchData.messageMetadata ? fetchData.messageMetadata.cid ? fetchData.messageMetadata.cid.canonicalParticipantFbids : fetchData.messageMetadata.participantIds : []) || []),
563
- isGroup: (fetchData.message_sender.id != tid.toString())
564
- });
565
- globalCallback(null, {
566
- type: "message",
567
- senderID: utils.formatID(fetchData.message_sender.id),
568
- body: fetchData.message.text || "",
569
- threadID: utils.formatID(tid.toString()),
570
- messageID: fetchData.message_id,
571
- attachments: [{
572
- type: "share",
573
- ID: fetchData.extensible_attachment.legacy_attachment_id,
574
- url: fetchData.extensible_attachment.story_attachment.url,
575
-
576
- title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
577
- description: fetchData.extensible_attachment.story_attachment.description.text,
578
- source: fetchData.extensible_attachment.story_attachment.source,
579
-
580
- image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
581
- width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
582
- height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
583
- playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
584
- duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
585
-
586
- subattachments: fetchData.extensible_attachment.subattachments,
587
- properties: fetchData.extensible_attachment.story_attachment.properties,
588
- }],
589
- mentions: {},
590
- timestamp: parseInt(fetchData.timestamp_precise),
591
- participantIDs: (fetchData.participants || (fetchData.messageMetadata ? fetchData.messageMetadata.cid ? fetchData.messageMetadata.cid.canonicalParticipantFbids : fetchData.messageMetadata.participantIds : []) || []),
592
- isGroup: (fetchData.message_sender.id != tid.toString())
593
- });
594
- }
595
- }
596
- else log.error("forcedFetch", fetchData);
597
- })
598
- .catch((err) => log.error("forcedFetch", err));
599
- }
600
- break;
601
- case "ThreadName":
602
- case "ParticipantsAddedToGroupThread":
603
- case "ParticipantLeftGroupThread":
604
- var formattedEvent;
605
- try {
606
- formattedEvent = utils.formatDeltaEvent(v.delta);
607
- }
608
- catch (err) {
609
- return globalCallback({
610
- error: "Problem parsing message object. Please open an issue at https://github.com/Schmavery/facebook-chat-api/issues.",
611
- detail: err,
612
- res: v.delta,
613
- type: "parse_error"
614
- });
615
- }
616
- return (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) || !ctx.loggedIn ? undefined : (function () { globalCallback(null, formattedEvent); })();
617
- }
618
- }
619
-
620
- function markDelivery(ctx, api, threadID, messageID) {
621
- if (threadID && messageID) {
622
- api.markAsDelivered(threadID, messageID, (err) => {
623
- if (err) log.error("markAsDelivered", err);
624
- else {
625
- if (ctx.globalOptions.autoMarkRead) {
626
- api.markAsRead(threadID, (err) => {
627
- if (err) log.error("markAsDelivered", err);
628
- });
629
- }
630
- }
631
- });
632
- }
633
- }
634
-
635
- module.exports = function (defaultFuncs, api, ctx) {
636
- var globalCallback = identity;
637
- getSeqID = function getSeqID() {
638
- ctx.t_mqttCalled = false;
639
- defaultFuncs
640
- .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
641
- .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
642
- .then((resData) => {
643
- if (utils.getType(resData) != "Array") throw { error: "Not logged in", res: resData };
644
- if (resData && resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
645
- if (resData[resData.length - 1].successful_results === 0) throw { error: "getSeqId: there was no successful_results", res: resData };
646
- if (resData[0].o0.data.viewer.message_threads.sync_sequence_id) {
647
- ctx.lastSeqId = resData[0].o0.data.viewer.message_threads.sync_sequence_id;
648
- listenMqtt(defaultFuncs, api, ctx, globalCallback);
649
- }
650
- else throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
651
- })
652
- .catch((err) => {
653
- log.error("getSeqId", err);
654
- if (utils.getType(err) == "Object" && err.error === "Not logged in") ctx.loggedIn = false;
655
- return globalCallback(err);
656
- });
657
- };
658
-
659
- return function (callback) {
660
- class MessageEmitter extends EventEmitter {
661
- stopListening(callback) {
662
- callback = callback || (() => { });
663
- globalCallback = identity;
664
- if (ctx.mqttClient) {
665
- ctx.mqttClient.unsubscribe("/webrtc");
666
- ctx.mqttClient.unsubscribe("/rtc_multi");
667
- ctx.mqttClient.unsubscribe("/onevc");
668
- ctx.mqttClient.publish("/browser_close", "{}");
669
- ctx.mqttClient.end(false, function (...data) {
670
- callback(data);
671
- ctx.mqttClient = undefined;
672
- });
673
- }
674
- }
675
- }
676
-
677
- var msgEmitter = new MessageEmitter();
678
- globalCallback = (callback || function (error, message) {
679
- if (error) return msgEmitter.emit("error", error);
680
- msgEmitter.emit("message", message);
681
- });
682
-
683
- //Reset some stuff
684
- if (!ctx.firstListen) ctx.lastSeqId = null;
685
- ctx.syncToken = undefined;
686
- ctx.t_mqttCalled = false;
687
-
688
- //Same request as getThreadList
689
- form = {
690
- "av": ctx.globalOptions.pageID,
691
- "queries": JSON.stringify({
692
- "o0": {
693
- "doc_id": "3336396659757871",
694
- "query_params": {
695
- "limit": 1,
696
- "before": null,
697
- "tags": ["INBOX"],
698
- "includeDeliveryReceipts": false,
699
- "includeSeqID": true
700
- }
701
- }
702
- })
703
- };
704
-
705
- if (!ctx.firstListen || !ctx.lastSeqId) getSeqID();
706
- else listenMqtt(defaultFuncs, api, ctx, globalCallback);
707
- ctx.firstListen = false;
708
- return msgEmitter;
709
- };
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('ws');
7
+ const HttpsProxyAgent = require('https-proxy-agent');
8
+ const EventEmitter = require('events');
9
+ const Duplexify = require('duplexify');
10
+ const {
11
+ Transform
12
+ } = require('stream');
13
+ const debugSeq = false;
14
+ var identity = function() {};
15
+ var form = {};
16
+ var getSeqId = function() {};
17
+ const topics = ['/ls_req', '/ls_resp', '/legacy_web', '/webrtc', '/rtc_multi', '/onevc', '/br_sr', '/sr_res', '/t_ms', '/thread_typing', '/orca_typing_notifications', '/notify_disconnect', '/orca_presence', '/inbox', '/mercury', '/messaging_events', '/orca_message_notifications', '/pp', '/webrtc_response'];
18
+ let WebSocket_Global;
19
+ function buildProxy() {
20
+ const Proxy = new Transform({
21
+ objectMode: false,
22
+ transform(chunk, enc, next) {
23
+ if (WebSocket_Global.readyState !== WebSocket_Global.OPEN) {
24
+ return next();
25
+ }
26
+ let data;
27
+ if (typeof chunk === 'string') {
28
+ data = Buffer.from(chunk, 'utf8');
29
+ } else {
30
+ data = chunk;
31
+ }
32
+ WebSocket_Global.send(data);
33
+ next();
34
+ },
35
+ flush(done) {
36
+ WebSocket_Global.close();
37
+ done();
38
+ },
39
+ writev(chunks, cb) {
40
+ const buffers = chunks.map(({ chunk }) => {
41
+ if (typeof chunk === 'string') {
42
+ return Buffer.from(chunk, 'utf8');
43
+ }
44
+ return chunk;
45
+ });
46
+ this._write(Buffer.concat(buffers), 'binary', cb);
47
+ },
48
+ });
49
+ return Proxy;
50
+ }
51
+ function buildStream(options, WebSocket, Proxy) {
52
+ const Stream = Duplexify(undefined, undefined, options);
53
+ Stream.socket = WebSocket;
54
+ WebSocket.onclose = () => {
55
+ Stream.end();
56
+ Stream.destroy();
57
+ };
58
+ WebSocket.onerror = (err) => {
59
+ Stream.destroy(err);
60
+ };
61
+ WebSocket.onmessage = (event) => {
62
+ const data = event.data instanceof ArrayBuffer ? Buffer.from(event.data) : Buffer.from(event.data, 'utf8');
63
+ Stream.push(data);
64
+ };
65
+ WebSocket.onopen = () => {
66
+ Stream.setReadable(Proxy);
67
+ Stream.setWritable(Proxy);
68
+ Stream.emit('connect');
69
+ };
70
+ WebSocket_Global = WebSocket;
71
+ Proxy.on('close', () => WebSocket.close());
72
+ return Stream;
73
+ }
74
+ function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
75
+ const chatOn = ctx.globalOptions.online;
76
+ const foreground = false;
77
+ const sessionID = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER) + 1;
78
+ const GUID = utils.getGUID();
79
+ const username = {
80
+ u: ctx.userID,
81
+ s: sessionID,
82
+ chat_on: chatOn,
83
+ fg: foreground,
84
+ d: GUID,
85
+ ct: 'websocket',
86
+ aid: '219994525426954',
87
+ aids: null,
88
+ mqtt_sid: '',
89
+ cp: 3,
90
+ ecp: 10,
91
+ st: [],
92
+ pm: [],
93
+ dc: '',
94
+ no_auto_fg: true,
95
+ gas: null,
96
+ pack: [],
97
+ p: null,
98
+ php_override: ""
99
+ };
100
+ const cookies = ctx.jar.getCookies('https://www.facebook.com').join('; ');
101
+ let host;
102
+ if (ctx.mqttEndpoint) {
103
+ host = `${ctx.mqttEndpoint}&sid=${sessionID}&cid=${GUID}`;
104
+ } else if (ctx.region) {
105
+ host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLowerCase()}&sid=${sessionID}&cid=${GUID}`;
106
+ } else {
107
+ host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}&cid=${GUID}`;
108
+ }
109
+ const options = {
110
+ clientId: 'mqttwsclient',
111
+ protocolId: 'MQIsdp',
112
+ protocolVersion: 3,
113
+ username: JSON.stringify(username),
114
+ clean: true,
115
+ wsOptions: {
116
+ headers: {
117
+ Cookie: cookies,
118
+ Origin: 'https://www.facebook.com',
119
+ 'User-Agent': ctx.globalOptions.userAgent || 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.64 Safari/537.36',
120
+ Referer: 'https://www.facebook.com/',
121
+ Host: new URL(host).hostname,
122
+ },
123
+ origin: 'https://www.facebook.com',
124
+ protocolVersion: 13,
125
+ binaryType: 'arraybuffer',
126
+ },
127
+ keepalive: 60,
128
+ reschedulePings: true,
129
+ reconnectPeriod: 2000,
130
+ connectTimeout: 10000,
131
+ };
132
+ if (typeof ctx.globalOptions.proxy != "undefined") {
133
+ const agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
134
+ options.wsOptions.agent = agent;
135
+ }
136
+ ctx.mqttClient = new mqtt.Client(() => buildStream(options, new WebSocket(host, options.wsOptions), buildProxy()), options);
137
+ const mqttClient = ctx.mqttClient;
138
+ global.mqttClient = mqttClient;
139
+ mqttClient.on('error', function (err) {
140
+ log.error("listenMqtt", err);
141
+ mqttClient.end();
142
+ if (ctx.globalOptions.autoReconnect) {
143
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
144
+ } else {
145
+ utils.checkLiveCookie(ctx, defaultFuncs)
146
+ .then(res => {
147
+ globalCallback({
148
+ type: "stop_listen",
149
+ error: "Connection refused: Server unavailable"
150
+ }, null);
151
+ })
152
+ .catch(err => {
153
+ globalCallback({
154
+ type: "account_inactive",
155
+ error: "Maybe your account is blocked by facebook, please login and check at https://facebook.com"
156
+ }, null);
157
+ });
158
+ }
159
+ });
160
+
161
+ mqttClient.on('close', function () {
162
+
163
+ });
164
+
165
+ mqttClient.on('connect', function () {
166
+ topics.forEach(function (topicsub) {
167
+ mqttClient.subscribe(topicsub);
168
+ });
169
+
170
+ let topic;
171
+ const queue = {
172
+ sync_api_version: 10,
173
+ max_deltas_able_to_process: 1000,
174
+ delta_batch_size: 500,
175
+ encoding: "JSON",
176
+ entity_fbid: ctx.i_userID || ctx.userID
177
+ };
178
+
179
+ if (ctx.syncToken) {
180
+ topic = "/messenger_sync_get_diffs";
181
+ queue.last_seq_id = ctx.lastSeqId;
182
+ queue.sync_token = ctx.syncToken;
183
+ } else {
184
+ topic = "/messenger_sync_create_queue";
185
+ queue.initial_titan_sequence_id = ctx.lastSeqId;
186
+ queue.device_params = null;
187
+ }
188
+
189
+ mqttClient.publish(topic, JSON.stringify(queue), {
190
+ qos: 1,
191
+ retain: false
192
+ });
193
+ // set status online
194
+ // fix by NTKhang
195
+ mqttClient.publish("/foreground_state", JSON.stringify({
196
+ foreground: chatOn
197
+ }), {
198
+ qos: 1
199
+ });
200
+ mqttClient.publish("/set_client_settings", JSON.stringify({
201
+ make_user_available_when_in_foreground: true
202
+ }), {
203
+ qos: 1
204
+ });
205
+
206
+ const rTimeout = setTimeout(function () {
207
+ mqttClient.end();
208
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
209
+ }, 5000);
210
+
211
+ ctx.tmsWait = function () {
212
+ clearTimeout(rTimeout);
213
+ ctx.globalOptions.emitReady ? globalCallback({
214
+ type: "ready",
215
+ error: null
216
+ }) : "";
217
+ delete ctx.tmsWait;
218
+ };
219
+
220
+ });
221
+
222
+ mqttClient.on('message', function (topic, message, _packet) {
223
+ let jsonMessage = Buffer.isBuffer(message) ? Buffer.from(message).toString() : message;
224
+ try {
225
+ jsonMessage = JSON.parse(jsonMessage);
226
+ } catch (e) {
227
+ jsonMessage = {};
228
+ }
229
+
230
+ if (jsonMessage.type === "jewel_requests_add") {
231
+ globalCallback(null, {
232
+ type: "friend_request_received",
233
+ actorFbId: jsonMessage.from.toString(),
234
+ timestamp: Date.now().toString()
235
+ });
236
+ } else if (jsonMessage.type === "jewel_requests_remove_old") {
237
+ globalCallback(null, {
238
+ type: "friend_request_cancel",
239
+ actorFbId: jsonMessage.from.toString(),
240
+ timestamp: Date.now().toString()
241
+ });
242
+ } else if (topic === "/t_ms") {
243
+ if (ctx.tmsWait && typeof ctx.tmsWait == "function") {
244
+ ctx.tmsWait();
245
+ }
246
+
247
+ if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
248
+ ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
249
+ ctx.syncToken = jsonMessage.syncToken;
250
+ }
251
+
252
+ if (jsonMessage.lastIssuedSeqId) {
253
+ ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
254
+ }
255
+
256
+ //If it contains more than 1 delta
257
+ for (const i in jsonMessage.deltas) {
258
+ const delta = jsonMessage.deltas[i];
259
+ parseDelta(defaultFuncs, api, ctx, globalCallback, {
260
+ "delta": delta
261
+ });
262
+ }
263
+ } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
264
+ const typ = {
265
+ type: "typ",
266
+ isTyping: !!jsonMessage.state,
267
+ from: jsonMessage.sender_fbid.toString(),
268
+ threadID: utils.formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString())
269
+ };
270
+ (function () {
271
+ globalCallback(null, typ);
272
+ })();
273
+ } else if (topic === "/orca_presence") {
274
+ if (!ctx.globalOptions.updatePresence) {
275
+ for (const i in jsonMessage.list) {
276
+ const data = jsonMessage.list[i];
277
+ const userID = data["u"];
278
+
279
+ const presence = {
280
+ type: "presence",
281
+ userID: userID.toString(),
282
+ //Convert to ms
283
+ timestamp: data["l"] * 1000,
284
+ statuses: data["p"]
285
+ };
286
+ (function () {
287
+ globalCallback(null, presence);
288
+ })();
289
+ }
290
+ }
291
+ }
292
+
293
+ });
294
+
295
+ }
296
+
297
+ function parseDelta(defaultFuncs, api, ctx, globalCallback, v) {
298
+ if (v.delta.class == "NewMessage") {
299
+ //Not tested for pages
300
+ if (ctx.globalOptions.pageID &&
301
+ ctx.globalOptions.pageID != v.queue
302
+ )
303
+ return;
304
+
305
+ (function resolveAttachmentUrl(i) {
306
+ if (i == (v.delta.attachments || []).length) {
307
+ let fmtMsg;
308
+ try {
309
+ fmtMsg = utils.formatDeltaMessage(v);
310
+ } catch (err) {
311
+ return globalCallback({
312
+ error: "Problem parsing message object. Please open an issue at https://github.com/ntkhang03/fb-chat-api/issues.",
313
+ detail: err,
314
+ res: v,
315
+ type: "parse_error"
316
+ });
317
+ }
318
+ if (fmtMsg) {
319
+ if (ctx.globalOptions.autoMarkDelivery) {
320
+ markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
321
+ }
322
+ }
323
+ return !ctx.globalOptions.selfListen &&
324
+ (fmtMsg.senderID === ctx.i_userID || fmtMsg.senderID === ctx.userID) ?
325
+ undefined :
326
+ (function () {
327
+ globalCallback(null, fmtMsg);
328
+ })();
329
+ } else {
330
+ if (v.delta.attachments[i].mercury.attach_type == "photo") {
331
+ api.resolvePhotoUrl(
332
+ v.delta.attachments[i].fbid,
333
+ (err, url) => {
334
+ if (!err)
335
+ v.delta.attachments[
336
+ i
337
+ ].mercury.metadata.url = url;
338
+ return resolveAttachmentUrl(i + 1);
339
+ }
340
+ );
341
+ } else {
342
+ return resolveAttachmentUrl(i + 1);
343
+ }
344
+ }
345
+ })(0);
346
+ }
347
+
348
+ if (v.delta.class == "ClientPayload") {
349
+ const clientPayload = utils.decodeClientPayload(
350
+ v.delta.payload
351
+ );
352
+
353
+ if (clientPayload && clientPayload.deltas) {
354
+ for (const i in clientPayload.deltas) {
355
+ const delta = clientPayload.deltas[i];
356
+ if (delta.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
357
+ (function () {
358
+ globalCallback(null, {
359
+ type: "message_reaction",
360
+ threadID: (delta.deltaMessageReaction.threadKey
361
+ .threadFbId ?
362
+ delta.deltaMessageReaction.threadKey.threadFbId : delta.deltaMessageReaction.threadKey
363
+ .otherUserFbId).toString(),
364
+ messageID: delta.deltaMessageReaction.messageId,
365
+ reaction: delta.deltaMessageReaction.reaction,
366
+ senderID: delta.deltaMessageReaction.senderId == 0 ? delta.deltaMessageReaction.userId.toString() : delta.deltaMessageReaction.senderId.toString(),
367
+ userID: (delta.deltaMessageReaction.userId || delta.deltaMessageReaction.senderId).toString()
368
+ });
369
+ })();
370
+ } else if (delta.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
371
+ (function () {
372
+ globalCallback(null, {
373
+ type: "message_unsend",
374
+ threadID: (delta.deltaRecallMessageData.threadKey.threadFbId ?
375
+ delta.deltaRecallMessageData.threadKey.threadFbId : delta.deltaRecallMessageData.threadKey
376
+ .otherUserFbId).toString(),
377
+ messageID: delta.deltaRecallMessageData.messageID,
378
+ senderID: delta.deltaRecallMessageData.senderID.toString(),
379
+ deletionTimestamp: delta.deltaRecallMessageData.deletionTimestamp,
380
+ timestamp: delta.deltaRecallMessageData.timestamp
381
+ });
382
+ })();
383
+ } else if (delta.deltaRemoveMessage && !!ctx.globalOptions.listenEvents) {
384
+ (function () {
385
+ globalCallback(null, {
386
+ type: "message_self_delete",
387
+ threadID: (delta.deltaRemoveMessage.threadKey.threadFbId ?
388
+ delta.deltaRemoveMessage.threadKey.threadFbId : delta.deltaRemoveMessage.threadKey
389
+ .otherUserFbId).toString(),
390
+ messageID: delta.deltaRemoveMessage.messageIds.length == 1 ? delta.deltaRemoveMessage.messageIds[0] : delta.deltaRemoveMessage.messageIds,
391
+ senderID: api.getCurrentUserID(),
392
+ deletionTimestamp: delta.deltaRemoveMessage.deletionTimestamp,
393
+ timestamp: delta.deltaRemoveMessage.timestamp
394
+ });
395
+ })();
396
+ } else if (delta.deltaMessageReply) {
397
+ //Mention block - #1
398
+ let mdata =
399
+ delta.deltaMessageReply.message === undefined ? [] :
400
+ delta.deltaMessageReply.message.data === undefined ? [] :
401
+ delta.deltaMessageReply.message.data.prng === undefined ? [] :
402
+ JSON.parse(delta.deltaMessageReply.message.data.prng);
403
+ let m_id = mdata.map(u => u.i);
404
+ let m_offset = mdata.map(u => u.o);
405
+ let m_length = mdata.map(u => u.l);
406
+
407
+ const mentions = {};
408
+
409
+ for (let i = 0; i < m_id.length; i++) {
410
+ mentions[m_id[i]] = (delta.deltaMessageReply.message.body || "").substring(
411
+ m_offset[i],
412
+ m_offset[i] + m_length[i]
413
+ );
414
+ }
415
+ //Mention block - 1#
416
+ const callbackToReturn = {
417
+ type: "message_reply",
418
+ threadID: (delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ?
419
+ delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.message.messageMetadata.threadKey
420
+ .otherUserFbId).toString(),
421
+ messageID: delta.deltaMessageReply.message.messageMetadata.messageId,
422
+ senderID: delta.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
423
+ attachments: (delta.deltaMessageReply.message.attachments || []).map(function (att) {
424
+ const mercury = JSON.parse(att.mercuryJSON);
425
+ Object.assign(att, mercury);
426
+ return att;
427
+ }).map(att => {
428
+ let x;
429
+ try {
430
+ x = utils._formatAttachment(att);
431
+ } catch (ex) {
432
+ x = att;
433
+ x.error = ex;
434
+ x.type = "unknown";
435
+ }
436
+ return x;
437
+ }),
438
+ body: delta.deltaMessageReply.message.body || "",
439
+ isGroup: !!delta.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
440
+ mentions: mentions,
441
+ timestamp: delta.deltaMessageReply.message.messageMetadata.timestamp,
442
+ participantIDs: (delta.deltaMessageReply.message.messageMetadata.cid.canonicalParticipantFbids || delta.deltaMessageReply.message.participants || []).map(e => e.toString())
443
+ };
444
+
445
+ if (delta.deltaMessageReply.repliedToMessage) {
446
+ //Mention block - #2
447
+ mdata =
448
+ delta.deltaMessageReply.repliedToMessage === undefined ? [] :
449
+ delta.deltaMessageReply.repliedToMessage.data === undefined ? [] :
450
+ delta.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] :
451
+ JSON.parse(delta.deltaMessageReply.repliedToMessage.data.prng);
452
+ m_id = mdata.map(u => u.i);
453
+ m_offset = mdata.map(u => u.o);
454
+ m_length = mdata.map(u => u.l);
455
+
456
+ const rmentions = {};
457
+
458
+ for (let i = 0; i < m_id.length; i++) {
459
+ rmentions[m_id[i]] = (delta.deltaMessageReply.repliedToMessage.body || "").substring(
460
+ m_offset[i],
461
+ m_offset[i] + m_length[i]
462
+ );
463
+ }
464
+ //Mention block - 2#
465
+ callbackToReturn.messageReply = {
466
+ threadID: (delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ?
467
+ delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey
468
+ .otherUserFbId).toString(),
469
+ messageID: delta.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
470
+ senderID: delta.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
471
+ attachments: delta.deltaMessageReply.repliedToMessage.attachments.map(function (att) {
472
+ const mercury = JSON.parse(att.mercuryJSON);
473
+ Object.assign(att, mercury);
474
+ return att;
475
+ }).map(att => {
476
+ let x;
477
+ try {
478
+ x = utils._formatAttachment(att);
479
+ } catch (ex) {
480
+ x = att;
481
+ x.error = ex;
482
+ x.type = "unknown";
483
+ }
484
+ return x;
485
+ }),
486
+ body: delta.deltaMessageReply.repliedToMessage.body || "",
487
+ isGroup: !!delta.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
488
+ mentions: rmentions,
489
+ timestamp: delta.deltaMessageReply.repliedToMessage.messageMetadata.timestamp
490
+ };
491
+ } else if (delta.deltaMessageReply.replyToMessageId) {
492
+ return defaultFuncs
493
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
494
+ "av": ctx.globalOptions.pageID,
495
+ "queries": JSON.stringify({
496
+ "o0": {
497
+ //Using the same doc_id as forcedFetch
498
+ "doc_id": "2848441488556444",
499
+ "query_params": {
500
+ "thread_and_message_id": {
501
+ "thread_id": callbackToReturn.threadID,
502
+ "message_id": delta.deltaMessageReply.replyToMessageId.id
503
+ }
504
+ }
505
+ }
506
+ })
507
+ })
508
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
509
+ .then((resData) => {
510
+ if (resData[resData.length - 1].error_results > 0) {
511
+ throw resData[0].o0.errors;
512
+ }
513
+
514
+ if (resData[resData.length - 1].successful_results === 0) {
515
+ throw {
516
+ error: "forcedFetch: there was no successful_results",
517
+ res: resData
518
+ };
519
+ }
520
+
521
+ const fetchData = resData[0].o0.data.message;
522
+
523
+ const mobj = {};
524
+ for (const n in fetchData.message.ranges) {
525
+ mobj[fetchData.message.ranges[n].entity.id] = (fetchData.message.text || "").substr(fetchData.message.ranges[n].offset, fetchData.message.ranges[n].length);
526
+ }
527
+
528
+ callbackToReturn.messageReply = {
529
+ threadID: callbackToReturn.threadID,
530
+ messageID: fetchData.message_id,
531
+ senderID: fetchData.message_sender.id.toString(),
532
+ attachments: fetchData.message.blob_attachment.map(att => {
533
+ let x;
534
+ try {
535
+ x = utils._formatAttachment({
536
+ blob_attachment: att
537
+ });
538
+ } catch (ex) {
539
+ x = att;
540
+ x.error = ex;
541
+ x.type = "unknown";
542
+ }
543
+ return x;
544
+ }),
545
+ body: fetchData.message.text || "",
546
+ isGroup: callbackToReturn.isGroup,
547
+ mentions: mobj,
548
+ timestamp: parseInt(fetchData.timestamp_precise)
549
+ };
550
+ })
551
+ .catch((err) => {
552
+ log.error("forcedFetch", err);
553
+ })
554
+ .finally(function () {
555
+ if (ctx.globalOptions.autoMarkDelivery) {
556
+ markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
557
+ } !ctx.globalOptions.selfListen &&
558
+ (callbackToReturn.senderID === ctx.i_userID || callbackToReturn.senderID === ctx.userID) ?
559
+ undefined :
560
+ (function () {
561
+ globalCallback(null, callbackToReturn);
562
+ })();
563
+ });
564
+ } else {
565
+ callbackToReturn.delta = delta;
566
+ }
567
+
568
+ if (ctx.globalOptions.autoMarkDelivery) {
569
+ markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
570
+ }
571
+
572
+ return !ctx.globalOptions.selfListen &&
573
+ (callbackToReturn.senderID === ctx.i_userID || callbackToReturn.senderID === ctx.userID) ?
574
+ undefined :
575
+ (function () {
576
+ globalCallback(null, callbackToReturn);
577
+ })();
578
+ }
579
+ }
580
+ return;
581
+ }
582
+ }
583
+
584
+ if (v.delta.class !== "NewMessage" &&
585
+ !ctx.globalOptions.listenEvents
586
+ )
587
+ return;
588
+
589
+ switch (v.delta.class) {
590
+ case "ReadReceipt":
591
+ var fmtMsg;
592
+ try {
593
+ fmtMsg = utils.formatDeltaReadReceipt(v.delta);
594
+ } catch (err) {
595
+ return globalCallback({
596
+ error: "Problem parsing message object. Please open an issue at https://github.com/ntkhang03/fb-chat-api/issues.",
597
+ detail: err,
598
+ res: v.delta,
599
+ type: "parse_error"
600
+ });
601
+ }
602
+ return (function () {
603
+ globalCallback(null, fmtMsg);
604
+ })();
605
+ case "AdminTextMessage":
606
+ switch (v.delta.type) {
607
+ case "change_thread_theme":
608
+ case "change_thread_nickname":
609
+ case "change_thread_icon":
610
+ case "change_thread_admins":
611
+ case "group_poll":
612
+ case "joinable_group_link_mode_change":
613
+ case "magic_words":
614
+ case "change_thread_approval_mode":
615
+ case "messenger_call_log":
616
+ case "participant_joined_group_call":
617
+ var fmtMsg;
618
+ try {
619
+ fmtMsg = utils.formatDeltaEvent(v.delta);
620
+ } catch (err) {
621
+ return globalCallback({
622
+ error: "Problem parsing message object. Please open an issue at https://github.com/ntkhang03/fb-chat-api/issues.",
623
+ detail: err,
624
+ res: v.delta,
625
+ type: "parse_error"
626
+ });
627
+ }
628
+ return (function () {
629
+ globalCallback(null, fmtMsg);
630
+ })();
631
+ default:
632
+ return;
633
+ }
634
+ //For group images
635
+ case "ForcedFetch":
636
+ if (!v.delta.threadKey) return;
637
+ var mid = v.delta.messageId;
638
+ var tid = v.delta.threadKey.threadFbId;
639
+ if (mid && tid) {
640
+ const form = {
641
+ "av": ctx.globalOptions.pageID,
642
+ "queries": JSON.stringify({
643
+ "o0": {
644
+ //This doc_id is valid as of March 25, 2020
645
+ "doc_id": "2848441488556444",
646
+ "query_params": {
647
+ "thread_and_message_id": {
648
+ "thread_id": tid.toString(),
649
+ "message_id": mid
650
+ }
651
+ }
652
+ }
653
+ })
654
+ };
655
+
656
+ defaultFuncs
657
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form)
658
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs))
659
+ .then((resData) => {
660
+ if (resData[resData.length - 1].error_results > 0) {
661
+ throw resData[0].o0.errors;
662
+ }
663
+
664
+ if (resData[resData.length - 1].successful_results === 0) {
665
+ throw {
666
+ error: "forcedFetch: there was no successful_results",
667
+ res: resData
668
+ };
669
+ }
670
+
671
+ const fetchData = resData[0].o0.data.message;
672
+
673
+ if (utils.getType(fetchData) == "Object") {
674
+ log.info("forcedFetch", fetchData);
675
+ switch (fetchData.__typename) {
676
+ case "ThreadImageMessage":
677
+ (!ctx.globalOptions.selfListenEvent && (fetchData.message_sender.id.toString() === ctx.i_userID || fetchData.message_sender.id.toString() === ctx.userID)) || !ctx.loggedIn ?
678
+ undefined :
679
+ (function () {
680
+ globalCallback(null, {
681
+ type: "event",
682
+ threadID: utils.formatID(tid.toString()),
683
+ messageID: fetchData.message_id,
684
+ logMessageType: "log:thread-image",
685
+ logMessageData: {
686
+ attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
687
+ width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
688
+ height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
689
+ url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
690
+ },
691
+ logMessageBody: fetchData.snippet,
692
+ timestamp: fetchData.timestamp_precise,
693
+ author: fetchData.message_sender.id
694
+ });
695
+ })();
696
+ break;
697
+ case "UserMessage":
698
+ log.info("ff-Return", {
699
+ type: "message",
700
+ senderID: utils.formatID(fetchData.message_sender.id),
701
+ body: fetchData.message.text || "",
702
+ threadID: utils.formatID(tid.toString()),
703
+ messageID: fetchData.message_id,
704
+ attachments: [{
705
+ type: "share",
706
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
707
+ url: fetchData.extensible_attachment.story_attachment.url,
708
+
709
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
710
+ description: fetchData.extensible_attachment.story_attachment.description.text,
711
+ source: fetchData.extensible_attachment.story_attachment.source,
712
+
713
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
714
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
715
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
716
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
717
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
718
+
719
+ subattachments: fetchData.extensible_attachment.subattachments,
720
+ properties: fetchData.extensible_attachment.story_attachment.properties
721
+ }],
722
+ mentions: {},
723
+ timestamp: parseInt(fetchData.timestamp_precise),
724
+ participantIDs: (fetchData.participants || (fetchData.messageMetadata ? fetchData.messageMetadata.cid ? fetchData.messageMetadata.cid.canonicalParticipantFbids : fetchData.messageMetadata.participantIds : []) || []),
725
+ isGroup: (fetchData.message_sender.id != tid.toString())
726
+ });
727
+ globalCallback(null, {
728
+ type: "message",
729
+ senderID: utils.formatID(fetchData.message_sender.id),
730
+ body: fetchData.message.text || "",
731
+ threadID: utils.formatID(tid.toString()),
732
+ messageID: fetchData.message_id,
733
+ attachments: [{
734
+ type: "share",
735
+ ID: fetchData.extensible_attachment.legacy_attachment_id,
736
+ url: fetchData.extensible_attachment.story_attachment.url,
737
+
738
+ title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
739
+ description: fetchData.extensible_attachment.story_attachment.description.text,
740
+ source: fetchData.extensible_attachment.story_attachment.source,
741
+
742
+ image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
743
+ width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
744
+ height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
745
+ playable: (fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false,
746
+ duration: (fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0,
747
+
748
+ subattachments: fetchData.extensible_attachment.subattachments,
749
+ properties: fetchData.extensible_attachment.story_attachment.properties
750
+ }],
751
+ mentions: {},
752
+ timestamp: parseInt(fetchData.timestamp_precise),
753
+ participantIDs: (fetchData.participants || (fetchData.messageMetadata ? fetchData.messageMetadata.cid ? fetchData.messageMetadata.cid.canonicalParticipantFbids : fetchData.messageMetadata.participantIds : []) || []),
754
+ isGroup: (fetchData.message_sender.id != tid.toString())
755
+ });
756
+ }
757
+ } else {
758
+ log.error("forcedFetch", fetchData);
759
+ }
760
+ })
761
+ .catch((err) => {
762
+ log.error("forcedFetch", err);
763
+ });
764
+ }
765
+ break;
766
+ case "ThreadName":
767
+ case "ParticipantsAddedToGroupThread":
768
+ case "ParticipantLeftGroupThread":
769
+ case "ApprovalQueue":
770
+ var formattedEvent;
771
+ try {
772
+ formattedEvent = utils.formatDeltaEvent(v.delta);
773
+ } catch (err) {
774
+ return globalCallback({
775
+ error: "Problem parsing message object. Please open an issue at https://github.com/ntkhang03/fb-chat-api/issues.",
776
+ detail: err,
777
+ res: v.delta,
778
+ type: "parse_error"
779
+ });
780
+ }
781
+ return (!ctx.globalOptions.selfListenEvent && (formattedEvent.author.toString() === ctx.i_userID || formattedEvent.author.toString() === ctx.userID)) || !ctx.loggedIn ?
782
+ undefined :
783
+ (function () {
784
+ globalCallback(null, formattedEvent);
785
+ })();
786
+ }
787
+ }
788
+
789
+ function markDelivery(ctx, api, threadID, messageID) {
790
+ if (threadID && messageID) {
791
+ api.markAsDelivered(threadID, messageID, (err) => {
792
+ if (err) {
793
+ log.error("markAsDelivered", err);
794
+ } else {
795
+ if (ctx.globalOptions.autoMarkRead) {
796
+ api.markAsRead(threadID, (err) => {
797
+ if (err) {
798
+ log.error("markAsDelivered", err);
799
+ }
800
+ });
801
+ }
802
+ }
803
+ });
804
+ }
805
+ }
806
+
807
+ module.exports = function (defaultFuncs, api, ctx) {
808
+ let globalCallback = identity;
809
+ let sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
810
+ getSeqId = function getSeqId() {
811
+ ctx.t_mqttCalled = false;
812
+ async function attemptRequest(retries = 3) {
813
+ try {
814
+ if (!ctx.fb_dtsg) {
815
+ const dtsg = await api.getFreshDtsg();
816
+ if (!dtsg) {
817
+ if (retries > 0) {
818
+ console.log("Failed to get fb_dtsg, retrying...");
819
+ await sleep(2000);
820
+ return attemptRequest(retries - 1);
821
+ }
822
+ throw {
823
+ error: "Could not obtain fb_dtsg after multiple attempts"
824
+ };
825
+ }
826
+ ctx.fb_dtsg = dtsg;
827
+ }
828
+ const form = {
829
+ av: ctx.userID,
830
+ fb_dtsg: ctx.fb_dtsg,
831
+ queries: JSON.stringify({
832
+ o0: {
833
+ doc_id: '3336396659757871',
834
+ query_params: {
835
+ limit: 1,
836
+ before: null,
837
+ tags: ['INBOX'],
838
+ includeDeliveryReceipts: false,
839
+ includeSeqID: true
840
+ }
841
+ }
842
+ }),
843
+ __user: ctx.userID,
844
+ __a: '1',
845
+ __req: '8',
846
+ __hs: '19577.HYP:comet_pkg.2.1..2.1',
847
+ dpr: '1',
848
+ fb_api_caller_class: 'RelayModern',
849
+ fb_api_req_friendly_name: 'MessengerGraphQLThreadlistFetcher'
850
+ };
851
+ const headers = {
852
+ 'Content-Type': 'application/x-www-form-urlencoded',
853
+ 'Referer': 'https://www.facebook.com/',
854
+ 'Origin': 'https://www.facebook.com',
855
+ 'sec-fetch-site': 'same-origin',
856
+ 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36',
857
+ 'Cookie': ctx.jar.getCookieString('https://www.facebook.com'),
858
+ 'accept': '*/*',
859
+ 'accept-encoding': 'gzip, deflate, br'
860
+ };
861
+
862
+ const resData = await defaultFuncs
863
+ .post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form, {
864
+ headers
865
+ })
866
+ .then(utils.parseAndCheckLogin(ctx, defaultFuncs));
867
+ if (debugSeq) {
868
+ console.log('GraphQL SeqID Response:', JSON.stringify(resData, null, 2));
869
+ }
870
+
871
+ if (resData.error === 1357004 || resData.error === 1357001) {
872
+ if (retries > 0) {
873
+ console.log("Session error, refreshing token and retrying...");
874
+ ctx.fb_dtsg = null;
875
+ await sleep(2000);
876
+ return attemptRequest(retries - 1);
877
+ }
878
+ throw {
879
+ error: "Session refresh failed after retries"
880
+ };
881
+ }
882
+
883
+ if (!Array.isArray(resData)) {
884
+ throw {
885
+ error: "Invalid response format",
886
+ res: resData
887
+ };
888
+ }
889
+
890
+ const seqID = resData[0]?.o0?.data?.viewer?.message_threads?.sync_sequence_id;
891
+ if (!seqID) {
892
+ throw {
893
+ error: "Missing sync_sequence_id",
894
+ res: resData
895
+ };
896
+ }
897
+
898
+ ctx.lastSeqId = seqID;
899
+ if (debugSeq) {
900
+ console.log('Got SeqID:', ctx.lastSeqId);
901
+ }
902
+
903
+ return listenMqtt(defaultFuncs, api, ctx, globalCallback);
904
+
905
+ } catch (err) {
906
+ if (retries > 0) {
907
+ console.log("Request failed, retrying...");
908
+
909
+ return attemptRequest(retries - 1);
910
+ }
911
+ throw err;
912
+ }
913
+ }
914
+
915
+ return attemptRequest()
916
+ .catch((err) => {
917
+ log.error("getSeqId", err);
918
+ if (utils.getType(err) == "Object" && err.error === "Not logged in") ctx.loggedIn = false;
919
+ return globalCallback(err);
920
+ });
921
+ }
922
+ return function (callback) {
923
+ class MessageEmitter extends EventEmitter {
924
+ stopListening(callback) {
925
+
926
+ callback = callback || (() => { });
927
+ globalCallback = identity;
928
+ if (ctx.mqttClient) {
929
+ ctx.mqttClient.unsubscribe("/webrtc");
930
+ ctx.mqttClient.unsubscribe("/rtc_multi");
931
+ ctx.mqttClient.unsubscribe("/onevc");
932
+ ctx.mqttClient.publish("/browser_close", "{}");
933
+ ctx.mqttClient.end(false, function (...data) {
934
+ callback(data);
935
+ ctx.mqttClient = undefined;
936
+ });
937
+ }
938
+ }
939
+
940
+ async stopListeningAsync() {
941
+ return new Promise((resolve) => {
942
+ this.stopListening(resolve);
943
+ });
944
+ }
945
+ }
946
+
947
+ const msgEmitter = new MessageEmitter();
948
+ globalCallback = (callback || function (error, message) {
949
+ if (error) {
950
+ return msgEmitter.emit("error", error);
951
+ }
952
+ msgEmitter.emit("message", message);
953
+ });
954
+
955
+ if (!ctx.firstListen)
956
+ ctx.lastSeqId = null;
957
+ ctx.syncToken = undefined;
958
+ ctx.t_mqttCalled = false;
959
+
960
+ if (!ctx.firstListen || !ctx.lastSeqId) {
961
+ getSeqId(defaultFuncs, api, ctx, globalCallback);
962
+ } else {
963
+ listenMqtt(defaultFuncs, api, ctx, globalCallback);
964
+ }
965
+
966
+ api.stopListening = msgEmitter.stopListening;
967
+ api.stopListeningAsync = msgEmitter.stopListeningAsync;
968
+ return msgEmitter;
969
+ };
710
970
  };