stfca 1.0.2 → 1.0.3

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 (141) hide show
  1. package/CHANGELOG.md +32 -0
  2. package/README.md +45 -2
  3. package/checkUpdate.js +109 -0
  4. package/index.js +460 -1
  5. package/package.json +22 -46
  6. package/src/{api/action/addExternalModule.js → addExternalModule.js} +25 -25
  7. package/src/addUserToGroup.js +115 -0
  8. package/src/changeAdminStatus.js +103 -0
  9. package/src/changeArchivedStatus.js +55 -0
  10. package/src/{api/action/changeAvatar.js → changeAvatar.js} +136 -137
  11. package/src/changeAvatarV2.js +86 -0
  12. package/src/changeAvt.js +85 -0
  13. package/src/{api/action/changeBio.js → changeBio.js} +76 -75
  14. package/src/{api/messaging/changeBlockedStatus.js → changeBlockedStatus.js} +49 -48
  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 +79 -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 +59 -0
  23. package/src/createCommentPost.js +230 -0
  24. package/src/{api/messaging/createNewGroup.js → createNewGroup.js} +88 -88
  25. package/src/createPoll.js +71 -0
  26. package/src/createPost.js +276 -0
  27. package/src/{api/messaging/deleteMessage.js → deleteMessage.js} +56 -56
  28. package/src/{api/messaging/deleteThread.js → deleteThread.js} +56 -56
  29. package/src/editMessage.js +68 -0
  30. package/src/editMessageOld.js +67 -0
  31. package/src/follow.js +74 -0
  32. package/src/forwardAttachment.js +60 -0
  33. package/src/getAccess.js +112 -0
  34. package/src/getAvatarUser.js +78 -0
  35. package/src/{api/action/getCurrentUserID.js → getCurrentUserID.js} +7 -7
  36. package/src/{api/messaging/getEmojiUrl.js → getEmojiUrl.js} +2 -2
  37. package/src/{api/messaging/getFriendsList.js → getFriendsList.js} +83 -82
  38. package/src/{api/messaging/getMessage.js → getMessage.js} +847 -829
  39. package/src/getRegion.js +7 -0
  40. package/src/{api/threads/getThreadHistory.js → getThreadHistory.js} +680 -664
  41. package/src/getThreadHistoryDeprecated.js +71 -0
  42. package/src/getThreadInfo.js +232 -0
  43. package/src/getThreadInfoDeprecated.js +56 -0
  44. package/src/getThreadList.js +213 -0
  45. package/src/getThreadListDeprecated.js +46 -0
  46. package/src/getThreadPictures.js +59 -0
  47. package/src/getUID.js +119 -0
  48. package/src/{api/users/getUserID.js → getUserID.js} +61 -65
  49. package/src/getUserInfo.js +66 -0
  50. package/src/handleFriendRequest.js +46 -0
  51. package/src/handleMessageRequest.js +47 -0
  52. package/src/httpGet.js +49 -0
  53. package/src/httpPost.js +48 -0
  54. package/src/listenMqtt.js +833 -0
  55. package/src/logout.js +75 -0
  56. package/src/markAsDelivered.js +47 -0
  57. package/src/markAsRead.js +70 -0
  58. package/src/markAsReadAll.js +40 -0
  59. package/src/markAsSeen.js +48 -0
  60. package/src/muteThread.js +45 -0
  61. package/src/refreshFb_dtsg.js +89 -0
  62. package/src/removeUserFromGroup.js +79 -0
  63. package/src/{api/messaging/resolvePhotoUrl.js → resolvePhotoUrl.js} +45 -43
  64. package/src/{api/messaging/searchForThread.js → searchForThread.js} +53 -52
  65. package/src/searchStickers.js +53 -0
  66. package/src/sendMessage.js +379 -0
  67. package/src/{api/messaging/sendMessageMqtt.js → sendMessageMqtt.js} +322 -323
  68. package/src/sendTypingIndicator.js +101 -0
  69. package/src/sendTypingIndicatorV2.js +28 -0
  70. package/src/setMessageReaction.js +122 -0
  71. package/src/setMessageReactionMqtt.js +62 -0
  72. package/src/{api/action/setPostReaction.js → setPostReaction.js} +112 -106
  73. package/src/setStoryReaction.js +64 -0
  74. package/src/setTitle.js +90 -0
  75. package/src/shareContact.js +110 -0
  76. package/src/shareLink.js +59 -0
  77. package/src/stopListenMqtt.js +23 -0
  78. package/src/{api/messaging/threadColors.js → threadColors.js} +131 -128
  79. package/src/{api/action/unfriend.js → unfriend.js} +52 -54
  80. package/src/unsendMessage.js +45 -0
  81. package/src/{api/messaging/uploadAttachment.js → uploadAttachment.js} +93 -95
  82. package/utils.js +2876 -0
  83. package/LICENSE-MIT +0 -4
  84. package/index.d.ts +0 -615
  85. package/module/config.js +0 -33
  86. package/module/login.js +0 -48
  87. package/module/loginHelper.js +0 -722
  88. package/module/options.js +0 -44
  89. package/src/api/action/handleFriendRequest.js +0 -57
  90. package/src/api/action/logout.js +0 -76
  91. package/src/api/action/refreshFb_dtsg.js +0 -71
  92. package/src/api/http/httpGet.js +0 -46
  93. package/src/api/http/httpPost.js +0 -52
  94. package/src/api/http/postFormData.js +0 -47
  95. package/src/api/messaging/addUserToGroup.js +0 -68
  96. package/src/api/messaging/changeAdminStatus.js +0 -122
  97. package/src/api/messaging/changeArchivedStatus.js +0 -55
  98. package/src/api/messaging/changeGroupImage.js +0 -90
  99. package/src/api/messaging/changeNickname.js +0 -70
  100. package/src/api/messaging/changeThreadColor.js +0 -79
  101. package/src/api/messaging/changeThreadEmoji.js +0 -106
  102. package/src/api/messaging/createPoll.js +0 -43
  103. package/src/api/messaging/editMessage.js +0 -68
  104. package/src/api/messaging/forwardAttachment.js +0 -51
  105. package/src/api/messaging/handleMessageRequest.js +0 -65
  106. package/src/api/messaging/markAsDelivered.js +0 -57
  107. package/src/api/messaging/markAsRead.js +0 -88
  108. package/src/api/messaging/markAsReadAll.js +0 -49
  109. package/src/api/messaging/markAsSeen.js +0 -61
  110. package/src/api/messaging/muteThread.js +0 -50
  111. package/src/api/messaging/removeUserFromGroup.js +0 -105
  112. package/src/api/messaging/sendMessage.js +0 -379
  113. package/src/api/messaging/sendTypingIndicator.js +0 -67
  114. package/src/api/messaging/setMessageReaction.js +0 -75
  115. package/src/api/messaging/setTitle.js +0 -119
  116. package/src/api/messaging/shareContact.js +0 -49
  117. package/src/api/messaging/unsendMessage.js +0 -81
  118. package/src/api/socket/core/connectMqtt.js +0 -179
  119. package/src/api/socket/core/getSeqID.js +0 -25
  120. package/src/api/socket/core/getTaskResponseData.js +0 -22
  121. package/src/api/socket/core/markDelivery.js +0 -12
  122. package/src/api/socket/core/parseDelta.js +0 -351
  123. package/src/api/socket/detail/buildStream.js +0 -208
  124. package/src/api/socket/detail/constants.js +0 -24
  125. package/src/api/socket/listenMqtt.js +0 -133
  126. package/src/api/threads/getThreadInfo.js +0 -358
  127. package/src/api/threads/getThreadList.js +0 -248
  128. package/src/api/threads/getThreadPictures.js +0 -78
  129. package/src/api/users/getUserInfo.js +0 -319
  130. package/src/api/users/getUserInfoV2.js +0 -133
  131. package/src/core/sendReqMqtt.js +0 -63
  132. package/src/database/models/index.js +0 -49
  133. package/src/database/models/thread.js +0 -31
  134. package/src/database/models/user.js +0 -32
  135. package/src/database/threadData.js +0 -98
  136. package/src/database/userData.js +0 -89
  137. package/src/utils/client.js +0 -214
  138. package/src/utils/constants.js +0 -23
  139. package/src/utils/format.js +0 -1111
  140. package/src/utils/headers.js +0 -41
  141. package/src/utils/request.js +0 -215
