kzi 1.0.0

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