tempmail-sdk 1.1.2 → 1.1.3
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/dist/index.js +4 -2
- package/dist/providers/awamail.js +4 -3
- package/dist/providers/chatgpt-org-uk.d.ts +1 -1
- package/dist/providers/chatgpt-org-uk.js +119 -17
- package/dist/providers/dropmail.js +3 -2
- package/dist/providers/guerrillamail.js +4 -3
- package/dist/providers/linshi-email.js +4 -3
- package/dist/providers/mail-tm.js +7 -6
- package/dist/providers/maildrop.js +3 -2
- package/dist/providers/temp-mail-io.js +5 -4
- package/dist/providers/tempmail-la.js +4 -3
- package/dist/providers/tempmail-lol.js +4 -3
- package/dist/providers/tempmail.js +4 -3
- package/dist/retry.d.ts +2 -10
- package/dist/retry.js +41 -10
- package/package.json +1 -1
- package/src/index.ts +2 -1
- package/src/providers/awamail.ts +3 -2
- package/src/providers/chatgpt-org-uk.ts +131 -18
- package/src/providers/dropmail.ts +2 -1
- package/src/providers/guerrillamail.ts +3 -2
- package/src/providers/linshi-email.ts +3 -2
- package/src/providers/mail-tm.ts +6 -5
- package/src/providers/maildrop.ts +2 -1
- package/src/providers/temp-mail-io.ts +4 -3
- package/src/providers/tempmail-la.ts +3 -2
- package/src/providers/tempmail-lol.ts +3 -2
- package/src/providers/tempmail.ts +3 -2
- package/src/retry.ts +42 -9
- package/test/example.ts +9 -1
package/dist/retry.js
CHANGED
|
@@ -17,8 +17,13 @@ const DEFAULT_RETRY_OPTIONS = {
|
|
|
17
17
|
};
|
|
18
18
|
/**
|
|
19
19
|
* 默认重试判断
|
|
20
|
-
*
|
|
21
|
-
*
|
|
20
|
+
* 以下错误类型会触发重试:
|
|
21
|
+
* - 网络连接错误(fetch failed, ECONNREFUSED, ECONNRESET 等)
|
|
22
|
+
* - 超时错误(timeout, abort)
|
|
23
|
+
* - DNS 解析失败
|
|
24
|
+
* - HTTP 429 限流
|
|
25
|
+
* - HTTP 4xx/5xx 服务端错误(含状态码的错误消息)
|
|
26
|
+
* 仅 SDK 内部的参数校验类错误不重试
|
|
22
27
|
*/
|
|
23
28
|
function defaultShouldRetry(error) {
|
|
24
29
|
if (!error)
|
|
@@ -40,7 +45,7 @@ function defaultShouldRetry(error) {
|
|
|
40
45
|
if (message.includes('429') || message.includes('too many requests') || message.includes('rate limit')) {
|
|
41
46
|
return true;
|
|
42
47
|
}
|
|
43
|
-
/* HTTP 4xx/5xx
|
|
48
|
+
/* HTTP 4xx/5xx 错误(含状态码的错误消息)→ 重试 */
|
|
44
49
|
const statusMatch = message.match(/:\s*(\d{3})/);
|
|
45
50
|
if (statusMatch) {
|
|
46
51
|
const status = parseInt(statusMatch[1], 10);
|
|
@@ -56,9 +61,9 @@ function sleep(ms) {
|
|
|
56
61
|
}
|
|
57
62
|
/**
|
|
58
63
|
* 带重试的异步操作执行器
|
|
59
|
-
* - 自动重试可恢复的错误(网络错误、超时、5xx)
|
|
64
|
+
* - 自动重试可恢复的错误(网络错误、超时、HTTP 4xx/5xx)
|
|
60
65
|
* - 指数退避避免过度请求
|
|
61
|
-
* - 不可恢复的错误(
|
|
66
|
+
* - 不可恢复的错误(SDK 内部参数校验错误等)直接抛出不重试
|
|
62
67
|
*
|
|
63
68
|
* @param fn 要执行的异步操作
|
|
64
69
|
* @param options 重试配置
|
|
@@ -103,13 +108,39 @@ async function withRetry(fn, options) {
|
|
|
103
108
|
* @param init fetch 选项
|
|
104
109
|
* @param timeoutMs 超时时间(毫秒)
|
|
105
110
|
*/
|
|
106
|
-
|
|
111
|
+
/**
|
|
112
|
+
* 缓存的全局配置快照,避免每次请求都读取
|
|
113
|
+
* 仅在 setConfig 被调用时失效(通过 configVersion 比对)
|
|
114
|
+
*/
|
|
115
|
+
let _cachedFetchConfig = null;
|
|
116
|
+
/**
|
|
117
|
+
* 获取缓存的 fetch 配置
|
|
118
|
+
*/
|
|
119
|
+
function getFetchConfig() {
|
|
107
120
|
const config = (0, config_1.getConfig)();
|
|
108
|
-
|
|
121
|
+
/* 简单的引用比对即可,getConfig 在未变更时返回同一对象 */
|
|
122
|
+
if (!_cachedFetchConfig || _cachedFetchConfig.fetchFn !== (config.customFetch || fetch) || _cachedFetchConfig.timeout !== (config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout)) {
|
|
123
|
+
_cachedFetchConfig = {
|
|
124
|
+
fetchFn: config.customFetch || fetch,
|
|
125
|
+
timeout: config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout,
|
|
126
|
+
version: 0,
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
return _cachedFetchConfig;
|
|
130
|
+
}
|
|
131
|
+
async function fetchWithTimeout(url, init, timeoutMs) {
|
|
132
|
+
const { fetchFn, timeout: defaultTimeout } = getFetchConfig();
|
|
133
|
+
const effectiveTimeout = timeoutMs ?? defaultTimeout;
|
|
109
134
|
const controller = new AbortController();
|
|
110
135
|
const timeoutId = setTimeout(() => controller.abort(), effectiveTimeout);
|
|
111
|
-
/*
|
|
112
|
-
|
|
136
|
+
/*
|
|
137
|
+
* 如果调用方已提供 signal,需要同时监听两个信号(调用方 + 超时)
|
|
138
|
+
* 任一触发则中断请求
|
|
139
|
+
*/
|
|
140
|
+
const externalSignal = init?.signal;
|
|
141
|
+
if (externalSignal) {
|
|
142
|
+
externalSignal.addEventListener('abort', () => controller.abort(), { once: true });
|
|
143
|
+
}
|
|
113
144
|
try {
|
|
114
145
|
const response = await fetchFn(url, {
|
|
115
146
|
...init,
|
|
@@ -127,4 +158,4 @@ async function fetchWithTimeout(url, init, timeoutMs) {
|
|
|
127
158
|
clearTimeout(timeoutId);
|
|
128
159
|
}
|
|
129
160
|
}
|
|
130
|
-
//# 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 * 网络错误、超时、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 * - 自动重试可恢复的错误（网络错误、超时、5xx）\n * - 指数退避避免过度请求\n * - 不可恢复的错误（4xx 参数错误等）直接抛出不重试\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 */\nexport async function fetchWithTimeout(\n  url: string,\n  init?: RequestInit,\n  timeoutMs?: number,\n): Promise<Response> {\n  const config = getConfig();\n  const effectiveTimeout = timeoutMs ?? config.timeout ?? DEFAULT_RETRY_OPTIONS.timeout;\n  const controller = new AbortController();\n  const timeoutId = setTimeout(() => controller.abort(), effectiveTimeout);\n\n  /* 使用自定义 fetch 或原生 fetch */\n  const fetchFn = config.customFetch || fetch;\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"]}
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"retry.js","sourceRoot":"","sources":["../src/retry.ts"],"names":[],"mappings":";AAAA;;;GAGG;;AAwFH,8BAiCC;AAgCD,4CAiCC;AAxLD,qCAAkC;AAClC,qCAAqC;AAkBrC,MAAM,qBAAqB,GAA2B;IACpD,UAAU,EAAE,CAAC;IACb,YAAY,EAAE,IAAI;IAClB,QAAQ,EAAE,IAAI;IACd,OAAO,EAAE,KAAK;IACd,WAAW,EAAE,kBAAkB;CAChC,CAAC;AAEF;;;;;;;;;GASG;AACH,SAAS,kBAAkB,CAAC,KAAU;IACpC,IAAI,CAAC,KAAK;QAAE,OAAO,KAAK,CAAC;IAEzB,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,KAAK,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;IAEnE,iBAAiB;IACjB,IAAI,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAC;QAChC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC;QAC3B,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAC;QAChC,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAC;QAC9B,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC;QAC7B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC;QAC3B,OAAO,CAAC,QAAQ,CAAC,gBAAgB,CAAC;QAClC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC;QACvB,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QAC9B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,sBAAsB;IACtB,IAAI,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAC,EAAE,CAAC;QACvG,OAAO,IAAI,CAAC;IACd,CAAC;IAED,oCAAoC;IACpC,MAAM,WAAW,GAAG,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;IACjD,IAAI,WAAW,EAAE,CAAC;QAChB,MAAM,MAAM,GAAG,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAC5C,OAAO,MAAM,IAAI,GAAG,CAAC;IACvB,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;GAEG;AACH,SAAS,KAAK,CAAC,EAAU;IACvB,OAAO,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;AACzD,CAAC;AAED;;;;;;;;GAQG;AACI,KAAK,UAAU,SAAS,CAAI,EAAoB,EAAE,OAAsB;IAC7E,MAAM,IAAI,GAAG,EAAE,GAAG,qBAAqB,EAAE,GAAG,OAAO,EAAE,CAAC;IACtD,IAAI,SAAc,CAAC;IAEnB,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,IAAI,IAAI,CAAC,UAAU,EAAE,OAAO,EAAE,EAAE,CAAC;QAC5D,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,EAAE,EAAE,CAAC;YAC1B,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;gBAChB,eAAM,CAAC,IAAI,CAAC,KAAK,OAAO,GAAG,CAAC,QAAQ,CAAC,CAAC;YACxC,CAAC;YACD,OAAO,MAAM,CAAC;QAChB,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,SAAS,GAAG,KAAK,CAAC;YAClB,MAAM,QAAQ,GAAG,KAAK,CAAC,OAAO,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC;YAEhD,6BAA6B;YAC7B,IAAI,OAAO,IAAI,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC3D,IAAI,OAAO,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,GAAG,CAAC,EAAE,CAAC;oBACtD,eAAM,CAAC,KAAK,CAAC,MAAM,IAAI,CAAC,UAAU,WAAW,QAAQ,EAAE,CAAC,CAAC;gBAC3D,CAAC;qBAAM,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC;oBACpC,eAAM,CAAC,KAAK,CAAC,YAAY,QAAQ,EAAE,CAAC,CAAC;gBACvC,CAAC;gBACD,MAAM,KAAK,CAAC;YACd,CAAC;YAED,YAAY;YACZ,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;YAChF,eAAM,CAAC,IAAI,CAAC,SAAS,QAAQ,KAAK,KAAK,SAAS,OAAO,GAAG,CAAC,SAAS,CAAC,CAAC;YACtE,MAAM,KAAK,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC;IACH,CAAC;IAED,MAAM,SAAS,CAAC;AAClB,CAAC;AAED;;;;;;;GAOG;AACH;;;GAGG;AACH,IAAI,kBAAkB,GAAuE,IAAI,CAAC;AAElG;;GAEG;AACH,SAAS,cAAc;IACrB,MAAM,MAAM,GAAG,IAAA,kBAAS,GAAE,CAAC;IAC3B,qCAAqC;IACrC,IAAI,CAAC,kBAAkB,IAAI,kBAAkB,CAAC,OAAO,KAAK,CAAC,MAAM,CAAC,WAAW,IAAI,KAAK,CAAC,IAAI,kBAAkB,CAAC,OAAO,KAAK,CAAC,MAAM,CAAC,OAAO,IAAI,qBAAqB,CAAC,OAAO,CAAC,EAAE,CAAC;QAC5K,kBAAkB,GAAG;YACnB,OAAO,EAAE,MAAM,CAAC,WAAW,IAAI,KAAK;YACpC,OAAO,EAAE,MAAM,CAAC,OAAO,IAAI,qBAAqB,CAAC,OAAO;YACxD,OAAO,EAAE,CAAC;SACX,CAAC;IACJ,CAAC;IACD,OAAO,kBAAkB,CAAC;AAC5B,CAAC;AAEM,KAAK,UAAU,gBAAgB,CACpC,GAAW,EACX,IAAkB,EAClB,SAAkB;IAElB,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,cAAc,EAAE,GAAG,cAAc,EAAE,CAAC;IAC9D,MAAM,gBAAgB,GAAG,SAAS,IAAI,cAAc,CAAC;IACrD,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;IACzC,MAAM,SAAS,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,gBAAgB,CAAC,CAAC;IAEzE;;;OAGG;IACH,MAAM,cAAc,GAAG,IAAI,EAAE,MAAM,CAAC;IACpC,IAAI,cAAc,EAAE,CAAC;QACnB,cAAc,CAAC,gBAAgB,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;IACrF,CAAC;IAED,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,OAAO,CAAC,GAAG,EAAE;YAClC,GAAG,IAAI;YACP,MAAM,EAAE,UAAU,CAAC,MAAM;SAC1B,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC;IAClB,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,IAAI,KAAK,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;YAChC,MAAM,IAAI,KAAK,CAAC,yBAAyB,gBAAgB,OAAO,GAAG,EAAE,CAAC,CAAC;QACzE,CAAC;QACD,MAAM,KAAK,CAAC;IACd,CAAC;YAAS,CAAC;QACT,YAAY,CAAC,SAAS,CAAC,CAAC;IAC1B,CAAC;AACH,CAAC","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"]}
|
package/package.json
CHANGED
package/src/index.ts
CHANGED
|
@@ -264,7 +264,8 @@ async function getEmailsOnce(channel: Channel, email: string, token?: string): P
|
|
|
264
264
|
if (!token) throw new Error('internal error: token missing for tempmail-lol');
|
|
265
265
|
return tempmailLol.getEmails(token, email);
|
|
266
266
|
case 'chatgpt-org-uk':
|
|
267
|
-
|
|
267
|
+
if (!token) throw new Error('internal error: token missing for chatgpt-org-uk');
|
|
268
|
+
return chatgptOrgUk.getEmails(token, email);
|
|
268
269
|
case 'tempmail-la':
|
|
269
270
|
return tempmailLa.getEmails(email);
|
|
270
271
|
case 'temp-mail-io':
|
package/src/providers/awamail.ts
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'awamail';
|
|
5
6
|
const BASE_URL = 'https://awamail.com/welcome';
|
|
@@ -37,7 +38,7 @@ function extractSessionCookie(response: Response): string {
|
|
|
37
38
|
* 需要保存响应中的 Set-Cookie (awamail_session) 用于后续获取邮件
|
|
38
39
|
*/
|
|
39
40
|
export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
40
|
-
const response = await
|
|
41
|
+
const response = await fetchWithTimeout(`${BASE_URL}/change_mailbox`, {
|
|
41
42
|
method: 'POST',
|
|
42
43
|
headers: {
|
|
43
44
|
...DEFAULT_HEADERS,
|
|
@@ -77,7 +78,7 @@ export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
|
77
78
|
* 返回: { success, data: { emails: [...], latest: {...} } }
|
|
78
79
|
*/
|
|
79
80
|
export async function getEmails(token: string, email: string): Promise<Email[]> {
|
|
80
|
-
const response = await
|
|
81
|
+
const response = await fetchWithTimeout(`${BASE_URL}/get_emails`, {
|
|
81
82
|
method: 'GET',
|
|
82
83
|
headers: {
|
|
83
84
|
...DEFAULT_HEADERS,
|
|
@@ -1,47 +1,104 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'chatgpt-org-uk';
|
|
5
6
|
const BASE_URL = 'https://mail.chatgpt.org.uk/api';
|
|
7
|
+
const HOME_URL = 'https://mail.chatgpt.org.uk/';
|
|
6
8
|
|
|
7
9
|
const DEFAULT_HEADERS = {
|
|
8
10
|
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/143.0.0.0 Safari/537.36',
|
|
9
|
-
'Content-Type': 'application/json',
|
|
10
11
|
'Accept': '*/*',
|
|
11
12
|
'Referer': 'https://mail.chatgpt.org.uk/',
|
|
12
|
-
'
|
|
13
|
-
'sec-ch-ua-mobile': '?0',
|
|
14
|
-
'sec-ch-ua-platform': '"Windows"',
|
|
13
|
+
'Origin': 'https://mail.chatgpt.org.uk',
|
|
15
14
|
'DNT': '1',
|
|
16
15
|
};
|
|
17
16
|
|
|
18
|
-
|
|
19
|
-
const
|
|
17
|
+
function extractGmSid(response: Response): string {
|
|
18
|
+
const setCookie = response.headers.get('set-cookie') || '';
|
|
19
|
+
const match = setCookie.match(/gm_sid=([^;]+)/);
|
|
20
|
+
return match ? match[1] : '';
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async function fetchGmSid(): Promise<string> {
|
|
24
|
+
const response = await fetchWithTimeout(HOME_URL, {
|
|
20
25
|
method: 'GET',
|
|
21
26
|
headers: DEFAULT_HEADERS,
|
|
22
27
|
});
|
|
23
28
|
|
|
24
29
|
if (!response.ok) {
|
|
25
|
-
throw new Error(`Failed to
|
|
30
|
+
throw new Error(`Failed to fetch gm_sid: ${response.status}`);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const gmSid = extractGmSid(response);
|
|
34
|
+
if (!gmSid) {
|
|
35
|
+
throw new Error('Failed to extract gm_sid cookie');
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
return gmSid;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
async function fetchGmSidWithRetry(): Promise<string> {
|
|
42
|
+
try {
|
|
43
|
+
return await fetchGmSid();
|
|
44
|
+
} catch (error: any) {
|
|
45
|
+
const message = String(error?.message || error || '').toLowerCase();
|
|
46
|
+
if (message.includes('401') || message.includes('extract gm_sid')) {
|
|
47
|
+
return await fetchGmSid();
|
|
48
|
+
}
|
|
49
|
+
throw error;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
async function fetchInboxToken(email: string, gmSid: string): Promise<string> {
|
|
54
|
+
const response = await fetchWithTimeout(`${BASE_URL}/inbox-token`, {
|
|
55
|
+
method: 'POST',
|
|
56
|
+
headers: {
|
|
57
|
+
...DEFAULT_HEADERS,
|
|
58
|
+
'Content-Type': 'application/json',
|
|
59
|
+
'Cookie': `gm_sid=${gmSid}`,
|
|
60
|
+
},
|
|
61
|
+
body: JSON.stringify({ email }),
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
if (!response.ok) {
|
|
65
|
+
throw new Error(`Failed to get inbox token: ${response.status}`);
|
|
26
66
|
}
|
|
27
67
|
|
|
28
68
|
const data = await response.json();
|
|
29
|
-
|
|
30
|
-
if (!
|
|
31
|
-
throw new Error('Failed to
|
|
69
|
+
const token = data?.auth?.token;
|
|
70
|
+
if (!token) {
|
|
71
|
+
throw new Error('Failed to get inbox token');
|
|
32
72
|
}
|
|
33
73
|
|
|
34
|
-
return
|
|
35
|
-
channel: CHANNEL,
|
|
36
|
-
email: data.data.email,
|
|
37
|
-
};
|
|
74
|
+
return token;
|
|
38
75
|
}
|
|
39
76
|
|
|
40
|
-
|
|
77
|
+
async function fetchInboxTokenWithRetry(email: string): Promise<string> {
|
|
78
|
+
const gmSid = await fetchGmSidWithRetry();
|
|
79
|
+
try {
|
|
80
|
+
return await fetchInboxToken(email, gmSid);
|
|
81
|
+
} catch (error: any) {
|
|
82
|
+
const message = String(error?.message || error || '').toLowerCase();
|
|
83
|
+
if (message.includes('401')) {
|
|
84
|
+
const refreshedGmSid = await fetchGmSidWithRetry();
|
|
85
|
+
return await fetchInboxToken(email, refreshedGmSid);
|
|
86
|
+
}
|
|
87
|
+
throw error;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
async function fetchEmails(token: string, email: string): Promise<Email[]> {
|
|
92
|
+
if (!token) {
|
|
93
|
+
throw new Error('internal error: token missing for chatgpt-org-uk');
|
|
94
|
+
}
|
|
41
95
|
const encodedEmail = encodeURIComponent(email);
|
|
42
|
-
const response = await
|
|
96
|
+
const response = await fetchWithTimeout(`${BASE_URL}/emails?email=${encodedEmail}`, {
|
|
43
97
|
method: 'GET',
|
|
44
|
-
headers:
|
|
98
|
+
headers: {
|
|
99
|
+
...DEFAULT_HEADERS,
|
|
100
|
+
'x-inbox-token': token,
|
|
101
|
+
},
|
|
45
102
|
});
|
|
46
103
|
|
|
47
104
|
if (!response.ok) {
|
|
@@ -49,7 +106,7 @@ export async function getEmails(email: string): Promise<Email[]> {
|
|
|
49
106
|
}
|
|
50
107
|
|
|
51
108
|
const data = await response.json();
|
|
52
|
-
|
|
109
|
+
|
|
53
110
|
if (!data.success) {
|
|
54
111
|
throw new Error('Failed to get emails');
|
|
55
112
|
}
|
|
@@ -57,3 +114,59 @@ export async function getEmails(email: string): Promise<Email[]> {
|
|
|
57
114
|
const rawEmails = data.data?.emails || [];
|
|
58
115
|
return rawEmails.map((raw: any) => normalizeEmail(raw, email));
|
|
59
116
|
}
|
|
117
|
+
|
|
118
|
+
async function fetchEmailsWithRetry(email: string): Promise<Email[]> {
|
|
119
|
+
const token = await fetchInboxTokenWithRetry(email);
|
|
120
|
+
try {
|
|
121
|
+
return await fetchEmails(token, email);
|
|
122
|
+
} catch (error: any) {
|
|
123
|
+
const message = String(error?.message || error || '').toLowerCase();
|
|
124
|
+
if (message.includes('401')) {
|
|
125
|
+
const refreshedToken = await fetchInboxTokenWithRetry(email);
|
|
126
|
+
return await fetchEmails(refreshedToken, email);
|
|
127
|
+
}
|
|
128
|
+
throw error;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
133
|
+
const response = await fetchWithTimeout(`${BASE_URL}/generate-email`, {
|
|
134
|
+
method: 'GET',
|
|
135
|
+
headers: DEFAULT_HEADERS,
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
if (!response.ok) {
|
|
139
|
+
throw new Error(`Failed to generate email: ${response.status}`);
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
const data = await response.json();
|
|
143
|
+
|
|
144
|
+
if (!data.success) {
|
|
145
|
+
throw new Error('Failed to generate email');
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
const email = data.data.email;
|
|
149
|
+
const token = await fetchInboxTokenWithRetry(email);
|
|
150
|
+
|
|
151
|
+
return {
|
|
152
|
+
channel: CHANNEL,
|
|
153
|
+
email,
|
|
154
|
+
token,
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
export async function getEmails(token: string, email: string): Promise<Email[]> {
|
|
159
|
+
if (!token) {
|
|
160
|
+
throw new Error('internal error: token missing for chatgpt-org-uk');
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
try {
|
|
164
|
+
return await fetchEmails(token, email);
|
|
165
|
+
} catch (error: any) {
|
|
166
|
+
const message = String(error?.message || error || '').toLowerCase();
|
|
167
|
+
if (message.includes('401')) {
|
|
168
|
+
return await fetchEmailsWithRetry(email);
|
|
169
|
+
}
|
|
170
|
+
throw error;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'dropmail';
|
|
5
6
|
const BASE_URL = 'https://dropmail.me/api/graphql/MY_TOKEN';
|
|
@@ -29,7 +30,7 @@ async function graphqlRequest(query: string, variables?: Record<string, any>): P
|
|
|
29
30
|
params.set('variables', JSON.stringify(variables));
|
|
30
31
|
}
|
|
31
32
|
|
|
32
|
-
const response = await
|
|
33
|
+
const response = await fetchWithTimeout(BASE_URL, {
|
|
33
34
|
method: 'POST',
|
|
34
35
|
headers: DEFAULT_HEADERS,
|
|
35
36
|
body: params.toString(),
|
|
@@ -10,6 +10,7 @@
|
|
|
10
10
|
|
|
11
11
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
12
12
|
import { normalizeEmail } from '../normalize';
|
|
13
|
+
import { fetchWithTimeout } from '../retry';
|
|
13
14
|
|
|
14
15
|
const CHANNEL: Channel = 'guerrillamail';
|
|
15
16
|
const BASE_URL = 'https://api.guerrillamail.com/ajax.php';
|
|
@@ -20,7 +21,7 @@ const BASE_URL = 'https://api.guerrillamail.com/ajax.php';
|
|
|
20
21
|
* 返回 email_addr + sid_token(用于后续获取邮件)
|
|
21
22
|
*/
|
|
22
23
|
export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
23
|
-
const response = await
|
|
24
|
+
const response = await fetchWithTimeout(`${BASE_URL}?f=get_email_address&lang=en`, {
|
|
24
25
|
method: 'GET',
|
|
25
26
|
headers: {
|
|
26
27
|
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
|
|
@@ -51,7 +52,7 @@ export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
|
51
52
|
* 返回 list 数组,每个元素包含 mail_id, mail_from, mail_subject, mail_body 等
|
|
52
53
|
*/
|
|
53
54
|
export async function getEmails(token: string, email: string): Promise<Email[]> {
|
|
54
|
-
const response = await
|
|
55
|
+
const response = await fetchWithTimeout(`${BASE_URL}?f=check_email&seq=0&sid_token=${encodeURIComponent(token)}`, {
|
|
55
56
|
method: 'GET',
|
|
56
57
|
headers: {
|
|
57
58
|
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'linshi-email';
|
|
5
6
|
const BASE_URL = 'https://www.linshi-email.com/api/v1';
|
|
@@ -17,7 +18,7 @@ const DEFAULT_HEADERS = {
|
|
|
17
18
|
};
|
|
18
19
|
|
|
19
20
|
export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
20
|
-
const response = await
|
|
21
|
+
const response = await fetchWithTimeout(`${BASE_URL}/email/${API_KEY}`, {
|
|
21
22
|
method: 'POST',
|
|
22
23
|
headers: DEFAULT_HEADERS,
|
|
23
24
|
body: JSON.stringify({}),
|
|
@@ -43,7 +44,7 @@ export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
|
43
44
|
export async function getEmails(email: string): Promise<Email[]> {
|
|
44
45
|
const encodedEmail = encodeURIComponent(email);
|
|
45
46
|
const timestamp = Date.now();
|
|
46
|
-
const response = await
|
|
47
|
+
const response = await fetchWithTimeout(`${BASE_URL}/refreshmessage/${API_KEY}/${encodedEmail}?t=${timestamp}`, {
|
|
47
48
|
method: 'GET',
|
|
48
49
|
headers: DEFAULT_HEADERS,
|
|
49
50
|
});
|
package/src/providers/mail-tm.ts
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'mail-tm';
|
|
5
6
|
const BASE_URL = 'https://api.mail.tm';
|
|
@@ -26,7 +27,7 @@ function randomString(length: number): string {
|
|
|
26
27
|
* API: GET /domains
|
|
27
28
|
*/
|
|
28
29
|
async function getDomains(): Promise<string[]> {
|
|
29
|
-
const response = await
|
|
30
|
+
const response = await fetchWithTimeout(`${BASE_URL}/domains`, {
|
|
30
31
|
method: 'GET',
|
|
31
32
|
headers: DEFAULT_HEADERS,
|
|
32
33
|
});
|
|
@@ -51,7 +52,7 @@ async function getDomains(): Promise<string[]> {
|
|
|
51
52
|
* API: POST /accounts
|
|
52
53
|
*/
|
|
53
54
|
async function createAccount(address: string, password: string): Promise<any> {
|
|
54
|
-
const response = await
|
|
55
|
+
const response = await fetchWithTimeout(`${BASE_URL}/accounts`, {
|
|
55
56
|
method: 'POST',
|
|
56
57
|
headers: { ...DEFAULT_HEADERS, 'Content-Type': 'application/ld+json' },
|
|
57
58
|
body: JSON.stringify({ address, password }),
|
|
@@ -70,7 +71,7 @@ async function createAccount(address: string, password: string): Promise<any> {
|
|
|
70
71
|
* API: POST /token
|
|
71
72
|
*/
|
|
72
73
|
async function getToken(address: string, password: string): Promise<string> {
|
|
73
|
-
const response = await
|
|
74
|
+
const response = await fetchWithTimeout(`${BASE_URL}/token`, {
|
|
74
75
|
method: 'POST',
|
|
75
76
|
headers: DEFAULT_HEADERS,
|
|
76
77
|
body: JSON.stringify({ address, password }),
|
|
@@ -144,7 +145,7 @@ function flattenMessage(msg: any, recipientEmail: string): any {
|
|
|
144
145
|
*/
|
|
145
146
|
export async function getEmails(token: string, email: string): Promise<Email[]> {
|
|
146
147
|
// 1. 获取邮件列表
|
|
147
|
-
const listResponse = await
|
|
148
|
+
const listResponse = await fetchWithTimeout(`${BASE_URL}/messages`, {
|
|
148
149
|
method: 'GET',
|
|
149
150
|
headers: {
|
|
150
151
|
...DEFAULT_HEADERS,
|
|
@@ -167,7 +168,7 @@ export async function getEmails(token: string, email: string): Promise<Email[]>
|
|
|
167
168
|
// 2. 并行获取每封邮件的详情(含 text/html/attachments)
|
|
168
169
|
const detailPromises = messages.map(async (msg: any) => {
|
|
169
170
|
try {
|
|
170
|
-
const detailResponse = await
|
|
171
|
+
const detailResponse = await fetchWithTimeout(`${BASE_URL}/messages/${msg.id}`, {
|
|
171
172
|
method: 'GET',
|
|
172
173
|
headers: {
|
|
173
174
|
...DEFAULT_HEADERS,
|
|
@@ -10,6 +10,7 @@
|
|
|
10
10
|
*/
|
|
11
11
|
|
|
12
12
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
13
|
+
import { fetchWithTimeout } from '../retry';
|
|
13
14
|
|
|
14
15
|
const CHANNEL: Channel = 'maildrop';
|
|
15
16
|
const GRAPHQL_URL = 'https://api.maildrop.cc/graphql';
|
|
@@ -113,7 +114,7 @@ async function graphqlRequest(
|
|
|
113
114
|
query: string,
|
|
114
115
|
variables: Record<string, string> = {},
|
|
115
116
|
): Promise<any> {
|
|
116
|
-
const response = await
|
|
117
|
+
const response = await fetchWithTimeout(GRAPHQL_URL, {
|
|
117
118
|
method: 'POST',
|
|
118
119
|
headers: {
|
|
119
120
|
'Content-Type': 'application/json',
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'temp-mail-io';
|
|
5
6
|
const BASE_URL = 'https://api.internal.temp-mail.io/api/v3';
|
|
@@ -18,7 +19,7 @@ async function fetchCorsHeader(): Promise<string> {
|
|
|
18
19
|
if (cachedCorsHeader) return cachedCorsHeader;
|
|
19
20
|
|
|
20
21
|
try {
|
|
21
|
-
const response = await
|
|
22
|
+
const response = await fetchWithTimeout(PAGE_URL, {
|
|
22
23
|
headers: {
|
|
23
24
|
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/144.0.0.0 Safari/537.36',
|
|
24
25
|
},
|
|
@@ -61,7 +62,7 @@ async function getApiHeaders(): Promise<Record<string, string>> {
|
|
|
61
62
|
*/
|
|
62
63
|
export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
63
64
|
const headers = await getApiHeaders();
|
|
64
|
-
const response = await
|
|
65
|
+
const response = await fetchWithTimeout(`${BASE_URL}/email/new`, {
|
|
65
66
|
method: 'POST',
|
|
66
67
|
headers,
|
|
67
68
|
body: JSON.stringify({ min_name_length: 10, max_name_length: 10 }),
|
|
@@ -91,7 +92,7 @@ export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
|
91
92
|
*/
|
|
92
93
|
export async function getEmails(email: string): Promise<Email[]> {
|
|
93
94
|
const headers = await getApiHeaders();
|
|
94
|
-
const response = await
|
|
95
|
+
const response = await fetchWithTimeout(`${BASE_URL}/email/${email}/messages`, {
|
|
95
96
|
method: 'GET',
|
|
96
97
|
headers,
|
|
97
98
|
});
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'tempmail-la';
|
|
5
6
|
const BASE_URL = 'https://tempmail.la/api';
|
|
@@ -31,7 +32,7 @@ const DEFAULT_HEADERS: Record<string, string> = {
|
|
|
31
32
|
* 返回: { code: 0, data: { mailId, address, type, startAt, endAt } }
|
|
32
33
|
*/
|
|
33
34
|
export async function generateEmail(): Promise<InternalEmailInfo> {
|
|
34
|
-
const response = await
|
|
35
|
+
const response = await fetchWithTimeout(`${BASE_URL}/mail/create`, {
|
|
35
36
|
method: 'POST',
|
|
36
37
|
headers: DEFAULT_HEADERS,
|
|
37
38
|
body: JSON.stringify({ turnstile: '' }),
|
|
@@ -69,7 +70,7 @@ export async function getEmails(email: string): Promise<Email[]> {
|
|
|
69
70
|
|
|
70
71
|
// 支持分页,循环获取所有邮件
|
|
71
72
|
while (hasMore) {
|
|
72
|
-
const response: Response = await
|
|
73
|
+
const response: Response = await fetchWithTimeout(`${BASE_URL}/mail/box`, {
|
|
73
74
|
method: 'POST',
|
|
74
75
|
headers: DEFAULT_HEADERS,
|
|
75
76
|
body: JSON.stringify({ address: email, cursor }),
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'tempmail-lol';
|
|
5
6
|
const BASE_URL = 'https://api.tempmail.lol/v2';
|
|
@@ -15,7 +16,7 @@ const DEFAULT_HEADERS = {
|
|
|
15
16
|
};
|
|
16
17
|
|
|
17
18
|
export async function generateEmail(domain: string | null = null): Promise<InternalEmailInfo> {
|
|
18
|
-
const response = await
|
|
19
|
+
const response = await fetchWithTimeout(`${BASE_URL}/inbox/create`, {
|
|
19
20
|
method: 'POST',
|
|
20
21
|
headers: DEFAULT_HEADERS,
|
|
21
22
|
body: JSON.stringify({ domain, captcha: null }),
|
|
@@ -39,7 +40,7 @@ export async function generateEmail(domain: string | null = null): Promise<Inter
|
|
|
39
40
|
}
|
|
40
41
|
|
|
41
42
|
export async function getEmails(token: string, recipientEmail: string = ''): Promise<Email[]> {
|
|
42
|
-
const response = await
|
|
43
|
+
const response = await fetchWithTimeout(`${BASE_URL}/inbox?token=${encodeURIComponent(token)}`, {
|
|
43
44
|
method: 'GET',
|
|
44
45
|
headers: DEFAULT_HEADERS,
|
|
45
46
|
});
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { InternalEmailInfo, Email, Channel } from '../types';
|
|
2
2
|
import { normalizeEmail } from '../normalize';
|
|
3
|
+
import { fetchWithTimeout } from '../retry';
|
|
3
4
|
|
|
4
5
|
const CHANNEL: Channel = 'tempmail';
|
|
5
6
|
const BASE_URL = 'https://api.tempmail.ing/api';
|
|
@@ -15,7 +16,7 @@ const DEFAULT_HEADERS = {
|
|
|
15
16
|
};
|
|
16
17
|
|
|
17
18
|
export async function generateEmail(duration: number = 30): Promise<InternalEmailInfo> {
|
|
18
|
-
const response = await
|
|
19
|
+
const response = await fetchWithTimeout(`${BASE_URL}/generate`, {
|
|
19
20
|
method: 'POST',
|
|
20
21
|
headers: DEFAULT_HEADERS,
|
|
21
22
|
body: JSON.stringify({ duration }),
|
|
@@ -41,7 +42,7 @@ export async function generateEmail(duration: number = 30): Promise<InternalEmai
|
|
|
41
42
|
|
|
42
43
|
export async function getEmails(email: string): Promise<Email[]> {
|
|
43
44
|
const encodedEmail = encodeURIComponent(email);
|
|
44
|
-
const response = await
|
|
45
|
+
const response = await fetchWithTimeout(`${BASE_URL}/emails/${encodedEmail}`, {
|
|
45
46
|
method: 'GET',
|
|
46
47
|
headers: DEFAULT_HEADERS,
|
|
47
48
|
});
|