stfca 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 (82) hide show
  1. package/LICENSE-MIT +4 -0
  2. package/README.md +325 -0
  3. package/index.d.ts +615 -0
  4. package/index.js +1 -0
  5. package/module/config.js +33 -0
  6. package/module/login.js +48 -0
  7. package/module/loginHelper.js +722 -0
  8. package/module/options.js +44 -0
  9. package/package.json +69 -0
  10. package/src/api/action/addExternalModule.js +25 -0
  11. package/src/api/action/changeAvatar.js +137 -0
  12. package/src/api/action/changeBio.js +75 -0
  13. package/src/api/action/getCurrentUserID.js +7 -0
  14. package/src/api/action/handleFriendRequest.js +57 -0
  15. package/src/api/action/logout.js +76 -0
  16. package/src/api/action/refreshFb_dtsg.js +71 -0
  17. package/src/api/action/setPostReaction.js +106 -0
  18. package/src/api/action/unfriend.js +54 -0
  19. package/src/api/http/httpGet.js +46 -0
  20. package/src/api/http/httpPost.js +52 -0
  21. package/src/api/http/postFormData.js +47 -0
  22. package/src/api/messaging/addUserToGroup.js +68 -0
  23. package/src/api/messaging/changeAdminStatus.js +122 -0
  24. package/src/api/messaging/changeArchivedStatus.js +55 -0
  25. package/src/api/messaging/changeBlockedStatus.js +48 -0
  26. package/src/api/messaging/changeGroupImage.js +90 -0
  27. package/src/api/messaging/changeNickname.js +70 -0
  28. package/src/api/messaging/changeThreadColor.js +79 -0
  29. package/src/api/messaging/changeThreadEmoji.js +106 -0
  30. package/src/api/messaging/createNewGroup.js +88 -0
  31. package/src/api/messaging/createPoll.js +43 -0
  32. package/src/api/messaging/deleteMessage.js +56 -0
  33. package/src/api/messaging/deleteThread.js +56 -0
  34. package/src/api/messaging/editMessage.js +68 -0
  35. package/src/api/messaging/forwardAttachment.js +51 -0
  36. package/src/api/messaging/getEmojiUrl.js +29 -0
  37. package/src/api/messaging/getFriendsList.js +82 -0
  38. package/src/api/messaging/getMessage.js +829 -0
  39. package/src/api/messaging/handleMessageRequest.js +65 -0
  40. package/src/api/messaging/markAsDelivered.js +57 -0
  41. package/src/api/messaging/markAsRead.js +88 -0
  42. package/src/api/messaging/markAsReadAll.js +49 -0
  43. package/src/api/messaging/markAsSeen.js +61 -0
  44. package/src/api/messaging/muteThread.js +50 -0
  45. package/src/api/messaging/removeUserFromGroup.js +105 -0
  46. package/src/api/messaging/resolvePhotoUrl.js +43 -0
  47. package/src/api/messaging/searchForThread.js +52 -0
  48. package/src/api/messaging/sendMessage.js +379 -0
  49. package/src/api/messaging/sendMessageMqtt.js +323 -0
  50. package/src/api/messaging/sendTypingIndicator.js +67 -0
  51. package/src/api/messaging/setMessageReaction.js +75 -0
  52. package/src/api/messaging/setTitle.js +119 -0
  53. package/src/api/messaging/shareContact.js +49 -0
  54. package/src/api/messaging/threadColors.js +128 -0
  55. package/src/api/messaging/unsendMessage.js +81 -0
  56. package/src/api/messaging/uploadAttachment.js +95 -0
  57. package/src/api/socket/core/connectMqtt.js +179 -0
  58. package/src/api/socket/core/getSeqID.js +25 -0
  59. package/src/api/socket/core/getTaskResponseData.js +22 -0
  60. package/src/api/socket/core/markDelivery.js +12 -0
  61. package/src/api/socket/core/parseDelta.js +351 -0
  62. package/src/api/socket/detail/buildStream.js +208 -0
  63. package/src/api/socket/detail/constants.js +24 -0
  64. package/src/api/socket/listenMqtt.js +133 -0
  65. package/src/api/threads/getThreadHistory.js +664 -0
  66. package/src/api/threads/getThreadInfo.js +358 -0
  67. package/src/api/threads/getThreadList.js +248 -0
  68. package/src/api/threads/getThreadPictures.js +78 -0
  69. package/src/api/users/getUserID.js +65 -0
  70. package/src/api/users/getUserInfo.js +319 -0
  71. package/src/api/users/getUserInfoV2.js +133 -0
  72. package/src/core/sendReqMqtt.js +63 -0
  73. package/src/database/models/index.js +49 -0
  74. package/src/database/models/thread.js +31 -0
  75. package/src/database/models/user.js +32 -0
  76. package/src/database/threadData.js +98 -0
  77. package/src/database/userData.js +89 -0
  78. package/src/utils/client.js +214 -0
  79. package/src/utils/constants.js +23 -0
  80. package/src/utils/format.js +1111 -0
  81. package/src/utils/headers.js +41 -0
  82. package/src/utils/request.js +215 -0
