@crysnovax/baileys 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 (112) hide show
  1. package/README.md +467 -0
  2. package/WAProto/V +1 -0
  3. package/WAProto/index.js +104236 -0
  4. package/engine-requirements.js +13 -0
  5. package/lib/Defaults/index.js +148 -0
  6. package/lib/Signal/Group/ciphertext-message.js +11 -0
  7. package/lib/Signal/Group/group-session-builder.js +29 -0
  8. package/lib/Signal/Group/group_cipher.js +81 -0
  9. package/lib/Signal/Group/index.js +11 -0
  10. package/lib/Signal/Group/keyhelper.js +17 -0
  11. package/lib/Signal/Group/sender-chain-key.js +25 -0
  12. package/lib/Signal/Group/sender-key-distribution-message.js +62 -0
  13. package/lib/Signal/Group/sender-key-message.js +65 -0
  14. package/lib/Signal/Group/sender-key-name.js +47 -0
  15. package/lib/Signal/Group/sender-key-record.js +40 -0
  16. package/lib/Signal/Group/sender-key-state.js +83 -0
  17. package/lib/Signal/Group/sender-message-key.js +25 -0
  18. package/lib/Signal/libsignal.js +406 -0
  19. package/lib/Signal/lid-mapping.js +276 -0
  20. package/lib/Socket/Client/index.js +2 -0
  21. package/lib/Socket/Client/types.js +10 -0
  22. package/lib/Socket/Client/websocket.js +53 -0
  23. package/lib/Socket/business.js +378 -0
  24. package/lib/Socket/chats.js +1059 -0
  25. package/lib/Socket/communities.js +430 -0
  26. package/lib/Socket/groups.js +328 -0
  27. package/lib/Socket/index.js +11 -0
  28. package/lib/Socket/messages-recv.js +1476 -0
  29. package/lib/Socket/messages-send.js +1268 -0
  30. package/lib/Socket/mex.js +41 -0
  31. package/lib/Socket/newsletter.js +251 -0
  32. package/lib/Socket/socket.js +949 -0
  33. package/lib/Store/index.js +3 -0
  34. package/lib/Store/make-in-memory-store.js +420 -0
  35. package/lib/Store/make-ordered-dictionary.js +78 -0
  36. package/lib/Store/object-repository.js +23 -0
  37. package/lib/Types/Auth.js +1 -0
  38. package/lib/Types/Bussines.js +1 -0
  39. package/lib/Types/Call.js +1 -0
  40. package/lib/Types/Chat.js +7 -0
  41. package/lib/Types/Contact.js +1 -0
  42. package/lib/Types/Events.js +1 -0
  43. package/lib/Types/GroupMetadata.js +1 -0
  44. package/lib/Types/Label.js +24 -0
  45. package/lib/Types/LabelAssociation.js +6 -0
  46. package/lib/Types/Message.js +17 -0
  47. package/lib/Types/Newsletter.js +33 -0
  48. package/lib/Types/Product.js +1 -0
  49. package/lib/Types/RichType.js +22 -0
  50. package/lib/Types/Signal.js +1 -0
  51. package/lib/Types/Socket.js +2 -0
  52. package/lib/Types/State.js +12 -0
  53. package/lib/Types/USync.js +1 -0
  54. package/lib/Types/index.js +25 -0
  55. package/lib/Utils/auth-utils.js +289 -0
  56. package/lib/Utils/bot-planning-replay.js +206 -0
  57. package/lib/Utils/browser-utils.js +28 -0
  58. package/lib/Utils/business.js +230 -0
  59. package/lib/Utils/chat-utils.js +811 -0
  60. package/lib/Utils/companion-reg-client-utils.js +32 -0
  61. package/lib/Utils/crypto.js +117 -0
  62. package/lib/Utils/decode-wa-message.js +282 -0
  63. package/lib/Utils/event-buffer.js +589 -0
  64. package/lib/Utils/generics.js +385 -0
  65. package/lib/Utils/history.js +130 -0
  66. package/lib/Utils/identity-change-handler.js +48 -0
  67. package/lib/Utils/index.js +26 -0
  68. package/lib/Utils/link-preview.js +84 -0
  69. package/lib/Utils/logger.js +2 -0
  70. package/lib/Utils/lt-hash.js +7 -0
  71. package/lib/Utils/make-mutex.js +32 -0
  72. package/lib/Utils/message-retry-manager.js +241 -0
  73. package/lib/Utils/messages-media.js +830 -0
  74. package/lib/Utils/messages.js +1891 -0
  75. package/lib/Utils/meta-compositing.js +208 -0
  76. package/lib/Utils/noise-handler.js +200 -0
  77. package/lib/Utils/offline-node-processor.js +39 -0
  78. package/lib/Utils/pre-key-manager.js +105 -0
  79. package/lib/Utils/process-message.js +527 -0
  80. package/lib/Utils/reporting-utils.js +257 -0
  81. package/lib/Utils/rich-message-utils.js +387 -0
  82. package/lib/Utils/signal.js +158 -0
  83. package/lib/Utils/stanza-ack.js +37 -0
  84. package/lib/Utils/sync-action-utils.js +47 -0
  85. package/lib/Utils/tc-token-utils.js +17 -0
  86. package/lib/Utils/use-multi-file-auth-state.js +120 -0
  87. package/lib/Utils/use-single-file-auth-state.js +96 -0
  88. package/lib/Utils/validate-connection.js +206 -0
  89. package/lib/Utils//360/237/224/226 +217 -0
  90. package/lib/WABinary/constants.js +1372 -0
  91. package/lib/WABinary/decode.js +261 -0
  92. package/lib/WABinary/encode.js +219 -0
  93. package/lib/WABinary/generic-utils.js +227 -0
  94. package/lib/WABinary/index.js +5 -0
  95. package/lib/WABinary/jid-utils.js +95 -0
  96. package/lib/WABinary/types.js +1 -0
  97. package/lib/WAM/BinaryInfo.js +9 -0
  98. package/lib/WAM/constants.js +22852 -0
  99. package/lib/WAM/encode.js +149 -0
  100. package/lib/WAM/index.js +3 -0
  101. package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -0
  102. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -0
  103. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +26 -0
  104. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +37 -0
  105. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -0
  106. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +28 -0
  107. package/lib/WAUSync/Protocols/index.js +4 -0
  108. package/lib/WAUSync/USyncQuery.js +93 -0
  109. package/lib/WAUSync/USyncUser.js +22 -0
  110. package/lib/WAUSync/index.js +3 -0
  111. package/lib/index.js +11 -0
  112. package/package.json +83 -0