@@ -1,81 +0,0 @@
1
- "use strict";
2
-
3
- const { generateOfflineThreadingID } = require("../../utils/format.js");
4
- const log = require("npmlog");
5
-
6
- module.exports = function (defaultFuncs, api, ctx) {
7
- return function unsendMessage(messageID, threadID, callback) {
8
- return new Promise((resolve, reject) => {
9
- if (!ctx.mqttClient) {
10
- const err = new Error("Not connected to MQTT");
11
- callback?.(err);
12
- return reject(err);
13
- }
14
- const reqID = ++ctx.wsReqNumber;
15
- const taskID = ++ctx.wsTaskNumber;
16
- const taskPayload = {
17
- message_id: messageID,
18
- thread_key: threadID,
19
- sync_group: 1,
20
- };
21
- const task = {
22
- failure_count: null,
23
- label: "33",
24
- payload: JSON.stringify(taskPayload),
25
- queue_name: "unsend_message",
26
- task_id: taskID,
27
- };
28
- const content = {
29
- app_id: "2220391788200892",
30
- payload: JSON.stringify({
31
- tasks: [task],
32
- epoch_id: parseInt(generateOfflineThreadingID()),
33
- version_id: "25393437286970779",
34
- }),
35
- request_id: reqID,
36
- type: 3,
37
- };
38
- try {
39
- ctx.mqttClient.publish("/ls_req", JSON.stringify(content), {
40
- qos: 1,
41
- retain: false
42
- });
43
- } catch (err) {
44
- log.error("unsendMessage (MQTT publish failed)", err);
45
- callback?.(err);
46
- return reject(err);
47
- }
48
- const handleRes = (topic, message) => {
49
- if (topic !== "/ls_resp") return;
50
- let jsonMsg;
51
- try {
52
- jsonMsg = JSON.parse(message.toString());
53
- jsonMsg.payload = JSON.parse(jsonMsg.payload);
54
- } catch (err) {
55
- return;
56
- }
57
- if (jsonMsg.request_id !== reqID) return;
58
- ctx.mqttClient.removeListener("message", handleRes);
59
- try {
60
- const msgID = jsonMsg.payload.step?.[1]?.[2]?.[2]?.[1]?.[2];
61
- const msgReplace = jsonMsg.payload.step?.[1]?.[2]?.[2]?.[1]?.[4];
62
- if (msgID && msgReplace) {
63
- const bodies = {
64
- body: msgReplace,
65
- messageID: msgID
66
- };
67
- callback?.(null, bodies);
68
- return resolve(bodies);
69
- } else {
70
- callback?.(null, { success: true });
71
- return resolve({ success: true });
72
- }
73
- } catch (err) {
74
- callback?.(null, { success: true });
75
- return resolve({ success: true });
76
- }
77
- };
78
- ctx.mqttClient.on("message", handleRes);
79
- });
80
- };
81
- };
@@ -1,179 +0,0 @@
1
- "use strict";
2
- const { formatID } = require("../../../utils/format");
3
- module.exports = function createListenMqtt(deps) {
4
- const { WebSocket, mqtt, HttpsProxyAgent, buildStream, buildProxy, topics, parseDelta, getTaskResponseData } = deps;
5
- return function listenMqtt(defaultFuncs, api, ctx, globalCallback) {
6
- const chatOn = ctx.globalOptions.online;
7
- const foreground = false;
8
- const sessionID = Math.floor(Math.random() * Number.MAX_SAFE_INTEGER) + 1;
9
- const username = {
10
- u: ctx.userID,
11
- s: sessionID,
12
- chat_on: chatOn,
13
- fg: foreground,
14
- d: ctx.clientId,
15
- ct: "websocket",
16
- aid: 219994525426954,
17
- aids: null,
18
- mqtt_sid: "",
19
- cp: 3,
20
- ecp: 10,
21
- st: [],
22
- pm: [],
23
- dc: "",
24
- no_auto_fg: true,
25
- gas: null,
26
- pack: [],
27
- p: null,
28
- php_override: ""
29
- };
30
- const cookies = api.getCookies();
31
- let host;
32
- if (ctx.mqttEndpoint) {
33
- host = `${ctx.mqttEndpoint}&sid=${sessionID}&cid=${ctx.clientId}`;
34
- } else if (ctx.region) {
35
- host = `wss://edge-chat.facebook.com/chat?region=${ctx.region.toLowerCase()}&sid=${sessionID}&cid=${ctx.clientId}`;
36
- } else {
37
- host = `wss://edge-chat.facebook.com/chat?sid=${sessionID}&cid=${ctx.clientId}`;
38
- }
39
- const options = {
40
- clientId: "mqttwsclient",
41
- protocolId: "MQIsdp",
42
- protocolVersion: 3,
43
- username: JSON.stringify(username),
44
- clean: true,
45
- wsOptions: {
46
- headers: {
47
- Cookie: cookies,
48
- Origin: "https://www.facebook.com",
49
- "User-Agent": ctx.globalOptions.userAgent || "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/134.0.0.0 Safari/537.36",
50
- Referer: "https://www.facebook.com/",
51
- Host: "edge-chat.facebook.com",
52
- Connection: "Upgrade",
53
- Pragma: "no-cache",
54
- "Cache-Control": "no-cache",
55
- Upgrade: "websocket",
56
- "Sec-WebSocket-Version": "13",
57
- "Accept-Encoding": "gzip, deflate, br",
58
- "Accept-Language": "vi,en;q=0.9",
59
- "Sec-WebSocket-Extensions": "permessage-deflate; client_max_window_bits"
60
- },
61
- origin: "https://www.facebook.com",
62
- protocolVersion: 13,
63
- binaryType: "arraybuffer"
64
- },
65
- keepalive: 30,
66
- reschedulePings: true,
67
- reconnectPeriod: 1000,
68
- connectTimeout: 5000
69
- };
70
- if (ctx.globalOptions.proxy !== undefined) {
71
- const agent = new HttpsProxyAgent(ctx.globalOptions.proxy);
72
- options.wsOptions.agent = agent;
73
- }
74
- ctx.mqttClient = new mqtt.Client(() => buildStream(options, new WebSocket(host, options.wsOptions), buildProxy()), options);
75
- const mqttClient = ctx.mqttClient;
76
- global.mqttClient = mqttClient;
77
- mqttClient.on("error", function (err) {
78
- const msg = String(err && err.message ? err.message : err || "");
79
- if (ctx._ending && /No subscription existed/i.test(msg)) {
80
- console.warn("[FCA-WARN] MQTT ignore unsubscribe error during shutdown");
81
- return;
82
- }
83
- console.error(`[FCA-ERROR] MQTT error: ${msg}`);
84
- // mqttClient.end();
85
- // console.warn("[FCA-WARN] MQTT autoReconnect listenMqtt() in 2000ms");
86
- // setTimeout(() => listenMqtt(defaultFuncs, api, ctx, globalCallback), 2000);
87
- process.exit(1);
88
- });
89
- mqttClient.on("connect", function () {
90
- if (process.env.OnStatus === undefined) {
91
- console.log("[FCA-INFO] ST FCA connected");
92
- process.env.OnStatus = true;
93
- }
94
- topics.forEach(topicsub => mqttClient.subscribe(topicsub));
95
- let topic;
96
- const queue = { sync_api_version: 11, max_deltas_able_to_process: 100, delta_batch_size: 500, encoding: "JSON", entity_fbid: ctx.userID, initial_titan_sequence_id: ctx.lastSeqId, device_params: null };
97
- if (ctx.syncToken) {
98
- topic = "/messenger_sync_get_diffs";
99
- queue.last_seq_id = ctx.lastSeqId;
100
- queue.sync_token = ctx.syncToken;
101
- } else {
102
- topic = "/messenger_sync_create_queue";
103
- queue.initial_titan_sequence_id = ctx.lastSeqId;
104
- queue.device_params = null;
105
- }
106
- mqttClient.publish(topic, JSON.stringify(queue), { qos: 1, retain: false });
107
- mqttClient.publish("/foreground_state", JSON.stringify({ foreground: chatOn }), { qos: 1 });
108
- mqttClient.publish("/set_client_settings", JSON.stringify({ make_user_available_when_in_foreground: true }), { qos: 1 });
109
- const rTimeout = setTimeout(function () {
110
- mqttClient.end();
111
- listenMqtt(defaultFuncs, api, ctx, globalCallback);
112
- }, 5000);
113
- ctx.tmsWait = function () {
114
- clearTimeout(rTimeout);
115
- ctx.globalOptions.emitReady ? globalCallback({ type: "ready", error: null }) : "";
116
- delete ctx.tmsWait;
117
- };
118
- });
119
- mqttClient.on("message", function (topic, message, _packet) {
120
- try {
121
- let jsonMessage = Buffer.isBuffer(message) ? Buffer.from(message).toString() : message;
122
- try {
123
- jsonMessage = JSON.parse(jsonMessage);
124
- } catch (e) {
125
- jsonMessage = {};
126
- }
127
- if (jsonMessage.type === "jewel_requests_add") {
128
- globalCallback(null, { type: "friend_request_received", actorFbId: jsonMessage.from.toString(), timestamp: Date.now().toString() });
129
- } else if (jsonMessage.type === "jewel_requests_remove_old") {
130
- globalCallback(null, { type: "friend_request_cancel", actorFbId: jsonMessage.from.toString(), timestamp: Date.now().toString() });
131
- } else if (topic === "/t_ms") {
132
- if (ctx.tmsWait && typeof ctx.tmsWait == "function") {
133
- ctx.tmsWait();
134
- }
135
- if (jsonMessage.firstDeltaSeqId && jsonMessage.syncToken) {
136
- ctx.lastSeqId = jsonMessage.firstDeltaSeqId;
137
- ctx.syncToken = jsonMessage.syncToken;
138
- }
139
- if (jsonMessage.lastIssuedSeqId) {
140
- ctx.lastSeqId = parseInt(jsonMessage.lastIssuedSeqId);
141
- }
142
- for (const i in jsonMessage.deltas) {
143
- const delta = jsonMessage.deltas[i];
144
- parseDelta(defaultFuncs, api, ctx, globalCallback, { delta });
145
- }
146
- } else if (topic === "/thread_typing" || topic === "/orca_typing_notifications") {
147
- const typ = { type: "typ", isTyping: !!jsonMessage.state, from: jsonMessage.sender_fbid.toString(), threadID: formatID((jsonMessage.thread || jsonMessage.sender_fbid).toString()) };
148
- globalCallback(null, typ);
149
- } else if (topic === "/orca_presence") {
150
- if (!ctx.globalOptions.updatePresence) {
151
- for (const i in jsonMessage.list) {
152
- const data = jsonMessage.list[i];
153
- const userID = data["u"];
154
- const presence = { type: "presence", userID: userID.toString(), timestamp: data["l"] * 1000, statuses: data["p"] };
155
- globalCallback(null, presence);
156
- }
157
- }
158
- } else if (topic == "/ls_resp") {
159
- const parsedPayload = JSON.parse(jsonMessage.payload);
160
- const reqID = jsonMessage.request_id;
161
- if (ctx["tasks"].has(reqID)) {
162
- const taskData = ctx["tasks"].get(reqID);
163
- const { type: taskType, callback: taskCallback } = taskData;
164
- const taskRespData = getTaskResponseData(taskType, parsedPayload);
165
- if (taskRespData == null) {
166
- taskCallback("error", null);
167
- } else {
168
- taskCallback(null, Object.assign({ type: taskType, reqID: reqID }, taskRespData));
169
- }
170
- }
171
- }
172
- } catch (ex) {
173
- return;
174
- }
175
- });
176
- mqttClient.on("close", function () { });
177
- mqttClient.on("disconnect", () => { });
178
- };
179
- };
@@ -1,25 +0,0 @@
1
- "use strict";
2
- const { getType } = require("../../../utils/format");
3
- const { parseAndCheckLogin } = require("../../../utils/client");
4
- module.exports = function createGetSeqID(deps) {
5
- const { listenMqtt } = deps;
6
- return function getSeqID(defaultFuncs, api, ctx, globalCallback, form) {
7
- ctx.t_mqttCalled = false;
8
- return defaultFuncs.post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form).then(parseAndCheckLogin(ctx, defaultFuncs)).then(resData => {
9
- if (getType(resData) !== "Array") throw { error: "Not logged in", res: resData };
10
- if (!Array.isArray(resData) || !resData.length) return;
11
- const lastRes = resData[resData.length - 1];
12
- if (lastRes && lastRes.successful_results === 0) return;
13
- const syncSeqId = resData[0] && resData[0].o0 && resData[0].o0.data && resData[0].o0.data.viewer && resData[0].o0.data.viewer.message_threads && resData[0].o0.data.viewer.message_threads.sync_sequence_id;
14
- if (syncSeqId) {
15
- ctx.lastSeqId = syncSeqId;
16
- listenMqtt(defaultFuncs, api, ctx, globalCallback);
17
- } else {
18
- throw { error: "getSeqId: no sync_sequence_id found.", res: resData };
19
- }
20
- }).catch(err => {
21
- if (getType(err) === "Object" && err.error === "Not logged in") ctx.loggedIn = false;
22
- return globalCallback(err);
23
- });
24
- };
25
- };
@@ -1,22 +0,0 @@
1
- "use strict";
2
- module.exports = function getTaskResponseData(taskType, payload) {
3
- try {
4
- switch (taskType) {
5
- case "send_message_mqtt":
6
- return {
7
- type: taskType,
8
- threadID: payload.step[1][2][2][1][2],
9
- messageID: payload.step[1][2][2][1][3],
10
- payload: payload.step[1][2]
11
- };
12
- case "set_message_reaction":
13
- return { mid: payload.step[1][2][2][1][4] };
14
- case "edit_message":
15
- return { mid: payload.step[1][2][2][1][2] };
16
- default:
17
- return null;
18
- }
19
- } catch (e) {
20
- return null;
21
- }
22
- };
@@ -1,12 +0,0 @@
1
- "use strict";
2
- module.exports = function markDelivery(ctx, api, threadID, messageID) {
3
- if (threadID && messageID) {
4
- api.markAsDelivered(threadID, messageID, err => {
5
- if (err) {} else {
6
- if (ctx.globalOptions.autoMarkRead) {
7
- api.markAsRead(threadID, err2 => {});
8
- }
9
- }
10
- });
11
- }
12
- };
@@ -1,351 +0,0 @@
1
- "use strict";
2
- const { formatDeltaEvent, formatMessage, _formatAttachment, formatDeltaMessage, formatDeltaReadReceipt, formatID, getType, decodeClientPayload } = require("../../../utils/format");
3
- module.exports = function createParseDelta(deps) {
4
- const { markDelivery, parseAndCheckLogin } = deps;
5
- return function parseDelta(defaultFuncs, api, ctx, globalCallback, { delta }) {
6
- if (delta.class === "NewMessage") {
7
- const resolveAttachmentUrl = i => {
8
- if (!delta.attachments || i === delta.attachments.length || getType(delta.attachments) !== "Array") {
9
- let fmtMsg;
10
- try {
11
- fmtMsg = formatDeltaMessage(delta);
12
- } catch (err) {
13
- return;
14
- }
15
- if (fmtMsg) {
16
- if (ctx.globalOptions.autoMarkDelivery) {
17
- markDelivery(ctx, api, fmtMsg.threadID, fmtMsg.messageID);
18
- }
19
- if (!ctx.globalOptions.selfListen && fmtMsg.senderID === ctx.userID) return;
20
- globalCallback(null, fmtMsg);
21
- }
22
- } else {
23
- const attachment = delta.attachments[i];
24
- if (attachment.mercury.attach_type === "photo") {
25
- api.resolvePhotoUrl(attachment.fbid, (err, url) => {
26
- if (!err) attachment.mercury.metadata.url = url;
27
- resolveAttachmentUrl(i + 1);
28
- });
29
- } else {
30
- resolveAttachmentUrl(i + 1);
31
- }
32
- }
33
- };
34
- resolveAttachmentUrl(0);
35
- } else if (delta.class === "ClientPayload") {
36
- const clientPayload = decodeClientPayload(delta.payload);
37
- if (clientPayload && clientPayload.deltas) {
38
- for (const d of clientPayload.deltas) {
39
- if (d.deltaMessageReaction && !!ctx.globalOptions.listenEvents) {
40
- const messageReaction = {
41
- type: "message_reaction",
42
- threadID: (d.deltaMessageReaction.threadKey.threadFbId ? d.deltaMessageReaction.threadKey.threadFbId : d.deltaMessageReaction.threadKey.otherUserFbId).toString(),
43
- messageID: d.deltaMessageReaction.messageId,
44
- reaction: d.deltaMessageReaction.reaction,
45
- senderID: d.deltaMessageReaction.senderId.toString(),
46
- userID: d.deltaMessageReaction.userId.toString()
47
- };
48
- globalCallback(null, messageReaction);
49
- } else if (d.deltaRecallMessageData && !!ctx.globalOptions.listenEvents) {
50
- const messageUnsend = {
51
- type: "message_unsend",
52
- threadID: (d.deltaRecallMessageData.threadKey.threadFbId ? d.deltaRecallMessageData.threadKey.threadFbId : d.deltaRecallMessageData.threadKey.otherUserFbId).toString(),
53
- messageID: d.deltaRecallMessageData.messageID,
54
- senderID: d.deltaRecallMessageData.senderID.toString(),
55
- deletionTimestamp: d.deltaRecallMessageData.deletionTimestamp,
56
- timestamp: d.deltaRecallMessageData.timestamp
57
- };
58
- globalCallback(null, messageUnsend);
59
- } else if (d.deltaMessageReply) {
60
- const mdata = d.deltaMessageReply.message === undefined ? [] : d.deltaMessageReply.message.data === undefined ? [] : d.deltaMessageReply.message.data.prng === undefined ? [] : JSON.parse(d.deltaMessageReply.message.data.prng);
61
- const m_id = mdata.map(u => u.i);
62
- const m_offset = mdata.map(u => u.o);
63
- const m_length = mdata.map(u => u.l);
64
- const mentions = {};
65
- for (let i = 0; i < m_id.length; i++) {
66
- mentions[m_id[i]] = (d.deltaMessageReply.message.body || "").substring(m_offset[i], m_offset[i] + m_length[i]);
67
- }
68
- const callbackToReturn = {
69
- type: "message_reply",
70
- threadID: (d.deltaMessageReply.message.messageMetadata.threadKey.threadFbId ? d.deltaMessageReply.message.messageMetadata.threadKey.threadFbId : d.deltaMessageReply.message.messageMetadata.threadKey.otherUserFbId).toString(),
71
- messageID: d.deltaMessageReply.message.messageMetadata.messageId,
72
- senderID: d.deltaMessageReply.message.messageMetadata.actorFbId.toString(),
73
- attachments: (d.deltaMessageReply.message.attachments || []).map(att => {
74
- const mercury = JSON.parse(att.mercuryJSON);
75
- Object.assign(att, mercury);
76
- return att;
77
- }).map(att => {
78
- let x;
79
- try {
80
- x = _formatAttachment(att);
81
- } catch (ex) {
82
- x = att;
83
- x.error = ex;
84
- x.type = "unknown";
85
- }
86
- return x;
87
- }),
88
- args: (d.deltaMessageReply.message.body || "").trim().split(/\s+/),
89
- body: d.deltaMessageReply.message.body || "",
90
- isGroup: !!d.deltaMessageReply.message.messageMetadata.threadKey.threadFbId,
91
- mentions,
92
- timestamp: parseInt(d.deltaMessageReply.message.messageMetadata.timestamp),
93
- participantIDs: (d.deltaMessageReply.message.participants || []).map(e => e.toString())
94
- };
95
- if (d.deltaMessageReply.repliedToMessage) {
96
- const mdata2 = d.deltaMessageReply.repliedToMessage === undefined ? [] : d.deltaMessageReply.repliedToMessage.data === undefined ? [] : d.deltaMessageReply.repliedToMessage.data.prng === undefined ? [] : JSON.parse(d.deltaMessageReply.repliedToMessage.data.prng);
97
- const m_id2 = mdata2.map(u => u.i);
98
- const m_offset2 = mdata2.map(u => u.o);
99
- const m_length2 = mdata2.map(u => u.l);
100
- const rmentions = {};
101
- for (let i = 0; i < m_id2.length; i++) {
102
- rmentions[m_id2[i]] = (d.deltaMessageReply.repliedToMessage.body || "").substring(m_offset2[i], m_offset2[i] + m_length2[i]);
103
- }
104
- callbackToReturn.messageReply = {
105
- threadID: (d.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId ? d.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId : d.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.otherUserFbId).toString(),
106
- messageID: d.deltaMessageReply.repliedToMessage.messageMetadata.messageId,
107
- senderID: d.deltaMessageReply.repliedToMessage.messageMetadata.actorFbId.toString(),
108
- attachments: d.deltaMessageReply.repliedToMessage.attachments.map(att => {
109
- let mercury;
110
- try {
111
- mercury = JSON.parse(att.mercuryJSON);
112
- Object.assign(att, mercury);
113
- } catch (ex) {
114
- mercury = {};
115
- }
116
- return att;
117
- }).map(att => {
118
- let x;
119
- try {
120
- x = _formatAttachment(att);
121
- } catch (ex) {
122
- x = att;
123
- x.error = ex;
124
- x.type = "unknown";
125
- }
126
- return x;
127
- }),
128
- args: (d.deltaMessageReply.repliedToMessage.body || "").trim().split(/\s+/),
129
- body: d.deltaMessageReply.repliedToMessage.body || "",
130
- isGroup: !!d.deltaMessageReply.repliedToMessage.messageMetadata.threadKey.threadFbId,
131
- mentions: rmentions,
132
- timestamp: parseInt(d.deltaMessageReply.repliedToMessage.messageMetadata.timestamp),
133
- participantIDs: (d.deltaMessageReply.repliedToMessage.participants || []).map(e => e.toString())
134
- };
135
- } else if (d.deltaMessageReply.replyToMessageId) {
136
- return defaultFuncs.post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, {
137
- av: ctx.globalOptions.pageID,
138
- queries: JSON.stringify({
139
- o0: {
140
- doc_id: "2848441488556444",
141
- query_params: {
142
- thread_and_message_id: {
143
- thread_id: callbackToReturn.threadID,
144
- message_id: d.deltaMessageReply.replyToMessageId.id
145
- }
146
- }
147
- }
148
- })
149
- }).then(parseAndCheckLogin(ctx, defaultFuncs)).then(resData => {
150
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
151
- if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
152
- const fetchData = resData[0].o0.data.message;
153
- const mobj = {};
154
- for (const n in fetchData.message.ranges) {
155
- mobj[fetchData.message.ranges[n].entity.id] = (fetchData.message.text || "").substr(fetchData.message.ranges[n].offset, fetchData.message.ranges[n].length);
156
- }
157
- callbackToReturn.messageReply = {
158
- type: "Message",
159
- threadID: callbackToReturn.threadID,
160
- messageID: fetchData.message_id,
161
- senderID: fetchData.message_sender.id.toString(),
162
- attachments: fetchData.message.blob_attachment.map(att => _formatAttachment({ blob_attachment: att })),
163
- args: (fetchData.message.text || "").trim().split(/\s+/) || [],
164
- body: fetchData.message.text || "",
165
- isGroup: callbackToReturn.isGroup,
166
- mentions: mobj,
167
- timestamp: parseInt(fetchData.timestamp_precise)
168
- };
169
- }).catch(err => {}).finally(() => {
170
- if (ctx.globalOptions.autoMarkDelivery) {
171
- markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
172
- }
173
- if (!ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID) return;
174
- globalCallback(null, callbackToReturn);
175
- });
176
- } else {
177
- callbackToReturn.delta = d;
178
- }
179
- if (ctx.globalOptions.autoMarkDelivery) {
180
- markDelivery(ctx, api, callbackToReturn.threadID, callbackToReturn.messageID);
181
- }
182
- if (!ctx.globalOptions.selfListen && callbackToReturn.senderID === ctx.userID) return;
183
- globalCallback(null, callbackToReturn);
184
- }
185
- }
186
- return;
187
- }
188
- }
189
- switch (delta.class) {
190
- case "ReadReceipt": {
191
- let fmtMsg;
192
- try {
193
- fmtMsg = formatDeltaReadReceipt(delta);
194
- } catch (err) {
195
- return;
196
- }
197
- globalCallback(null, fmtMsg);
198
- break;
199
- }
200
- case "AdminTextMessage": {
201
- switch (delta.type) {
202
- case "instant_game_dynamic_custom_update":
203
- case "accept_pending_thread":
204
- case "confirm_friend_request":
205
- case "shared_album_delete":
206
- case "shared_album_addition":
207
- case "pin_messages_v2":
208
- case "unpin_messages_v2":
209
- case "change_thread_theme":
210
- case "change_thread_nickname":
211
- case "change_thread_icon":
212
- case "change_thread_quick_reaction":
213
- case "change_thread_admins":
214
- case "group_poll":
215
- case "joinable_group_link_mode_change":
216
- case "magic_words":
217
- case "change_thread_approval_mode":
218
- case "messenger_call_log":
219
- case "participant_joined_group_call":
220
- case "rtc_call_log":
221
- case "update_vote": {
222
- let fmtMsg;
223
- try {
224
- fmtMsg = formatDeltaEvent(delta);
225
- } catch (err) {
226
- return;
227
- }
228
- globalCallback(null, fmtMsg);
229
- break;
230
- }
231
- }
232
- break;
233
- }
234
- case "ForcedFetch": {
235
- if (!delta.threadKey) return;
236
- const mid = delta.messageId;
237
- const tid = delta.threadKey.threadFbId;
238
- if (mid && tid) {
239
- const form = {
240
- av: ctx.globalOptions.pageID,
241
- queries: JSON.stringify({
242
- o0: {
243
- doc_id: "2848441488556444",
244
- query_params: {
245
- thread_and_message_id: {
246
- thread_id: tid.toString(),
247
- message_id: mid
248
- }
249
- }
250
- }
251
- })
252
- };
253
- defaultFuncs.post("https://www.facebook.com/api/graphqlbatch/", ctx.jar, form).then(parseAndCheckLogin(ctx, defaultFuncs)).then(resData => {
254
- if (resData[resData.length - 1].error_results > 0) throw resData[0].o0.errors;
255
- if (resData[resData.length - 1].successful_results === 0) throw { error: "forcedFetch: there was no successful_results", res: resData };
256
- const fetchData = resData[0].o0.data.message;
257
- if (getType(fetchData) === "Object") {
258
- switch (fetchData.__typename) {
259
- case "ThreadImageMessage":
260
- if ((!ctx.globalOptions.selfListen && fetchData.message_sender.id.toString() === ctx.userID) || !ctx.loggedIn) {} else {
261
- globalCallback(null, {
262
- type: "event",
263
- threadID: formatID(tid.toString()),
264
- logMessageType: "log:thread-image",
265
- logMessageData: {
266
- image: {
267
- attachmentID: fetchData.image_with_metadata && fetchData.image_with_metadata.legacy_attachment_id,
268
- width: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.x,
269
- height: fetchData.image_with_metadata && fetchData.image_with_metadata.original_dimensions.y,
270
- url: fetchData.image_with_metadata && fetchData.image_with_metadata.preview.uri
271
- }
272
- },
273
- logMessageBody: fetchData.snippet,
274
- timestamp: fetchData.timestamp_precise,
275
- author: fetchData.message_sender.id
276
- });
277
- }
278
- break;
279
- case "UserMessage": {
280
- const event = {
281
- type: "message",
282
- senderID: formatID(fetchData.message_sender.id),
283
- body: fetchData.message.text || "",
284
- threadID: formatID(tid.toString()),
285
- messageID: fetchData.message_id,
286
- attachments: [
287
- {
288
- type: "share",
289
- ID: fetchData.extensible_attachment.legacy_attachment_id,
290
- url: fetchData.extensible_attachment.story_attachment.url,
291
- title: fetchData.extensible_attachment.story_attachment.title_with_entities.text,
292
- description: fetchData.extensible_attachment.story_attachment.description.text,
293
- source: fetchData.extensible_attachment.story_attachment.source,
294
- image: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).uri,
295
- width: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).width,
296
- height: ((fetchData.extensible_attachment.story_attachment.media || {}).image || {}).height,
297
- playable: ((fetchData.extensible_attachment.story_attachment.media || {}).is_playable || false),
298
- duration: ((fetchData.extensible_attachment.story_attachment.media || {}).playable_duration_in_ms || 0),
299
- subattachments: fetchData.extensible_attachment.subattachments,
300
- properties: fetchData.extensible_attachment.story_attachment.properties
301
- }
302
- ],
303
- mentions: {},
304
- timestamp: parseInt(fetchData.timestamp_precise),
305
- isGroup: fetchData.message_sender.id !== tid.toString()
306
- };
307
- globalCallback(null, event);
308
- break;
309
- }
310
- default:
311
- break;
312
- }
313
- } else {
314
- return;
315
- }
316
- }).catch(err => {});
317
- }
318
- break;
319
- }
320
- case "ThreadName":
321
- case "ParticipantsAddedToGroupThread":
322
- case "ParticipantLeftGroupThread": {
323
- let formattedEvent;
324
- try {
325
- formattedEvent = formatDeltaEvent(delta);
326
- } catch (err) {
327
- return;
328
- }
329
- if (!ctx.globalOptions.selfListen && formattedEvent.author.toString() === ctx.userID) return;
330
- if (!ctx.loggedIn) return;
331
- globalCallback(null, formattedEvent);
332
- break;
333
- }
334
- case "NewMessage": {
335
- const hasLiveLocation = d => {
336
- const attachment = d.attachments && d.attachments[0] && d.attachments[0].mercury && d.attachments[0].mercury.extensible_attachment;
337
- const storyAttachment = attachment && attachment.story_attachment;
338
- return storyAttachment && storyAttachment.style_list && storyAttachment.style_list.includes("message_live_location");
339
- };
340
- if (delta.attachments && delta.attachments.length === 1 && hasLiveLocation(delta)) {
341
- delta.class = "UserLocation";
342
- try {
343
- const fmtMsg = formatDeltaEvent(delta);
344
- globalCallback(null, fmtMsg);
345
- } catch (err) {}
346
- }
347
- break;
348
- }
349
- }
350
- };
351
- };