@kurtucoben/baileys 0.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.
Files changed (196) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +431 -0
  3. package/WAProto/index.js +169661 -0
  4. package/engine-requirements.js +10 -0
  5. package/lib/Defaults/index.d.ts +53 -0
  6. package/lib/Defaults/index.js +147 -0
  7. package/lib/Defaults/kurtucoben-baileys-version.json +3 -0
  8. package/lib/Defaults/phonenumber-mcc.json +223 -0
  9. package/lib/Signal/Group/ciphertext-message.d.ts +9 -0
  10. package/lib/Signal/Group/ciphertext-message.js +15 -0
  11. package/lib/Signal/Group/group-session-builder.d.ts +14 -0
  12. package/lib/Signal/Group/group-session-builder.js +64 -0
  13. package/lib/Signal/Group/group_cipher.d.ts +17 -0
  14. package/lib/Signal/Group/group_cipher.js +96 -0
  15. package/lib/Signal/Group/index.d.ts +11 -0
  16. package/lib/Signal/Group/index.js +57 -0
  17. package/lib/Signal/Group/keyhelper.d.ts +10 -0
  18. package/lib/Signal/Group/keyhelper.js +55 -0
  19. package/lib/Signal/Group/queue-job.d.ts +1 -0
  20. package/lib/Signal/Group/queue-job.js +57 -0
  21. package/lib/Signal/Group/sender-chain-key.d.ts +13 -0
  22. package/lib/Signal/Group/sender-chain-key.js +34 -0
  23. package/lib/Signal/Group/sender-key-distribution-message.d.ts +16 -0
  24. package/lib/Signal/Group/sender-key-distribution-message.js +66 -0
  25. package/lib/Signal/Group/sender-key-message.d.ts +18 -0
  26. package/lib/Signal/Group/sender-key-message.js +69 -0
  27. package/lib/Signal/Group/sender-key-name.d.ts +17 -0
  28. package/lib/Signal/Group/sender-key-name.js +51 -0
  29. package/lib/Signal/Group/sender-key-record.d.ts +30 -0
  30. package/lib/Signal/Group/sender-key-record.js +53 -0
  31. package/lib/Signal/Group/sender-key-state.d.ts +38 -0
  32. package/lib/Signal/Group/sender-key-state.js +99 -0
  33. package/lib/Signal/Group/sender-message-key.d.ts +11 -0
  34. package/lib/Signal/Group/sender-message-key.js +29 -0
  35. package/lib/Signal/Group/telegram +1 -0
  36. package/lib/Signal/libsignal.d.ts +3 -0
  37. package/lib/Signal/libsignal.js +174 -0
  38. package/lib/Socket/Client/abstract-socket-client.d.ts +17 -0
  39. package/lib/Socket/Client/abstract-socket-client.js +13 -0
  40. package/lib/Socket/Client/index.d.ts +3 -0
  41. package/lib/Socket/Client/index.js +19 -0
  42. package/lib/Socket/Client/mobile-socket-client.d.ts +13 -0
  43. package/lib/Socket/Client/mobile-socket-client.js +65 -0
  44. package/lib/Socket/Client/web-socket-client.d.ts +12 -0
  45. package/lib/Socket/Client/web-socket-client.js +62 -0
  46. package/lib/Socket/business.d.ts +171 -0
  47. package/lib/Socket/business.js +260 -0
  48. package/lib/Socket/chats.d.ts +267 -0
  49. package/lib/Socket/chats.js +970 -0
  50. package/lib/Socket/dugong.js +480 -0
  51. package/lib/Socket/groups.d.ts +115 -0
  52. package/lib/Socket/groups.js +317 -0
  53. package/lib/Socket/index.d.ts +173 -0
  54. package/lib/Socket/index.js +11 -0
  55. package/lib/Socket/messages-recv.d.ts +161 -0
  56. package/lib/Socket/messages-recv.js +1110 -0
  57. package/lib/Socket/messages-send.d.ts +149 -0
  58. package/lib/Socket/messages-send.js +909 -0
  59. package/lib/Socket/newsletter.d.ts +134 -0
  60. package/lib/Socket/newsletter.js +315 -0
  61. package/lib/Socket/registration.d.ts +267 -0
  62. package/lib/Socket/registration.js +166 -0
  63. package/lib/Socket/socket.d.ts +43 -0
  64. package/lib/Socket/socket.js +665 -0
  65. package/lib/Socket/usync.d.ts +36 -0
  66. package/lib/Socket/usync.js +70 -0
  67. package/lib/Store/index.d.ts +3 -0
  68. package/lib/Store/index.js +10 -0
  69. package/lib/Store/make-cache-manager-store.d.ts +13 -0
  70. package/lib/Store/make-cache-manager-store.js +83 -0
  71. package/lib/Store/make-in-memory-store.d.ts +118 -0
  72. package/lib/Store/make-in-memory-store.js +427 -0
  73. package/lib/Store/make-ordered-dictionary.d.ts +13 -0
  74. package/lib/Store/make-ordered-dictionary.js +81 -0
  75. package/lib/Store/object-repository.d.ts +10 -0
  76. package/lib/Store/object-repository.js +27 -0
  77. package/lib/Types/Auth.d.ts +110 -0
  78. package/lib/Types/Auth.js +2 -0
  79. package/lib/Types/Call.d.ts +13 -0
  80. package/lib/Types/Call.js +2 -0
  81. package/lib/Types/Chat.d.ts +102 -0
  82. package/lib/Types/Chat.js +4 -0
  83. package/lib/Types/Contact.d.ts +19 -0
  84. package/lib/Types/Contact.js +2 -0
  85. package/lib/Types/Events.d.ts +157 -0
  86. package/lib/Types/Events.js +2 -0
  87. package/lib/Types/GroupMetadata.d.ts +55 -0
  88. package/lib/Types/GroupMetadata.js +2 -0
  89. package/lib/Types/Label.d.ts +35 -0
  90. package/lib/Types/Label.js +27 -0
  91. package/lib/Types/LabelAssociation.d.ts +29 -0
  92. package/lib/Types/LabelAssociation.js +9 -0
  93. package/lib/Types/Message.d.ts +273 -0
  94. package/lib/Types/Message.js +9 -0
  95. package/lib/Types/Newsletter.d.ts +103 -0
  96. package/lib/Types/Newsletter.js +38 -0
  97. package/lib/Types/Product.d.ts +78 -0
  98. package/lib/Types/Product.js +2 -0
  99. package/lib/Types/Signal.d.ts +57 -0
  100. package/lib/Types/Signal.js +2 -0
  101. package/lib/Types/Socket.d.ts +111 -0
  102. package/lib/Types/Socket.js +2 -0
  103. package/lib/Types/State.d.ts +27 -0
  104. package/lib/Types/State.js +2 -0
  105. package/lib/Types/USync.d.ts +25 -0
  106. package/lib/Types/USync.js +2 -0
  107. package/lib/Types/index.d.ts +57 -0
  108. package/lib/Types/index.js +42 -0
  109. package/lib/Types/telegram +1 -0
  110. package/lib/Utils/auth-utils.d.ts +18 -0
  111. package/lib/Utils/auth-utils.js +206 -0
  112. package/lib/Utils/baileys-event-stream.d.ts +16 -0
  113. package/lib/Utils/baileys-event-stream.js +63 -0
  114. package/lib/Utils/business.d.ts +22 -0
  115. package/lib/Utils/business.js +234 -0
  116. package/lib/Utils/chat-utils.d.ts +71 -0
  117. package/lib/Utils/chat-utils.js +729 -0
  118. package/lib/Utils/crypto.d.ts +41 -0
  119. package/lib/Utils/crypto.js +151 -0
  120. package/lib/Utils/decode-wa-message.d.ts +19 -0
  121. package/lib/Utils/decode-wa-message.js +198 -0
  122. package/lib/Utils/event-buffer.d.ts +35 -0
  123. package/lib/Utils/event-buffer.js +514 -0
  124. package/lib/Utils/generics.d.ts +92 -0
  125. package/lib/Utils/generics.js +423 -0
  126. package/lib/Utils/history.d.ts +15 -0
  127. package/lib/Utils/history.js +96 -0
  128. package/lib/Utils/index.d.ts +17 -0
  129. package/lib/Utils/index.js +33 -0
  130. package/lib/Utils/kurtucoben-connection.d.ts +11 -0
  131. package/lib/Utils/kurtucoben-connection.js +229 -0
  132. package/lib/Utils/link-preview.d.ts +21 -0
  133. package/lib/Utils/link-preview.js +93 -0
  134. package/lib/Utils/logger.d.ts +4 -0
  135. package/lib/Utils/logger.js +7 -0
  136. package/lib/Utils/lt-hash.d.ts +12 -0
  137. package/lib/Utils/lt-hash.js +51 -0
  138. package/lib/Utils/make-mutex.d.ts +7 -0
  139. package/lib/Utils/make-mutex.js +43 -0
  140. package/lib/Utils/messages-media.d.ts +116 -0
  141. package/lib/Utils/messages-media.js +819 -0
  142. package/lib/Utils/messages.d.ts +77 -0
  143. package/lib/Utils/messages.js +816 -0
  144. package/lib/Utils/noise-handler.d.ts +21 -0
  145. package/lib/Utils/noise-handler.js +155 -0
  146. package/lib/Utils/process-message.d.ts +41 -0
  147. package/lib/Utils/process-message.js +321 -0
  148. package/lib/Utils/signal.d.ts +32 -0
  149. package/lib/Utils/signal.js +153 -0
  150. package/lib/Utils/use-multi-file-auth-state.d.ts +13 -0
  151. package/lib/Utils/use-multi-file-auth-state.js +119 -0
  152. package/lib/WABinary/constants.d.ts +30 -0
  153. package/lib/WABinary/constants.js +40 -0
  154. package/lib/WABinary/decode.d.ts +7 -0
  155. package/lib/WABinary/decode.js +252 -0
  156. package/lib/WABinary/encode.d.ts +3 -0
  157. package/lib/WABinary/encode.js +265 -0
  158. package/lib/WABinary/generic-utils.d.ts +17 -0
  159. package/lib/WABinary/generic-utils.js +198 -0
  160. package/lib/WABinary/index.d.ts +5 -0
  161. package/lib/WABinary/index.js +21 -0
  162. package/lib/WABinary/jid-utils.d.ts +31 -0
  163. package/lib/WABinary/jid-utils.js +62 -0
  164. package/lib/WABinary/types.d.ts +18 -0
  165. package/lib/WABinary/types.js +2 -0
  166. package/lib/WAM/BinaryInfo.d.ts +17 -0
  167. package/lib/WAM/BinaryInfo.js +13 -0
  168. package/lib/WAM/constants.d.ts +38 -0
  169. package/lib/WAM/constants.js +15350 -0
  170. package/lib/WAM/encode.d.ts +3 -0
  171. package/lib/WAM/encode.js +155 -0
  172. package/lib/WAM/index.d.ts +3 -0
  173. package/lib/WAM/index.js +19 -0
  174. package/lib/WAUSync/Protocols/USyncContactProtocol.d.ts +9 -0
  175. package/lib/WAUSync/Protocols/USyncContactProtocol.js +31 -0
  176. package/lib/WAUSync/Protocols/USyncDeviceProtocol.d.ts +22 -0
  177. package/lib/WAUSync/Protocols/USyncDeviceProtocol.js +57 -0
  178. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.d.ts +12 -0
  179. package/lib/WAUSync/Protocols/USyncDisappearingModeProtocol.js +30 -0
  180. package/lib/WAUSync/Protocols/USyncStatusProtocol.d.ts +12 -0
  181. package/lib/WAUSync/Protocols/USyncStatusProtocol.js +42 -0
  182. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.d.ts +25 -0
  183. package/lib/WAUSync/Protocols/UsyncBotProfileProtocol.js +53 -0
  184. package/lib/WAUSync/Protocols/UsyncLIDProtocol.d.ts +8 -0
  185. package/lib/WAUSync/Protocols/UsyncLIDProtocol.js +24 -0
  186. package/lib/WAUSync/Protocols/index.d.ts +4 -0
  187. package/lib/WAUSync/Protocols/index.js +20 -0
  188. package/lib/WAUSync/USyncQuery.d.ts +28 -0
  189. package/lib/WAUSync/USyncQuery.js +89 -0
  190. package/lib/WAUSync/USyncUser.d.ts +12 -0
  191. package/lib/WAUSync/USyncUser.js +26 -0
  192. package/lib/WAUSync/index.d.ts +3 -0
  193. package/lib/WAUSync/index.js +19 -0
  194. package/lib/index.d.ts +12 -0
  195. package/lib/index.js +42 -0
  196. package/package.json +113 -0
