fca-horidai-remastered 1.0.0 → 2.0.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,228 @@
1
+ 'use strict';
2
+
3
+ var utils = require('../utils.js');
4
+ var log = require('npmlog');
5
+
6
+ module.exports = function (http, api, ctx) {
7
+ function handleUpload(msg, form) {
8
+ var cb;
9
+ var uploads = [];
10
+ var returnPromise = new Promise(function (resolve, reject) {
11
+ cb = error => error ? reject(error) : resolve();
12
+ });
13
+
14
+ for (let item of msg.attachments) {
15
+ if (!utils.isReadableStream(item))
16
+ return cb({ error: 'image should be a readable stream and not ' + utils.getType(image) });
17
+
18
+ var httpData = http
19
+ .postFormData('https://www.facebook.com/ajax/ufi/upload/', ctx.jar, {
20
+ profile_id: ctx.userID,
21
+ source: 19,
22
+ target_id: ctx.userID,
23
+ file: item
24
+ })
25
+ .then(utils.parseAndCheckLogin(ctx, http))
26
+ .then(function (res) {
27
+ if (res.errors || res.error || !res.payload)
28
+ throw res;
29
+
30
+ return {
31
+ media: {
32
+ id: res.payload.fbid
33
+ }
34
+ }
35
+ })
36
+ .catch(cb);
37
+
38
+ uploads.push(httpData);
39
+ }
40
+
41
+ Promise
42
+ .all(uploads)
43
+ .then(function (main) {
44
+ main.forEach(item => form.input.attachments.push(item));
45
+
46
+ return cb();
47
+ })
48
+ .catch(cb);
49
+
50
+ return returnPromise;
51
+ }
52
+
53
+ function handleURL(msg, form) {
54
+ if (typeof msg.url == 'string') {
55
+ form.input.attachments = [
56
+ {
57
+ link: {
58
+ external: {
59
+ url: msg.url
60
+ }
61
+ }
62
+ }
63
+ ];
64
+ }
65
+ }
66
+
67
+ function handleMentions(msg, form) {
68
+ for (let item of msg.mentions) {
69
+ var { tag, id, fromIndex } = item;
70
+
71
+ if (typeof tag != 'string')
72
+ throw 'Mention tag must be string';
73
+ if (!id)
74
+ throw 'id must be string';
75
+ var offset = msg.body.indexOf(tag, fromIndex || 0);
76
+ if (offset < 0)
77
+ throw 'Mention for "' + tag + '" not found in message string.';
78
+ form.input.message.ranges.push({
79
+ entity: { id },
80
+ length: tag.length,
81
+ offset
82
+ });
83
+ }
84
+ }
85
+
86
+ function handleSticker(msg, form) {
87
+ if (msg.sticker) {
88
+ form.input.attachments = [
89
+ {
90
+ media: {
91
+ id: msg.sticker
92
+ }
93
+ }
94
+ ];
95
+ }
96
+ }
97
+
98
+ function createContent(form) {
99
+ var cb;
100
+ var returnPromise = new Promise(function (resolve, reject) {
101
+ cb = (error, info) => info ? resolve(info) : reject(error);
102
+ });
103
+
104
+ http
105
+ .post('https://www.facebook.com/api/graphql/', ctx.jar, {
106
+ fb_api_caller_class: 'RelayModern',
107
+ fb_api_req_friendly_name: 'useCometUFICreateCommentMutation',
108
+ variables: JSON.stringify(form),
109
+ server_timestamps: !0,
110
+ doc_id: 6993516810709754
111
+ })
112
+ .then(utils.parseAndCheckLogin(ctx, http))
113
+ .then(function (res) {
114
+ if (res.errors)
115
+ throw res;
116
+
117
+ var res = res.data.comment_create;
118
+ var info = {
119
+ id: res.feedback_comment_edge.node.id,
120
+ url: res.feedback_comment_edge.node.feedback.url,
121
+ count: res.feedback.total_comment_count
122
+ }
123
+ return cb(null, info);
124
+ })
125
+ .catch(cb);
126
+
127
+ return returnPromise;
128
+ }
129
+
130
+ return function createCommentPost(msg, postID, callback, replyCommentID) {
131
+ var cb;
132
+ var returnPromise = new Promise(function (resolve, reject) {
133
+ cb = (error, info) => info ? resolve(info) : reject(error);
134
+ });
135
+
136
+ if (typeof msg == 'function') {
137
+ var error = 'Message must be a string or object!!';
138
+ log.error('createCommentPost', error);
139
+ return msg(error);
140
+ }
141
+ if (typeof postID == 'function') {
142
+ var error = 'postID must be a string!!';
143
+ log.error('createCommentPost', error);
144
+ return postID(error);
145
+ }
146
+ if (typeof callback == 'string') {
147
+ replyCommentID = callback;
148
+ callback = null;
149
+ }
150
+ if (typeof callback == 'function')
151
+ cb = calback;
152
+
153
+ var MessageType = utils.getType(msg);
154
+
155
+ if (MessageType == 'String')
156
+ msg = {
157
+ body: msg,
158
+ attachments: [],
159
+ mentions: [],
160
+ sticker: null,
161
+ url: null
162
+ }
163
+ else if (MessageType == 'Object') {
164
+ msg.mentions ? !Array.isArray(msg.mentions) ? msg.mentions = [msg.mentions] : null : msg.mentions = [];
165
+ msg.attachments ? !Array.isArray(msg.attachments) ? msg.attachments = [msg.attachments] : null : msg.attachments = [];
166
+ isNaN(msg.sticker) ? msg.sticker = null : null;
167
+ msg.body ? typeof msg.body == 'object' ? msg.body = JSON.stringify(msg.body) : null : msg.body = '';
168
+ } else {
169
+ var error = 'Message must be a string or object!!';
170
+ log.error('createCommentPost', error);
171
+ return cb(error);
172
+ }
173
+ if (typeof postID != 'string') {
174
+ var error = 'postID must be a string!!';
175
+ log.error('createCommentPost', error);
176
+ return cb(error);
177
+ }
178
+
179
+ if (typeof replyCommentID != 'string')
180
+ replyCommentID = null;
181
+
182
+ var form = {
183
+ feedLocation: 'NEWSFEED',
184
+ feedbackSource: 1,
185
+ groupID: null,
186
+ input: {
187
+ client_mutation_id: Math.round(Math.random() * 19).toString(),
188
+ actor_id: ctx.userID,
189
+ attachments: [],
190
+ feedback_id: Buffer.from('feedback:' + postID).toString('base64'),
191
+ formatting_style: null,
192
+ message: {
193
+ ranges: [],
194
+ text: msg.body
195
+ },
196
+ reply_comment_parent_fbid: replyCommentID ? isNaN(replyCommentID) ? replyCommentID : Buffer.from('comment:' + postID + '_' + replyCommentID).toString('base64') : null,
197
+ reply_target_clicked: !!replyCommentID,
198
+ attribution_id_v2:
199
+ 'CometHomeRoot.react,comet.home,via_cold_start,'
200
+ + Date.now()
201
+ + ',156248,4748854339,,',
202
+ vod_video_timestamp: null,
203
+ feedback_referrer: '/',
204
+ is_tracking_encrypted: !0,
205
+ tracking: [],
206
+ feedback_source: 'NEWS_FEED',
207
+ idempotence_token: 'client:' + utils.getGUID(),
208
+ session_id: utils.getGUID()
209
+ },
210
+ inviteShortLinkKey: null,
211
+ renderLocation: null,
212
+ scale: 1,
213
+ useDefaultActor: !1,
214
+ focusCommentID: null
215
+ }
216
+ handleUpload(msg, form)
217
+ .then(_ => handleURL(msg, form))
218
+ .then(_ => handleMentions(msg, form))
219
+ .then(_ => handleSticker(msg, form))
220
+ .then(_ => createContent(form))
221
+ .then(info => cb(null, info))
222
+ .catch(function (err) {
223
+ log.error('createCommentPost', err);
224
+ return cb(null, err);
225
+ })
226
+ return returnPromise;
227
+ }
228
+ }
@@ -0,0 +1,56 @@
1
+ 'use strict';
2
+
3
+ const { generateOfflineThreadingID, getCurrentTimestamp } = require('../utils');
4
+
5
+ function isCallable(func) {
6
+ try {
7
+ Reflect.apply(func, null, []);
8
+ return true;
9
+ } catch (error) {
10
+ return false;
11
+ }
12
+ }
13
+
14
+ module.exports = function (defaultFuncs, api, ctx) {
15
+ return function createPollMqtt(title, options, threadID, callback) {
16
+ if (!ctx.mqttClient) {
17
+ throw new Error('Not connected to MQTT');
18
+ }
19
+
20
+ ctx.wsReqNumber += 1;
21
+ ctx.wsTaskNumber += 1;
22
+
23
+ const taskPayload = {
24
+ question_text: title,
25
+ thread_key: threadID,
26
+ options: options,
27
+ sync_group: 1,
28
+ };
29
+
30
+ const task = {
31
+ failure_count: null,
32
+ label: '163',
33
+ payload: JSON.stringify(taskPayload),
34
+ queue_name: 'poll_creation',
35
+ task_id: ctx.wsTaskNumber,
36
+ };
37
+
38
+ const content = {
39
+ app_id: '2220391788200892',
40
+ payload: JSON.stringify({
41
+ data_trace_id: null,
42
+ epoch_id: parseInt(generateOfflineThreadingID()),
43
+ tasks: [task],
44
+ version_id: '7158486590867448',
45
+ }),
46
+ request_id: ctx.wsReqNumber,
47
+ type: 3,
48
+ };
49
+
50
+ if (isCallable(callback)) {
51
+ // to be implemented
52
+ }
53
+
54
+ ctx.mqttClient.publish('/ls_req', JSON.stringify(content), { qos: 1, retain: false });
55
+ };
56
+ };
@@ -0,0 +1,277 @@
1
+ 'use strict';
2
+
3
+ var utils = require('../utils');
4
+ var log = require('npmlog');
5
+
6
+ module.exports = function (http, api, ctx) {
7
+ function handleUpload(msg, form) {
8
+ var cb;
9
+ var rt = new Promise(function (resolve, reject) {
10
+ cb = error => error ? reject(error) : resolve();
11
+ });
12
+
13
+ if (!msg.attachment) cb();
14
+ else {
15
+ msg.attachment = Array.isArray(msg.attachment) ? msg.attachment : [msg.attachment];
16
+ let uploads = [];
17
+ for (let attachment of msg.attachment) {
18
+ if (!utils.isReadableStream(attachment))
19
+ cb('Attachment should be a readable stream, not ' + utils.getType(attachment));
20
+
21
+ var vari = {
22
+ source: 8,
23
+ profile_id: ctx.userID,
24
+ waterfallxapp: 'comet',
25
+ farr: attachment,
26
+ upload_id: 'jsc_c_6'
27
+ }
28
+ var main = http
29
+ .postFormData('https://upload.facebook.com/ajax/react_composer/attachments/photo/upload', ctx.jar, vari)
30
+ .then(utils.parseAndCheckLogin(ctx, http))
31
+ .then(function (res) {
32
+ if (res.error || res.errors)
33
+ throw res;
34
+
35
+ return res.payload;
36
+ });
37
+
38
+ uploads.push(main);
39
+ }
40
+
41
+ Promise
42
+ .all(uploads)
43
+ .then(function (res) {
44
+ for (let payload of res) {
45
+ if (!payload) break;
46
+ form.input.attachments.push({
47
+ photo: {
48
+ id: payload.photoID
49
+ }
50
+ });
51
+ }
52
+
53
+ return cb();
54
+ })
55
+ .catch(cb);
56
+ }
57
+
58
+ return rt;
59
+ }
60
+
61
+ function handleUrl(msg, form) {
62
+ var cb;
63
+ var rt = new Promise(function (resolve, reject) {
64
+ cb = error => error ? reject(error) : resolve();
65
+ });
66
+
67
+ if (!msg.url) cb();
68
+ else {
69
+ var vari = {
70
+ feedLocation: "FEED_COMPOSER",
71
+ focusCommentID: null,
72
+ goodwillCampaignId: "",
73
+ goodwillCampaignMediaIds: [],
74
+ goodwillContentType: null,
75
+ params: {
76
+ url: msg.url
77
+ },
78
+ privacySelectorRenderLocation: "COMET_COMPOSER",
79
+ renderLocation: "composer_preview",
80
+ parentStoryID: null,
81
+ scale: 1,
82
+ useDefaultActor: false,
83
+ shouldIncludeStoryAttachment: false,
84
+ __relay_internal__pv__IsWorkUserrelayprovider: false,
85
+ __relay_internal__pv__IsMergQAPollsrelayprovider: false
86
+ }
87
+
88
+ http
89
+ .post('https://www.facebook.com/api/graphql/', ctx.jar, {
90
+ fb_api_req_friendly_name: 'ComposerLinkAttachmentPreviewQuery',
91
+ variables: JSON.stringify(vari),
92
+ server_timestamps: true,
93
+ doc_id: 6549975235094234
94
+ })
95
+ .then(utils.parseAndCheckLogin(ctx, http))
96
+ .then(function (res) {
97
+ var res = (res[0] || res).data.link_preview;
98
+ if (JSON.parse(res.share_scrape_data).share_type == 400)
99
+ throw { error: 'url is not accepted' }
100
+
101
+ form.input.attachments.push({
102
+ link: {
103
+ share_scrape_data: res.share_scrape_data
104
+ }
105
+ });
106
+
107
+ return cb();
108
+ })
109
+ .catch(cb);
110
+ }
111
+
112
+ return rt;
113
+ }
114
+
115
+ function handleMention(msg, form) {
116
+ if (!msg.mentions) return;
117
+
118
+ msg.mentions = Array.isArray(msg.mentions) ? msg.mentions : [msg.mentions];
119
+ for (let mention of msg.mentions) {
120
+ var { id, tag, fromIndex } = mention;
121
+
122
+ if (typeof tag != 'string')
123
+ throw 'Mention tag must be string';
124
+ if (!id)
125
+ throw 'id must be string';
126
+ var offset = msg.body.indexOf(tag, fromIndex || 0);
127
+ if (offset < 0)
128
+ throw 'Mention for "' + tag + '" not found in message string.';
129
+ form.input.message.ranges.push({
130
+ entity: { id },
131
+ length: tag.length,
132
+ offset
133
+ });
134
+ }
135
+ }
136
+
137
+ function createContent(vari) {
138
+ var cb;
139
+ var rt = new Promise(function (resolve, reject) {
140
+ cb = (error, postData) => error ? reject(error) : resolve(postData);
141
+ });
142
+
143
+ var form = {
144
+ fb_api_req_friendly_name: 'ComposerStoryCreateMutation',
145
+ variables: JSON.stringify(vari),
146
+ server_timestamps: true,
147
+ doc_id: 6255089511280268
148
+ }
149
+
150
+ http
151
+ .post('https://www.facebook.com/api/graphql/', ctx.jar, form)
152
+ .then(utils.parseAndCheckLogin(ctx, http))
153
+ .then(res => cb(null, res))
154
+ .catch(cb);
155
+
156
+ return rt;
157
+ }
158
+
159
+ return function createPost(msg, callback) {
160
+ var cb;
161
+ var rt = new Promise(function (resolve, reject) {
162
+ cb = (error, url) => url ? resolve(url) : reject(error);
163
+ });
164
+
165
+ if (typeof msg == 'function') {
166
+ var error = 'Msg must be a string or object and not function';
167
+ log.error('createPost', error);
168
+ return msg(error);
169
+ }
170
+ if (typeof callback == 'function') cb = callback;
171
+
172
+ var typeMsg = utils.getType(msg);
173
+ if (!['Object', 'String'].includes(typeMsg)) {
174
+ var error = 'Msg must be a string or object and not ' + typeMsg;
175
+ log.error('createPost', error);
176
+ return cb(error);
177
+ } else if (typeMsg == 'String') msg = { body: msg };
178
+ msg.allowUserID = msg.allowUserID ? !Array.isArray(msg.allowUserID) ? [msg.allowUserID] : msg.allowUserID : null;
179
+
180
+ var sessionID = utils.getGUID();
181
+ var base = [
182
+ 'EVERYONE',
183
+ 'FRIENDS',
184
+ 'SELF'
185
+ ];
186
+ var form = {
187
+ input: {
188
+ composer_entry_point: !msg.groupID && msg.url ? 'share_modal' : "inline_composer",
189
+ composer_source_surface: !msg.groupID && msg.url ? 'feed_story' : msg.groupID ? "group" : "timeline",
190
+ composer_type: !msg.groupID && msg.url ? 'share' : msg.groupID ? "group" : "timeline",
191
+ idempotence_token: sessionID + "_FEED",
192
+ source: "WWW",
193
+ attachments: [],
194
+ audience: msg.groupID ? {
195
+ to_id: msg.groupID
196
+ } : {
197
+ privacy: {
198
+ allow: msg.allowUserID ? msg.allowUserID : [],
199
+ base_state: msg.allowUserID && msg.allowUserID.length > 0 ? base[2] : (base[msg.baseState - 1] || base[0]),
200
+ deny: [],
201
+ tag_expansion_state: "UNSPECIFIED"
202
+ }
203
+ },
204
+ message: {
205
+ ranges: [],
206
+ text: msg.body ? typeof msg.body == 'object' ? JSON.stringify(msg.body, null, 2) : msg.body : ''
207
+ },
208
+ with_tags_ids: [],
209
+ inline_activities: [],
210
+ explicit_place_id: 0,
211
+ text_format_preset_id: 0,
212
+ logging: {
213
+ composer_session_id: sessionID
214
+ },
215
+ navigation_data: {
216
+ attribution_id_v2: msg.groupID ? "CometGroupDiscussionRoot.react,comet.group,tap_search_bar," + Date.now() + ",909538,2361831622," : "ProfileCometTimelineListViewRoot.react,comet.profile.timeline.list,via_cold_start," + Date.now() + ",796829,190055527696468,"
217
+ },
218
+ is_tracking_encrypted: !!msg.url,
219
+ tracking: [],
220
+ event_share_metadata: {
221
+ surface: "newsfeed"
222
+ },
223
+ actor_id: ctx.globalOptions.pageID || ctx.userID,
224
+ client_mutation_id: Math.round(Math.random() * 19).toString()
225
+ },
226
+ displayCommentsFeedbackContext: null,
227
+ displayCommentsContextEnableComment: null,
228
+ displayCommentsContextIsAdPreview: null,
229
+ displayCommentsContextIsAggregatedShare: null,
230
+ displayCommentsContextIsStorySet: null,
231
+ feedLocation: msg.groupID ? "GROUP" : "TIMELINE",
232
+ feedbackSource: 0,
233
+ focusCommentID: null,
234
+ gridMediaWidth: 230,
235
+ groupID: null,
236
+ scale: 1,
237
+ privacySelectorRenderLocation: "COMET_STREAM",
238
+ renderLocation: msg.groupID ? "group" : "timeline",
239
+ useDefaultActor: false,
240
+ inviteShortLinkKey: null,
241
+ isFeed: false,
242
+ isFundraiser: false,
243
+ isFunFactPost: false,
244
+ isGroup: !!msg.groupID,
245
+ isEvent: false,
246
+ isTimeline: !msg.groupID,
247
+ isSocialLearning: false,
248
+ isPageNewsFeed: !!ctx.globalOptions.pageID,
249
+ isProfileReviews: false,
250
+ isWorkSharedDraft: false,
251
+ UFI2CommentsProvider_commentsKey: msg.groupID ? "CometGroupDiscussionRootSuccessQuery" : "ProfileCometTimelineRoute",
252
+ hashtag: null,
253
+ canUserManageOffers: false,
254
+ __relay_internal__pv__CometUFIIsRTAEnabledrelayprovider: false,
255
+ __relay_internal__pv__IsWorkUserrelayprovider: false,
256
+ __relay_internal__pv__IsMergQAPollsrelayprovider: false,
257
+ __relay_internal__pv__StoriesArmadilloReplyEnabledrelayprovider: false,
258
+ __relay_internal__pv__StoriesRingrelayprovider: false
259
+ }
260
+
261
+ handleUpload(msg, form)
262
+ .then(_ => handleUrl(msg, form))
263
+ .then(_ => handleMention(msg, form))
264
+ .then(_ => createContent(form))
265
+ .then(function (res) {
266
+ if (res.error || res.errors) throw res;
267
+
268
+ return cb(null, (res[0] || res).data.story_create.story.url);
269
+ })
270
+ .catch(function (err) {
271
+ log.error('createPost', err);
272
+ return cb(err);
273
+ });
274
+
275
+ return rt;
276
+ }
277
+ }
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+
3
+ const utils = require('../utils');
4
+ const log = require('npmlog');
5
+
6
+ module.exports = function (http, api, ctx) {
7
+ return async function createPostGroup(text, groupID, callback) {
8
+ try {
9
+ const formData = {
10
+ input: {
11
+ composer_entry_point: "hosted_inline_composer",
12
+ composer_source_surface: "group",
13
+ composer_type: "group",
14
+ logging: {
15
+ composer_session_id: utils.getGUID()
16
+ },
17
+ source: "WWW",
18
+ message: {
19
+ ranges: [],
20
+ text: text
21
+ },
22
+ with_tags_ids: null,
23
+ inline_activities: [],
24
+ explicit_place_id: "0",
25
+ text_format_preset_id: "0",
26
+ navigation_data: {
27
+ attribution_id_v2: `CometGroupDiscussionRoot.react,comet.group,unexpected,${Date.now()},582916,2361831622,,;GroupsCometJoinsRoot.react,comet.groups.joins,unexpected,1723903257951,878430,,,;GroupsCometCrossGroupFeedRoot.react,comet.groups.feed,tap_bookmark,${Date.now()},406054,2361831622,,`
28
+ },
29
+ tracking: [null],
30
+ event_share_metadata: {
31
+ surface: "newsfeed"
32
+ },
33
+ audience: {
34
+ to_id: groupID
35
+ },
36
+ actor_id: ctx.userID,
37
+ client_mutation_id: Math.floor(Math.random() * 17)
38
+ },
39
+ feedLocation: "GROUP",
40
+ feedbackSource: 0,
41
+ focusCommentID: null,
42
+ gridMediaWidth: null,
43
+ groupID: null,
44
+ scale: 1,
45
+ privacySelectorRenderLocation: "COMET_STREAM",
46
+ checkPhotosToReelsUpsellEligibility: false,
47
+ renderLocation: "group",
48
+ useDefaultActor: false,
49
+ inviteShortLinkKey: null,
50
+ isFeed: false,
51
+ isFundraiser: false,
52
+ isFunFactPost: false,
53
+ isGroup: true,
54
+ isEvent: false,
55
+ isTimeline: false,
56
+ isSocialLearning: false,
57
+ isPageNewsFeed: false,
58
+ isProfileReviews: false,
59
+ isWorkSharedDraft: false,
60
+ hashtag: null,
61
+ canUserManageOffers: false
62
+ };
63
+
64
+ const form = {
65
+ av: ctx.userID,
66
+ fb_api_req_friendly_name: "ComposerStoryCreateMutation",
67
+ fb_api_caller_class: "RelayModern",
68
+ doc_id: "7913168052133025",
69
+ variables: JSON.stringify(formData),
70
+ server_timestamps: true
71
+ };
72
+ const res = await http.post('https://www.facebook.com/api/graphql/', ctx.jar, form, null, null).then(utils.parseAndCheckLogin(ctx, http));
73
+ return callback(JSON.stringify(res?.[0]?.data?.story_create, null, 2) || { error: "Không thể tạo bài viết" });
74
+ } catch (err) {
75
+ log.error('createPostGroup', err);
76
+ return callback(err);
77
+ }
78
+ };
79
+ };
@@ -0,0 +1,60 @@
1
+ 'use strict';
2
+
3
+ const { generateOfflineThreadingID, getCurrentTimestamp } = require('../utils');
4
+
5
+ function isCallable(func) {
6
+ try {
7
+ Reflect.apply(func, null, []);
8
+ return true;
9
+ } catch (error) {
10
+ return false;
11
+ }
12
+ }
13
+
14
+ module.exports = function (defaultFuncs, api, ctx) {
15
+ return function forwardMessage(messageID, threadID, callback) {
16
+ if (!ctx.mqttClient) {
17
+ throw new Error('Not connected to MQTT');
18
+ }
19
+
20
+ ctx.wsReqNumber += 1;
21
+ ctx.wsTaskNumber += 1;
22
+
23
+ const taskPayload = {
24
+ thread_id: threadID,
25
+ otid: parseInt(generateOfflineThreadingID()),
26
+ source: 65544,
27
+ send_type: 5,
28
+ sync_group: 1,
29
+ forwarded_msg_id: messageID,
30
+ strip_forwarded_msg_caption: 0,
31
+ initiating_source: 1,
32
+ };
33
+
34
+ const task = {
35
+ failure_count: null,
36
+ label: '46',
37
+ payload: JSON.stringify(taskPayload),
38
+ queue_name: `${threadID}`,
39
+ task_id: ctx.wsTaskNumber,
40
+ };
41
+
42
+ const content = {
43
+ app_id: '2220391788200892',
44
+ payload: JSON.stringify({
45
+ data_trace_id: null,
46
+ epoch_id: parseInt(generateOfflineThreadingID()),
47
+ tasks: [task],
48
+ version_id: '25095469420099952',
49
+ }),
50
+ request_id: ctx.wsReqNumber,
51
+ type: 3,
52
+ };
53
+
54
+ if (isCallable(callback)) {
55
+ // to be implemented
56
+ }
57
+
58
+ ctx.mqttClient.publish('/ls_req', JSON.stringify(content), { qos: 1, retain: false });
59
+ };
60
+ };