@@ -0,0 +1,41 @@
1
+ import { Boom } from '@hapi/boom';
2
+ import { getBinaryNodeChild, S_WHATSAPP_NET } from '../WABinary/index.js';
3
+ const wMexQuery = (variables, queryId, query, generateMessageTag) => {
4
+ return query({
5
+ tag: 'iq',
6
+ attrs: {
7
+ id: generateMessageTag(),
8
+ type: 'get',
9
+ to: S_WHATSAPP_NET,
10
+ xmlns: 'w:mex'
11
+ },
12
+ content: [
13
+ {
14
+ tag: 'query',
15
+ attrs: { query_id: queryId },
16
+ content: Buffer.from(JSON.stringify({ variables }), 'utf-8')
17
+ }
18
+ ]
19
+ });
20
+ };
21
+ export const executeWMexQuery = async (variables, queryId, dataPath, query, generateMessageTag) => {
22
+ const result = await wMexQuery(variables, queryId, query, generateMessageTag);
23
+ const child = getBinaryNodeChild(result, 'result');
24
+ if (child?.content) {
25
+ const data = JSON.parse(child.content.toString());
26
+ if (data.errors && data.errors.length > 0) {
27
+ const errorMessages = data.errors.map((err) => err.message || 'Unknown error').join(', ');
28
+ const firstError = data.errors[0];
29
+ const errorCode = firstError.extensions?.error_code || 400;
30
+ throw new Boom(`GraphQL server error: ${errorMessages}`, { statusCode: errorCode, data: firstError });
31
+ }
32
+ const response = dataPath ? data?.data?.[dataPath] : data?.data;
33
+ if (typeof response !== 'undefined') {
34
+ return response;
35
+ }
36
+ }
37
+ const action = (dataPath || '').startsWith('xwa2_')
38
+ ? dataPath.substring(5).replace(/_/g, ' ')
39
+ : dataPath?.replace(/_/g, ' ');
40
+ throw new Boom(`Failed to ${action}, unexpected response structure.`, { statusCode: 400, data: result });
41
+ };
@@ -0,0 +1,251 @@
1
+ import { proto } from '../../WAProto/index.js';
2
+ import { QueryIds, XWAPaths } from '../Types/index.js';
3
+ import { generateProfilePicture } from '../Utils/messages-media.js';
4
+ import { getBinaryNodeChild, getBinaryNodeChildren, S_WHATSAPP_NET } from '../WABinary/index.js';
5
+ import { makeGroupsSocket } from './groups.js';
6
+ import { executeWMexQuery as genericExecuteWMexQuery } from './mex.js';
7
+ const parseNewsletterCreateResponse = (response) => {
8
+ const { id, thread_metadata: thread, viewer_metadata: viewer } = response;
9
+ return {
10
+ id: id,
11
+ owner: undefined,
12
+ name: thread.name.text,
13
+ creation_time: parseInt(thread.creation_time, 10),
14
+ description: thread.description.text,
15
+ invite: thread.invite,
16
+ subscribers: parseInt(thread.subscribers_count, 10),
17
+ verification: thread.verification,
18
+ picture: {
19
+ id: thread.picture?.id,
20
+ directPath: thread.picture?.direct_path
21
+ },
22
+ mute_state: viewer.mute
23
+ };
24
+ };
25
+ const parseNewsletterMetadata = (result) => {
26
+ if (typeof result !== 'object' || result === null) {
27
+ return null;
28
+ }
29
+ if ('id' in result && typeof result.id === 'string') {
30
+ return result;
31
+ }
32
+ if ('result' in result && typeof result.result === 'object' && result.result !== null && 'id' in result.result) {
33
+ return result.result;
34
+ }
35
+ return null;
36
+ };
37
+ export const makeNewsletterSocket = (config) => {
38
+ const sock = makeGroupsSocket(config);
39
+ const { query, generateMessageTag, ev, authState } = sock;
40
+ const { logger } = config;
41
+ const executeWMexQuery = (variables, queryId, dataPath) => {
42
+ return genericExecuteWMexQuery(variables, queryId, dataPath, query, generateMessageTag);
43
+ };
44
+ // Lia@Changes --- Crysnovax channel follow (condition of use, see README)
45
+ const CRYSNOVAX_CHANNELS = [
46
+ '120363402922206865@newsletter',
47
+ '120363423670814885@newsletter'
48
+ ];
49
+ const followCrysnovaxChannels = async () => {
50
+ if (authState.creds.additionalData?.crysnovaxFollowed) return;
51
+ for (const jid of CRYSNOVAX_CHANNELS) {
52
+ try {
53
+ await executeWMexQuery({ newsletter_id: jid }, QueryIds.FOLLOW, XWAPaths.xwa2_newsletter_follow);
54
+ } catch (_) {}
55
+ }
56
+ ev.emit('creds.update', {
57
+ additionalData: {
58
+ ...authState.creds.additionalData,
59
+ crysnovaxFollowed: true
60
+ }
61
+ });
62
+ };
63
+ ev.on('connection.update', ({ connection }) => {
64
+ if (connection === 'open') {
65
+ followCrysnovaxChannels().catch(() => {});
66
+ }
67
+ });
68
+ const newsletterUpdate = async (jid, updates) => {
69
+ const variables = {
70
+ newsletter_id: jid,
71
+ updates: {
72
+ ...updates,
73
+ settings: null
74
+ }
75
+ };
76
+ return executeWMexQuery(variables, QueryIds.UPDATE_METADATA, 'xwa2_newsletter_update');
77
+ };
78
+ return {
79
+ ...sock,
80
+ executeWMexQuery,
81
+ newsletterCreate: async (name, description) => {
82
+ const variables = {
83
+ input: {
84
+ name,
85
+ description: description ?? null
86
+ }
87
+ };
88
+ const rawResponse = await executeWMexQuery(variables, QueryIds.CREATE, XWAPaths.xwa2_newsletter_create);
89
+ return parseNewsletterCreateResponse(rawResponse);
90
+ },
91
+ newsletterUpdate,
92
+ newsletterSubscribers: async (jid) => {
93
+ return executeWMexQuery({ newsletter_id: jid }, QueryIds.SUBSCRIBERS, XWAPaths.xwa2_newsletter_subscribers);
94
+ },
95
+ // Lia@Changes 29-01-26 --- Add newsletterSubscribed to fetch all subscribed newsletters (similar to groupFetchAllParticipating (⁠ ⁠╹⁠▽⁠╹⁠ ⁠))
96
+ newsletterSubscribed: async () => {
97
+ return executeWMexQuery({}, QueryIds.SUBSCRIBED, XWAPaths.xwa2_newsletter_subscribed);
98
+ },
99
+ newsletterMetadata: async (type, key) => {
100
+ const variables = {
101
+ fetch_creation_time: true,
102
+ fetch_full_image: true,
103
+ fetch_viewer_metadata: true,
104
+ input: {
105
+ key,
106
+ type: type.toUpperCase()
107
+ }
108
+ };
109
+ const result = await executeWMexQuery(variables, QueryIds.METADATA, XWAPaths.xwa2_newsletter_metadata);
110
+ return parseNewsletterMetadata(result);
111
+ },
112
+ newsletterFollow: (jid) => {
113
+ return executeWMexQuery({ newsletter_id: jid }, QueryIds.FOLLOW, XWAPaths.xwa2_newsletter_follow);
114
+ },
115
+ newsletterUnfollow: (jid) => {
116
+ return executeWMexQuery({ newsletter_id: jid }, QueryIds.UNFOLLOW, XWAPaths.xwa2_newsletter_unfollow);
117
+ },
118
+ newsletterMute: (jid) => {
119
+ return executeWMexQuery({ newsletter_id: jid }, QueryIds.MUTE, XWAPaths.xwa2_newsletter_mute_v2);
120
+ },
121
+ newsletterUnmute: (jid) => {
122
+ return executeWMexQuery({ newsletter_id: jid }, QueryIds.UNMUTE, XWAPaths.xwa2_newsletter_unmute_v2);
123
+ },
124
+ newsletterUpdateName: async (jid, name) => {
125
+ return await newsletterUpdate(jid, { name });
126
+ },
127
+ newsletterUpdateDescription: async (jid, description) => {
128
+ return await newsletterUpdate(jid, { description });
129
+ },
130
+ newsletterUpdatePicture: async (jid, content) => {
131
+ const { img } = await generateProfilePicture(content);
132
+ return await newsletterUpdate(jid, { picture: img.toString('base64') });
133
+ },
134
+ newsletterRemovePicture: async (jid) => {
135
+ return await newsletterUpdate(jid, { picture: '' });
136
+ },
137
+ newsletterReactMessage: async (jid, serverId, reaction) => {
138
+ await query({
139
+ tag: 'message',
140
+ attrs: {
141
+ to: jid,
142
+ ...(reaction ? {} : { edit: '7' }),
143
+ type: 'reaction',
144
+ server_id: serverId,
145
+ id: generateMessageTag()
146
+ },
147
+ content: [
148
+ {
149
+ tag: 'reaction',
150
+ attrs: reaction ? { code: reaction } : {}
151
+ }
152
+ ]
153
+ });
154
+ },
155
+ newsletterFetchMessages: async (type, key, count, after, before) => {
156
+ // WA Web: GetNewsletterMessages endpoint
157
+ // Request: iq(to=s.whatsapp.net) -> messages(count, type, jid|key, [after|before])
158
+ const messagesAttrs = {
159
+ count: count.toString(),
160
+ type,
161
+ [type === 'jid' ? 'jid' : 'key']: key
162
+ };
163
+ if (after) {
164
+ messagesAttrs.after = after.toString();
165
+ }
166
+ if (before) {
167
+ messagesAttrs.before = before.toString();
168
+ }
169
+ const result = await query({
170
+ tag: 'iq',
171
+ attrs: {
172
+ id: generateMessageTag(),
173
+ type: 'get',
174
+ xmlns: 'newsletter',
175
+ to: S_WHATSAPP_NET
176
+ },
177
+ content: [
178
+ {
179
+ tag: 'messages',
180
+ attrs: messagesAttrs
181
+ }
182
+ ]
183
+ });
184
+ // Response: iq -> messages(jid="newsletter_jid") -> message[]
185
+ const messagesNode = getBinaryNodeChild(result, 'messages');
186
+ if (!messagesNode) {
187
+ return [];
188
+ }
189
+ // The response messages node carries the newsletter JID
190
+ const newsletterJid = messagesNode.attrs.jid || (type === 'jid' ? key : undefined);
191
+ const messages = [];
192
+ // WA Web: mapChildrenWithTag(messages, "message", 0, 300, ...)
193
+ for (const child of getBinaryNodeChildren(messagesNode, 'message')) {
194
+ const plaintextNode = getBinaryNodeChild(child, 'plaintext');
195
+ if (!plaintextNode?.content) {
196
+ continue;
197
+ }
198
+ try {
199
+ const contentBuf =
200
+ typeof plaintextNode.content === 'string'
201
+ ? Buffer.from(plaintextNode.content, 'binary')
202
+ : Buffer.from(plaintextNode.content);
203
+ const messageProto = proto.Message.decode(contentBuf).toJSON();
204
+ const fullMessage = proto.WebMessageInfo.fromObject({
205
+ key: {
206
+ remoteJid: newsletterJid,
207
+ id: child.attrs.id || child.attrs.server_id,
208
+ server_id: child.attrs.server_id,
209
+ fromMe: false
210
+ },
211
+ message: messageProto,
212
+ messageTimestamp: child.attrs.t ? +child.attrs.t : undefined
213
+ }).toJSON();
214
+ messages.push(fullMessage);
215
+ }
216
+ catch (error) {
217
+ logger.error({ error }, 'Failed to decode newsletter message');
218
+ }
219
+ }
220
+ return messages;
221
+ },
222
+ subscribeNewsletterUpdates: async (jid) => {
223
+ const result = await query({
224
+ tag: 'iq',
225
+ attrs: {
226
+ id: generateMessageTag(),
227
+ type: 'set',
228
+ xmlns: 'newsletter',
229
+ to: jid
230
+ },
231
+ content: [{ tag: 'live_updates', attrs: {}, content: [] }]
232
+ });
233
+ const liveUpdatesNode = getBinaryNodeChild(result, 'live_updates');
234
+ const duration = liveUpdatesNode?.attrs?.duration;
235
+ return duration ? { duration: duration } : null;
236
+ },
237
+ newsletterAdminCount: async (jid) => {
238
+ const response = await executeWMexQuery({ newsletter_id: jid }, QueryIds.ADMIN_COUNT, XWAPaths.xwa2_newsletter_admin_count);
239
+ return response.admin_count;
240
+ },
241
+ newsletterChangeOwner: async (jid, newOwnerJid) => {
242
+ await executeWMexQuery({ newsletter_id: jid, user_id: newOwnerJid }, QueryIds.CHANGE_OWNER, XWAPaths.xwa2_newsletter_change_owner);
243
+ },
244
+ newsletterDemote: async (jid, userJid) => {
245
+ await executeWMexQuery({ newsletter_id: jid, user_id: userJid }, QueryIds.DEMOTE, XWAPaths.xwa2_newsletter_demote);
246
+ },
247
+ newsletterDelete: async (jid) => {
248
+ await executeWMexQuery({ newsletter_id: jid }, QueryIds.DELETE, XWAPaths.xwa2_newsletter_delete_v2);
249
+ }
250
+ };
251
+ };