@alannxd/baileys 6.0.6 → 6.0.9
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 +1 -1
- package/README.md +341 -286
- package/WAProto/GenerateStatics.sh +3 -0
- package/WAProto/WAProto.proto +6902 -0
- package/WAProto/fix-imports.js +85 -0
- package/WAProto/index.d.ts +79257 -0
- package/WAProto/index.js +205861 -60565
- package/engine-requirements.js +1 -1
- package/lib/Defaults/index.js +119 -136
- package/lib/Signal/Group/ciphertext-message.js +2 -5
- package/lib/Signal/Group/group-session-builder.js +7 -41
- package/lib/Signal/Group/group_cipher.js +37 -51
- package/lib/Signal/Group/index.js +12 -57
- package/lib/Signal/Group/keyhelper.js +7 -44
- package/lib/Signal/Group/sender-chain-key.js +7 -15
- package/lib/Signal/Group/sender-key-distribution-message.js +8 -11
- package/lib/Signal/Group/sender-key-message.js +9 -12
- package/lib/Signal/Group/sender-key-name.js +2 -5
- package/lib/Signal/Group/sender-key-record.js +9 -21
- package/lib/Signal/Group/sender-key-state.js +27 -42
- package/lib/Signal/Group/sender-message-key.js +4 -7
- package/lib/Signal/libsignal.js +347 -90
- package/lib/Signal/lid-mapping.js +277 -0
- package/lib/Socket/Client/index.js +3 -19
- package/lib/Socket/Client/types.js +11 -0
- package/lib/Socket/Client/websocket.js +54 -0
- package/lib/Socket/business.js +162 -43
- package/lib/Socket/chats.js +627 -427
- package/lib/Socket/communities.js +90 -80
- package/lib/Socket/groups.js +154 -161
- package/lib/Socket/index.js +11 -10
- package/lib/Socket/luxu.js +315 -469
- package/lib/Socket/messages-recv.js +1421 -615
- package/lib/Socket/messages-send.js +1150 -799
- package/lib/Socket/mex.js +42 -0
- package/lib/Socket/newsletter.js +152 -204
- package/lib/Socket/socket.js +544 -313
- package/lib/Store/index.js +10 -10
- package/lib/Store/keyed-db.js +108 -0
- package/lib/Store/make-cache-manager-store.js +43 -41
- package/lib/Store/make-in-memory-store.js +112 -341
- package/lib/Store/make-ordered-dictionary.js +14 -20
- package/lib/Store/object-repository.js +11 -6
- package/lib/Types/Auth.js +2 -2
- package/lib/Types/Bussines.js +2 -0
- package/lib/Types/Call.js +2 -2
- package/lib/Types/Chat.js +8 -4
- package/lib/Types/Contact.js +2 -2
- package/lib/Types/Events.js +2 -2
- package/lib/Types/GroupMetadata.js +2 -2
- package/lib/Types/Label.js +3 -5
- package/lib/Types/LabelAssociation.js +3 -5
- package/lib/Types/Message.js +11 -9
- package/lib/Types/Mex.js +37 -0
- package/lib/Types/Product.js +2 -2
- package/lib/Types/Signal.js +2 -2
- package/lib/Types/Socket.js +3 -2
- package/lib/Types/State.js +56 -2
- package/lib/Types/USync.js +2 -2
- package/lib/Types/index.js +15 -31
- package/lib/Utils/auth-utils.js +239 -143
- package/lib/Utils/browser-utils.js +48 -0
- package/lib/Utils/business.js +66 -69
- package/lib/Utils/chat-utils.js +396 -253
- package/lib/Utils/companion-reg-client-utils.js +35 -0
- package/lib/Utils/crypto.js +57 -90
- package/lib/Utils/decode-wa-message.js +236 -84
- package/lib/Utils/event-buffer.js +185 -77
- package/lib/Utils/generics.js +189 -209
- package/lib/Utils/history.js +93 -55
- package/lib/Utils/identity-change-handler.js +50 -0
- package/lib/Utils/index.js +23 -33
- package/lib/Utils/link-preview.js +16 -24
- package/lib/Utils/logger.js +3 -7
- package/lib/Utils/lt-hash.js +3 -46
- package/lib/Utils/make-mutex.js +24 -34
- package/lib/Utils/message-composer.js +273 -0
- package/lib/Utils/message-retry-manager.js +265 -0
- package/lib/Utils/messages-media.js +451 -482
- package/lib/Utils/messages.js +795 -369
- package/lib/Utils/noise-handler.js +145 -99
- package/lib/Utils/offline-node-processor.js +40 -0
- package/lib/Utils/pre-key-manager.js +106 -0
- package/lib/Utils/process-message.js +459 -150
- package/lib/Utils/reporting-utils.js +258 -0
- package/lib/Utils/signal.js +120 -72
- package/lib/Utils/stanza-ack.js +38 -0
- package/lib/Utils/sync-action-utils.js +49 -0
- package/lib/Utils/tc-token-utils.js +163 -0
- package/lib/Utils/use-multi-file-auth-state.js +29 -27
- package/lib/Utils/validate-connection.js +73 -99
- package/lib/WABinary/constants.js +1281 -20
- package/lib/WABinary/decode.js +52 -42
- package/lib/WABinary/encode.js +110 -155
- package/lib/WABinary/generic-utils.js +55 -49
- package/lib/WABinary/index.js +6 -21
- package/lib/WABinary/jid-utils.js +76 -40
- package/lib/WABinary/types.js +2 -2
- package/lib/WAM/BinaryInfo.js +2 -5
- package/lib/WAM/constants.js +19071 -11568
- package/lib/WAM/encode.js +17 -22
- package/lib/WAM/index.js +4 -19
- package/lib/WAUSync/Protocols/USyncContactProtocol.js +33 -13
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +11 -14
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +9 -12
- package/lib/WAUSync/Protocols/USyncStatusProtocol.js +9 -13
- package/lib/WAUSync/Protocols/USyncUsernameProtocol.js +25 -0
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +20 -22
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +13 -8
- package/lib/WAUSync/Protocols/index.js +6 -20
- package/lib/WAUSync/USyncQuery.js +44 -35
- package/lib/WAUSync/USyncUser.js +10 -5
- package/lib/WAUSync/index.js +4 -19
- package/lib/index.js +13 -36
- package/package.json +85 -51
- package/WAProto/fix-import.js +0 -29
- package/lib/Defaults/baileys-version.json +0 -3
- package/lib/Defaults/index.d.ts +0 -53
- package/lib/Defaults/phonenumber-mcc.json +0 -223
- package/lib/Signal/Group/ciphertext-message.d.ts +0 -9
- package/lib/Signal/Group/group-session-builder.d.ts +0 -14
- package/lib/Signal/Group/group_cipher.d.ts +0 -17
- package/lib/Signal/Group/index.d.ts +0 -11
- package/lib/Signal/Group/keyhelper.d.ts +0 -10
- package/lib/Signal/Group/queue-job.d.ts +0 -1
- package/lib/Signal/Group/queue-job.js +0 -57
- package/lib/Signal/Group/sender-chain-key.d.ts +0 -13
- package/lib/Signal/Group/sender-key-distribution-message.d.ts +0 -16
- package/lib/Signal/Group/sender-key-message.d.ts +0 -18
- package/lib/Signal/Group/sender-key-name.d.ts +0 -17
- package/lib/Signal/Group/sender-key-record.d.ts +0 -30
- package/lib/Signal/Group/sender-key-state.d.ts +0 -38
- package/lib/Signal/Group/sender-message-key.d.ts +0 -11
- package/lib/Signal/libsignal.d.ts +0 -3
- package/lib/Socket/Client/abstract-socket-client.d.ts +0 -17
- package/lib/Socket/Client/abstract-socket-client.js +0 -13
- package/lib/Socket/Client/index.d.ts +0 -3
- package/lib/Socket/Client/mobile-socket-client.d.ts +0 -13
- package/lib/Socket/Client/mobile-socket-client.js +0 -65
- package/lib/Socket/Client/web-socket-client.d.ts +0 -12
- package/lib/Socket/Client/web-socket-client.js +0 -62
- package/lib/Socket/business.d.ts +0 -171
- package/lib/Socket/chats.d.ts +0 -267
- package/lib/Socket/communities.d.ts +0 -180
- package/lib/Socket/groups.d.ts +0 -115
- package/lib/Socket/index.d.ts +0 -173
- package/lib/Socket/luxu.d.ts +0 -266
- package/lib/Socket/messages-recv.d.ts +0 -161
- package/lib/Socket/messages-send.d.ts +0 -183
- package/lib/Socket/newsletter.d.ts +0 -134
- package/lib/Socket/registration.d.ts +0 -267
- package/lib/Socket/registration.js +0 -166
- package/lib/Socket/socket.d.ts +0 -44
- package/lib/Socket/usync.d.ts +0 -36
- package/lib/Socket/usync.js +0 -70
- package/lib/Store/index.d.ts +0 -3
- package/lib/Store/make-cache-manager-store.d.ts +0 -13
- package/lib/Store/make-in-memory-store.d.ts +0 -118
- package/lib/Store/make-ordered-dictionary.d.ts +0 -13
- package/lib/Store/object-repository.d.ts +0 -10
- package/lib/Types/Auth.d.ts +0 -110
- package/lib/Types/Call.d.ts +0 -13
- package/lib/Types/Chat.d.ts +0 -102
- package/lib/Types/Contact.d.ts +0 -19
- package/lib/Types/Events.d.ts +0 -157
- package/lib/Types/GroupMetadata.d.ts +0 -55
- package/lib/Types/Label.d.ts +0 -35
- package/lib/Types/LabelAssociation.d.ts +0 -29
- package/lib/Types/Message.d.ts +0 -273
- package/lib/Types/Newsletter.d.ts +0 -103
- package/lib/Types/Newsletter.js +0 -38
- package/lib/Types/Product.d.ts +0 -78
- package/lib/Types/Signal.d.ts +0 -57
- package/lib/Types/Socket.d.ts +0 -111
- package/lib/Types/State.d.ts +0 -27
- package/lib/Types/USync.d.ts +0 -25
- package/lib/Types/index.d.ts +0 -57
- package/lib/Utils/auth-utils.d.ts +0 -18
- package/lib/Utils/baileys-event-stream.d.ts +0 -16
- package/lib/Utils/baileys-event-stream.js +0 -63
- package/lib/Utils/business.d.ts +0 -22
- package/lib/Utils/chat-utils.d.ts +0 -71
- package/lib/Utils/crypto.d.ts +0 -41
- package/lib/Utils/decode-wa-message.d.ts +0 -19
- package/lib/Utils/event-buffer.d.ts +0 -35
- package/lib/Utils/generics.d.ts +0 -92
- package/lib/Utils/history.d.ts +0 -15
- package/lib/Utils/index.d.ts +0 -17
- package/lib/Utils/link-preview.d.ts +0 -21
- package/lib/Utils/logger.d.ts +0 -4
- package/lib/Utils/lt-hash.d.ts +0 -12
- package/lib/Utils/make-mutex.d.ts +0 -7
- package/lib/Utils/messages-media.d.ts +0 -116
- package/lib/Utils/messages.d.ts +0 -77
- package/lib/Utils/noise-handler.d.ts +0 -21
- package/lib/Utils/process-message.d.ts +0 -41
- package/lib/Utils/signal.d.ts +0 -32
- package/lib/Utils/use-multi-file-auth-state.d.ts +0 -13
- package/lib/Utils/validate-connection.d.ts +0 -11
- package/lib/WABinary/constants.d.ts +0 -30
- package/lib/WABinary/decode.d.ts +0 -7
- package/lib/WABinary/encode.d.ts +0 -3
- package/lib/WABinary/generic-utils.d.ts +0 -17
- package/lib/WABinary/index.d.ts +0 -5
- package/lib/WABinary/jid-utils.d.ts +0 -31
- package/lib/WABinary/types.d.ts +0 -18
- package/lib/WAM/BinaryInfo.d.ts +0 -17
- package/lib/WAM/constants.d.ts +0 -38
- package/lib/WAM/encode.d.ts +0 -3
- package/lib/WAM/index.d.ts +0 -3
- package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts +0 -9
- package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts +0 -22
- package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts +0 -12
- package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts +0 -12
- package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts +0 -25
- package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts +0 -8
- package/lib/WAUSync/Protocols/index.d.ts +0 -4
- package/lib/WAUSync/USyncQuery.d.ts +0 -28
- package/lib/WAUSync/USyncUser.d.ts +0 -12
- package/lib/index.d.ts +0 -12
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { getBinaryNodeChild, getBinaryNodeChildren, isHostedLidUser, isHostedPnUser, isJidMetaAI, isLidUser, isPnUser, jidNormalizedUser } from '../WABinary/index.js';
|
|
2
|
+
// Same phone-number pattern as WABinary's isJidBot, applied against the user
|
|
3
|
+
// part so the check is invariant to @c.us ↔ @s.whatsapp.net normalization.
|
|
4
|
+
const BOT_PHONE_REGEX = /^1313555\d{4}$|^131655500\d{2}$/;
|
|
5
|
+
/**
|
|
6
|
+
* Mirrors WA Web's `Wid.isRegularUser()` (user ∧ ¬PSA ∧ ¬Bot). Used to gate tctoken
|
|
7
|
+
* storage against malformed notifications — WA Web filters server-side but we
|
|
8
|
+
* defend here for parity with `WAWebSetTcTokenChatAction.handleIncomingTcToken`.
|
|
9
|
+
* Works for both pre- and post-normalized JIDs (`@c.us` vs `@s.whatsapp.net`).
|
|
10
|
+
*/
|
|
11
|
+
function isRegularUser(jid) {
|
|
12
|
+
if (!jid)
|
|
13
|
+
return false;
|
|
14
|
+
const user = jid.split('@')[0] ?? '';
|
|
15
|
+
if (user === '0')
|
|
16
|
+
return false; // PSA
|
|
17
|
+
if (BOT_PHONE_REGEX.test(user))
|
|
18
|
+
return false; // Bot by phone pattern
|
|
19
|
+
if (isJidMetaAI(jid))
|
|
20
|
+
return false; // MetaAI (@bot server)
|
|
21
|
+
return !!(isPnUser(jid) || isLidUser(jid) || isHostedPnUser(jid) || isHostedLidUser(jid) || jid.endsWith('@c.us'));
|
|
22
|
+
}
|
|
23
|
+
const TC_TOKEN_BUCKET_DURATION = 604800; // 7 days
|
|
24
|
+
const TC_TOKEN_NUM_BUCKETS = 4; // ~28-day rolling window
|
|
25
|
+
/** Sentinel key under `tctoken` store holding a JSON array of tracked storage JIDs for cross-session pruning. */
|
|
26
|
+
export const TC_TOKEN_INDEX_KEY = '__index';
|
|
27
|
+
/** Read the persisted tctoken JID index and return its entries (never contains the sentinel key itself). */
|
|
28
|
+
export async function readTcTokenIndex(keys) {
|
|
29
|
+
const data = await keys.get('tctoken', [TC_TOKEN_INDEX_KEY]);
|
|
30
|
+
const entry = data[TC_TOKEN_INDEX_KEY];
|
|
31
|
+
if (!entry?.token?.length)
|
|
32
|
+
return [];
|
|
33
|
+
try {
|
|
34
|
+
const parsed = JSON.parse(Buffer.from(entry.token).toString());
|
|
35
|
+
if (!Array.isArray(parsed))
|
|
36
|
+
return [];
|
|
37
|
+
return parsed.filter((j) => typeof j === 'string' && j.length > 0 && j !== TC_TOKEN_INDEX_KEY);
|
|
38
|
+
}
|
|
39
|
+
catch {
|
|
40
|
+
return [];
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
/** Build a SignalDataSet fragment that writes the merged index (persisted ∪ added) under the sentinel key. */
|
|
44
|
+
export async function buildMergedTcTokenIndexWrite(keys, addedJids) {
|
|
45
|
+
const persisted = await readTcTokenIndex(keys);
|
|
46
|
+
const merged = new Set(persisted);
|
|
47
|
+
for (const jid of addedJids) {
|
|
48
|
+
if (jid && jid !== TC_TOKEN_INDEX_KEY)
|
|
49
|
+
merged.add(jid);
|
|
50
|
+
}
|
|
51
|
+
return {
|
|
52
|
+
[TC_TOKEN_INDEX_KEY]: { token: Buffer.from(JSON.stringify([...merged])) }
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
// WA Web has separate sender/receiver AB props for these but they're identical today
|
|
56
|
+
export function isTcTokenExpired(timestamp) {
|
|
57
|
+
if (timestamp === null || timestamp === undefined)
|
|
58
|
+
return true;
|
|
59
|
+
const ts = typeof timestamp === 'string' ? parseInt(timestamp) : timestamp;
|
|
60
|
+
if (isNaN(ts))
|
|
61
|
+
return true;
|
|
62
|
+
const now = Math.floor(Date.now() / 1000);
|
|
63
|
+
const currentBucket = Math.floor(now / TC_TOKEN_BUCKET_DURATION);
|
|
64
|
+
const cutoffBucket = currentBucket - (TC_TOKEN_NUM_BUCKETS - 1);
|
|
65
|
+
const cutoffTimestamp = cutoffBucket * TC_TOKEN_BUCKET_DURATION;
|
|
66
|
+
return ts < cutoffTimestamp;
|
|
67
|
+
}
|
|
68
|
+
export function shouldSendNewTcToken(senderTimestamp) {
|
|
69
|
+
if (senderTimestamp === undefined)
|
|
70
|
+
return true;
|
|
71
|
+
const now = Math.floor(Date.now() / 1000);
|
|
72
|
+
const currentBucket = Math.floor(now / TC_TOKEN_BUCKET_DURATION);
|
|
73
|
+
const senderBucket = Math.floor(senderTimestamp / TC_TOKEN_BUCKET_DURATION);
|
|
74
|
+
return currentBucket > senderBucket;
|
|
75
|
+
}
|
|
76
|
+
/** Resolve JID to LID for tctoken storage (WA Web stores under LID) */
|
|
77
|
+
export async function resolveTcTokenJid(jid, getLIDForPN) {
|
|
78
|
+
if (isLidUser(jid))
|
|
79
|
+
return jid;
|
|
80
|
+
const lid = await getLIDForPN(jid);
|
|
81
|
+
return lid ?? jid;
|
|
82
|
+
}
|
|
83
|
+
/** Resolve target JID for issuing privacy token based on AB prop 14303 */
|
|
84
|
+
export async function resolveIssuanceJid(jid, issueToLid, getLIDForPN, getPNForLID) {
|
|
85
|
+
if (issueToLid) {
|
|
86
|
+
if (isLidUser(jid))
|
|
87
|
+
return jid;
|
|
88
|
+
const lid = await getLIDForPN(jid);
|
|
89
|
+
return lid ?? jid;
|
|
90
|
+
}
|
|
91
|
+
if (!isLidUser(jid))
|
|
92
|
+
return jid;
|
|
93
|
+
if (getPNForLID) {
|
|
94
|
+
const pn = await getPNForLID(jid);
|
|
95
|
+
return pn ?? jid;
|
|
96
|
+
}
|
|
97
|
+
return jid;
|
|
98
|
+
}
|
|
99
|
+
export async function buildTcTokenFromJid({ authState, jid, baseContent = [], getLIDForPN }) {
|
|
100
|
+
try {
|
|
101
|
+
const storageJid = await resolveTcTokenJid(jid, getLIDForPN);
|
|
102
|
+
const tcTokenData = await authState.keys.get('tctoken', [storageJid]);
|
|
103
|
+
const entry = tcTokenData?.[storageJid];
|
|
104
|
+
const tcTokenBuffer = entry?.token;
|
|
105
|
+
if (!tcTokenBuffer?.length || isTcTokenExpired(entry?.timestamp)) {
|
|
106
|
+
if (tcTokenBuffer) {
|
|
107
|
+
// Preserve senderTimestamp so shouldSendNewTcToken() keeps its dedupe state
|
|
108
|
+
// after we drop the unusable peer token. Only wipe the record entirely when
|
|
109
|
+
// there's nothing worth keeping.
|
|
110
|
+
const cleared = entry?.senderTimestamp !== undefined
|
|
111
|
+
? { token: Buffer.alloc(0), senderTimestamp: entry.senderTimestamp }
|
|
112
|
+
: null;
|
|
113
|
+
await authState.keys.set({ tctoken: { [storageJid]: cleared } });
|
|
114
|
+
}
|
|
115
|
+
return baseContent.length > 0 ? baseContent : undefined;
|
|
116
|
+
}
|
|
117
|
+
baseContent.push({
|
|
118
|
+
tag: 'tctoken',
|
|
119
|
+
attrs: {},
|
|
120
|
+
content: tcTokenBuffer
|
|
121
|
+
});
|
|
122
|
+
return baseContent;
|
|
123
|
+
}
|
|
124
|
+
catch (error) {
|
|
125
|
+
return baseContent.length > 0 ? baseContent : undefined;
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
export async function storeTcTokensFromIqResult({ result, fallbackJid, keys, getLIDForPN, onNewJidStored }) {
|
|
129
|
+
const tokensNode = getBinaryNodeChild(result, 'tokens');
|
|
130
|
+
if (!tokensNode)
|
|
131
|
+
return;
|
|
132
|
+
const tokenNodes = getBinaryNodeChildren(tokensNode, 'token');
|
|
133
|
+
for (const tokenNode of tokenNodes) {
|
|
134
|
+
if (tokenNode.attrs.type !== 'trusted_contact' || !(tokenNode.content instanceof Uint8Array)) {
|
|
135
|
+
continue;
|
|
136
|
+
}
|
|
137
|
+
// In notifications tokenNode.attrs.jid is your own device JID, not the sender's
|
|
138
|
+
const rawJid = jidNormalizedUser(fallbackJid || tokenNode.attrs.jid);
|
|
139
|
+
if (!isRegularUser(rawJid))
|
|
140
|
+
continue;
|
|
141
|
+
const storageJid = await resolveTcTokenJid(rawJid, getLIDForPN);
|
|
142
|
+
const existingTcData = await keys.get('tctoken', [storageJid]);
|
|
143
|
+
const existingEntry = existingTcData[storageJid];
|
|
144
|
+
const existingTs = existingEntry?.timestamp ? Number(existingEntry.timestamp) : 0;
|
|
145
|
+
const incomingTs = tokenNode.attrs.t ? Number(tokenNode.attrs.t) : 0;
|
|
146
|
+
// timestamp-less tokens would be immediately expired
|
|
147
|
+
if (!incomingTs)
|
|
148
|
+
continue;
|
|
149
|
+
if (existingTs > 0 && existingTs > incomingTs)
|
|
150
|
+
continue;
|
|
151
|
+
await keys.set({
|
|
152
|
+
tctoken: {
|
|
153
|
+
[storageJid]: {
|
|
154
|
+
...existingEntry,
|
|
155
|
+
token: Buffer.from(tokenNode.content),
|
|
156
|
+
timestamp: tokenNode.attrs.t
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
});
|
|
160
|
+
onNewJidStored?.(storageJid);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
//# sourceMappingURL=tc-token-utils.js.map
|
|
@@ -1,17 +1,19 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
1
|
+
import { Mutex } from 'async-mutex';
|
|
2
|
+
import { mkdir, readFile, stat, unlink, writeFile } from 'fs/promises';
|
|
3
|
+
import { join } from 'path';
|
|
4
|
+
import { proto } from '../../WAProto/index.js';
|
|
5
|
+
import { initAuthCreds } from './auth-utils.js';
|
|
6
|
+
import { BufferJSON } from './generics.js';
|
|
7
|
+
// We need to lock files due to the fact that we are using async functions to read and write files
|
|
8
|
+
// https://github.com/WhiskeySockets/Baileys/issues/794
|
|
9
|
+
// https://github.com/nodejs/node/issues/26338
|
|
10
|
+
// Use a Map to store mutexes for each file path
|
|
10
11
|
const fileLocks = new Map();
|
|
12
|
+
// Get or create a mutex for a specific file path
|
|
11
13
|
const getFileLock = (path) => {
|
|
12
14
|
let mutex = fileLocks.get(path);
|
|
13
15
|
if (!mutex) {
|
|
14
|
-
mutex = new
|
|
16
|
+
mutex = new Mutex();
|
|
15
17
|
fileLocks.set(path, mutex);
|
|
16
18
|
}
|
|
17
19
|
return mutex;
|
|
@@ -23,13 +25,14 @@ const getFileLock = (path) => {
|
|
|
23
25
|
* Again, I wouldn't endorse this for any production level use other than perhaps a bot.
|
|
24
26
|
* Would recommend writing an auth state for use with a proper SQL or No-SQL DB
|
|
25
27
|
* */
|
|
26
|
-
const useMultiFileAuthState = async (folder) => {
|
|
28
|
+
export const useMultiFileAuthState = async (folder) => {
|
|
29
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
27
30
|
const writeData = async (data, file) => {
|
|
28
|
-
const filePath =
|
|
31
|
+
const filePath = join(folder, fixFileName(file));
|
|
29
32
|
const mutex = getFileLock(filePath);
|
|
30
33
|
return mutex.acquire().then(async (release) => {
|
|
31
34
|
try {
|
|
32
|
-
await
|
|
35
|
+
await writeFile(filePath, JSON.stringify(data, BufferJSON.replacer));
|
|
33
36
|
}
|
|
34
37
|
finally {
|
|
35
38
|
release();
|
|
@@ -38,12 +41,12 @@ const useMultiFileAuthState = async (folder) => {
|
|
|
38
41
|
};
|
|
39
42
|
const readData = async (file) => {
|
|
40
43
|
try {
|
|
41
|
-
const filePath =
|
|
44
|
+
const filePath = join(folder, fixFileName(file));
|
|
42
45
|
const mutex = getFileLock(filePath);
|
|
43
46
|
return await mutex.acquire().then(async (release) => {
|
|
44
47
|
try {
|
|
45
|
-
const data = await
|
|
46
|
-
return JSON.parse(data,
|
|
48
|
+
const data = await readFile(filePath, { encoding: 'utf-8' });
|
|
49
|
+
return JSON.parse(data, BufferJSON.reviver);
|
|
47
50
|
}
|
|
48
51
|
finally {
|
|
49
52
|
release();
|
|
@@ -56,33 +59,32 @@ const useMultiFileAuthState = async (folder) => {
|
|
|
56
59
|
};
|
|
57
60
|
const removeData = async (file) => {
|
|
58
61
|
try {
|
|
59
|
-
const filePath =
|
|
62
|
+
const filePath = join(folder, fixFileName(file));
|
|
60
63
|
const mutex = getFileLock(filePath);
|
|
61
64
|
return mutex.acquire().then(async (release) => {
|
|
62
65
|
try {
|
|
63
|
-
await
|
|
66
|
+
await unlink(filePath);
|
|
64
67
|
}
|
|
65
|
-
catch
|
|
68
|
+
catch {
|
|
66
69
|
}
|
|
67
70
|
finally {
|
|
68
71
|
release();
|
|
69
72
|
}
|
|
70
73
|
});
|
|
71
74
|
}
|
|
72
|
-
catch
|
|
73
|
-
}
|
|
75
|
+
catch { }
|
|
74
76
|
};
|
|
75
|
-
const folderInfo = await
|
|
77
|
+
const folderInfo = await stat(folder).catch(() => { });
|
|
76
78
|
if (folderInfo) {
|
|
77
79
|
if (!folderInfo.isDirectory()) {
|
|
78
80
|
throw new Error(`found something that is not a directory at ${folder}, either delete it or specify a different location`);
|
|
79
81
|
}
|
|
80
82
|
}
|
|
81
83
|
else {
|
|
82
|
-
await
|
|
84
|
+
await mkdir(folder, { recursive: true });
|
|
83
85
|
}
|
|
84
|
-
const fixFileName = (file) =>
|
|
85
|
-
const creds = await readData('creds.json') ||
|
|
86
|
+
const fixFileName = (file) => file?.replace(/\//g, '__')?.replace(/:/g, '-');
|
|
87
|
+
const creds = (await readData('creds.json')) || initAuthCreds();
|
|
86
88
|
return {
|
|
87
89
|
state: {
|
|
88
90
|
creds,
|
|
@@ -92,7 +94,7 @@ const useMultiFileAuthState = async (folder) => {
|
|
|
92
94
|
await Promise.all(ids.map(async (id) => {
|
|
93
95
|
let value = await readData(`${type}-${id}.json`);
|
|
94
96
|
if (type === 'app-state-sync-key' && value) {
|
|
95
|
-
value =
|
|
97
|
+
value = proto.Message.AppStateSyncKeyData.fromObject(value);
|
|
96
98
|
}
|
|
97
99
|
data[id] = value;
|
|
98
100
|
}));
|
|
@@ -116,4 +118,4 @@ const useMultiFileAuthState = async (folder) => {
|
|
|
116
118
|
}
|
|
117
119
|
};
|
|
118
120
|
};
|
|
119
|
-
|
|
121
|
+
//# sourceMappingURL=use-multi-file-auth-state.js.map
|
|
@@ -1,81 +1,78 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
const crypto_2 = require("./crypto");
|
|
10
|
-
const generics_1 = require("./generics");
|
|
11
|
-
const signal_1 = require("./signal");
|
|
12
|
-
|
|
1
|
+
import { Boom } from '@hapi/boom';
|
|
2
|
+
import { createHash } from 'crypto';
|
|
3
|
+
import { proto } from '../../WAProto/index.js';
|
|
4
|
+
import { KEY_BUNDLE_TYPE, WA_ADV_ACCOUNT_SIG_PREFIX, WA_ADV_DEVICE_SIG_PREFIX, WA_ADV_HOSTED_ACCOUNT_SIG_PREFIX } from '../Defaults/index.js';
|
|
5
|
+
import { getBinaryNodeChild, jidDecode, S_WHATSAPP_NET } from '../WABinary/index.js';
|
|
6
|
+
import { Curve, hmacSign } from './crypto.js';
|
|
7
|
+
import { encodeBigEndian } from './generics.js';
|
|
8
|
+
import { createSignalIdentity } from './signal.js';
|
|
13
9
|
const getUserAgent = (config) => {
|
|
14
10
|
return {
|
|
15
11
|
appVersion: {
|
|
16
12
|
primary: config.version[0],
|
|
17
13
|
secondary: config.version[1],
|
|
18
|
-
tertiary: config.version[2]
|
|
14
|
+
tertiary: config.version[2]
|
|
19
15
|
},
|
|
20
|
-
platform:
|
|
21
|
-
releaseChannel:
|
|
16
|
+
platform: proto.ClientPayload.UserAgent.Platform.MACOS,
|
|
17
|
+
releaseChannel: proto.ClientPayload.UserAgent.ReleaseChannel.RELEASE,
|
|
22
18
|
osVersion: '0.1',
|
|
23
19
|
device: 'Desktop',
|
|
24
20
|
osBuildNumber: '0.1',
|
|
25
21
|
localeLanguageIso6391: 'en',
|
|
26
22
|
mnc: '000',
|
|
27
23
|
mcc: '000',
|
|
28
|
-
localeCountryIso31661Alpha2: config.countryCode
|
|
24
|
+
localeCountryIso31661Alpha2: config.countryCode
|
|
29
25
|
};
|
|
30
26
|
};
|
|
31
|
-
|
|
32
27
|
const PLATFORM_MAP = {
|
|
33
|
-
'Mac OS':
|
|
34
|
-
|
|
28
|
+
'Mac OS': proto.ClientPayload.WebInfo.WebSubPlatform.DARWIN,
|
|
29
|
+
Windows: proto.ClientPayload.WebInfo.WebSubPlatform.WIN32
|
|
35
30
|
};
|
|
36
|
-
|
|
37
31
|
const getWebInfo = (config) => {
|
|
38
|
-
let webSubPlatform =
|
|
39
|
-
if (config.syncFullHistory &&
|
|
32
|
+
let webSubPlatform = proto.ClientPayload.WebInfo.WebSubPlatform.WEB_BROWSER;
|
|
33
|
+
if (config.syncFullHistory &&
|
|
34
|
+
PLATFORM_MAP[config.browser[0]] &&
|
|
35
|
+
config.browser[1] === 'Desktop') {
|
|
40
36
|
webSubPlatform = PLATFORM_MAP[config.browser[0]];
|
|
41
37
|
}
|
|
42
38
|
return { webSubPlatform };
|
|
43
39
|
};
|
|
44
|
-
|
|
45
40
|
const getClientPayload = (config) => {
|
|
46
41
|
const payload = {
|
|
47
|
-
connectType:
|
|
48
|
-
connectReason:
|
|
49
|
-
userAgent: getUserAgent(config)
|
|
42
|
+
connectType: proto.ClientPayload.ConnectType.WIFI_UNKNOWN,
|
|
43
|
+
connectReason: proto.ClientPayload.ConnectReason.USER_ACTIVATED,
|
|
44
|
+
userAgent: getUserAgent(config)
|
|
50
45
|
};
|
|
51
46
|
payload.webInfo = getWebInfo(config);
|
|
47
|
+
if (config.pushName) {
|
|
48
|
+
payload.pushName = config.pushName;
|
|
49
|
+
}
|
|
52
50
|
return payload;
|
|
53
51
|
};
|
|
54
|
-
|
|
55
|
-
const
|
|
56
|
-
const { user, device } = (0, WABinary_1.jidDecode)(userJid);
|
|
52
|
+
export const generateLoginNode = (userJid, config) => {
|
|
53
|
+
const { user, device } = jidDecode(userJid);
|
|
57
54
|
const payload = {
|
|
58
55
|
...getClientPayload(config),
|
|
59
56
|
passive: true,
|
|
60
57
|
pull: true,
|
|
61
58
|
username: +user,
|
|
62
59
|
device: device,
|
|
60
|
+
// TODO: investigate (hard set as false atm)
|
|
63
61
|
lidDbMigrated: false
|
|
64
62
|
};
|
|
65
|
-
return
|
|
63
|
+
return proto.ClientPayload.fromObject(payload);
|
|
66
64
|
};
|
|
67
|
-
exports.generateLoginNode = generateLoginNode;
|
|
68
|
-
|
|
69
65
|
const getPlatformType = (platform) => {
|
|
70
66
|
const platformType = platform.toUpperCase();
|
|
71
|
-
return
|
|
67
|
+
return (proto.DeviceProps.PlatformType[platformType] ||
|
|
68
|
+
proto.DeviceProps.PlatformType.CHROME);
|
|
72
69
|
};
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
70
|
+
export const generateRegistrationNode = ({ registrationId, signedPreKey, signedIdentityKey }, config) => {
|
|
71
|
+
// the app version needs to be md5 hashed
|
|
72
|
+
// and passed in
|
|
73
|
+
const appVersionBuf = createHash('md5')
|
|
74
|
+
.update(config.version.join('.')) // join as string
|
|
77
75
|
.digest();
|
|
78
|
-
|
|
79
76
|
const companion = {
|
|
80
77
|
os: config.browser[0],
|
|
81
78
|
platformType: getPlatformType(config.browser[1]),
|
|
@@ -103,9 +100,7 @@ const generateRegistrationNode = ({ registrationId, signedPreKey, signedIdentity
|
|
|
103
100
|
tertiary: 7
|
|
104
101
|
}
|
|
105
102
|
};
|
|
106
|
-
|
|
107
|
-
const companionProto = WAProto_1.proto.DeviceProps.encode(companion).finish();
|
|
108
|
-
|
|
103
|
+
const companionProto = proto.DeviceProps.encode(companion).finish();
|
|
109
104
|
const registerPayload = {
|
|
110
105
|
...getClientPayload(config),
|
|
111
106
|
passive: false,
|
|
@@ -113,77 +108,63 @@ const generateRegistrationNode = ({ registrationId, signedPreKey, signedIdentity
|
|
|
113
108
|
devicePairingData: {
|
|
114
109
|
buildHash: appVersionBuf,
|
|
115
110
|
deviceProps: companionProto,
|
|
116
|
-
eRegid:
|
|
117
|
-
eKeytype:
|
|
111
|
+
eRegid: encodeBigEndian(registrationId),
|
|
112
|
+
eKeytype: KEY_BUNDLE_TYPE,
|
|
118
113
|
eIdent: signedIdentityKey.public,
|
|
119
|
-
eSkeyId:
|
|
114
|
+
eSkeyId: encodeBigEndian(signedPreKey.keyId, 3),
|
|
120
115
|
eSkeyVal: signedPreKey.keyPair.public,
|
|
121
|
-
eSkeySig: signedPreKey.signature
|
|
122
|
-
}
|
|
116
|
+
eSkeySig: signedPreKey.signature
|
|
117
|
+
}
|
|
123
118
|
};
|
|
124
|
-
return
|
|
119
|
+
return proto.ClientPayload.fromObject(registerPayload);
|
|
125
120
|
};
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
const configureSuccessfulPairing = (stanza, { advSecretKey, signedIdentityKey, signalIdentities }) => {
|
|
121
|
+
export const configureSuccessfulPairing = (stanza, { advSecretKey, signedIdentityKey, signalIdentities }) => {
|
|
129
122
|
const msgId = stanza.attrs.id;
|
|
130
|
-
const pairSuccessNode =
|
|
131
|
-
const deviceIdentityNode =
|
|
132
|
-
const platformNode =
|
|
133
|
-
const deviceNode =
|
|
134
|
-
const businessNode =
|
|
135
|
-
|
|
123
|
+
const pairSuccessNode = getBinaryNodeChild(stanza, 'pair-success');
|
|
124
|
+
const deviceIdentityNode = getBinaryNodeChild(pairSuccessNode, 'device-identity');
|
|
125
|
+
const platformNode = getBinaryNodeChild(pairSuccessNode, 'platform');
|
|
126
|
+
const deviceNode = getBinaryNodeChild(pairSuccessNode, 'device');
|
|
127
|
+
const businessNode = getBinaryNodeChild(pairSuccessNode, 'biz');
|
|
136
128
|
if (!deviceIdentityNode || !deviceNode) {
|
|
137
|
-
throw new
|
|
129
|
+
throw new Boom('Missing device-identity or device in pair success node', { data: stanza });
|
|
138
130
|
}
|
|
139
|
-
|
|
140
131
|
const bizName = businessNode?.attrs.name;
|
|
141
132
|
const jid = deviceNode.attrs.jid;
|
|
142
133
|
const lid = deviceNode.attrs.lid;
|
|
143
|
-
|
|
144
|
-
const { details, hmac, accountType } = WAProto_1.proto.ADVSignedDeviceIdentityHMAC.decode(deviceIdentityNode.content);
|
|
145
|
-
|
|
134
|
+
const { details, hmac, accountType } = proto.ADVSignedDeviceIdentityHMAC.decode(deviceIdentityNode.content);
|
|
146
135
|
let hmacPrefix = Buffer.from([]);
|
|
147
|
-
if (accountType !== undefined && accountType ===
|
|
148
|
-
hmacPrefix =
|
|
136
|
+
if (accountType !== undefined && accountType === proto.ADVEncryptionType.HOSTED) {
|
|
137
|
+
hmacPrefix = WA_ADV_HOSTED_ACCOUNT_SIG_PREFIX;
|
|
149
138
|
}
|
|
150
|
-
|
|
151
|
-
const advSign = (0, crypto_2.hmacSign)(Buffer.concat([hmacPrefix, details]), Buffer.from(advSecretKey, 'base64'));
|
|
139
|
+
const advSign = hmacSign(Buffer.concat([hmacPrefix, details]), Buffer.from(advSecretKey, 'base64'));
|
|
152
140
|
if (Buffer.compare(hmac, advSign) !== 0) {
|
|
153
|
-
throw new
|
|
141
|
+
throw new Boom('Invalid account signature');
|
|
154
142
|
}
|
|
155
|
-
|
|
156
|
-
const account = WAProto_1.proto.ADVSignedDeviceIdentity.decode(details);
|
|
143
|
+
const account = proto.ADVSignedDeviceIdentity.decode(details);
|
|
157
144
|
const { accountSignatureKey, accountSignature, details: deviceDetails } = account;
|
|
158
|
-
|
|
159
|
-
const
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
? Buffer.from([0x06, 0x05])
|
|
163
|
-
: Buffer.from([0x06, 0x00]);
|
|
145
|
+
const deviceIdentity = proto.ADVDeviceIdentity.decode(deviceDetails);
|
|
146
|
+
const accountSignaturePrefix = deviceIdentity.deviceType === proto.ADVEncryptionType.HOSTED
|
|
147
|
+
? WA_ADV_HOSTED_ACCOUNT_SIG_PREFIX
|
|
148
|
+
: WA_ADV_ACCOUNT_SIG_PREFIX;
|
|
164
149
|
const accountMsg = Buffer.concat([accountSignaturePrefix, deviceDetails, signedIdentityKey.public]);
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
throw new boom_1.Boom('Failed to verify account signature');
|
|
150
|
+
if (!Curve.verify(accountSignatureKey, accountMsg, accountSignature)) {
|
|
151
|
+
throw new Boom('Failed to verify account signature');
|
|
168
152
|
}
|
|
169
|
-
|
|
170
153
|
const deviceMsg = Buffer.concat([
|
|
171
|
-
|
|
154
|
+
WA_ADV_DEVICE_SIG_PREFIX,
|
|
172
155
|
deviceDetails,
|
|
173
156
|
signedIdentityKey.public,
|
|
174
157
|
accountSignatureKey
|
|
175
158
|
]);
|
|
176
|
-
account.deviceSignature =
|
|
177
|
-
|
|
178
|
-
const
|
|
179
|
-
const accountEnc = (0, exports.encodeSignedDeviceIdentity)(account, false);
|
|
180
|
-
|
|
159
|
+
account.deviceSignature = Curve.sign(signedIdentityKey.private, deviceMsg);
|
|
160
|
+
const identity = createSignalIdentity(lid, accountSignatureKey);
|
|
161
|
+
const accountEnc = encodeSignedDeviceIdentity(account, false);
|
|
181
162
|
const reply = {
|
|
182
163
|
tag: 'iq',
|
|
183
164
|
attrs: {
|
|
184
|
-
to:
|
|
165
|
+
to: S_WHATSAPP_NET,
|
|
185
166
|
type: 'result',
|
|
186
|
-
id: msgId
|
|
167
|
+
id: msgId
|
|
187
168
|
},
|
|
188
169
|
content: [
|
|
189
170
|
{
|
|
@@ -199,31 +180,24 @@ const configureSuccessfulPairing = (stanza, { advSecretKey, signedIdentityKey, s
|
|
|
199
180
|
}
|
|
200
181
|
]
|
|
201
182
|
};
|
|
202
|
-
|
|
203
183
|
const authUpdate = {
|
|
204
184
|
account,
|
|
205
185
|
me: { id: jid, name: bizName, lid },
|
|
206
|
-
signalIdentities: [
|
|
207
|
-
...(signalIdentities || []),
|
|
208
|
-
identity
|
|
209
|
-
],
|
|
186
|
+
signalIdentities: [...(signalIdentities || []), identity],
|
|
210
187
|
platform: platformNode?.attrs.name
|
|
211
188
|
};
|
|
212
|
-
|
|
213
189
|
return {
|
|
214
190
|
creds: authUpdate,
|
|
215
191
|
reply
|
|
216
192
|
};
|
|
217
193
|
};
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
const encodeSignedDeviceIdentity = (account, includeSignatureKey) => {
|
|
194
|
+
export const encodeSignedDeviceIdentity = (account, includeSignatureKey) => {
|
|
221
195
|
account = { ...account };
|
|
196
|
+
// set to null if we are not to include the signature key
|
|
197
|
+
// or if we are including the signature key but it is empty
|
|
222
198
|
if (!includeSignatureKey || !account.accountSignatureKey?.length) {
|
|
223
199
|
account.accountSignatureKey = null;
|
|
224
200
|
}
|
|
225
|
-
return
|
|
226
|
-
.encode(account)
|
|
227
|
-
.finish();
|
|
201
|
+
return proto.ADVSignedDeviceIdentity.encode(account).finish();
|
|
228
202
|
};
|
|
229
|
-
|
|
203
|
+
//# sourceMappingURL=validate-connection.js.map
|