tempmail-sdk 1.1.8 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,341 @@
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
+ exports.generateEmail = generateEmail;
7
+ exports.getEmails = getEmails;
8
+ const ws_1 = __importDefault(require("ws"));
9
+ const normalize_1 = require("../normalize");
10
+ const CHANNEL = 'tempmail-cn';
11
+ const DEFAULT_HOST = 'tempmail.cn';
12
+ const CONNECT_TIMEOUT_MS = 15000;
13
+ const HANDSHAKE_WAIT_MS = 1000;
14
+ const INITIAL_SYNC_WAIT_MS = 80;
15
+ const SOCKET_IO_VERSIONS = [4, 3];
16
+ const DEFAULT_HEADERS = {
17
+ 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/146.0.0.0 Safari/537.36 Edg/146.0.0.0',
18
+ 'Accept-Language': 'zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6',
19
+ 'Cache-Control': 'no-cache',
20
+ DNT: '1',
21
+ Pragma: 'no-cache',
22
+ };
23
+ const mailboxes = new Map();
24
+ function getState(email) {
25
+ const key = email.trim().toLowerCase();
26
+ let st = mailboxes.get(key);
27
+ if (!st) {
28
+ st = { emails: [], seenIds: new Set(), ws: null };
29
+ mailboxes.set(key, st);
30
+ }
31
+ return st;
32
+ }
33
+ function sleep(ms) {
34
+ return new Promise((resolve) => setTimeout(resolve, ms));
35
+ }
36
+ function normalizeHost(domain) {
37
+ const raw = String(domain || '').trim();
38
+ if (!raw) {
39
+ return DEFAULT_HOST;
40
+ }
41
+ let host = raw;
42
+ if (/^https?:\/\//i.test(host)) {
43
+ try {
44
+ host = new URL(host).host;
45
+ }
46
+ catch {
47
+ host = raw;
48
+ }
49
+ }
50
+ else if (host.includes('/')) {
51
+ try {
52
+ host = new URL(`https://${host}`).host;
53
+ }
54
+ catch {
55
+ host = host.split('/')[0];
56
+ }
57
+ }
58
+ host = host.replace(/^\.+|\.+$/g, '');
59
+ const at = host.lastIndexOf('@');
60
+ if (at >= 0) {
61
+ host = host.slice(at + 1);
62
+ }
63
+ return host || DEFAULT_HOST;
64
+ }
65
+ function splitAddress(email) {
66
+ const trimmed = email.trim();
67
+ const at = trimmed.indexOf('@');
68
+ if (at <= 0 || at === trimmed.length - 1) {
69
+ throw new Error('tempmail-cn: invalid email address');
70
+ }
71
+ return {
72
+ local: trimmed.slice(0, at),
73
+ host: normalizeHost(trimmed.slice(at + 1)),
74
+ };
75
+ }
76
+ function socketUrl(host, eio) {
77
+ return `wss://${host}/socket.io/?EIO=${eio}&transport=websocket`;
78
+ }
79
+ function socketHeaders(host) {
80
+ return {
81
+ ...DEFAULT_HEADERS,
82
+ Origin: `https://${host}`,
83
+ Referer: `https://${host}/`,
84
+ };
85
+ }
86
+ function sendEvent(ws, event, payload) {
87
+ ws.send(`42${JSON.stringify([event, payload])}`);
88
+ }
89
+ function parseEventPacket(packet) {
90
+ if (!packet.startsWith('42')) {
91
+ return null;
92
+ }
93
+ try {
94
+ const decoded = JSON.parse(packet.slice(2));
95
+ if (!Array.isArray(decoded) || typeof decoded[0] !== 'string') {
96
+ return null;
97
+ }
98
+ return { event: decoded[0], payload: decoded[1] };
99
+ }
100
+ catch {
101
+ return null;
102
+ }
103
+ }
104
+ function stableMessageId(raw, recipientEmail) {
105
+ const headers = raw?.headers || {};
106
+ return String(raw?.id ??
107
+ raw?.messageId ??
108
+ headers['message-id'] ??
109
+ headers.messageId ??
110
+ `${headers.from || raw?.from || ''}\n${headers.subject || raw?.subject || ''}\n${headers.date || raw?.date || ''}\n${recipientEmail}`);
111
+ }
112
+ function flattenMail(raw, recipientEmail) {
113
+ const headers = raw?.headers || {};
114
+ return {
115
+ id: stableMessageId(raw, recipientEmail),
116
+ from: headers.from || raw?.from || '',
117
+ to: recipientEmail,
118
+ subject: headers.subject || raw?.subject || '',
119
+ text: raw?.text || raw?.body || '',
120
+ html: raw?.html || '',
121
+ date: headers.date || raw?.date || '',
122
+ isRead: false,
123
+ attachments: Array.isArray(raw?.attachments) ? raw.attachments : [],
124
+ };
125
+ }
126
+ async function connectSocket(host) {
127
+ let lastError = null;
128
+ for (const version of SOCKET_IO_VERSIONS) {
129
+ try {
130
+ return await new Promise((resolve, reject) => {
131
+ const ws = new ws_1.default(socketUrl(host, version), {
132
+ handshakeTimeout: CONNECT_TIMEOUT_MS,
133
+ headers: socketHeaders(host),
134
+ });
135
+ let timer;
136
+ let sentConnect = false;
137
+ let settled = false;
138
+ const cleanup = () => {
139
+ if (timer) {
140
+ clearTimeout(timer);
141
+ timer = undefined;
142
+ }
143
+ ws.off('message', onMessage);
144
+ ws.off('error', onError);
145
+ ws.off('close', onClose);
146
+ };
147
+ const resolveOnce = () => {
148
+ if (settled)
149
+ return;
150
+ settled = true;
151
+ cleanup();
152
+ resolve(ws);
153
+ };
154
+ const rejectOnce = (err) => {
155
+ if (settled)
156
+ return;
157
+ settled = true;
158
+ cleanup();
159
+ try {
160
+ ws.close();
161
+ }
162
+ catch {
163
+ /* ignore close errors */
164
+ }
165
+ reject(err);
166
+ };
167
+ const armHandshakeFallback = () => {
168
+ if (timer)
169
+ return;
170
+ timer = setTimeout(resolveOnce, HANDSHAKE_WAIT_MS);
171
+ };
172
+ const onMessage = (data) => {
173
+ const packet = data.toString();
174
+ if (packet === '2') {
175
+ try {
176
+ ws.send('3');
177
+ }
178
+ catch {
179
+ /* ignore send errors during probing */
180
+ }
181
+ return;
182
+ }
183
+ if (!sentConnect) {
184
+ if (!packet.startsWith('0')) {
185
+ rejectOnce(new Error(`tempmail-cn: unexpected open packet for EIO=${version}`));
186
+ return;
187
+ }
188
+ sentConnect = true;
189
+ try {
190
+ ws.send('40');
191
+ }
192
+ catch (error) {
193
+ rejectOnce(error instanceof Error ? error : new Error(String(error)));
194
+ return;
195
+ }
196
+ armHandshakeFallback();
197
+ return;
198
+ }
199
+ if (packet.startsWith('40')) {
200
+ resolveOnce();
201
+ }
202
+ };
203
+ const onError = (error) => rejectOnce(error);
204
+ const onClose = () => rejectOnce(new Error(`tempmail-cn: websocket closed during EIO=${version} handshake`));
205
+ ws.on('message', onMessage);
206
+ ws.once('error', onError);
207
+ ws.once('close', onClose);
208
+ });
209
+ }
210
+ catch (error) {
211
+ lastError = error instanceof Error ? error : new Error(String(error));
212
+ }
213
+ }
214
+ throw lastError || new Error('tempmail-cn: websocket handshake failed');
215
+ }
216
+ async function requestShortId(host) {
217
+ const ws = await connectSocket(host);
218
+ return await new Promise((resolve, reject) => {
219
+ const timer = setTimeout(() => {
220
+ cleanup();
221
+ try {
222
+ ws.close();
223
+ }
224
+ catch {
225
+ /* ignore close errors */
226
+ }
227
+ reject(new Error('tempmail-cn: timed out waiting for shortid'));
228
+ }, CONNECT_TIMEOUT_MS);
229
+ const cleanup = () => {
230
+ clearTimeout(timer);
231
+ ws.off('message', onMessage);
232
+ ws.off('error', onError);
233
+ ws.off('close', onClose);
234
+ };
235
+ const finish = (value) => {
236
+ cleanup();
237
+ try {
238
+ ws.close();
239
+ }
240
+ catch {
241
+ /* ignore close errors */
242
+ }
243
+ resolve(value);
244
+ };
245
+ const fail = (error) => {
246
+ cleanup();
247
+ try {
248
+ ws.close();
249
+ }
250
+ catch {
251
+ /* ignore close errors */
252
+ }
253
+ reject(error);
254
+ };
255
+ const onMessage = (data) => {
256
+ const packet = data.toString();
257
+ if (packet === '2') {
258
+ ws.send('3');
259
+ return;
260
+ }
261
+ const decoded = parseEventPacket(packet);
262
+ if (!decoded || decoded.event !== 'shortid' || typeof decoded.payload !== 'string') {
263
+ return;
264
+ }
265
+ finish(decoded.payload);
266
+ };
267
+ const onError = (error) => fail(error);
268
+ const onClose = () => fail(new Error('tempmail-cn: websocket closed before shortid arrived'));
269
+ ws.on('message', onMessage);
270
+ ws.once('error', onError);
271
+ ws.once('close', onClose);
272
+ sendEvent(ws, 'request shortid', true);
273
+ });
274
+ }
275
+ async function ensureMailbox(email) {
276
+ const st = getState(email);
277
+ if (st.ws?.readyState === ws_1.default.OPEN) {
278
+ return;
279
+ }
280
+ if (st.connectPromise) {
281
+ return st.connectPromise;
282
+ }
283
+ const { local, host } = splitAddress(email);
284
+ st.connectPromise = (async () => {
285
+ try {
286
+ const ws = await connectSocket(host);
287
+ st.ws = ws;
288
+ const detach = () => {
289
+ if (st.ws === ws) {
290
+ st.ws = null;
291
+ }
292
+ st.connectPromise = undefined;
293
+ };
294
+ ws.on('message', (data) => {
295
+ const packet = data.toString();
296
+ if (packet === '2') {
297
+ try {
298
+ ws.send('3');
299
+ }
300
+ catch {
301
+ /* ignore late pong failures */
302
+ }
303
+ return;
304
+ }
305
+ const decoded = parseEventPacket(packet);
306
+ if (!decoded || decoded.event !== 'mail' || !decoded.payload) {
307
+ return;
308
+ }
309
+ const normalized = (0, normalize_1.normalizeEmail)(flattenMail(decoded.payload, email), email);
310
+ if (!normalized.id || st.seenIds.has(normalized.id)) {
311
+ return;
312
+ }
313
+ st.seenIds.add(normalized.id);
314
+ st.emails.push(normalized);
315
+ });
316
+ ws.on('close', detach);
317
+ ws.on('error', detach);
318
+ sendEvent(ws, 'set shortid', local);
319
+ await sleep(INITIAL_SYNC_WAIT_MS);
320
+ }
321
+ catch (error) {
322
+ st.connectPromise = undefined;
323
+ throw error;
324
+ }
325
+ })();
326
+ return st.connectPromise;
327
+ }
328
+ async function generateEmail(domain) {
329
+ const host = normalizeHost(domain);
330
+ const shortid = await requestShortId(host);
331
+ return {
332
+ channel: CHANNEL,
333
+ email: `${shortid}@${host}`,
334
+ };
335
+ }
336
+ async function getEmails(email) {
337
+ await ensureMailbox(email);
338
+ const st = getState(email);
339
+ return [...st.emails];
340
+ }
341
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tempmail-cn.js","sourceRoot":"","sources":["../../src/providers/tempmail-cn.ts"],"names":[],"mappings":"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","sourcesContent":["import WebSocket from 'ws';\nimport { InternalEmailInfo, Email, Channel } from '../types';\nimport { normalizeEmail } from '../normalize';\n\nconst CHANNEL: Channel = 'tempmail-cn';\nconst DEFAULT_HOST = 'tempmail.cn';\nconst CONNECT_TIMEOUT_MS = 15000;\nconst HANDSHAKE_WAIT_MS = 1000;\nconst INITIAL_SYNC_WAIT_MS = 80;\nconst SOCKET_IO_VERSIONS = [4, 3];\n\nconst DEFAULT_HEADERS: Record<string, string> = {\n  'User-Agent':\n    'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/146.0.0.0 Safari/537.36 Edg/146.0.0.0',\n  'Accept-Language': 'zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6',\n  'Cache-Control': 'no-cache',\n  DNT: '1',\n  Pragma: 'no-cache',\n};\n\ntype BoxState = {\n  emails: Email[];\n  seenIds: Set<string>;\n  ws: WebSocket | null;\n  connectPromise?: Promise<void>;\n};\n\nconst mailboxes = new Map<string, BoxState>();\n\nfunction getState(email: string): BoxState {\n  const key = email.trim().toLowerCase();\n  let st = mailboxes.get(key);\n  if (!st) {\n    st = { emails: [], seenIds: new Set(), ws: null };\n    mailboxes.set(key, st);\n  }\n  return st;\n}\n\nfunction sleep(ms: number): Promise<void> {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\nfunction normalizeHost(domain?: string | null): string {\n  const raw = String(domain || '').trim();\n  if (!raw) {\n    return DEFAULT_HOST;\n  }\n\n  let host = raw;\n  if (/^https?:\\/\\//i.test(host)) {\n    try {\n      host = new URL(host).host;\n    } catch {\n      host = raw;\n    }\n  } else if (host.includes('/')) {\n    try {\n      host = new URL(`https://${host}`).host;\n    } catch {\n      host = host.split('/')[0];\n    }\n  }\n\n  host = host.replace(/^\\.+|\\.+$/g, '');\n  const at = host.lastIndexOf('@');\n  if (at >= 0) {\n    host = host.slice(at + 1);\n  }\n  return host || DEFAULT_HOST;\n}\n\nfunction splitAddress(email: string): { local: string; host: string } {\n  const trimmed = email.trim();\n  const at = trimmed.indexOf('@');\n  if (at <= 0 || at === trimmed.length - 1) {\n    throw new Error('tempmail-cn: invalid email address');\n  }\n  return {\n    local: trimmed.slice(0, at),\n    host: normalizeHost(trimmed.slice(at + 1)),\n  };\n}\n\nfunction socketUrl(host: string, eio: number): string {\n  return `wss://${host}/socket.io/?EIO=${eio}&transport=websocket`;\n}\n\nfunction socketHeaders(host: string): Record<string, string> {\n  return {\n    ...DEFAULT_HEADERS,\n    Origin: `https://${host}`,\n    Referer: `https://${host}/`,\n  };\n}\n\nfunction sendEvent(ws: WebSocket, event: string, payload: unknown): void {\n  ws.send(`42${JSON.stringify([event, payload])}`);\n}\n\nfunction parseEventPacket(packet: string): { event: string; payload: any } | null {\n  if (!packet.startsWith('42')) {\n    return null;\n  }\n  try {\n    const decoded = JSON.parse(packet.slice(2));\n    if (!Array.isArray(decoded) || typeof decoded[0] !== 'string') {\n      return null;\n    }\n    return { event: decoded[0], payload: decoded[1] };\n  } catch {\n    return null;\n  }\n}\n\nfunction stableMessageId(raw: any, recipientEmail: string): string {\n  const headers = raw?.headers || {};\n  return String(\n    raw?.id ??\n      raw?.messageId ??\n      headers['message-id'] ??\n      headers.messageId ??\n      `${headers.from || raw?.from || ''}\\n${headers.subject || raw?.subject || ''}\\n${headers.date || raw?.date || ''}\\n${recipientEmail}`,\n  );\n}\n\nfunction flattenMail(raw: any, recipientEmail: string): any {\n  const headers = raw?.headers || {};\n  return {\n    id: stableMessageId(raw, recipientEmail),\n    from: headers.from || raw?.from || '',\n    to: recipientEmail,\n    subject: headers.subject || raw?.subject || '',\n    text: raw?.text || raw?.body || '',\n    html: raw?.html || '',\n    date: headers.date || raw?.date || '',\n    isRead: false,\n    attachments: Array.isArray(raw?.attachments) ? raw.attachments : [],\n  };\n}\n\nasync function connectSocket(host: string): Promise<WebSocket> {\n  let lastError: Error | null = null;\n\n  for (const version of SOCKET_IO_VERSIONS) {\n    try {\n      return await new Promise<WebSocket>((resolve, reject) => {\n        const ws = new WebSocket(socketUrl(host, version), {\n          handshakeTimeout: CONNECT_TIMEOUT_MS,\n          headers: socketHeaders(host),\n        });\n\n        let timer: NodeJS.Timeout | undefined;\n        let sentConnect = false;\n        let settled = false;\n\n        const cleanup = () => {\n          if (timer) {\n            clearTimeout(timer);\n            timer = undefined;\n          }\n          ws.off('message', onMessage);\n          ws.off('error', onError);\n          ws.off('close', onClose);\n        };\n\n        const resolveOnce = () => {\n          if (settled) return;\n          settled = true;\n          cleanup();\n          resolve(ws);\n        };\n\n        const rejectOnce = (err: Error) => {\n          if (settled) return;\n          settled = true;\n          cleanup();\n          try {\n            ws.close();\n          } catch {\n            /* ignore close errors */\n          }\n          reject(err);\n        };\n\n        const armHandshakeFallback = () => {\n          if (timer) return;\n          timer = setTimeout(resolveOnce, HANDSHAKE_WAIT_MS);\n        };\n\n        const onMessage = (data: WebSocket.RawData) => {\n          const packet = data.toString();\n          if (packet === '2') {\n            try {\n              ws.send('3');\n            } catch {\n              /* ignore send errors during probing */\n            }\n            return;\n          }\n          if (!sentConnect) {\n            if (!packet.startsWith('0')) {\n              rejectOnce(new Error(`tempmail-cn: unexpected open packet for EIO=${version}`));\n              return;\n            }\n            sentConnect = true;\n            try {\n              ws.send('40');\n            } catch (error) {\n              rejectOnce(error instanceof Error ? error : new Error(String(error)));\n              return;\n            }\n            armHandshakeFallback();\n            return;\n          }\n          if (packet.startsWith('40')) {\n            resolveOnce();\n          }\n        };\n\n        const onError = (error: Error) => rejectOnce(error);\n        const onClose = () => rejectOnce(new Error(`tempmail-cn: websocket closed during EIO=${version} handshake`));\n\n        ws.on('message', onMessage);\n        ws.once('error', onError);\n        ws.once('close', onClose);\n      });\n    } catch (error) {\n      lastError = error instanceof Error ? error : new Error(String(error));\n    }\n  }\n\n  throw lastError || new Error('tempmail-cn: websocket handshake failed');\n}\n\nasync function requestShortId(host: string): Promise<string> {\n  const ws = await connectSocket(host);\n\n  return await new Promise<string>((resolve, reject) => {\n    const timer = setTimeout(() => {\n      cleanup();\n      try {\n        ws.close();\n      } catch {\n        /* ignore close errors */\n      }\n      reject(new Error('tempmail-cn: timed out waiting for shortid'));\n    }, CONNECT_TIMEOUT_MS);\n\n    const cleanup = () => {\n      clearTimeout(timer);\n      ws.off('message', onMessage);\n      ws.off('error', onError);\n      ws.off('close', onClose);\n    };\n\n    const finish = (value: string) => {\n      cleanup();\n      try {\n        ws.close();\n      } catch {\n        /* ignore close errors */\n      }\n      resolve(value);\n    };\n\n    const fail = (error: Error) => {\n      cleanup();\n      try {\n        ws.close();\n      } catch {\n        /* ignore close errors */\n      }\n      reject(error);\n    };\n\n    const onMessage = (data: WebSocket.RawData) => {\n      const packet = data.toString();\n      if (packet === '2') {\n        ws.send('3');\n        return;\n      }\n      const decoded = parseEventPacket(packet);\n      if (!decoded || decoded.event !== 'shortid' || typeof decoded.payload !== 'string') {\n        return;\n      }\n      finish(decoded.payload);\n    };\n\n    const onError = (error: Error) => fail(error);\n    const onClose = () => fail(new Error('tempmail-cn: websocket closed before shortid arrived'));\n\n    ws.on('message', onMessage);\n    ws.once('error', onError);\n    ws.once('close', onClose);\n\n    sendEvent(ws, 'request shortid', true);\n  });\n}\n\nasync function ensureMailbox(email: string): Promise<void> {\n  const st = getState(email);\n  if (st.ws?.readyState === WebSocket.OPEN) {\n    return;\n  }\n  if (st.connectPromise) {\n    return st.connectPromise;\n  }\n\n  const { local, host } = splitAddress(email);\n\n  st.connectPromise = (async () => {\n    try {\n      const ws = await connectSocket(host);\n      st.ws = ws;\n\n      const detach = () => {\n        if (st.ws === ws) {\n          st.ws = null;\n        }\n        st.connectPromise = undefined;\n      };\n\n      ws.on('message', (data: WebSocket.RawData) => {\n        const packet = data.toString();\n        if (packet === '2') {\n          try {\n            ws.send('3');\n          } catch {\n            /* ignore late pong failures */\n          }\n          return;\n        }\n\n        const decoded = parseEventPacket(packet);\n        if (!decoded || decoded.event !== 'mail' || !decoded.payload) {\n          return;\n        }\n\n        const normalized = normalizeEmail(flattenMail(decoded.payload, email), email);\n        if (!normalized.id || st.seenIds.has(normalized.id)) {\n          return;\n        }\n\n        st.seenIds.add(normalized.id);\n        st.emails.push(normalized);\n      });\n\n      ws.on('close', detach);\n      ws.on('error', detach);\n\n      sendEvent(ws, 'set shortid', local);\n      await sleep(INITIAL_SYNC_WAIT_MS);\n    } catch (error) {\n      st.connectPromise = undefined;\n      throw error;\n    }\n  })();\n\n  return st.connectPromise;\n}\n\nexport async function generateEmail(domain?: string | null): Promise<InternalEmailInfo> {\n  const host = normalizeHost(domain);\n  const shortid = await requestShortId(host);\n  return {\n    channel: CHANNEL,\n    email: `${shortid}@${host}`,\n  };\n}\n\nexport async function getEmails(email: string): Promise<Email[]> {\n  await ensureMailbox(email);\n  const st = getState(email);\n  return [...st.emails];\n}\n"]}
package/dist/retry.d.ts CHANGED
@@ -26,5 +26,18 @@ export interface RetryOptions {
26
26
  * @param fn 要执行的异步操作
27
27
  * @param options 重试配置
28
28
  */
29
+ export type RetryWithAttemptsResult<T> = {
30
+ ok: true;
31
+ value: T;
32
+ attempts: number;
33
+ } | {
34
+ ok: false;
35
+ error: unknown;
36
+ attempts: number;
37
+ };
38
+ /**
39
+ * 与 withRetry 相同,额外返回尝试次数(成功或最终失败时均有效)
40
+ */
41
+ export declare function withRetryWithAttempts<T>(fn: () => Promise<T>, options?: RetryOptions): Promise<RetryWithAttemptsResult<T>>;
29
42
  export declare function withRetry<T>(fn: () => Promise<T>, options?: RetryOptions): Promise<T>;
30
43
  export declare function fetchWithTimeout(url: string, init?: RequestInit, timeoutMs?: number): Promise<Response>;
package/dist/retry.js CHANGED
@@ -4,6 +4,7 @@
4
4
  * 提供请求重试、超时控制、指数退避等错误恢复机制
5
5
  */
6
6
  Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.withRetryWithAttempts = withRetryWithAttempts;
7
8
  exports.withRetry = withRetry;
8
9
  exports.fetchWithTimeout = fetchWithTimeout;
9
10
  const logger_1 = require("./logger");
@@ -60,24 +61,19 @@ function sleep(ms) {
60
61
  return new Promise(resolve => setTimeout(resolve, ms));
61
62
  }
62
63
  /**
63
- * 带重试的异步操作执行器
64
- * - 自动重试可恢复的错误(网络错误、超时、HTTP 4xx/5xx)
65
- * - 指数退避避免过度请求
66
- * - 不可恢复的错误(SDK 内部参数校验错误等)直接抛出不重试
67
- *
68
- * @param fn 要执行的异步操作
69
- * @param options 重试配置
64
+ * 与 withRetry 相同,额外返回尝试次数(成功或最终失败时均有效)
70
65
  */
71
- async function withRetry(fn, options) {
66
+ async function withRetryWithAttempts(fn, options) {
72
67
  const opts = { ...DEFAULT_RETRY_OPTIONS, ...options };
73
68
  let lastError;
74
69
  for (let attempt = 0; attempt <= opts.maxRetries; attempt++) {
70
+ const attempts = attempt + 1;
75
71
  try {
76
72
  const result = await fn();
77
73
  if (attempt > 0) {
78
74
  logger_1.logger.info(`第 ${attempt + 1} 次尝试成功`);
79
75
  }
80
- return result;
76
+ return { ok: true, value: result, attempts };
81
77
  }
82
78
  catch (error) {
83
79
  lastError = error;
@@ -90,7 +86,7 @@ async function withRetry(fn, options) {
90
86
  else if (!opts.shouldRetry(error)) {
91
87
  logger_1.logger.debug(`不可重试的错误: ${errorMsg}`);
92
88
  }
93
- throw error;
89
+ return { ok: false, error, attempts };
94
90
  }
95
91
  /* 指数退避等待 */
96
92
  const delay = Math.min(opts.initialDelay * Math.pow(2, attempt), opts.maxDelay);
@@ -98,7 +94,13 @@ async function withRetry(fn, options) {
98
94
  await sleep(delay);
99
95
  }
100
96
  }
101
- throw lastError;
97
+ return { ok: false, error: lastError, attempts: opts.maxRetries + 1 };
98
+ }
99
+ async function withRetry(fn, options) {
100
+ const r = await withRetryWithAttempts(fn, options);
101
+ if (r.ok)
102
+ return r.value;
103
+ throw r.error;
102
104
  }
103
105
  /**
104
106
  * 带超时控制的 fetch 包装
@@ -158,4 +160,4 @@ async function fetchWithTimeout(url, init, timeoutMs) {
158
160
  clearTimeout(timeoutId);
159
161
  }
160
162
  }
161
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"retry.js","sourceRoot":"","sources":["../src/retry.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * 通用重试工具\n * 提供请求重试、超时控制、指数退避等错误恢复机制\n */\n\nimport { logger } from './logger';\nimport { getConfig } from './config';\n\n/**\n * 重试配置选项\n */\nexport interface RetryOptions {\n  /** 最大重试次数（不含首次请求），默认 2 */\n  maxRetries?: number;\n  /** 初始重试延迟（毫秒），默认 1000 */\n  initialDelay?: number;\n  /** 最大重试延迟（毫秒），默认 5000 */\n  maxDelay?: number;\n  /** 请求超时时间（毫秒），默认 15000 */\n  timeout?: number;\n  /** 是否对该错误进行重试的判断函数 */\n  shouldRetry?: (error: any) => boolean;\n}\n\nconst DEFAULT_RETRY_OPTIONS: Required<RetryOptions> = {\n  maxRetries: 2,\n  initialDelay: 1000,\n  maxDelay: 5000,\n  timeout: 15000,\n  shouldRetry: defaultShouldRetry,\n};\n\n/**\n * 默认重试判断\n * 以下错误类型会触发重试：\n * - 网络连接错误（fetch failed, ECONNREFUSED, ECONNRESET 等）\n * - 超时错误（timeout, abort）\n * - DNS 解析失败\n * - HTTP 429 限流\n * - HTTP 4xx/5xx 服务端错误（含状态码的错误消息）\n * 仅 SDK 内部的参数校验类错误不重试\n */\nfunction defaultShouldRetry(error: any): boolean {\n  if (!error) return false;\n\n  const message = String(error.message || error || '').toLowerCase();\n\n  /* 网络级别错误 → 重试 */\n  if (message.includes('fetch failed') ||\n      message.includes('network') ||\n      message.includes('econnrefused') ||\n      message.includes('econnreset') ||\n      message.includes('etimedout') ||\n      message.includes('timeout') ||\n      message.includes('socket hang up') ||\n      message.includes('dns') ||\n      message.includes('abort')) {\n    return true;\n  }\n\n  /* HTTP 429 限流 → 重试 */\n  if (message.includes('429') || message.includes('too many requests') || message.includes('rate limit')) {\n    return true;\n  }\n\n  /* HTTP 4xx/5xx 错误（含状态码的错误消息）→ 重试 */\n  const statusMatch = message.match(/:\\s*(\\d{3})/);\n  if (statusMatch) {\n    const status = parseInt(statusMatch[1], 10);\n    return status >= 400;\n  }\n\n  return false;\n}\n\n/**\n * 休眠指定毫秒\n */\nfunction sleep(ms: number): Promise<void> {\n  return new Promise(resolve => setTimeout(resolve, ms));\n}\n\n/**\n * 带重试的异步操作执行器\n * - 自动重试可恢复的错误（网络错误、超时、HTTP 4xx/5xx）\n * - 指数退避避免过度请求\n * - 不可恢复的错误（SDK 内部参数校验错误等）直接抛出不重试\n *\n * @param fn 要执行的异步操作\n * @param options 重试配置\n */\nexport async function withRetry<T>(fn: () => Promise<T>, options?: RetryOptions): Promise<T> {\n  const opts = { ...DEFAULT_RETRY_OPTIONS, ...options };\n  let lastError: any;\n\n  for (let attempt = 0; attempt <= opts.maxRetries; attempt++) {\n    try {\n      const result = await fn();\n      if (attempt > 0) {\n        logger.info(`第 ${attempt + 1} 次尝试成功`);\n      }\n      return result;\n    } catch (error: any) {\n      lastError = error;\n      const errorMsg = error.message || String(error);\n\n      /* 最后一次尝试失败或不可重试的错误 → 直接抛出 */\n      if (attempt >= opts.maxRetries || !opts.shouldRetry(error)) {\n        if (attempt >= opts.maxRetries && opts.maxRetries > 0) {\n          logger.error(`重试 ${opts.maxRetries} 次后仍失败: ${errorMsg}`);\n        } else if (!opts.shouldRetry(error)) {\n          logger.debug(`不可重试的错误: ${errorMsg}`);\n        }\n        throw error;\n      }\n\n      /* 指数退避等待 */\n      const delay = Math.min(opts.initialDelay * Math.pow(2, attempt), opts.maxDelay);\n      logger.warn(`请求失败 (${errorMsg})，${delay}ms 后第 ${attempt + 2} 次重试...`);\n      await sleep(delay);\n    }\n  }\n\n  throw lastError;\n}\n\n/**\n * 带超时控制的 fetch 包装\n * 在原生 fetch 基础上添加超时中断能力\n *\n * @param url 请求 URL\n * @param init fetch 选项\n * @param timeoutMs 超时时间（毫秒）\n */\n/**\n * 缓存的全局配置快照，避免每次请求都读取\n * 仅在 setConfig 被调用时失效（通过 configVersion 比对）\n */\nlet _cachedFetchConfig: { fetchFn: typeof fetch; timeout: number; version: number } | null = null;\n\n/**\n * 获取缓存的 fetch 配置\n */\nfunction getFetchConfig(): { fetchFn: typeof fetch; timeout: number } {\n  const config = getConfig();\n  /* 简单的引用比对即可，getConfig 在未变更时返回同一对象 */\n  if (!_cachedFetchConfig || _cachedFetchConfig.fetchFn !== (config.customFetch || fetch) || _cachedFetchConfig.timeout !== (config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout)) {\n    _cachedFetchConfig = {\n      fetchFn: config.customFetch || fetch,\n      timeout: config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout,\n      version: 0,\n    };\n  }\n  return _cachedFetchConfig;\n}\n\nexport async function fetchWithTimeout(\n  url: string,\n  init?: RequestInit,\n  timeoutMs?: number,\n): Promise<Response> {\n  const { fetchFn, timeout: defaultTimeout } = getFetchConfig();\n  const effectiveTimeout = timeoutMs ?? defaultTimeout;\n  const controller = new AbortController();\n  const timeoutId = setTimeout(() => controller.abort(), effectiveTimeout);\n\n  /*\n   * 如果调用方已提供 signal，需要同时监听两个信号（调用方 + 超时）\n   * 任一触发则中断请求\n   */\n  const externalSignal = init?.signal;\n  if (externalSignal) {\n    externalSignal.addEventListener('abort', () => controller.abort(), { once: true });\n  }\n\n  try {\n    const response = await fetchFn(url, {\n      ...init,\n      signal: controller.signal,\n    });\n    return response;\n  } catch (error: any) {\n    if (error.name === 'AbortError') {\n      throw new Error(`Request timeout after ${effectiveTimeout}ms: ${url}`);\n    }\n    throw error;\n  } finally {\n    clearTimeout(timeoutId);\n  }\n}\n"]}
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"retry.js","sourceRoot":"","sources":["../src/retry.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * 通用重试工具\n * 提供请求重试、超时控制、指数退避等错误恢复机制\n */\n\nimport { logger } from './logger';\nimport { getConfig } from './config';\n\n/**\n * 重试配置选项\n */\nexport interface RetryOptions {\n  /** 最大重试次数（不含首次请求），默认 2 */\n  maxRetries?: number;\n  /** 初始重试延迟（毫秒），默认 1000 */\n  initialDelay?: number;\n  /** 最大重试延迟（毫秒），默认 5000 */\n  maxDelay?: number;\n  /** 请求超时时间（毫秒），默认 15000 */\n  timeout?: number;\n  /** 是否对该错误进行重试的判断函数 */\n  shouldRetry?: (error: any) => boolean;\n}\n\nconst DEFAULT_RETRY_OPTIONS: Required<RetryOptions> = {\n  maxRetries: 2,\n  initialDelay: 1000,\n  maxDelay: 5000,\n  timeout: 15000,\n  shouldRetry: defaultShouldRetry,\n};\n\n/**\n * 默认重试判断\n * 以下错误类型会触发重试：\n * - 网络连接错误（fetch failed, ECONNREFUSED, ECONNRESET 等）\n * - 超时错误（timeout, abort）\n * - DNS 解析失败\n * - HTTP 429 限流\n * - HTTP 4xx/5xx 服务端错误（含状态码的错误消息）\n * 仅 SDK 内部的参数校验类错误不重试\n */\nfunction defaultShouldRetry(error: any): boolean {\n  if (!error) return false;\n\n  const message = String(error.message || error || '').toLowerCase();\n\n  /* 网络级别错误 → 重试 */\n  if (message.includes('fetch failed') ||\n      message.includes('network') ||\n      message.includes('econnrefused') ||\n      message.includes('econnreset') ||\n      message.includes('etimedout') ||\n      message.includes('timeout') ||\n      message.includes('socket hang up') ||\n      message.includes('dns') ||\n      message.includes('abort')) {\n    return true;\n  }\n\n  /* HTTP 429 限流 → 重试 */\n  if (message.includes('429') || message.includes('too many requests') || message.includes('rate limit')) {\n    return true;\n  }\n\n  /* HTTP 4xx/5xx 错误（含状态码的错误消息）→ 重试 */\n  const statusMatch = message.match(/:\\s*(\\d{3})/);\n  if (statusMatch) {\n    const status = parseInt(statusMatch[1], 10);\n    return status >= 400;\n  }\n\n  return false;\n}\n\n/**\n * 休眠指定毫秒\n */\nfunction sleep(ms: number): Promise<void> {\n  return new Promise(resolve => setTimeout(resolve, ms));\n}\n\n/**\n * 带重试的异步操作执行器\n * - 自动重试可恢复的错误（网络错误、超时、HTTP 4xx/5xx）\n * - 指数退避避免过度请求\n * - 不可恢复的错误（SDK 内部参数校验错误等）直接抛出不重试\n *\n * @param fn 要执行的异步操作\n * @param options 重试配置\n */\nexport type RetryWithAttemptsResult<T> =\n  | { ok: true; value: T; attempts: number }\n  | { ok: false; error: unknown; attempts: number };\n\n/**\n * 与 withRetry 相同，额外返回尝试次数（成功或最终失败时均有效）\n */\nexport async function withRetryWithAttempts<T>(\n  fn: () => Promise<T>,\n  options?: RetryOptions,\n): Promise<RetryWithAttemptsResult<T>> {\n  const opts = { ...DEFAULT_RETRY_OPTIONS, ...options };\n  let lastError: any;\n\n  for (let attempt = 0; attempt <= opts.maxRetries; attempt++) {\n    const attempts = attempt + 1;\n    try {\n      const result = await fn();\n      if (attempt > 0) {\n        logger.info(`第 ${attempt + 1} 次尝试成功`);\n      }\n      return { ok: true, value: result, attempts };\n    } catch (error: any) {\n      lastError = error;\n      const errorMsg = error.message || String(error);\n\n      /* 最后一次尝试失败或不可重试的错误 → 直接抛出 */\n      if (attempt >= opts.maxRetries || !opts.shouldRetry(error)) {\n        if (attempt >= opts.maxRetries && opts.maxRetries > 0) {\n          logger.error(`重试 ${opts.maxRetries} 次后仍失败: ${errorMsg}`);\n        } else if (!opts.shouldRetry(error)) {\n          logger.debug(`不可重试的错误: ${errorMsg}`);\n        }\n        return { ok: false, error, attempts };\n      }\n\n      /* 指数退避等待 */\n      const delay = Math.min(opts.initialDelay * Math.pow(2, attempt), opts.maxDelay);\n      logger.warn(`请求失败 (${errorMsg})，${delay}ms 后第 ${attempt + 2} 次重试...`);\n      await sleep(delay);\n    }\n  }\n\n  return { ok: false, error: lastError, attempts: opts.maxRetries + 1 };\n}\n\nexport async function withRetry<T>(fn: () => Promise<T>, options?: RetryOptions): Promise<T> {\n  const r = await withRetryWithAttempts(fn, options);\n  if (r.ok) return r.value;\n  throw r.error;\n}\n\n/**\n * 带超时控制的 fetch 包装\n * 在原生 fetch 基础上添加超时中断能力\n *\n * @param url 请求 URL\n * @param init fetch 选项\n * @param timeoutMs 超时时间（毫秒）\n */\n/**\n * 缓存的全局配置快照，避免每次请求都读取\n * 仅在 setConfig 被调用时失效（通过 configVersion 比对）\n */\nlet _cachedFetchConfig: { fetchFn: typeof fetch; timeout: number; version: number } | null = null;\n\n/**\n * 获取缓存的 fetch 配置\n */\nfunction getFetchConfig(): { fetchFn: typeof fetch; timeout: number } {\n  const config = getConfig();\n  /* 简单的引用比对即可，getConfig 在未变更时返回同一对象 */\n  if (!_cachedFetchConfig || _cachedFetchConfig.fetchFn !== (config.customFetch || fetch) || _cachedFetchConfig.timeout !== (config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout)) {\n    _cachedFetchConfig = {\n      fetchFn: config.customFetch || fetch,\n      timeout: config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout,\n      version: 0,\n    };\n  }\n  return _cachedFetchConfig;\n}\n\nexport async function fetchWithTimeout(\n  url: string,\n  init?: RequestInit,\n  timeoutMs?: number,\n): Promise<Response> {\n  const { fetchFn, timeout: defaultTimeout } = getFetchConfig();\n  const effectiveTimeout = timeoutMs ?? defaultTimeout;\n  const controller = new AbortController();\n  const timeoutId = setTimeout(() => controller.abort(), effectiveTimeout);\n\n  /*\n   * 如果调用方已提供 signal，需要同时监听两个信号（调用方 + 超时）\n   * 任一触发则中断请求\n   */\n  const externalSignal = init?.signal;\n  if (externalSignal) {\n    externalSignal.addEventListener('abort', () => controller.abort(), { once: true });\n  }\n\n  try {\n    const response = await fetchFn(url, {\n      ...init,\n      signal: controller.signal,\n    });\n    return response;\n  } catch (error: any) {\n    if (error.name === 'AbortError') {\n      throw new Error(`Request timeout after ${effectiveTimeout}ms: ${url}`);\n    }\n    throw error;\n  } finally {\n    clearTimeout(timeoutId);\n  }\n}\n"]}
@@ -0,0 +1 @@
1
+ export declare function reportTelemetry(operation: string, channel: string, success: boolean, attemptCount: number, channelsTried: number, error: string): void;