@skyzopedia/baileys-mod 3.0.2
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.
- package/LICENSE +21 -0
- package/WAProto/WAProto.proto +5311 -0
- package/WAProto/index.js +94091 -0
- package/lib/Defaults/index.js +123 -0
- package/lib/KeyDB/BinarySearch.js +20 -0
- package/lib/KeyDB/KeyedDB.js +167 -0
- package/lib/KeyDB/index.js +4 -0
- package/lib/Signal/Group/ciphertext-message.js +13 -0
- package/lib/Signal/Group/group-session-builder.js +32 -0
- package/lib/Signal/Group/group_cipher.js +84 -0
- package/lib/Signal/Group/index.js +13 -0
- package/lib/Signal/Group/keyhelper.js +20 -0
- package/lib/Signal/Group/sender-chain-key.js +28 -0
- package/lib/Signal/Group/sender-key-distribution-message.js +65 -0
- package/lib/Signal/Group/sender-key-message.js +68 -0
- package/lib/Signal/Group/sender-key-name.js +52 -0
- package/lib/Signal/Group/sender-key-record.js +43 -0
- package/lib/Signal/Group/sender-key-state.js +86 -0
- package/lib/Signal/Group/sender-message-key.js +28 -0
- package/lib/Signal/libsignal.js +324 -0
- package/lib/Signal/lid-mapping.js +155 -0
- package/lib/Socket/Client/index.js +4 -0
- package/lib/Socket/Client/types.js +13 -0
- package/lib/Socket/Client/websocket.js +52 -0
- package/lib/Socket/business.js +377 -0
- package/lib/Socket/chats.js +881 -0
- package/lib/Socket/communities.js +413 -0
- package/lib/Socket/groups.js +312 -0
- package/lib/Socket/index.js +16 -0
- package/lib/Socket/messages-recv.js +1163 -0
- package/lib/Socket/messages-send.js +1082 -0
- package/lib/Socket/mex.js +45 -0
- package/lib/Socket/newsletter.js +259 -0
- package/lib/Socket/socket.js +781 -0
- package/lib/Store/index.js +6 -0
- package/lib/Store/make-cache-manager-store.js +75 -0
- package/lib/Store/make-in-memory-store.js +290 -0
- package/lib/Store/make-ordered-dictionary.js +79 -0
- package/lib/Store/object-repository.js +25 -0
- package/lib/Types/Auth.js +3 -0
- package/lib/Types/Bussines.js +3 -0
- package/lib/Types/Call.js +3 -0
- package/lib/Types/Chat.js +9 -0
- package/lib/Types/Contact.js +3 -0
- package/lib/Types/Events.js +3 -0
- package/lib/Types/GroupMetadata.js +3 -0
- package/lib/Types/Label.js +25 -0
- package/lib/Types/LabelAssociation.js +7 -0
- package/lib/Types/Message.js +12 -0
- package/lib/Types/Newsletter.js +33 -0
- package/lib/Types/Newsletter.js.bak +33 -0
- package/lib/Types/Product.js +3 -0
- package/lib/Types/Signal.js +3 -0
- package/lib/Types/Socket.js +4 -0
- package/lib/Types/State.js +11 -0
- package/lib/Types/USync.js +3 -0
- package/lib/Types/index.js +28 -0
- package/lib/Utils/auth-utils.js +219 -0
- package/lib/Utils/baileys-event-stream.js +44 -0
- package/lib/Utils/browser-utils.js +17 -0
- package/lib/Utils/business.js +233 -0
- package/lib/Utils/chat-utils.js +752 -0
- package/lib/Utils/crypto.js +130 -0
- package/lib/Utils/decode-wa-message.js +267 -0
- package/lib/Utils/event-buffer.js +528 -0
- package/lib/Utils/generics.js +355 -0
- package/lib/Utils/history.js +87 -0
- package/lib/Utils/index.js +21 -0
- package/lib/Utils/link-preview.js +81 -0
- package/lib/Utils/logger.js +5 -0
- package/lib/Utils/lt-hash.js +45 -0
- package/lib/Utils/make-mutex.js +36 -0
- package/lib/Utils/message-retry-manager.js +113 -0
- package/lib/Utils/messages-media.js +601 -0
- package/lib/Utils/messages.js +776 -0
- package/lib/Utils/noise-handler.js +144 -0
- package/lib/Utils/pre-key-manager.js +85 -0
- package/lib/Utils/process-message.js +341 -0
- package/lib/Utils/signal.js +161 -0
- package/lib/Utils/use-multi-file-auth-state.js +111 -0
- package/lib/Utils/validate-connection.js +200 -0
- package/lib/WABinary/constants.js +1303 -0
- package/lib/WABinary/decode.js +240 -0
- package/lib/WABinary/encode.js +218 -0
- package/lib/WABinary/generic-utils.js +113 -0
- package/lib/WABinary/index.js +7 -0
- package/lib/WABinary/jid-utils.js +93 -0
- package/lib/WABinary/types.js +3 -0
- package/lib/WAM/BinaryInfo.js +11 -0
- package/lib/WAM/constants.js +22853 -0
- package/lib/WAM/encode.js +154 -0
- package/lib/WAM/index.js +5 -0
- package/lib/WAUSync/Protocols/USyncContactProtocol.js +30 -0
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +53 -0
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +29 -0
- package/lib/WAUSync/Protocols/USyncStatusProtocol.js +39 -0
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +53 -0
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +30 -0
- package/lib/WAUSync/Protocols/index.js +6 -0
- package/lib/WAUSync/USyncQuery.js +90 -0
- package/lib/WAUSync/USyncUser.js +24 -0
- package/lib/WAUSync/index.js +5 -0
- package/lib/index.js +15 -0
- package/package.json +102 -0
|
@@ -0,0 +1,355 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
import { getAllBinaryNodeChildren, jidDecode } from "../WABinary/index.js";
|
|
3
|
+
import { DisconnectReason } from "../Types/index.js";
|
|
4
|
+
import { createHash, randomBytes } from "crypto";
|
|
5
|
+
import { proto } from "../../WAProto/index.js";
|
|
6
|
+
import { version } from "../Defaults/index.js"
|
|
7
|
+
import { sha256 } from "./crypto.js";
|
|
8
|
+
import { Boom } from "@hapi/boom";
|
|
9
|
+
//=======================================================//
|
|
10
|
+
export const BufferJSON = {
|
|
11
|
+
replacer: (k, value) => {
|
|
12
|
+
if (Buffer.isBuffer(value) || value instanceof Uint8Array || value?.type === "Buffer") {
|
|
13
|
+
return { type: "Buffer", data: Buffer.from(value?.data || value).toString("base64") };
|
|
14
|
+
}
|
|
15
|
+
return value;
|
|
16
|
+
},
|
|
17
|
+
reviver: (_, value) => {
|
|
18
|
+
if (typeof value === "object" && value !== null && value.type === "Buffer" && typeof value.data === "string") {
|
|
19
|
+
return Buffer.from(value.data, "base64");
|
|
20
|
+
}
|
|
21
|
+
if (typeof value === "object" && value !== null && !Array.isArray(value)) {
|
|
22
|
+
const keys = Object.keys(value);
|
|
23
|
+
if (keys.length > 0 && keys.every(k => !isNaN(parseInt(k, 10)))) {
|
|
24
|
+
const values = Object.values(value);
|
|
25
|
+
if (values.every(v => typeof v === "number")) {
|
|
26
|
+
return Buffer.from(values);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
return value;
|
|
31
|
+
}
|
|
32
|
+
};
|
|
33
|
+
//=======================================================//
|
|
34
|
+
export const getKeyAuthor = (key, meId = "me") => (key?.fromMe ? meId : key?.participant || key?.remoteJid) || "";
|
|
35
|
+
export const writeRandomPadMax16 = (msg) => {
|
|
36
|
+
const pad = randomBytes(1);
|
|
37
|
+
const padLength = (pad[0] & 0x0f) + 1;
|
|
38
|
+
return Buffer.concat([msg, Buffer.alloc(padLength, padLength)]);
|
|
39
|
+
};
|
|
40
|
+
//=======================================================//
|
|
41
|
+
export const unpadRandomMax16 = (e) => {
|
|
42
|
+
const t = new Uint8Array(e);
|
|
43
|
+
if (0 === t.length) {
|
|
44
|
+
throw new Error("unpadPkcs7 given empty bytes");
|
|
45
|
+
}
|
|
46
|
+
var r = t[t.length - 1];
|
|
47
|
+
if (r > t.length) {
|
|
48
|
+
throw new Error(`unpad given ${t.length} bytes, but pad is ${r}`);
|
|
49
|
+
}
|
|
50
|
+
return new Uint8Array(t.buffer, t.byteOffset, t.length - r);
|
|
51
|
+
};
|
|
52
|
+
//=======================================================//
|
|
53
|
+
export const generateParticipantHashV2 = (participants) => {
|
|
54
|
+
participants.sort();
|
|
55
|
+
const sha256Hash = sha256(Buffer.from(participants.join(""))).toString("base64");
|
|
56
|
+
return "2:" + sha256Hash.slice(0, 6);
|
|
57
|
+
};
|
|
58
|
+
//=======================================================//
|
|
59
|
+
export const encodeWAMessage = (message) => writeRandomPadMax16(proto.Message.encode(message).finish());
|
|
60
|
+
export const generateRegistrationId = () => {
|
|
61
|
+
return Uint16Array.from(randomBytes(2))[0] & 16383;
|
|
62
|
+
};
|
|
63
|
+
//=======================================================//
|
|
64
|
+
export const encodeBigEndian = (e, t = 4) => {
|
|
65
|
+
let r = e;
|
|
66
|
+
const a = new Uint8Array(t);
|
|
67
|
+
for (let i = t - 1; i >= 0; i--) {
|
|
68
|
+
a[i] = 255 & r;
|
|
69
|
+
r >>>= 8;
|
|
70
|
+
}
|
|
71
|
+
return a;
|
|
72
|
+
};
|
|
73
|
+
export const toNumber = (t) => typeof t === "object" && t ? ("toNumber" in t ? t.toNumber() : t.low) : t || 0;
|
|
74
|
+
//=======================================================//
|
|
75
|
+
export const unixTimestampSeconds = (date = new Date()) => Math.floor(date.getTime() / 1000);
|
|
76
|
+
export const debouncedTimeout = (intervalMs = 1000, task) => {
|
|
77
|
+
let timeout;
|
|
78
|
+
return {
|
|
79
|
+
start: (newIntervalMs, newTask) => {
|
|
80
|
+
task = newTask || task;
|
|
81
|
+
intervalMs = newIntervalMs || intervalMs;
|
|
82
|
+
timeout && clearTimeout(timeout);
|
|
83
|
+
timeout = setTimeout(() => task?.(), intervalMs);
|
|
84
|
+
},
|
|
85
|
+
cancel: () => {
|
|
86
|
+
timeout && clearTimeout(timeout);
|
|
87
|
+
timeout = undefined;
|
|
88
|
+
},
|
|
89
|
+
setTask: (newTask) => (task = newTask),
|
|
90
|
+
setInterval: (newInterval) => (intervalMs = newInterval)
|
|
91
|
+
};
|
|
92
|
+
};
|
|
93
|
+
//=======================================================//
|
|
94
|
+
export const delay = (ms) => delayCancellable(ms).delay;
|
|
95
|
+
export const delayCancellable = (ms) => {
|
|
96
|
+
const stack = new Error().stack;
|
|
97
|
+
let timeout;
|
|
98
|
+
let reject;
|
|
99
|
+
const delay = new Promise((resolve, _reject) => {
|
|
100
|
+
timeout = setTimeout(resolve, ms);
|
|
101
|
+
reject = _reject;
|
|
102
|
+
});
|
|
103
|
+
const cancel = () => {
|
|
104
|
+
clearTimeout(timeout);
|
|
105
|
+
reject(new Boom("Cancelled", {
|
|
106
|
+
statusCode: 500,
|
|
107
|
+
data: {
|
|
108
|
+
stack
|
|
109
|
+
}
|
|
110
|
+
}));
|
|
111
|
+
};
|
|
112
|
+
return { delay, cancel };
|
|
113
|
+
};
|
|
114
|
+
//=======================================================//
|
|
115
|
+
export async function promiseTimeout(ms, promise) {
|
|
116
|
+
return new Promise(promise);
|
|
117
|
+
}
|
|
118
|
+
//=======================================================//
|
|
119
|
+
export const generateMessageIDV2 = (userId) => {
|
|
120
|
+
const data = Buffer.alloc(8 + 20 + 16);
|
|
121
|
+
data.writeBigUInt64BE(BigInt(Math.floor(Date.now() / 1000)));
|
|
122
|
+
if (userId) {
|
|
123
|
+
const id = jidDecode(userId);
|
|
124
|
+
if (id?.user) {
|
|
125
|
+
data.write(id.user, 8);
|
|
126
|
+
data.write("@c.us", 8 + id.user.length);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
const random = randomBytes(16);
|
|
130
|
+
random.copy(data, 28);
|
|
131
|
+
const hash = createHash("sha256").update(data).digest();
|
|
132
|
+
return "3EB0" + hash.toString("hex").toUpperCase().substring(0, 18);
|
|
133
|
+
};
|
|
134
|
+
//=======================================================//
|
|
135
|
+
export const generateMessageID = () => "3EB0" + randomBytes(18).toString("hex").toUpperCase();
|
|
136
|
+
export function bindWaitForEvent(ev, event) {
|
|
137
|
+
return async (check, timeoutMs) => {
|
|
138
|
+
let listener;
|
|
139
|
+
let closeListener;
|
|
140
|
+
await promiseTimeout(timeoutMs, (resolve, reject) => {
|
|
141
|
+
closeListener = ({ connection, lastDisconnect }) => {
|
|
142
|
+
if (connection === "close") {
|
|
143
|
+
reject(lastDisconnect?.error || new Boom("Connection Closed", { statusCode: DisconnectReason.connectionClosed }));
|
|
144
|
+
}
|
|
145
|
+
};
|
|
146
|
+
ev.on("connection.update", closeListener);
|
|
147
|
+
listener = async (update) => {
|
|
148
|
+
if (await check(update)) {
|
|
149
|
+
resolve();
|
|
150
|
+
}
|
|
151
|
+
};
|
|
152
|
+
ev.on(event, listener);
|
|
153
|
+
}).finally(() => {
|
|
154
|
+
ev.off(event, listener);
|
|
155
|
+
ev.off("connection.update", closeListener);
|
|
156
|
+
});
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
export const bindWaitForConnectionUpdate = (ev) => bindWaitForEvent(ev, "connection.update");
|
|
160
|
+
//=======================================================//
|
|
161
|
+
export const fetchLatestBaileysVersion = async (options = {}) => {
|
|
162
|
+
const URL = "https://raw.githubusercontent.com/WhiskeySockets/Baileys/master/src/Defaults/index.ts";
|
|
163
|
+
try {
|
|
164
|
+
const response = await fetch(URL, {
|
|
165
|
+
dispatcher: options.dispatcher,
|
|
166
|
+
method: "GET",
|
|
167
|
+
headers: options.headers
|
|
168
|
+
});
|
|
169
|
+
if (!response.ok) {
|
|
170
|
+
throw new Boom(`Failed to fetch latest Baileys version: ${response.statusText}`, { statusCode: response.status });
|
|
171
|
+
}
|
|
172
|
+
const text = await response.text();
|
|
173
|
+
const lines = text.split("\n");
|
|
174
|
+
const versionLine = lines[6];
|
|
175
|
+
const versionMatch = versionLine.match(/const version = \[(\d+),\s*(\d+),\s*(\d+)\]/);
|
|
176
|
+
if (versionMatch) {
|
|
177
|
+
const version = [parseInt(versionMatch[1]), parseInt(versionMatch[2]), parseInt(versionMatch[3])];
|
|
178
|
+
return {
|
|
179
|
+
version,
|
|
180
|
+
isLatest: true
|
|
181
|
+
};
|
|
182
|
+
}
|
|
183
|
+
else {
|
|
184
|
+
throw new Error("Could not parse version from Defaults/index.ts");
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
catch (error) {
|
|
188
|
+
return {
|
|
189
|
+
version: version,
|
|
190
|
+
isLatest: false,
|
|
191
|
+
error
|
|
192
|
+
};
|
|
193
|
+
}
|
|
194
|
+
};
|
|
195
|
+
//=======================================================//
|
|
196
|
+
export const fetchLatestWaWebVersion = async (options = {}) => {
|
|
197
|
+
try {
|
|
198
|
+
const defaultHeaders = {
|
|
199
|
+
"sec-fetch-site": "none",
|
|
200
|
+
"user-agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/131.0.0.0 Safari/537.36"
|
|
201
|
+
};
|
|
202
|
+
const headers = { ...defaultHeaders, ...options.headers };
|
|
203
|
+
const response = await fetch("https://web.whatsapp.com/sw.js", {
|
|
204
|
+
...options,
|
|
205
|
+
method: "GET",
|
|
206
|
+
headers
|
|
207
|
+
});
|
|
208
|
+
if (!response.ok) {
|
|
209
|
+
throw new Boom(`Failed to fetch sw.js: ${response.statusText}`, { statusCode: response.status });
|
|
210
|
+
}
|
|
211
|
+
const data = await response.text();
|
|
212
|
+
const regex = /\\?"client_revision\\?":\s*(\d+)/;
|
|
213
|
+
const match = data.match(regex);
|
|
214
|
+
if (!match?.[1]) {
|
|
215
|
+
return {
|
|
216
|
+
version: version,
|
|
217
|
+
isLatest: false,
|
|
218
|
+
error: {
|
|
219
|
+
message: "Could not find client revision in the fetched content"
|
|
220
|
+
}
|
|
221
|
+
};
|
|
222
|
+
}
|
|
223
|
+
const clientRevision = match[1];
|
|
224
|
+
return {
|
|
225
|
+
version: [2, 3000, +clientRevision],
|
|
226
|
+
isLatest: true
|
|
227
|
+
};
|
|
228
|
+
}
|
|
229
|
+
catch (error) {
|
|
230
|
+
return {
|
|
231
|
+
version: version,
|
|
232
|
+
isLatest: false,
|
|
233
|
+
error
|
|
234
|
+
};
|
|
235
|
+
}
|
|
236
|
+
};
|
|
237
|
+
//=======================================================//
|
|
238
|
+
export const generateMdTagPrefix = () => {
|
|
239
|
+
const bytes = randomBytes(4);
|
|
240
|
+
return `${bytes.readUInt16BE()}.${bytes.readUInt16BE(2)}-`;
|
|
241
|
+
};
|
|
242
|
+
//=======================================================//
|
|
243
|
+
const STATUS_MAP = {
|
|
244
|
+
"sender": proto.WebMessageInfo.Status.SERVER_ACK,
|
|
245
|
+
"played": proto.WebMessageInfo.Status.PLAYED,
|
|
246
|
+
"read": proto.WebMessageInfo.Status.READ,
|
|
247
|
+
"read-self": proto.WebMessageInfo.Status.READ
|
|
248
|
+
};
|
|
249
|
+
//=======================================================//
|
|
250
|
+
export const getStatusFromReceiptType = (type) => {
|
|
251
|
+
const status = STATUS_MAP[type];
|
|
252
|
+
if (typeof type === "undefined") {
|
|
253
|
+
return proto.WebMessageInfo.Status.DELIVERY_ACK;
|
|
254
|
+
}
|
|
255
|
+
return status;
|
|
256
|
+
};
|
|
257
|
+
//=======================================================//
|
|
258
|
+
const CODE_MAP = {
|
|
259
|
+
conflict: DisconnectReason.connectionReplaced
|
|
260
|
+
};
|
|
261
|
+
//=======================================================//
|
|
262
|
+
export const getErrorCodeFromStreamError = (node) => {
|
|
263
|
+
const [reasonNode] = getAllBinaryNodeChildren(node);
|
|
264
|
+
let reason = reasonNode?.tag || "unknown";
|
|
265
|
+
const statusCode = +(node.attrs.code || CODE_MAP[reason] || DisconnectReason.badSession);
|
|
266
|
+
if (statusCode === DisconnectReason.restartRequired) {
|
|
267
|
+
reason = "restart required";
|
|
268
|
+
}
|
|
269
|
+
return {
|
|
270
|
+
reason,
|
|
271
|
+
statusCode
|
|
272
|
+
};
|
|
273
|
+
};
|
|
274
|
+
//=======================================================//
|
|
275
|
+
export const getCallStatusFromNode = ({ tag, attrs }) => {
|
|
276
|
+
let status;
|
|
277
|
+
switch (tag) {
|
|
278
|
+
case "offer":
|
|
279
|
+
case "offer_notice":
|
|
280
|
+
status = "offer";
|
|
281
|
+
break;
|
|
282
|
+
case "terminate":
|
|
283
|
+
if (attrs.reason === "timeout") {
|
|
284
|
+
status = "timeout";
|
|
285
|
+
}
|
|
286
|
+
else {
|
|
287
|
+
status = "terminate";
|
|
288
|
+
}
|
|
289
|
+
break;
|
|
290
|
+
case "reject":
|
|
291
|
+
status = "reject";
|
|
292
|
+
break;
|
|
293
|
+
case "accept":
|
|
294
|
+
status = "accept";
|
|
295
|
+
break;
|
|
296
|
+
default:
|
|
297
|
+
status = "ringing";
|
|
298
|
+
break;
|
|
299
|
+
}
|
|
300
|
+
return status;
|
|
301
|
+
};
|
|
302
|
+
//=======================================================//
|
|
303
|
+
const UNEXPECTED_SERVER_CODE_TEXT = "Unexpected server response: ";
|
|
304
|
+
export const getCodeFromWSError = (error) => {
|
|
305
|
+
let statusCode = 500;
|
|
306
|
+
if (error?.message?.includes(UNEXPECTED_SERVER_CODE_TEXT)) {
|
|
307
|
+
const code = +error?.message.slice(UNEXPECTED_SERVER_CODE_TEXT.length);
|
|
308
|
+
if (!Number.isNaN(code) && code >= 400) {
|
|
309
|
+
statusCode = code;
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
else if (
|
|
313
|
+
error?.code?.startsWith("E") ||
|
|
314
|
+
error?.message?.includes("timed out")) {
|
|
315
|
+
statusCode = 408;
|
|
316
|
+
}
|
|
317
|
+
return statusCode;
|
|
318
|
+
};
|
|
319
|
+
//=======================================================//
|
|
320
|
+
export const isWABusinessPlatform = (platform) => {
|
|
321
|
+
return platform === "smbi" || platform === "smba";
|
|
322
|
+
};
|
|
323
|
+
//=======================================================//
|
|
324
|
+
export function trimUndefined(obj) {
|
|
325
|
+
for (const key in obj) {
|
|
326
|
+
if (typeof obj[key] === "undefined") {
|
|
327
|
+
delete obj[key];
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
return obj;
|
|
331
|
+
}
|
|
332
|
+
//=======================================================//
|
|
333
|
+
const CROCKFORD_CHARACTERS = "123456789ABCDEFGHJKLMNPQRSTVWXYZ";
|
|
334
|
+
export function bytesToCrockford(buffer) {
|
|
335
|
+
let value = 0;
|
|
336
|
+
let bitCount = 0;
|
|
337
|
+
const crockford = [];
|
|
338
|
+
for (const element of buffer) {
|
|
339
|
+
value = (value << 8) | (element & 0xff);
|
|
340
|
+
bitCount += 8;
|
|
341
|
+
while (bitCount >= 5) {
|
|
342
|
+
crockford.push(CROCKFORD_CHARACTERS.charAt((value >>> (bitCount - 5)) & 31));
|
|
343
|
+
bitCount -= 5;
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
if (bitCount > 0) {
|
|
347
|
+
crockford.push(CROCKFORD_CHARACTERS.charAt((value << (5 - bitCount)) & 31));
|
|
348
|
+
}
|
|
349
|
+
return crockford.join("");
|
|
350
|
+
}
|
|
351
|
+
//=======================================================//
|
|
352
|
+
export function encodeNewsletterMessage(message) {
|
|
353
|
+
return proto.Message.encode(message).finish();
|
|
354
|
+
}
|
|
355
|
+
//=======================================================//
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
import { downloadContentFromMessage } from "./messages-media.js";
|
|
3
|
+
import { normalizeMessageContent } from "./messages.js";
|
|
4
|
+
import { WAMessageStubType } from "../Types/index.js";
|
|
5
|
+
import { proto } from "../../WAProto/index.js";
|
|
6
|
+
import { toNumber } from "./generics.js";
|
|
7
|
+
import { promisify } from "util";
|
|
8
|
+
import { inflate } from "zlib";
|
|
9
|
+
//=======================================================//
|
|
10
|
+
const inflatePromise = promisify(inflate);
|
|
11
|
+
export const downloadHistory = async (msg, options) => {
|
|
12
|
+
const stream = await downloadContentFromMessage(msg, "md-msg-hist", { options });
|
|
13
|
+
const bufferArray = [];
|
|
14
|
+
for await (const chunk of stream) {
|
|
15
|
+
bufferArray.push(chunk);
|
|
16
|
+
}
|
|
17
|
+
let buffer = Buffer.concat(bufferArray);
|
|
18
|
+
buffer = await inflatePromise(buffer);
|
|
19
|
+
const syncData = proto.HistorySync.decode(buffer);
|
|
20
|
+
return syncData;
|
|
21
|
+
};
|
|
22
|
+
//=======================================================//
|
|
23
|
+
export const processHistoryMessage = (item) => {
|
|
24
|
+
const messages = [];
|
|
25
|
+
const contacts = [];
|
|
26
|
+
const chats = [];
|
|
27
|
+
switch (item.syncType) {
|
|
28
|
+
case proto.HistorySync.HistorySyncType.INITIAL_BOOTSTRAP:
|
|
29
|
+
case proto.HistorySync.HistorySyncType.RECENT:
|
|
30
|
+
case proto.HistorySync.HistorySyncType.FULL:
|
|
31
|
+
case proto.HistorySync.HistorySyncType.ON_DEMAND:
|
|
32
|
+
for (const chat of item.conversations) {
|
|
33
|
+
contacts.push({
|
|
34
|
+
id: chat.id,
|
|
35
|
+
name: chat.name || undefined,
|
|
36
|
+
lid: chat.lidJid || undefined,
|
|
37
|
+
phoneNumber: chat.pnJid || undefined
|
|
38
|
+
});
|
|
39
|
+
const msgs = chat.messages || [];
|
|
40
|
+
delete chat.messages;
|
|
41
|
+
for (const item of msgs) {
|
|
42
|
+
const message = item.message;
|
|
43
|
+
messages.push(message);
|
|
44
|
+
if (!chat.messages?.length) {
|
|
45
|
+
chat.messages = [{ message }];
|
|
46
|
+
}
|
|
47
|
+
if (!message.key.fromMe && !chat.lastMessageRecvTimestamp) {
|
|
48
|
+
chat.lastMessageRecvTimestamp = toNumber(message.messageTimestamp);
|
|
49
|
+
}
|
|
50
|
+
if ((message.messageStubType === WAMessageStubType.BIZ_PRIVACY_MODE_TO_BSP ||
|
|
51
|
+
message.messageStubType === WAMessageStubType.BIZ_PRIVACY_MODE_TO_FB) &&
|
|
52
|
+
message.messageStubParameters?.[0]) {
|
|
53
|
+
contacts.push({
|
|
54
|
+
id: message.key.participant || message.key.remoteJid,
|
|
55
|
+
verifiedName: message.messageStubParameters?.[0]
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
chats.push({ ...chat });
|
|
60
|
+
}
|
|
61
|
+
break;
|
|
62
|
+
case proto.HistorySync.HistorySyncType.PUSH_NAME:
|
|
63
|
+
for (const c of item.pushnames) {
|
|
64
|
+
contacts.push({ id: c.id, notify: c.pushname });
|
|
65
|
+
}
|
|
66
|
+
break;
|
|
67
|
+
}
|
|
68
|
+
return {
|
|
69
|
+
chats,
|
|
70
|
+
contacts,
|
|
71
|
+
messages,
|
|
72
|
+
syncType: item.syncType,
|
|
73
|
+
progress: item.progress
|
|
74
|
+
};
|
|
75
|
+
};
|
|
76
|
+
//=======================================================//
|
|
77
|
+
export const downloadAndProcessHistorySyncNotification = async (msg, options) => {
|
|
78
|
+
const historyMsg = await downloadHistory(msg, options);
|
|
79
|
+
return processHistoryMessage(historyMsg);
|
|
80
|
+
};
|
|
81
|
+
//=======================================================//
|
|
82
|
+
export const getHistoryMsg = (message) => {
|
|
83
|
+
const normalizedContent = !!message ? normalizeMessageContent(message) : undefined;
|
|
84
|
+
const anyHistoryMsg = normalizedContent?.protocolMessage?.historySyncNotification;
|
|
85
|
+
return anyHistoryMsg;
|
|
86
|
+
};
|
|
87
|
+
//=======================================================//
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
export * from "./use-multi-file-auth-state.js";
|
|
3
|
+
export * from "./message-retry-manager.js";
|
|
4
|
+
export * from "./baileys-event-stream.js";
|
|
5
|
+
export * from "./validate-connection.js";
|
|
6
|
+
export * from "./decode-wa-message.js";
|
|
7
|
+
export * from "./process-message.js";
|
|
8
|
+
export * from "./messages-media.js";
|
|
9
|
+
export * from "./noise-handler.js";
|
|
10
|
+
export * from "./browser-utils.js";
|
|
11
|
+
export * from "./link-preview.js";
|
|
12
|
+
export * from "./event-buffer.js";
|
|
13
|
+
export * from "./auth-utils.js";
|
|
14
|
+
export * from "./chat-utils.js";
|
|
15
|
+
export * from "./generics.js";
|
|
16
|
+
export * from "./messages.js";
|
|
17
|
+
export * from "./history.js";
|
|
18
|
+
export * from "./lt-hash.js";
|
|
19
|
+
export * from "./crypto.js";
|
|
20
|
+
export * from "./signal.js";
|
|
21
|
+
//=======================================================//
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
import { extractImageThumb, getHttpStream } from "./messages-media.js";
|
|
3
|
+
import { prepareWAMessageMedia } from "./messages.js";
|
|
4
|
+
//=======================================================//
|
|
5
|
+
const THUMBNAIL_WIDTH_PX = 192;
|
|
6
|
+
//=======================================================//
|
|
7
|
+
const getCompressedJpegThumbnail = async (url, { thumbnailWidth, fetchOpts }) => {
|
|
8
|
+
const stream = await getHttpStream(url, fetchOpts);
|
|
9
|
+
const result = await extractImageThumb(stream, thumbnailWidth);
|
|
10
|
+
return result;
|
|
11
|
+
};
|
|
12
|
+
//=======================================================//
|
|
13
|
+
export const getUrlInfo = async (text, opts = {
|
|
14
|
+
thumbnailWidth: THUMBNAIL_WIDTH_PX,
|
|
15
|
+
fetchOpts: { timeout: 3000 }
|
|
16
|
+
}) => {
|
|
17
|
+
try {
|
|
18
|
+
const retries = 0;
|
|
19
|
+
const maxRetry = 5;
|
|
20
|
+
const { getLinkPreview } = await import("link-preview-js");
|
|
21
|
+
let previewLink = text;
|
|
22
|
+
if (!text.startsWith("https://") && !text.startsWith("http://")) {
|
|
23
|
+
previewLink = "https://" + previewLink;
|
|
24
|
+
}
|
|
25
|
+
const info = await getLinkPreview(previewLink, {
|
|
26
|
+
...opts.fetchOpts,
|
|
27
|
+
followRedirects: "follow",
|
|
28
|
+
handleRedirects: (baseURL, forwardedURL) => {
|
|
29
|
+
const urlObj = new URL(baseURL);
|
|
30
|
+
const forwardedURLObj = new URL(forwardedURL);
|
|
31
|
+
if (retries >= maxRetry) {
|
|
32
|
+
return false;
|
|
33
|
+
}
|
|
34
|
+
if (forwardedURLObj.hostname === urlObj.hostname ||
|
|
35
|
+
forwardedURLObj.hostname === "www." + urlObj.hostname ||
|
|
36
|
+
"www." + forwardedURLObj.hostname === urlObj.hostname) {
|
|
37
|
+
retries + 1;
|
|
38
|
+
return true;
|
|
39
|
+
}
|
|
40
|
+
else {
|
|
41
|
+
return false;
|
|
42
|
+
}
|
|
43
|
+
},
|
|
44
|
+
headers: opts.fetchOpts?.headers
|
|
45
|
+
});
|
|
46
|
+
if (info && "title" in info && info.title) {
|
|
47
|
+
const [image] = info.images;
|
|
48
|
+
const urlInfo = {
|
|
49
|
+
"canonical-url": info.url,
|
|
50
|
+
"matched-text": text,
|
|
51
|
+
title: info.title,
|
|
52
|
+
description: info.description,
|
|
53
|
+
originalThumbnailUrl: image
|
|
54
|
+
};
|
|
55
|
+
if (opts.uploadImage) {
|
|
56
|
+
const { imageMessage } = await prepareWAMessageMedia({ image: { url: image } }, {
|
|
57
|
+
upload: opts.uploadImage,
|
|
58
|
+
mediaTypeOverride: "thumbnail-link",
|
|
59
|
+
options: opts.fetchOpts
|
|
60
|
+
});
|
|
61
|
+
urlInfo.jpegThumbnail = imageMessage?.jpegThumbnail ? Buffer.from(imageMessage.jpegThumbnail) : undefined;
|
|
62
|
+
urlInfo.highQualityThumbnail = imageMessage || undefined;
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
try {
|
|
66
|
+
urlInfo.jpegThumbnail = image ? (await getCompressedJpegThumbnail(image, opts)).buffer : undefined;
|
|
67
|
+
}
|
|
68
|
+
catch (error) {
|
|
69
|
+
opts.logger?.debug({ err: error.stack, url: previewLink }, "error in generating thumbnail");
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
return urlInfo;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
catch (error) {
|
|
76
|
+
if (!error.message.includes("receive a valid")) {
|
|
77
|
+
throw error;
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
};
|
|
81
|
+
//=======================================================//
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
import P from "pino";
|
|
3
|
+
//=======================================================//
|
|
4
|
+
export default P({ timestamp: () => `,"time":"${new Date().toJSON()}"` });
|
|
5
|
+
//=======================================================//
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
import { hkdf } from "./crypto.js";
|
|
3
|
+
//=======================================================//
|
|
4
|
+
const o = 128;
|
|
5
|
+
class LTHash {
|
|
6
|
+
constructor(e) {
|
|
7
|
+
this.salt = e;
|
|
8
|
+
}
|
|
9
|
+
async add(e, t) {
|
|
10
|
+
for (const item of t) {
|
|
11
|
+
e = await this._addSingle(e, item);
|
|
12
|
+
}
|
|
13
|
+
return e;
|
|
14
|
+
}
|
|
15
|
+
async subtract(e, t) {
|
|
16
|
+
for (const item of t) {
|
|
17
|
+
e = await this._subtractSingle(e, item);
|
|
18
|
+
}
|
|
19
|
+
return e;
|
|
20
|
+
}
|
|
21
|
+
async subtractThenAdd(e, addList, subtractList) {
|
|
22
|
+
const subtracted = await this.subtract(e, subtractList);
|
|
23
|
+
return this.add(subtracted, addList);
|
|
24
|
+
}
|
|
25
|
+
async _addSingle(e, t) {
|
|
26
|
+
const derived = new Uint8Array(await hkdf(Buffer.from(t), o, { info: this.salt })).buffer;
|
|
27
|
+
return this.performPointwiseWithOverflow(e, derived, (a, b) => a + b);
|
|
28
|
+
}
|
|
29
|
+
async _subtractSingle(e, t) {
|
|
30
|
+
const derived = new Uint8Array(await hkdf(Buffer.from(t), o, { info: this.salt })).buffer;
|
|
31
|
+
return this.performPointwiseWithOverflow(e, derived, (a, b) => a - b);
|
|
32
|
+
}
|
|
33
|
+
performPointwiseWithOverflow(e, t, op) {
|
|
34
|
+
const n = new DataView(e);
|
|
35
|
+
const i = new DataView(t);
|
|
36
|
+
const out = new ArrayBuffer(n.byteLength);
|
|
37
|
+
const s = new DataView(out);
|
|
38
|
+
for (let offset = 0; offset < n.byteLength; offset += 2) {
|
|
39
|
+
s.setUint16(offset, op(n.getUint16(offset, true), i.getUint16(offset, true)), true);
|
|
40
|
+
}
|
|
41
|
+
return out;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
export const LT_HASH_ANTI_TAMPERING = new LTHash("WhatsApp Patch Integrity");
|
|
45
|
+
//=======================================================//
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
//=======================================================//
|
|
2
|
+
export const makeMutex = () => {
|
|
3
|
+
let task = Promise.resolve();
|
|
4
|
+
let taskTimeout;
|
|
5
|
+
return {
|
|
6
|
+
mutex(code) {
|
|
7
|
+
task = (async () => {
|
|
8
|
+
try {
|
|
9
|
+
await task;
|
|
10
|
+
}
|
|
11
|
+
catch { }
|
|
12
|
+
try {
|
|
13
|
+
const result = await code();
|
|
14
|
+
return result;
|
|
15
|
+
}
|
|
16
|
+
finally {
|
|
17
|
+
clearTimeout(taskTimeout);
|
|
18
|
+
}
|
|
19
|
+
})();
|
|
20
|
+
return task;
|
|
21
|
+
}
|
|
22
|
+
};
|
|
23
|
+
};
|
|
24
|
+
//=======================================================//
|
|
25
|
+
export const makeKeyedMutex = () => {
|
|
26
|
+
const map = {};
|
|
27
|
+
return {
|
|
28
|
+
mutex(key, task) {
|
|
29
|
+
if (!map[key]) {
|
|
30
|
+
map[key] = makeMutex();
|
|
31
|
+
}
|
|
32
|
+
return map[key].mutex(task);
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
};
|
|
36
|
+
//=======================================================//
|