@itsliaaa/baileys 0.1.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 (103) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +1078 -0
  3. package/WAProto/index.js +100441 -0
  4. package/engine-requirements.js +10 -0
  5. package/lib/Defaults/index.js +144 -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 +402 -0
  19. package/lib/Signal/lid-mapping.js +270 -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 +1048 -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 +1442 -0
  29. package/lib/Socket/messages-send.js +1153 -0
  30. package/lib/Socket/mex.js +41 -0
  31. package/lib/Socket/newsletter.js +227 -0
  32. package/lib/Socket/socket.js +936 -0
  33. package/lib/Store/index.js +3 -0
  34. package/lib/Store/make-in-memory-store.js +421 -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/Signal.js +1 -0
  50. package/lib/Types/Socket.js +2 -0
  51. package/lib/Types/State.js +12 -0
  52. package/lib/Types/USync.js +1 -0
  53. package/lib/Types/index.js +25 -0
  54. package/lib/Utils/auth-utils.js +289 -0
  55. package/lib/Utils/browser-utils.js +28 -0
  56. package/lib/Utils/business.js +230 -0
  57. package/lib/Utils/chat-utils.js +811 -0
  58. package/lib/Utils/crypto.js +117 -0
  59. package/lib/Utils/decode-wa-message.js +282 -0
  60. package/lib/Utils/event-buffer.js +573 -0
  61. package/lib/Utils/generics.js +385 -0
  62. package/lib/Utils/history.js +130 -0
  63. package/lib/Utils/identity-change-handler.js +48 -0
  64. package/lib/Utils/index.js +19 -0
  65. package/lib/Utils/link-preview.js +84 -0
  66. package/lib/Utils/logger.js +2 -0
  67. package/lib/Utils/lt-hash.js +7 -0
  68. package/lib/Utils/make-mutex.js +32 -0
  69. package/lib/Utils/message-retry-manager.js +224 -0
  70. package/lib/Utils/messages-media.js +789 -0
  71. package/lib/Utils/messages.js +1832 -0
  72. package/lib/Utils/noise-handler.js +200 -0
  73. package/lib/Utils/pre-key-manager.js +105 -0
  74. package/lib/Utils/process-message.js +527 -0
  75. package/lib/Utils/reporting-utils.js +257 -0
  76. package/lib/Utils/signal.js +158 -0
  77. package/lib/Utils/sync-action-utils.js +47 -0
  78. package/lib/Utils/tc-token-utils.js +17 -0
  79. package/lib/Utils/use-multi-file-auth-state.js +120 -0
  80. package/lib/Utils/validate-connection.js +206 -0
  81. package/lib/WABinary/constants.js +1300 -0
  82. package/lib/WABinary/decode.js +261 -0
  83. package/lib/WABinary/encode.js +219 -0
  84. package/lib/WABinary/generic-utils.js +197 -0
  85. package/lib/WABinary/index.js +5 -0
  86. package/lib/WABinary/jid-utils.js +95 -0
  87. package/lib/WABinary/types.js +1 -0
  88. package/lib/WAM/BinaryInfo.js +9 -0
  89. package/lib/WAM/constants.js +22852 -0
  90. package/lib/WAM/encode.js +149 -0
  91. package/lib/WAM/index.js +3 -0
  92. package/lib/WAUSync/Protocols/USyncContactProtocol.js +28 -0
  93. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -0
  94. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +26 -0
  95. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +37 -0
  96. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +50 -0
  97. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +28 -0
  98. package/lib/WAUSync/Protocols/index.js +4 -0
  99. package/lib/WAUSync/USyncQuery.js +93 -0
  100. package/lib/WAUSync/USyncUser.js +22 -0
  101. package/lib/WAUSync/index.js +3 -0
  102. package/lib/index.js +11 -0
  103. package/package.json +72 -0