@@ -0,0 +1,229 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.encodeSignedDeviceIdentity = exports.configureSuccessfulPairing = exports.generateRegistrationNode = exports.generateLoginNode = void 0;
4
+ const boom_1 = require("@hapi/boom");
5
+ const crypto_1 = require("crypto");
6
+ const WAProto_1 = require("../../WAProto");
7
+ const Defaults_1 = require("../Defaults");
8
+ const WABinary_1 = require("../WABinary");
9
+ const crypto_2 = require("./crypto");
10
+ const generics_1 = require("./generics");
11
+ const signal_1 = require("./signal");
12
+
13
+ const getUserAgent = (config) => {
14
+ return {
15
+ appVersion: {
16
+ primary: config.version[0],
17
+ secondary: config.version[1],
18
+ tertiary: config.version[2],
19
+ },
20
+ platform: WAProto_1.proto.ClientPayload.UserAgent.Platform.WEB,
21
+ releaseChannel: WAProto_1.proto.ClientPayload.UserAgent.ReleaseChannel.RELEASE,
22
+ osVersion: '0.1',
23
+ device: 'Desktop',
24
+ osBuildNumber: '0.1',
25
+ localeLanguageIso6391: 'en',
26
+ mnc: '000',
27
+ mcc: '000',
28
+ localeCountryIso31661Alpha2: config.countryCode || 'US'
29
+ };
30
+ };
31
+
32
+ const PLATFORM_MAP = {
33
+ 'Mac OS': WAProto_1.proto.ClientPayload.WebInfo.WebSubPlatform.DARWIN,
34
+ 'Windows': WAProto_1.proto.ClientPayload.WebInfo.WebSubPlatform.WIN32
35
+ };
36
+
37
+ const getWebInfo = (config) => {
38
+ let webSubPlatform = WAProto_1.proto.ClientPayload.WebInfo.WebSubPlatform.WEB_BROWSER;
39
+ if (config.syncFullHistory && PLATFORM_MAP[config.browser[0]] && config.browser[1] === 'Desktop') {
40
+ webSubPlatform = PLATFORM_MAP[config.browser[0]];
41
+ }
42
+ return { webSubPlatform };
43
+ };
44
+
45
+ const getClientPayload = (config) => {
46
+ const payload = {
47
+ connectType: WAProto_1.proto.ClientPayload.ConnectType.WIFI_UNKNOWN,
48
+ connectReason: WAProto_1.proto.ClientPayload.ConnectReason.USER_ACTIVATED,
49
+ userAgent: getUserAgent(config),
50
+ };
51
+ payload.webInfo = getWebInfo(config);
52
+ return payload;
53
+ };
54
+
55
+ const generateLoginNode = (userJid, config) => {
56
+ const { user, device } = (0, WABinary_1.jidDecode)(userJid);
57
+ const payload = {
58
+ ...getClientPayload(config),
59
+ passive: true,
60
+ pull: true,
61
+ username: +user,
62
+ device: device,
63
+ lidDbMigrated: false
64
+ };
65
+ return WAProto_1.proto.ClientPayload.fromObject(payload);
66
+ };
67
+ exports.generateLoginNode = generateLoginNode;
68
+
69
+ const getPlatformType = (platform) => {
70
+ const platformType = platform.toUpperCase();
71
+ return WAProto_1.proto.DeviceProps.PlatformType[platformType] || WAProto_1.proto.DeviceProps.PlatformType.CHROME;
72
+ };
73
+
74
+ const generateRegistrationNode = ({ registrationId, signedPreKey, signedIdentityKey }, config) => {
75
+ const appVersionBuf = (0, crypto_1.createHash)('md5')
76
+ .update(config.version.join('.'))
77
+ .digest();
78
+
79
+ const companion = {
80
+ os: config.browser[0],
81
+ platformType: getPlatformType(config.browser[1]),
82
+ requireFullSync: config.syncFullHistory,
83
+ historySyncConfig: {
84
+ storageQuotaMb: 10240,
85
+ inlineInitialPayloadInE2EeMsg: true,
86
+ recentSyncDaysLimit: undefined,
87
+ supportCallLogHistory: false,
88
+ supportBotUserAgentChatHistory: true,
89
+ supportCagReactionsAndPolls: true,
90
+ supportBizHostedMsg: true,
91
+ supportRecentSyncChunkMessageCountTuning: true,
92
+ supportHostedGroupMsg: true,
93
+ supportFbidBotChatHistory: true,
94
+ supportAddOnHistorySyncMigration: undefined,
95
+ supportMessageAssociation: true,
96
+ supportGroupHistory: false,
97
+ onDemandReady: undefined,
98
+ supportGuestChat: undefined
99
+ },
100
+ version: {
101
+ primary: 10,
102
+ secondary: 15,
103
+ tertiary: 7
104
+ }
105
+ };
106
+
107
+ const companionProto = WAProto_1.proto.DeviceProps.encode(companion).finish();
108
+
109
+ const registerPayload = {
110
+ ...getClientPayload(config),
111
+ passive: false,
112
+ pull: false,
113
+ devicePairingData: {
114
+ buildHash: appVersionBuf,
115
+ deviceProps: companionProto,
116
+ eRegid: (0, generics_1.encodeBigEndian)(registrationId),
117
+ eKeytype: Defaults_1.KEY_BUNDLE_TYPE,
118
+ eIdent: signedIdentityKey.public,
119
+ eSkeyId: (0, generics_1.encodeBigEndian)(signedPreKey.keyId, 3),
120
+ eSkeyVal: signedPreKey.keyPair.public,
121
+ eSkeySig: signedPreKey.signature,
122
+ },
123
+ };
124
+ return WAProto_1.proto.ClientPayload.fromObject(registerPayload);
125
+ };
126
+ exports.generateRegistrationNode = generateRegistrationNode;
127
+
128
+ const configureSuccessfulPairing = (stanza, { advSecretKey, signedIdentityKey, signalIdentities }) => {
129
+ const msgId = stanza.attrs.id;
130
+ const pairSuccessNode = (0, WABinary_1.getBinaryNodeChild)(stanza, 'pair-success');
131
+ const deviceIdentityNode = (0, WABinary_1.getBinaryNodeChild)(pairSuccessNode, 'device-identity');
132
+ const platformNode = (0, WABinary_1.getBinaryNodeChild)(pairSuccessNode, 'platform');
133
+ const deviceNode = (0, WABinary_1.getBinaryNodeChild)(pairSuccessNode, 'device');
134
+ const businessNode = (0, WABinary_1.getBinaryNodeChild)(pairSuccessNode, 'biz');
135
+
136
+ if (!deviceIdentityNode || !deviceNode) {
137
+ throw new boom_1.Boom('Missing device-identity or device in pair success node', { data: stanza });
138
+ }
139
+
140
+ const bizName = businessNode?.attrs.name;
141
+ const jid = deviceNode.attrs.jid;
142
+ const lid = deviceNode.attrs.lid;
143
+
144
+ const { details, hmac, accountType } = WAProto_1.proto.ADVSignedDeviceIdentityHMAC.decode(deviceIdentityNode.content);
145
+
146
+ let hmacPrefix = Buffer.from([]);
147
+ if (accountType !== undefined && accountType === WAProto_1.proto.ADVEncryptionType.HOSTED) {
148
+ hmacPrefix = Buffer.from([0x06, 0x05]);
149
+ }
150
+
151
+ const advSign = (0, crypto_2.hmacSign)(Buffer.concat([hmacPrefix, details]), Buffer.from(advSecretKey, 'base64'));
152
+ if (Buffer.compare(hmac, advSign) !== 0) {
153
+ throw new boom_1.Boom('Invalid account signature');
154
+ }
155
+
156
+ const account = WAProto_1.proto.ADVSignedDeviceIdentity.decode(details);
157
+ const { accountSignatureKey, accountSignature, details: deviceDetails } = account;
158
+
159
+ const deviceIdentity = WAProto_1.proto.ADVDeviceIdentity.decode(deviceDetails);
160
+
161
+ const accountSignaturePrefix = deviceIdentity.deviceType === WAProto_1.proto.ADVEncryptionType.HOSTED
162
+ ? Buffer.from([0x06, 0x05])
163
+ : Buffer.from([0x06, 0x00]);
164
+ const accountMsg = Buffer.concat([accountSignaturePrefix, deviceDetails, signedIdentityKey.public]);
165
+
166
+ if (!crypto_2.Curve.verify(accountSignatureKey, accountMsg, accountSignature)) {
167
+ throw new boom_1.Boom('Failed to verify account signature');
168
+ }
169
+
170
+ const deviceMsg = Buffer.concat([
171
+ Buffer.from([0x06, 0x01]),
172
+ deviceDetails,
173
+ signedIdentityKey.public,
174
+ accountSignatureKey
175
+ ]);
176
+ account.deviceSignature = crypto_2.Curve.sign(signedIdentityKey.private, deviceMsg);
177
+
178
+ const identity = (0, signal_1.createSignalIdentity)(jid, accountSignatureKey);
179
+ const accountEnc = (0, exports.encodeSignedDeviceIdentity)(account, false);
180
+
181
+ const reply = {
182
+ tag: 'iq',
183
+ attrs: {
184
+ to: WABinary_1.S_WHATSAPP_NET,
185
+ type: 'result',
186
+ id: msgId,
187
+ },
188
+ content: [
189
+ {
190
+ tag: 'pair-device-sign',
191
+ attrs: {},
192
+ content: [
193
+ {
194
+ tag: 'device-identity',
195
+ attrs: { 'key-index': deviceIdentity.keyIndex.toString() },
196
+ content: accountEnc
197
+ }
198
+ ]
199
+ }
200
+ ]
201
+ };
202
+
203
+ const authUpdate = {
204
+ account,
205
+ me: { id: jid, name: bizName, lid },
206
+ signalIdentities: [
207
+ ...(signalIdentities || []),
208
+ identity
209
+ ],
210
+ platform: platformNode?.attrs.name
211
+ };
212
+
213
+ return {
214
+ creds: authUpdate,
215
+ reply
216
+ };
217
+ };
218
+ exports.configureSuccessfulPairing = configureSuccessfulPairing;
219
+
220
+ const encodeSignedDeviceIdentity = (account, includeSignatureKey) => {
221
+ account = { ...account };
222
+ if (!includeSignatureKey || !account.accountSignatureKey?.length) {
223
+ account.accountSignatureKey = null;
224
+ }
225
+ return WAProto_1.proto.ADVSignedDeviceIdentity
226
+ .encode(account)
227
+ .finish();
228
+ };
229
+ exports.encodeSignedDeviceIdentity = encodeSignedDeviceIdentity;
@@ -0,0 +1,21 @@
1
+ import { AxiosRequestConfig } from 'axios';
2
+ import { Logger } from 'pino';
3
+ import { WAMediaUploadFunction, WAUrlInfo } from '../Types';
4
+ export type URLGenerationOptions = {
5
+ thumbnailWidth: number;
6
+ fetchOpts: {
7
+ /** Timeout in ms */
8
+ timeout: number;
9
+ proxyUrl?: string;
10
+ headers?: AxiosRequestConfig<{}>['headers'];
11
+ };
12
+ uploadImage?: WAMediaUploadFunction;
13
+ logger?: Logger;
14
+ };
15
+ /**
16
+ * Given a piece of text, checks for any URL present, generates link preview for the same and returns it
17
+ * Return undefined if the fetch failed or no URL was found
18
+ * @param text first matched URL in text
19
+ * @returns the URL info required to generate link preview
20
+ */
21
+ export declare const getUrlInfo: (text: string, opts?: URLGenerationOptions) => Promise<WAUrlInfo | undefined>;
@@ -0,0 +1,93 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getUrlInfo = void 0;
4
+ const messages_1 = require("./messages");
5
+ const messages_media_1 = require("./messages-media");
6
+ const THUMBNAIL_WIDTH_PX = 192;
7
+ /** Fetches an image and generates a thumbnail for it */
8
+ const getCompressedJpegThumbnail = async (url, { thumbnailWidth, fetchOpts }) => {
9
+ const stream = await (0, messages_media_1.getHttpStream)(url, fetchOpts);
10
+ const result = await (0, messages_media_1.extractImageThumb)(stream, thumbnailWidth);
11
+ return result;
12
+ };
13
+ /**
14
+ * Given a piece of text, checks for any URL present, generates link preview for the same and returns it
15
+ * Return undefined if the fetch failed or no URL was found
16
+ * @param text first matched URL in text
17
+ * @returns the URL info required to generate link preview
18
+ */
19
+ const getUrlInfo = async (text, opts = {
20
+ thumbnailWidth: THUMBNAIL_WIDTH_PX,
21
+ fetchOpts: { timeout: 3000 }
22
+ }) => {
23
+ var _a;
24
+ try {
25
+ // retries
26
+ const retries = 0;
27
+ const maxRetry = 5;
28
+ const { getLinkPreview } = await import('link-preview-js');
29
+ let previewLink = text;
30
+ if (!text.startsWith('https://') && !text.startsWith('http://')) {
31
+ previewLink = 'https://' + previewLink;
32
+ }
33
+ const info = await getLinkPreview(previewLink, {
34
+ ...opts.fetchOpts,
35
+ followRedirects: 'follow',
36
+ handleRedirects: (baseURL, forwardedURL) => {
37
+ const urlObj = new URL(baseURL);
38
+ const forwardedURLObj = new URL(forwardedURL);
39
+ if (retries >= maxRetry) {
40
+ return false;
41
+ }
42
+ if (forwardedURLObj.hostname === urlObj.hostname
43
+ || forwardedURLObj.hostname === 'www.' + urlObj.hostname
44
+ || 'www.' + forwardedURLObj.hostname === urlObj.hostname) {
45
+ retries + 1;
46
+ return true;
47
+ }
48
+ else {
49
+ return false;
50
+ }
51
+ },
52
+ headers: opts.fetchOpts
53
+ });
54
+ if (info && 'title' in info && info.title) {
55
+ const [image] = info.images;
56
+ const urlInfo = {
57
+ 'canonical-url': info.url,
58
+ 'matched-text': text,
59
+ title: info.title,
60
+ description: info.description,
61
+ originalThumbnailUrl: image
62
+ };
63
+ if (opts.uploadImage) {
64
+ const { imageMessage } = await (0, messages_1.prepareWAMessageMedia)({ image: { url: image } }, {
65
+ upload: opts.uploadImage,
66
+ mediaTypeOverride: 'thumbnail-link',
67
+ options: opts.fetchOpts
68
+ });
69
+ urlInfo.jpegThumbnail = (imageMessage === null || imageMessage === void 0 ? void 0 : imageMessage.jpegThumbnail)
70
+ ? Buffer.from(imageMessage.jpegThumbnail)
71
+ : undefined;
72
+ urlInfo.highQualityThumbnail = imageMessage || undefined;
73
+ }
74
+ else {
75
+ try {
76
+ urlInfo.jpegThumbnail = image
77
+ ? (await getCompressedJpegThumbnail(image, opts)).buffer
78
+ : undefined;
79
+ }
80
+ catch (error) {
81
+ (_a = opts.logger) === null || _a === void 0 ? void 0 : _a.debug({ err: error.stack, url: previewLink }, 'error in generating thumbnail');
82
+ }
83
+ }
84
+ return urlInfo;
85
+ }
86
+ }
87
+ catch (error) {
88
+ if (!error.message.includes('receive a valid')) {
89
+ throw error;
90
+ }
91
+ }
92
+ };
93
+ exports.getUrlInfo = getUrlInfo;
@@ -0,0 +1,4 @@
1
+ declare const _default: import("pino").Logger<{
2
+ timestamp: () => string;
3
+ }>;
4
+ export default _default;
@@ -0,0 +1,7 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const pino_1 = __importDefault(require("pino"));
7
+ exports.default = (0, pino_1.default)({ timestamp: () => `,"time":"${new Date().toJSON()}"` });
@@ -0,0 +1,12 @@
1
+ declare class d {
2
+ salt: string;
3
+ constructor(e: string);
4
+ add(e: any, t: any): any;
5
+ subtract(e: any, t: any): any;
6
+ subtractThenAdd(e: any, t: any, r: any): any;
7
+ _addSingle(e: any, t: any): ArrayBuffer;
8
+ _subtractSingle(e: any, t: any): ArrayBuffer;
9
+ performPointwiseWithOverflow(e: any, t: any, r: any): ArrayBuffer;
10
+ }
11
+ export declare const LT_HASH_ANTI_TAMPERING: d;
12
+ export {};
@@ -0,0 +1,51 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.LT_HASH_ANTI_TAMPERING = void 0;
4
+ const crypto_1 = require("./crypto");
5
+ /**
6
+ * LT Hash is a summation based hash algorithm that maintains the integrity of a piece of data
7
+ * over a series of mutations. You can add/remove mutations and it'll return a hash equal to
8
+ * if the same series of mutations was made sequentially.
9
+ */
10
+ const o = 128;
11
+ class d {
12
+ constructor(e) {
13
+ this.salt = e;
14
+ }
15
+ add(e, t) {
16
+ var r = this;
17
+ for (const item of t) {
18
+ e = r._addSingle(e, item);
19
+ }
20
+ return e;
21
+ }
22
+ subtract(e, t) {
23
+ var r = this;
24
+ for (const item of t) {
25
+ e = r._subtractSingle(e, item);
26
+ }
27
+ return e;
28
+ }
29
+ subtractThenAdd(e, t, r) {
30
+ var n = this;
31
+ return n.add(n.subtract(e, r), t);
32
+ }
33
+ _addSingle(e, t) {
34
+ var r = this;
35
+ const n = new Uint8Array((0, crypto_1.hkdf)(Buffer.from(t), o, { info: r.salt })).buffer;
36
+ return r.performPointwiseWithOverflow(e, n, ((e, t) => e + t));
37
+ }
38
+ _subtractSingle(e, t) {
39
+ var r = this;
40
+ const n = new Uint8Array((0, crypto_1.hkdf)(Buffer.from(t), o, { info: r.salt })).buffer;
41
+ return r.performPointwiseWithOverflow(e, n, ((e, t) => e - t));
42
+ }
43
+ performPointwiseWithOverflow(e, t, r) {
44
+ const n = new DataView(e), i = new DataView(t), a = new ArrayBuffer(n.byteLength), s = new DataView(a);
45
+ for (let e = 0; e < n.byteLength; e += 2) {
46
+ s.setUint16(e, r(n.getUint16(e, !0), i.getUint16(e, !0)), !0);
47
+ }
48
+ return a;
49
+ }
50
+ }
51
+ exports.LT_HASH_ANTI_TAMPERING = new d('WhatsApp Patch Integrity');
@@ -0,0 +1,7 @@
1
+ export declare const makeMutex: () => {
2
+ mutex<T>(code: () => T | Promise<T>): Promise<T>;
3
+ };
4
+ export type Mutex = ReturnType<typeof makeMutex>;
5
+ export declare const makeKeyedMutex: () => {
6
+ mutex<T>(key: string, task: () => T | Promise<T>): Promise<T>;
7
+ };
@@ -0,0 +1,43 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.makeKeyedMutex = exports.makeMutex = void 0;
4
+ const makeMutex = () => {
5
+ let task = Promise.resolve();
6
+ let taskTimeout;
7
+ return {
8
+ mutex(code) {
9
+ task = (async () => {
10
+ // wait for the previous task to complete
11
+ // if there is an error, we swallow so as to not block the queue
12
+ try {
13
+ await task;
14
+ }
15
+ catch (_a) { }
16
+ try {
17
+ // execute the current task
18
+ const result = await code();
19
+ return result;
20
+ }
21
+ finally {
22
+ clearTimeout(taskTimeout);
23
+ }
24
+ })();
25
+ // we replace the existing task, appending the new piece of execution to it
26
+ // so the next task will have to wait for this one to finish
27
+ return task;
28
+ },
29
+ };
30
+ };
31
+ exports.makeMutex = makeMutex;
32
+ const makeKeyedMutex = () => {
33
+ const map = {};
34
+ return {
35
+ mutex(key, task) {
36
+ if (!map[key]) {
37
+ map[key] = (0, exports.makeMutex)();
38
+ }
39
+ return map[key].mutex(task);
40
+ }
41
+ };
42
+ };
43
+ exports.makeKeyedMutex = makeKeyedMutex;
@@ -0,0 +1,116 @@
1
+ /// <reference types="node" />
2
+ /// <reference types="node" />
3
+ /// <reference types="node" />
4
+ import { Boom } from '@hapi/boom';
5
+ import { AxiosRequestConfig } from 'axios';
6
+ import type { Logger } from 'pino';
7
+ import { Readable, Transform } from 'stream';
8
+ import { URL } from 'url';
9
+ import { proto } from '../../WAProto';
10
+ import { DownloadableMessage, MediaConnInfo, MediaDecryptionKeyInfo, MediaType, SocketConfig, WAMediaUpload, WAMediaUploadFunction, WAMessageContent } from '../Types';
11
+ import { BinaryNode } from '../WABinary';
12
+ export declare const hkdfInfoKey: (type: MediaType) => string;
13
+ /** generates all the keys required to encrypt/decrypt & sign a media message */
14
+ export declare function getMediaKeys(buffer: Uint8Array | string | null | undefined, mediaType: MediaType): MediaDecryptionKeyInfo;
15
+ export declare const extractImageThumb: (bufferOrFilePath: Readable | Buffer | string, width?: number) => Promise<{
16
+ buffer: Buffer;
17
+ original: {
18
+ width: number | undefined;
19
+ height: number | undefined;
20
+ };
21
+ }>;
22
+ export declare const encodeBase64EncodedStringForUpload: (b64: string) => string;
23
+ export declare const generateProfilePicture: (mediaUpload: WAMediaUpload) => Promise<{
24
+ img: Buffer;
25
+ }>;
26
+ /** gets the SHA256 of the given media message */
27
+ export declare const mediaMessageSHA256B64: (message: WAMessageContent) => string | null | undefined;
28
+ export declare function getAudioDuration(buffer: Buffer | string | Readable): Promise<number | undefined>;
29
+ /**
30
+ referenced from and modifying https://github.com/wppconnect-team/wa-js/blob/main/src/chat/functions/prepareAudioWaveform.ts
31
+ */
32
+ export declare function getAudioWaveform(buffer: Buffer | string | Readable, logger?: Logger): Promise<Uint8Array | undefined>;
33
+ export declare const toReadable: (buffer: Buffer) => Readable;
34
+ export declare const toBuffer: (stream: Readable) => Promise<Buffer>;
35
+ export declare const getStream: (item: WAMediaUpload, opts?: AxiosRequestConfig) => Promise<{
36
+ readonly stream: Readable;
37
+ readonly type: "buffer";
38
+ } | {
39
+ readonly stream: Readable;
40
+ readonly type: "readable";
41
+ } | {
42
+ readonly stream: Readable;
43
+ readonly type: "remote";
44
+ } | {
45
+ readonly stream: import("fs").ReadStream;
46
+ readonly type: "file";
47
+ }>;
48
+ /** generates a thumbnail for a given media, if required */
49
+ export declare function generateThumbnail(file: string, mediaType: 'video' | 'image', options: {
50
+ logger?: Logger;
51
+ }): Promise<{
52
+ thumbnail: string | undefined;
53
+ originalImageDimensions: {
54
+ width: number;
55
+ height: number;
56
+ } | undefined;
57
+ }>;
58
+ export declare const getHttpStream: (url: string | URL, options?: AxiosRequestConfig & {
59
+ isStream?: true;
60
+ }) => Promise<Readable>;
61
+ type EncryptedStreamOptions = {
62
+ saveOriginalFileIfRequired?: boolean;
63
+ logger?: Logger;
64
+ opts?: AxiosRequestConfig;
65
+ };
66
+ export declare const prepareStream: (media: WAMediaUpload, mediaType: MediaType, { logger, saveOriginalFileIfRequired, opts }?: EncryptedStreamOptions) => Promise<{
67
+ mediaKey: undefined;
68
+ encWriteStream: Buffer;
69
+ fileLength: number;
70
+ fileSha256: Buffer;
71
+ fileEncSha256: undefined;
72
+ bodyPath: string | undefined;
73
+ didSaveToTmpPath: boolean;
74
+ }>;
75
+ export declare const encryptedStream: (media: WAMediaUpload, mediaType: MediaType, { logger, saveOriginalFileIfRequired, opts }?: EncryptedStreamOptions) => Promise<{
76
+ mediaKey: Buffer;
77
+ encWriteStream: Readable;
78
+ bodyPath: string | undefined;
79
+ mac: Buffer;
80
+ fileEncSha256: Buffer;
81
+ fileSha256: Buffer;
82
+ fileLength: number;
83
+ didSaveToTmpPath: boolean;
84
+ }>;
85
+ export type MediaDownloadOptions = {
86
+ startByte?: number;
87
+ endByte?: number;
88
+ options?: AxiosRequestConfig<any>;
89
+ };
90
+ export declare const getUrlFromDirectPath: (directPath: string) => string;
91
+ export declare const downloadContentFromMessage: ({ mediaKey, directPath, url }: DownloadableMessage, type: MediaType, opts?: MediaDownloadOptions) => Promise<Transform>;
92
+ /**
93
+ * Decrypts and downloads an AES256-CBC encrypted file given the keys.
94
+ * Assumes the SHA256 of the plaintext is appended to the end of the ciphertext
95
+ * */
96
+ export declare const downloadEncryptedContent: (downloadUrl: string, { cipherKey, iv }: MediaDecryptionKeyInfo, { startByte, endByte, options }?: MediaDownloadOptions) => Promise<Transform>;
97
+ export declare function extensionForMediaMessage(message: WAMessageContent): string;
98
+ export declare const getWAUploadToServer: ({ customUploadHosts, fetchAgent, logger, options }: SocketConfig, refreshMediaConn: (force: boolean) => Promise<MediaConnInfo>) => WAMediaUploadFunction;
99
+ /**
100
+ * Generate a binary node that will request the phone to re-upload the media & return the newly uploaded URL
101
+ */
102
+ export declare const encryptMediaRetryRequest: (key: proto.IMessageKey, mediaKey: Buffer | Uint8Array, meId: string) => BinaryNode;
103
+ export declare const decodeMediaRetryNode: (node: BinaryNode) => {
104
+ key: proto.IMessageKey;
105
+ media?: {
106
+ ciphertext: Uint8Array;
107
+ iv: Uint8Array;
108
+ } | undefined;
109
+ error?: Boom<any> | undefined;
110
+ };
111
+ export declare const decryptMediaRetryData: ({ ciphertext, iv }: {
112
+ ciphertext: Uint8Array;
113
+ iv: Uint8Array;
114
+ }, mediaKey: Uint8Array, msgId: string) => proto.MediaRetryNotification;
115
+ export declare const getStatusCodeForMediaRetry: (code: number) => any;
116
+ export {};