@@ -0,0 +1,54 @@
1
+ "use strict";
2
+
3
+ const log = require("npmlog");
4
+ const { parseAndCheckLogin } = require("../../utils/client");
5
+ module.exports = function(defaultFuncs, api, ctx) {
6
+ return function unfriend(userID, callback) {
7
+ let resolveFunc = function() {};
8
+ let rejectFunc = function() {};
9
+ const returnPromise = new Promise(function(resolve, reject) {
10
+ resolveFunc = resolve;
11
+ rejectFunc = reject;
12
+ });
13
+
14
+ if (!callback) {
15
+ callback = function(err, friendList) {
16
+ if (err) {
17
+ return rejectFunc(err);
18
+ }
19
+ resolveFunc(friendList);
20
+ };
21
+ }
22
+
23
+ const form = {
24
+ uid: userID,
25
+ unref: "bd_friends_tab",
26
+ floc: "friends_tab",
27
+ "nctr[_mod]":
28
+ "pagelet_timeline_app_collection_" +
29
+ ctx.userID +
30
+ ":2356318349:2"
31
+ };
32
+
33
+ defaultFuncs
34
+ .post(
35
+ "https://www.facebook.com/ajax/profile/removefriendconfirm.php",
36
+ ctx.jar,
37
+ form
38
+ )
39
+ .then(parseAndCheckLogin(ctx, defaultFuncs))
40
+ .then(function(resData) {
41
+ if (resData.error) {
42
+ throw resData;
43
+ }
44
+
45
+ return callback(null, true);
46
+ })
47
+ .catch(function(err) {
48
+ log.error("unfriend", err);
49
+ return callback(err);
50
+ });
51
+
52
+ return returnPromise;
53
+ };
54
+ };
@@ -0,0 +1,46 @@
1
+ "use strict";
2
+
3
+ const { getType } = require("../../utils/format.js");
4
+ const { get } = require("../../utils/request.js");
5
+
6
+ const httpGetFactory = function (defaultFuncs, api, ctx) {
7
+ return function httpGet(url, form, callback, notAPI) {
8
+ let resolveFunc = () => { };
9
+ let rejectFunc = () => { };
10
+
11
+ const returnPromise = new Promise((resolve, reject) => {
12
+ resolveFunc = resolve;
13
+ rejectFunc = reject;
14
+ });
15
+
16
+ if (
17
+ !callback &&
18
+ (getType(form) === "Function" || getType(form) === "AsyncFunction")
19
+ ) {
20
+ callback = form;
21
+ form = {};
22
+ }
23
+
24
+ form = form || {};
25
+
26
+ callback =
27
+ callback ||
28
+ function (err, data) {
29
+ if (err) return rejectFunc(err);
30
+ resolveFunc(data);
31
+ };
32
+
33
+ const executor = notAPI ? get : defaultFuncs.get;
34
+
35
+ executor(url, ctx.jar, form)
36
+ .then((resData) => callback(null, resData.data))
37
+ .catch(function (err) {
38
+ console.error("httpGet", err);
39
+ return callback(err);
40
+ });
41
+
42
+ return returnPromise;
43
+ };
44
+ };
45
+
46
+ module.exports = httpGetFactory;
@@ -0,0 +1,52 @@
1
+ "use strict";
2
+
3
+ const { post } = require("../../utils/request");
4
+ const { getType } = require("../../utils/format");
5
+
6
+ const httpPostFactory = function (defaultFuncs, api, ctx) {
7
+ return function httpPost(url, form, callback, notAPI) {
8
+ let resolveFunc = () => { };
9
+ let rejectFunc = () => { };
10
+
11
+ const returnPromise = new Promise((resolve, reject) => {
12
+ resolveFunc = resolve;
13
+ rejectFunc = reject;
14
+ });
15
+
16
+ if (
17
+ !callback &&
18
+ (getType(form) === "Function" || getType(form) === "AsyncFunction")
19
+ ) {
20
+ callback = form;
21
+ form = {};
22
+ }
23
+
24
+ form = form || {};
25
+
26
+ callback =
27
+ callback ||
28
+ function (err, data) {
29
+ if (err) return rejectFunc(err);
30
+ resolveFunc(data);
31
+ };
32
+
33
+ const executor = notAPI ? post : defaultFuncs.post;
34
+
35
+ executor(url, ctx.jar, form, ctx.globalOptions)
36
+ .then((resData) => {
37
+ let data = resData.data;
38
+ if (typeof data === "object") {
39
+ data = JSON.stringify(data, null, 2);
40
+ }
41
+ callback(null, data);
42
+ })
43
+ .catch((err) => {
44
+ console.error("httpPost", err);
45
+ return callback(err);
46
+ });
47
+
48
+ return returnPromise;
49
+ };
50
+ };
51
+
52
+ module.exports = httpPostFactory;
@@ -0,0 +1,47 @@
1
+ "use strict";
2
+
3
+ var log = require("npmlog");
4
+ const { getType } = require("../../utils/format");
5
+ const { parseAndCheckLogin } = require("../../utils/client");
6
+ module.exports = function(defaultFuncs, api, ctx) {
7
+ return function postFormData(url, form, callback) {
8
+ var resolveFunc = function() {};
9
+ var rejectFunc = function() {};
10
+
11
+ var returnPromise = new Promise(function(resolve, reject) {
12
+ resolveFunc = resolve;
13
+ rejectFunc = reject;
14
+ });
15
+
16
+ if (
17
+ !callback &&
18
+ (getType(form) == "Function" ||
19
+ getType(form) == "AsyncFunction")
20
+ ) {
21
+ callback = form;
22
+ form = {};
23
+ }
24
+
25
+ form = form || {};
26
+
27
+ callback =
28
+ callback ||
29
+ function(err, data) {
30
+ if (err) return rejectFunc(err);
31
+ resolveFunc(data);
32
+ };
33
+
34
+ defaultFuncs
35
+ .postFormData(url, ctx.jar, form, {})
36
+ .then(parseAndCheckLogin(ctx, defaultFuncs))
37
+ .then(function(resData) {
38
+ callback(null, resData);
39
+ })
40
+ .catch(function(err) {
41
+ log.error("postFormData", err);
42
+ return callback(err);
43
+ });
44
+
45
+ return returnPromise;
46
+ };
47
+ };
@@ -0,0 +1,68 @@
1
+ "use strict";
2
+
3
+ const { generateOfflineThreadingID, getType } = require("../../utils/format.js");
4
+ module.exports = function (defaultFuncs, api, ctx) {
5
+ return function addUserToGroup(userID, threadID, callback) {
6
+ return new Promise((resolve, reject) => {
7
+ if (!ctx.mqttClient) {
8
+ const err = new Error("Not connected to MQTT");
9
+ callback?.(err);
10
+ return reject(err);
11
+ }
12
+ if (getType(threadID) !== "Number" && getType(threadID) !== "String") {
13
+ const err = new Error("ThreadID should be of type Number or String.");
14
+ callback?.(err);
15
+ return reject(err);
16
+ }
17
+ if (getType(userID) !== "Array") userID = [userID];
18
+ const reqID = ++ctx.wsReqNumber;
19
+ const taskID = ++ctx.wsTaskNumber;
20
+
21
+ const payload = {
22
+ epoch_id: generateOfflineThreadingID(),
23
+ tasks: [
24
+ {
25
+ failure_count: null,
26
+ label: "23",
27
+ payload: JSON.stringify({
28
+ thread_key: threadID,
29
+ contact_ids: userID,
30
+ sync_group: 1
31
+ }),
32
+ queue_name: threadID.toString(),
33
+ task_id: taskID
34
+ }
35
+ ],
36
+ version_id: "24502707779384158"
37
+ };
38
+ const form = JSON.stringify({
39
+ app_id: "772021112871879",
40
+ payload: JSON.stringify(payload),
41
+ request_id: reqID,
42
+ type: 3
43
+ });
44
+ const handleRes = (topic, message) => {
45
+ if (topic !== "/ls_resp") return;
46
+ let jsonMsg;
47
+ try {
48
+ jsonMsg = JSON.parse(message.toString());
49
+ jsonMsg.payload = JSON.parse(jsonMsg.payload);
50
+ } catch {
51
+ return;
52
+ }
53
+ if (jsonMsg.request_id !== reqID) return;
54
+ ctx.mqttClient.removeListener("message", handleRes);
55
+ callback?.(null, { success: true, response: jsonMsg.payload });
56
+ resolve({ success: true, response: jsonMsg.payload });
57
+ };
58
+ ctx.mqttClient.on("message", handleRes);
59
+ ctx.mqttClient.publish("/ls_req", form, { qos: 1, retain: false }, (err) => {
60
+ if (err) {
61
+ ctx.mqttClient.removeListener("message", handleRes);
62
+ callback?.(err);
63
+ reject(err);
64
+ }
65
+ });
66
+ });
67
+ };
68
+ };
@@ -0,0 +1,122 @@
1
+ "use strict";
2
+
3
+ const { generateOfflineThreadingID, getType } = require("../../utils/format");
4
+
5
+ module.exports = function (defaultFuncs, api, ctx) {
6
+ function changeAdminStatusNoMqtt(threadID, adminID, adminStatus) {
7
+ if (getType(threadID) !== "String") throw { error: "changeAdminStatus: threadID must be a string" };
8
+ if (getType(adminID) !== "String" && getType(adminID) !== "Array") throw { error: "changeAdminStatus: adminID must be a string or an array" };
9
+ if (getType(adminStatus) !== "Boolean") throw { error: "changeAdminStatus: adminStatus must be true or false" };
10
+ let wsContent = {
11
+ request_id: 1,
12
+ type: 3,
13
+ payload: {
14
+ version_id: '3816854585040595',
15
+ tasks: [],
16
+ epoch_id: generateOfflineThreadingID(),
17
+ data_trace_id: null
18
+ },
19
+ app_id: '772021112871879'
20
+ }
21
+ if (getType(adminID) === "Array") {
22
+ for (let i = 0; i < adminID.length; i++) {
23
+ wsContent.payload.tasks.push({
24
+ label: '25',
25
+ payload: JSON.stringify({ thread_key: threadID, contact_id: adminID[i], is_admin: adminStatus }),
26
+ queue_name: 'admin_status',
27
+ task_id: i + 1,
28
+ failure_count: null
29
+ });
30
+ }
31
+ } else {
32
+ wsContent.payload.tasks.push({
33
+ label: '25',
34
+ payload: JSON.stringify({ thread_key: threadID, contact_id: adminID, is_admin: adminStatus }),
35
+ queue_name: 'admin_status',
36
+ task_id: 1,
37
+ failure_count: null
38
+ });
39
+ }
40
+ wsContent.payload = JSON.stringify(wsContent.payload);
41
+ return new Promise((resolve, reject) => ctx.mqttClient && ctx.mqttClient.publish('/ls_req', JSON.stringify(wsContent), {}, (err, _packet) => err ? reject(err) : resolve()));
42
+ };
43
+ function changeAdminStatusMqtt(threadID, adminID, adminStatus) {
44
+ if (!ctx.mqttClient) {
45
+ throw new Error("Not connected to MQTT");
46
+ }
47
+ if (getType(threadID) !== "String") {
48
+ throw { error: "changeAdminStatus: threadID must be a string" };
49
+ }
50
+ if (getType(adminID) !== "String" && getType(adminID) !== "Array") {
51
+ throw { error: "changeAdminStatus: adminID must be a string or an array" };
52
+ }
53
+ if (getType(adminStatus) !== "Boolean") {
54
+ throw { error: "changeAdminStatus: adminStatus must be true or false" };
55
+ }
56
+ const tasks = [];
57
+ const isAdmin = adminStatus ? 1 : 0;
58
+ const epochID = generateOfflineThreadingID();
59
+ if (getType(adminID) === "Array") {
60
+ adminID.forEach((id, index) => {
61
+ tasks.push({
62
+ failure_count: null,
63
+ label: "25",
64
+ payload: JSON.stringify({
65
+ thread_key: threadID,
66
+ contact_id: id,
67
+ is_admin: isAdmin
68
+ }),
69
+ queue_name: "admin_status",
70
+ task_id: index + 1
71
+ });
72
+ });
73
+ } else {
74
+ tasks.push({
75
+ failure_count: null,
76
+ label: "25",
77
+ payload: JSON.stringify({
78
+ thread_key: threadID,
79
+ contact_id: adminID,
80
+ is_admin: isAdmin
81
+ }),
82
+ queue_name: "admin_status",
83
+ task_id: 1
84
+ });
85
+ }
86
+ let count_req = 0
87
+ const form = JSON.stringify({
88
+ app_id: "2220391788200892",
89
+ payload: JSON.stringify({
90
+ epoch_id: epochID,
91
+ tasks: tasks,
92
+ version_id: "8798795233522156"
93
+ }),
94
+ request_id: ++count_req,
95
+ type: 3
96
+ });
97
+ return new Promise((resolve, reject) => {
98
+ if (ctx.mqttClient) {
99
+ ctx.mqttClient.publish("/ls_req", form, {}, (err, _packet) => {
100
+ if (err) {
101
+ reject(err);
102
+ } else {
103
+ resolve();
104
+ }
105
+ });
106
+ } else {
107
+ reject(new Error("MQTT client is not available"));
108
+ }
109
+ });
110
+ };
111
+ return function changeAdminStatus(threadID, adminID, adminStatus) {
112
+ if (ctx.mqttClient) {
113
+ try {
114
+ changeAdminStatusMqtt(threadID, adminID, adminStatus);
115
+ } catch (e) {
116
+ changeAdminStatusNoMqtt(threadID, adminID, adminStatus);
117
+ }
118
+ } else {
119
+ changeAdminStatusNoMqtt(threadID, adminID, adminStatus);
120
+ }
121
+ };
122
+ };
@@ -0,0 +1,55 @@
1
+ "use strict";
2
+
3
+ const log = require("npmlog");
4
+ const { formatID } = require("../../utils/format");
5
+ const { parseAndCheckLogin } = require("../../utils/client");
6
+ module.exports = function(defaultFuncs, api, ctx) {
7
+ return function changeArchivedStatus(threadOrThreads, archive, callback) {
8
+ let resolveFunc = function() {};
9
+ let rejectFunc = function() {};
10
+ const returnPromise = new Promise(function(resolve, reject) {
11
+ resolveFunc = resolve;
12
+ rejectFunc = reject;
13
+ });
14
+
15
+ if (!callback) {
16
+ callback = function(err) {
17
+ if (err) {
18
+ return rejectFunc(err);
19
+ }
20
+ resolveFunc();
21
+ };
22
+ }
23
+
24
+ const form = {};
25
+
26
+ if (Array.isArray(threadOrThreads)) {
27
+ for (let i = 0; i < threadOrThreads.length; i++) {
28
+ form["ids[" + formatID(threadOrThreads[i]) + "]"] = archive;
29
+ }
30
+ } else {
31
+ form["ids[" + formatID(threadOrThreads) + "]"] = archive;
32
+ }
33
+
34
+ defaultFuncs
35
+ .post(
36
+ "https://www.facebook.com/ajax/mercury/change_archived_status.php",
37
+ ctx.jar,
38
+ form
39
+ )
40
+ .then(parseAndCheckLogin(ctx, defaultFuncs))
41
+ .then(function(resData) {
42
+ if (resData.error) {
43
+ throw resData;
44
+ }
45
+
46
+ return callback();
47
+ })
48
+ .catch(function(err) {
49
+ log.error("changeArchivedStatus", err);
50
+ return callback(err);
51
+ });
52
+
53
+ return returnPromise;
54
+ };
55
+ };
@@ -0,0 +1,48 @@
1
+ "use strict";
2
+
3
+ const log = require("npmlog");
4
+ const { parseAndCheckLogin, saveCookies } = require("../../utils/client");
5
+ module.exports = function(defaultFuncs, api, ctx) {
6
+ return function changeBlockedStatus(userID, block, callback) {
7
+ let resolveFunc = function() {};
8
+ let rejectFunc = function() {};
9
+ const returnPromise = new Promise(function(resolve, reject) {
10
+ resolveFunc = resolve;
11
+ rejectFunc = reject;
12
+ });
13
+
14
+ if (!callback) {
15
+ callback = function(err) {
16
+ if (err) {
17
+ return rejectFunc(err);
18
+ }
19
+ resolveFunc();
20
+ };
21
+ }
22
+
23
+ defaultFuncs
24
+ .post(
25
+ `https://www.facebook.com/messaging/${
26
+ block ? "" : "un"
27
+ }block_messages/`,
28
+ ctx.jar,
29
+ {
30
+ fbid: userID
31
+ }
32
+ )
33
+ .then(saveCookies(ctx.jar))
34
+ .then(parseAndCheckLogin(ctx, defaultFuncs))
35
+ .then(function(resData) {
36
+ if (resData.error) {
37
+ throw resData;
38
+ }
39
+
40
+ return callback();
41
+ })
42
+ .catch(function(err) {
43
+ log.error("changeBlockedStatus", err);
44
+ return callback(err);
45
+ });
46
+ return returnPromise;
47
+ };
48
+ };
@@ -0,0 +1,90 @@
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
+ function handleUpload(image) {
8
+ const form = {
9
+ images_only: "true",
10
+ "attachment[]": image
11
+ };
12
+ return defaultFuncs
13
+ .postFormData("https://upload.facebook.com/ajax/mercury/upload.php", ctx.jar, form, {})
14
+ .then(parseAndCheckLogin(ctx, defaultFuncs))
15
+ .then(resData => {
16
+ if (resData.error) throw resData;
17
+ return resData.payload.metadata[0];
18
+ });
19
+ }
20
+ return function changeGroupImage(image, threadID, callback) {
21
+ return new Promise((resolve, reject) => {
22
+ if (!ctx.mqttClient) {
23
+ const err = new Error("Not connected to MQTT");
24
+ callback?.(err);
25
+ return reject(err);
26
+ }
27
+ if (!threadID || typeof threadID !== "string") {
28
+ const err = new Error("Invalid threadID");
29
+ callback?.(err);
30
+ return reject(err);
31
+ }
32
+ const reqID = ++ctx.wsReqNumber;
33
+ const taskID = ++ctx.wsTaskNumber;
34
+ const onResponse = (topic, message) => {
35
+ if (topic !== "/ls_resp") return;
36
+ let jsonMsg;
37
+ try {
38
+ jsonMsg = JSON.parse(message.toString());
39
+ jsonMsg.payload = JSON.parse(jsonMsg.payload);
40
+ } catch (err) {
41
+ return;
42
+ }
43
+ if (jsonMsg.request_id !== reqID) return;
44
+ ctx.mqttClient.removeListener("message", onResponse);
45
+ callback?.(null, { success: true, response: jsonMsg.payload });
46
+ return resolve({ success: true, response: jsonMsg.payload });
47
+ };
48
+ ctx.mqttClient.on("message", onResponse);
49
+ handleUpload(image)
50
+ .then(payload => {
51
+ const imageID = payload.image_id;
52
+ const taskPayload = {
53
+ thread_key: threadID,
54
+ image_id: imageID,
55
+ sync_group: 1
56
+ };
57
+
58
+ const mqttPayload = {
59
+ epoch_id: generateOfflineThreadingID(),
60
+ tasks: [
61
+ {
62
+ failure_count: null,
63
+ label: "37",
64
+ payload: JSON.stringify(taskPayload),
65
+ queue_name: "thread_image",
66
+ task_id: taskID
67
+ }
68
+ ],
69
+ version_id: "8798795233522156"
70
+ };
71
+ const request = {
72
+ app_id: "2220391788200892",
73
+ payload: JSON.stringify(mqttPayload),
74
+ request_id: reqID,
75
+ type: 3
76
+ };
77
+ ctx.mqttClient.publish("/ls_req", JSON.stringify(request), {
78
+ qos: 1,
79
+ retain: false
80
+ });
81
+ })
82
+ .catch(err => {
83
+ ctx.mqttClient.removeListener("message", onResponse);
84
+ log.error("changeGroupImageMqtt", err);
85
+ callback?.(err);
86
+ reject(err);
87
+ });
88
+ });
89
+ };
90
+ };
@@ -0,0 +1,70 @@
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 changeNickname(nickname, threadID, participantID, 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
+ if (!threadID || !participantID) {
15
+ const err = new Error("Missing required parameters");
16
+ callback?.(err);
17
+ return reject(err);
18
+ }
19
+ const reqID = ++ctx.wsReqNumber;
20
+ const taskID = ++ctx.wsTaskNumber;
21
+ const payload = {
22
+ epoch_id: generateOfflineThreadingID(),
23
+ tasks: [
24
+ {
25
+ failure_count: null,
26
+ label: "44",
27
+ payload: JSON.stringify({
28
+ thread_key: threadID,
29
+ contact_id: participantID,
30
+ nickname: nickname || "",
31
+ sync_group: 1
32
+ }),
33
+ queue_name: "thread_participant_nickname",
34
+ task_id: taskID
35
+ }
36
+ ],
37
+ version_id: "8798795233522156"
38
+ };
39
+ const request = {
40
+ app_id: "2220391788200892",
41
+ payload: JSON.stringify(payload),
42
+ request_id: reqID,
43
+ type: 3
44
+ };
45
+ const onResponse = (topic, message) => {
46
+ if (topic !== "/ls_resp") return;
47
+ let jsonMsg;
48
+ try {
49
+ jsonMsg = JSON.parse(message.toString());
50
+ jsonMsg.payload = JSON.parse(jsonMsg.payload);
51
+ } catch (err) {
52
+ return;
53
+ }
54
+ if (jsonMsg.request_id !== reqID) return;
55
+ ctx.mqttClient.removeListener("message", onResponse);
56
+ callback?.(null, { success: true, response: jsonMsg.payload });
57
+ return resolve({ success: true, response: jsonMsg.payload });
58
+ };
59
+ ctx.mqttClient.on("message", onResponse);
60
+ ctx.mqttClient.publish("/ls_req", JSON.stringify(request), { qos: 1, retain: false }, (err) => {
61
+ if (err) {
62
+ ctx.mqttClient.removeListener("message", onResponse);
63
+ log.error("changeNicknameMqtt", err);
64
+ callback?.(err);
65
+ return reject(err);
66
+ }
67
+ });
68
+ });
69
+ };
70
+ };