@@ -0,0 +1,328 @@
1
+ import { proto } from '../../WAProto/index.js';
2
+ import { WAMessageAddressingMode, WAMessageStubType } from '../Types/index.js';
3
+ import { generateMessageIDV2, unixTimestampSeconds } from '../Utils/index.js';
4
+ import { getBinaryNodeChild, getBinaryNodeChildren, getBinaryNodeChildString, isLidUser, isPnUser, jidEncode, jidNormalizedUser } from '../WABinary/index.js';
5
+ import { makeChatsSocket } from './chats.js';
6
+ export const makeGroupsSocket = (config) => {
7
+ const sock = makeChatsSocket(config);
8
+ const { authState, ev, query, upsertMessage } = sock;
9
+ const groupQuery = async (jid, type, content) => query({
10
+ tag: 'iq',
11
+ attrs: {
12
+ type,
13
+ xmlns: 'w:g2',
14
+ to: jid
15
+ },
16
+ content
17
+ });
18
+ const groupMetadata = async (jid) => {
19
+ const result = await groupQuery(jid, 'get', [{ tag: 'query', attrs: { request: 'interactive' } }]);
20
+ return extractGroupMetadata(result);
21
+ };
22
+ const groupFetchAllParticipating = async () => {
23
+ const result = await query({
24
+ tag: 'iq',
25
+ attrs: {
26
+ to: '@g.us',
27
+ xmlns: 'w:g2',
28
+ type: 'get'
29
+ },
30
+ content: [
31
+ {
32
+ tag: 'participating',
33
+ attrs: {},
34
+ content: [
35
+ { tag: 'participants', attrs: {} },
36
+ { tag: 'description', attrs: {} }
37
+ ]
38
+ }
39
+ ]
40
+ });
41
+ const data = Object.create(null);
42
+ const groupsChild = getBinaryNodeChild(result, 'groups');
43
+ if (groupsChild) {
44
+ const groups = getBinaryNodeChildren(groupsChild, 'group');
45
+ for (const groupNode of groups) {
46
+ const meta = extractGroupMetadata({
47
+ tag: 'result',
48
+ attrs: {},
49
+ content: [groupNode]
50
+ });
51
+ data[meta.id] = meta;
52
+ }
53
+ }
54
+ // TODO: properly parse LID / PN DATA
55
+ sock.ev.emit('groups.update', Object.values(data));
56
+ return data;
57
+ };
58
+ sock.ws.on('CB:ib,,dirty', async (node) => {
59
+ const { attrs } = getBinaryNodeChild(node, 'dirty');
60
+ if (attrs.type !== 'groups') {
61
+ return;
62
+ }
63
+ await groupFetchAllParticipating();
64
+ await sock.cleanDirtyBits('groups');
65
+ });
66
+ return {
67
+ ...sock,
68
+ extractGroupMetadata,
69
+ groupMetadata,
70
+ groupCreate: async (subject, participants) => {
71
+ const key = generateMessageIDV2();
72
+ const result = await groupQuery('@g.us', 'set', [
73
+ {
74
+ tag: 'create',
75
+ attrs: {
76
+ subject,
77
+ key
78
+ },
79
+ content: participants.map(jid => ({
80
+ tag: 'participant',
81
+ attrs: { jid }
82
+ }))
83
+ }
84
+ ]);
85
+ return extractGroupMetadata(result);
86
+ },
87
+ groupLeave: async (id) => {
88
+ await groupQuery('@g.us', 'set', [
89
+ {
90
+ tag: 'leave',
91
+ attrs: {},
92
+ content: [{ tag: 'group', attrs: { id } }]
93
+ }
94
+ ]);
95
+ },
96
+ groupUpdateSubject: async (jid, subject) => {
97
+ await groupQuery(jid, 'set', [
98
+ {
99
+ tag: 'subject',
100
+ attrs: {},
101
+ content: Buffer.from(subject, 'utf-8')
102
+ }
103
+ ]);
104
+ },
105
+ groupRequestParticipantsList: async (jid) => {
106
+ const result = await groupQuery(jid, 'get', [
107
+ {
108
+ tag: 'membership_approval_requests',
109
+ attrs: {}
110
+ }
111
+ ]);
112
+ const node = getBinaryNodeChild(result, 'membership_approval_requests');
113
+ const participants = getBinaryNodeChildren(node, 'membership_approval_request');
114
+ return participants.map(v => v.attrs);
115
+ },
116
+ groupRequestParticipantsUpdate: async (jid, participants, action) => {
117
+ const result = await groupQuery(jid, 'set', [
118
+ {
119
+ tag: 'membership_requests_action',
120
+ attrs: {},
121
+ content: [
122
+ {
123
+ tag: action,
124
+ attrs: {},
125
+ content: participants.map(jid => ({
126
+ tag: 'participant',
127
+ attrs: { jid }
128
+ }))
129
+ }
130
+ ]
131
+ }
132
+ ]);
133
+ const node = getBinaryNodeChild(result, 'membership_requests_action');
134
+ const nodeAction = getBinaryNodeChild(node, action);
135
+ const participantsAffected = getBinaryNodeChildren(nodeAction, 'participant');
136
+ return participantsAffected.map(p => {
137
+ return { status: p.attrs.error || '200', jid: p.attrs.jid };
138
+ });
139
+ },
140
+ groupParticipantsUpdate: async (jid, participants, action) => {
141
+ const result = await groupQuery(jid, 'set', [
142
+ {
143
+ tag: action,
144
+ attrs: {},
145
+ content: participants.map(jid => ({
146
+ tag: 'participant',
147
+ attrs: { jid }
148
+ }))
149
+ }
150
+ ]);
151
+ const node = getBinaryNodeChild(result, action);
152
+ const participantsAffected = getBinaryNodeChildren(node, 'participant');
153
+ return participantsAffected.map(p => {
154
+ return { status: p.attrs.error || '200', jid: p.attrs.jid, content: p };
155
+ });
156
+ },
157
+ groupUpdateDescription: async (jid, description) => {
158
+ const metadata = await groupMetadata(jid);
159
+ const prev = metadata.descId ?? null;
160
+ await groupQuery(jid, 'set', [
161
+ {
162
+ tag: 'description',
163
+ attrs: {
164
+ ...(description ? { id: generateMessageIDV2() } : { delete: 'true' }),
165
+ ...(prev ? { prev } : {})
166
+ },
167
+ content: description ? [{ tag: 'body', attrs: {}, content: Buffer.from(description, 'utf-8') }] : undefined
168
+ }
169
+ ]);
170
+ },
171
+ groupInviteCode: async (jid) => {
172
+ const result = await groupQuery(jid, 'get', [{ tag: 'invite', attrs: {} }]);
173
+ const inviteNode = getBinaryNodeChild(result, 'invite');
174
+ return inviteNode?.attrs.code;
175
+ },
176
+ groupRevokeInvite: async (jid) => {
177
+ const result = await groupQuery(jid, 'set', [{ tag: 'invite', attrs: {} }]);
178
+ const inviteNode = getBinaryNodeChild(result, 'invite');
179
+ return inviteNode?.attrs.code;
180
+ },
181
+ groupAcceptInvite: async (code) => {
182
+ const results = await groupQuery('@g.us', 'set', [{ tag: 'invite', attrs: { code } }]);
183
+ const result = getBinaryNodeChild(results, 'group');
184
+ return result?.attrs.jid;
185
+ },
186
+ /**
187
+ * revoke a v4 invite for someone
188
+ * @param groupJid group jid
189
+ * @param invitedJid jid of person you invited
190
+ * @returns true if successful
191
+ */
192
+ groupRevokeInviteV4: async (groupJid, invitedJid) => {
193
+ const result = await groupQuery(groupJid, 'set', [
194
+ { tag: 'revoke', attrs: {}, content: [{ tag: 'participant', attrs: { jid: invitedJid } }] }
195
+ ]);
196
+ return !!result;
197
+ },
198
+ /**
199
+ * accept a GroupInviteMessage
200
+ * @param key the key of the invite message, or optionally only provide the jid of the person who sent the invite
201
+ * @param inviteMessage the message to accept
202
+ */
203
+ groupAcceptInviteV4: ev.createBufferedFunction(async (key, inviteMessage) => {
204
+ key = typeof key === 'string' ? { remoteJid: key } : key;
205
+ const results = await groupQuery(inviteMessage.groupJid, 'set', [
206
+ {
207
+ tag: 'accept',
208
+ attrs: {
209
+ code: inviteMessage.inviteCode,
210
+ expiration: inviteMessage.inviteExpiration.toString(),
211
+ admin: key.remoteJid
212
+ }
213
+ }
214
+ ]);
215
+ // if we have the full message key
216
+ // update the invite message to be expired
217
+ if (key.id) {
218
+ // create new invite message that is expired
219
+ inviteMessage = proto.Message.GroupInviteMessage.fromObject(inviteMessage);
220
+ inviteMessage.inviteExpiration = 0;
221
+ inviteMessage.inviteCode = '';
222
+ ev.emit('messages.update', [
223
+ {
224
+ key,
225
+ update: {
226
+ message: {
227
+ groupInviteMessage: inviteMessage
228
+ }
229
+ }
230
+ }
231
+ ]);
232
+ }
233
+ // generate the group add message
234
+ await upsertMessage({
235
+ key: {
236
+ remoteJid: inviteMessage.groupJid,
237
+ id: generateMessageIDV2(sock.user?.id),
238
+ fromMe: false,
239
+ participant: key.remoteJid
240
+ },
241
+ messageStubType: WAMessageStubType.GROUP_PARTICIPANT_ADD,
242
+ messageStubParameters: [JSON.stringify(authState.creds.me)],
243
+ participant: key.remoteJid,
244
+ messageTimestamp: unixTimestampSeconds()
245
+ }, 'notify');
246
+ return results.attrs.from;
247
+ }),
248
+ groupGetInviteInfo: async (code) => {
249
+ const results = await groupQuery('@g.us', 'get', [{ tag: 'invite', attrs: { code } }]);
250
+ return extractGroupMetadata(results);
251
+ },
252
+ groupToggleEphemeral: async (jid, ephemeralExpiration) => {
253
+ const content = ephemeralExpiration
254
+ ? { tag: 'ephemeral', attrs: { expiration: ephemeralExpiration.toString() } }
255
+ : { tag: 'not_ephemeral', attrs: {} };
256
+ await groupQuery(jid, 'set', [content]);
257
+ },
258
+ groupSettingUpdate: async (jid, setting) => {
259
+ await groupQuery(jid, 'set', [{ tag: setting, attrs: {} }]);
260
+ },
261
+ groupMemberAddMode: async (jid, mode) => {
262
+ await groupQuery(jid, 'set', [{ tag: 'member_add_mode', attrs: {}, content: mode }]);
263
+ },
264
+ groupJoinApprovalMode: async (jid, mode) => {
265
+ await groupQuery(jid, 'set', [
266
+ { tag: 'membership_approval_mode', attrs: {}, content: [{ tag: 'group_join', attrs: { state: mode } }] }
267
+ ]);
268
+ },
269
+ groupFetchAllParticipating,
270
+ groupQuery
271
+ };
272
+ };
273
+ export const extractGroupMetadata = (result) => {
274
+ const group = getBinaryNodeChild(result, 'group');
275
+ const descChild = getBinaryNodeChild(group, 'description');
276
+ let desc;
277
+ let descId;
278
+ let descOwner;
279
+ let descOwnerPn;
280
+ let descTime;
281
+ if (descChild) {
282
+ desc = getBinaryNodeChildString(descChild, 'body');
283
+ descOwner = descChild.attrs.participant ? jidNormalizedUser(descChild.attrs.participant) : undefined;
284
+ descOwnerPn = descChild.attrs.participant_pn ? jidNormalizedUser(descChild.attrs.participant_pn) : undefined;
285
+ descTime = +descChild.attrs.t;
286
+ descId = descChild.attrs.id;
287
+ }
288
+ const groupId = group.attrs.id.includes('@') ? group.attrs.id : jidEncode(group.attrs.id, 'g.us');
289
+ const eph = getBinaryNodeChild(group, 'ephemeral')?.attrs.expiration;
290
+ const memberAddMode = getBinaryNodeChildString(group, 'member_add_mode') === 'all_member_add';
291
+ const metadata = {
292
+ id: groupId,
293
+ notify: group.attrs.notify,
294
+ addressingMode: group.attrs.addressing_mode === 'lid' ? WAMessageAddressingMode.LID : WAMessageAddressingMode.PN,
295
+ subject: group.attrs.subject,
296
+ subjectOwner: group.attrs.s_o,
297
+ subjectOwnerPn: group.attrs.s_o_pn,
298
+ subjectTime: +group.attrs.s_t,
299
+ size: group.attrs.size ? +group.attrs.size : getBinaryNodeChildren(group, 'participant').length,
300
+ creation: +group.attrs.creation,
301
+ owner: group.attrs.creator ? jidNormalizedUser(group.attrs.creator) : undefined,
302
+ ownerPn: group.attrs.creator_pn ? jidNormalizedUser(group.attrs.creator_pn) : undefined,
303
+ owner_country_code: group.attrs.creator_country_code,
304
+ desc,
305
+ descId,
306
+ descOwner,
307
+ descOwnerPn,
308
+ descTime,
309
+ linkedParent: getBinaryNodeChild(group, 'linked_parent')?.attrs.jid || undefined,
310
+ restrict: !!getBinaryNodeChild(group, 'locked'),
311
+ announce: !!getBinaryNodeChild(group, 'announcement'),
312
+ isCommunity: !!getBinaryNodeChild(group, 'parent'),
313
+ isCommunityAnnounce: !!getBinaryNodeChild(group, 'default_sub_group'),
314
+ joinApprovalMode: !!getBinaryNodeChild(group, 'membership_approval_mode'),
315
+ memberAddMode,
316
+ participants: getBinaryNodeChildren(group, 'participant').map(({ attrs }) => {
317
+ // TODO: Store LID MAPPINGS
318
+ return {
319
+ id: attrs.jid,
320
+ phoneNumber: isLidUser(attrs.jid) && isPnUser(attrs.phone_number) ? attrs.phone_number : undefined,
321
+ lid: isPnUser(attrs.jid) && isLidUser(attrs.lid) ? attrs.lid : undefined,
322
+ admin: (attrs.type || null)
323
+ };
324
+ }),
325
+ ephemeralDuration: eph ? +eph : undefined
326
+ };
327
+ return metadata;
328
+ };
@@ -0,0 +1,11 @@
1
+ import { DEFAULT_CONNECTION_CONFIG } from '../Defaults/index.js';
2
+ import { makeCommunitiesSocket } from './communities.js';
3
+ // export the last socket layer
4
+ const makeWASocket = (config) => {
5
+ const newConfig = {
6
+ ...DEFAULT_CONNECTION_CONFIG,
7
+ ...config
8
+ };
9
+ return makeCommunitiesSocket(newConfig);
10
+ };
11
+ export default makeWASocket;