fca-arif-babu 1.0.21

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