tempmail-sdk 1.0.1 → 1.0.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/README.md +162 -72
- package/demo/poll-emails.ts +25 -40
- package/dist/index.d.ts +96 -2
- package/dist/index.js +186 -21
- package/dist/logger.d.ts +73 -0
- package/dist/logger.js +100 -0
- package/dist/normalize.d.ts +12 -0
- package/dist/normalize.js +132 -0
- package/dist/providers/awamail.d.ts +15 -0
- package/dist/providers/awamail.js +91 -0
- package/dist/providers/chatgpt-org-uk.js +4 -2
- package/dist/providers/dropmail.d.ts +13 -0
- package/dist/providers/dropmail.js +86 -0
- package/dist/providers/linshi-email.js +4 -2
- package/dist/providers/mail-tm.d.ts +11 -0
- package/dist/providers/mail-tm.js +172 -0
- package/dist/providers/temp-mail-io.d.ts +13 -0
- package/dist/providers/temp-mail-io.js +99 -0
- package/dist/providers/tempmail-la.d.ts +15 -0
- package/dist/providers/tempmail-la.js +89 -0
- package/dist/providers/tempmail-lol.d.ts +1 -1
- package/dist/providers/tempmail-lol.js +5 -3
- package/dist/providers/tempmail.js +4 -2
- package/dist/retry.d.ts +38 -0
- package/dist/retry.js +121 -0
- package/dist/types.d.ts +118 -28
- package/dist/types.js +1 -1
- package/package.json +1 -1
- package/src/index.ts +182 -25
- package/src/logger.ts +106 -0
- package/src/normalize.ts +125 -0
- package/src/providers/awamail.ts +101 -0
- package/src/providers/chatgpt-org-uk.ts +3 -1
- package/src/providers/dropmail.ts +98 -0
- package/src/providers/linshi-email.ts +3 -1
- package/src/providers/mail-tm.ts +193 -0
- package/src/providers/temp-mail-io.ts +108 -0
- package/src/providers/tempmail-la.ts +99 -0
- package/src/providers/tempmail-lol.ts +4 -2
- package/src/providers/tempmail.ts +3 -1
- package/src/retry.ts +146 -0
- package/src/types.ts +120 -28
- package/test/example.ts +16 -1
package/dist/retry.js
ADDED
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* 通用重试工具
|
|
4
|
+
* 提供请求重试、超时控制、指数退避等错误恢复机制
|
|
5
|
+
*/
|
|
6
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
|
+
exports.withRetry = withRetry;
|
|
8
|
+
exports.fetchWithTimeout = fetchWithTimeout;
|
|
9
|
+
const logger_1 = require("./logger");
|
|
10
|
+
const DEFAULT_RETRY_OPTIONS = {
|
|
11
|
+
maxRetries: 2,
|
|
12
|
+
initialDelay: 1000,
|
|
13
|
+
maxDelay: 5000,
|
|
14
|
+
timeout: 15000,
|
|
15
|
+
shouldRetry: defaultShouldRetry,
|
|
16
|
+
};
|
|
17
|
+
/**
|
|
18
|
+
* 默认重试判断
|
|
19
|
+
* 网络错误、超时、HTTP 4xx/5xx 错误均可重试
|
|
20
|
+
* 仅参数校验类错误(由 SDK 内部抛出)不重试
|
|
21
|
+
*/
|
|
22
|
+
function defaultShouldRetry(error) {
|
|
23
|
+
if (!error)
|
|
24
|
+
return false;
|
|
25
|
+
const message = String(error.message || error || '').toLowerCase();
|
|
26
|
+
/* 网络级别错误 → 重试 */
|
|
27
|
+
if (message.includes('fetch failed') ||
|
|
28
|
+
message.includes('network') ||
|
|
29
|
+
message.includes('econnrefused') ||
|
|
30
|
+
message.includes('econnreset') ||
|
|
31
|
+
message.includes('etimedout') ||
|
|
32
|
+
message.includes('timeout') ||
|
|
33
|
+
message.includes('socket hang up') ||
|
|
34
|
+
message.includes('dns') ||
|
|
35
|
+
message.includes('abort')) {
|
|
36
|
+
return true;
|
|
37
|
+
}
|
|
38
|
+
/* HTTP 4xx/5xx 错误 → 重试 */
|
|
39
|
+
const statusMatch = message.match(/:\s*(\d{3})/);
|
|
40
|
+
if (statusMatch) {
|
|
41
|
+
const status = parseInt(statusMatch[1], 10);
|
|
42
|
+
return status >= 400;
|
|
43
|
+
}
|
|
44
|
+
return false;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* 休眠指定毫秒
|
|
48
|
+
*/
|
|
49
|
+
function sleep(ms) {
|
|
50
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* 带重试的异步操作执行器
|
|
54
|
+
* - 自动重试可恢复的错误(网络错误、超时、5xx)
|
|
55
|
+
* - 指数退避避免过度请求
|
|
56
|
+
* - 不可恢复的错误(4xx 参数错误等)直接抛出不重试
|
|
57
|
+
*
|
|
58
|
+
* @param fn 要执行的异步操作
|
|
59
|
+
* @param options 重试配置
|
|
60
|
+
*/
|
|
61
|
+
async function withRetry(fn, options) {
|
|
62
|
+
const opts = { ...DEFAULT_RETRY_OPTIONS, ...options };
|
|
63
|
+
let lastError;
|
|
64
|
+
for (let attempt = 0; attempt <= opts.maxRetries; attempt++) {
|
|
65
|
+
try {
|
|
66
|
+
const result = await fn();
|
|
67
|
+
if (attempt > 0) {
|
|
68
|
+
logger_1.logger.info(`第 ${attempt + 1} 次尝试成功`);
|
|
69
|
+
}
|
|
70
|
+
return result;
|
|
71
|
+
}
|
|
72
|
+
catch (error) {
|
|
73
|
+
lastError = error;
|
|
74
|
+
const errorMsg = error.message || String(error);
|
|
75
|
+
/* 最后一次尝试失败或不可重试的错误 → 直接抛出 */
|
|
76
|
+
if (attempt >= opts.maxRetries || !opts.shouldRetry(error)) {
|
|
77
|
+
if (attempt >= opts.maxRetries && opts.maxRetries > 0) {
|
|
78
|
+
logger_1.logger.error(`重试 ${opts.maxRetries} 次后仍失败: ${errorMsg}`);
|
|
79
|
+
}
|
|
80
|
+
else if (!opts.shouldRetry(error)) {
|
|
81
|
+
logger_1.logger.debug(`不可重试的错误: ${errorMsg}`);
|
|
82
|
+
}
|
|
83
|
+
throw error;
|
|
84
|
+
}
|
|
85
|
+
/* 指数退避等待 */
|
|
86
|
+
const delay = Math.min(opts.initialDelay * Math.pow(2, attempt), opts.maxDelay);
|
|
87
|
+
logger_1.logger.warn(`请求失败 (${errorMsg}),${delay}ms 后第 ${attempt + 2} 次重试...`);
|
|
88
|
+
await sleep(delay);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
throw lastError;
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* 带超时控制的 fetch 包装
|
|
95
|
+
* 在原生 fetch 基础上添加超时中断能力
|
|
96
|
+
*
|
|
97
|
+
* @param url 请求 URL
|
|
98
|
+
* @param init fetch 选项
|
|
99
|
+
* @param timeoutMs 超时时间(毫秒)
|
|
100
|
+
*/
|
|
101
|
+
async function fetchWithTimeout(url, init, timeoutMs = DEFAULT_RETRY_OPTIONS.timeout) {
|
|
102
|
+
const controller = new AbortController();
|
|
103
|
+
const timeoutId = setTimeout(() => controller.abort(), timeoutMs);
|
|
104
|
+
try {
|
|
105
|
+
const response = await fetch(url, {
|
|
106
|
+
...init,
|
|
107
|
+
signal: controller.signal,
|
|
108
|
+
});
|
|
109
|
+
return response;
|
|
110
|
+
}
|
|
111
|
+
catch (error) {
|
|
112
|
+
if (error.name === 'AbortError') {
|
|
113
|
+
throw new Error(`Request timeout after ${timeoutMs}ms: ${url}`);
|
|
114
|
+
}
|
|
115
|
+
throw error;
|
|
116
|
+
}
|
|
117
|
+
finally {
|
|
118
|
+
clearTimeout(timeoutId);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
//# 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';\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 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 = DEFAULT_RETRY_OPTIONS.timeout,\n): Promise<Response> {\n  const controller = new AbortController();\n  const timeoutId = setTimeout(() => controller.abort(), timeoutMs);\n\n  try {\n    const response = await fetch(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 ${timeoutMs}ms: ${url}`);\n    }\n    throw error;\n  } finally {\n    clearTimeout(timeoutId);\n  }\n}\n"]}
|
package/dist/types.d.ts
CHANGED
|
@@ -1,53 +1,143 @@
|
|
|
1
|
-
|
|
1
|
+
/**
|
|
2
|
+
* 支持的临时邮箱渠道标识
|
|
3
|
+
* 每个渠道对应一个第三方临时邮箱服务商
|
|
4
|
+
*/
|
|
5
|
+
export type Channel = 'tempmail' | 'linshi-email' | 'tempmail-lol' | 'chatgpt-org-uk' | 'tempmail-la' | 'temp-mail-io' | 'awamail' | 'mail-tm' | 'dropmail';
|
|
6
|
+
/**
|
|
7
|
+
* 创建临时邮箱后返回的邮箱信息
|
|
8
|
+
* 包含邮箱地址、认证令牌和生命周期信息
|
|
9
|
+
*/
|
|
2
10
|
export interface EmailInfo {
|
|
11
|
+
/** 创建该邮箱所使用的渠道 */
|
|
3
12
|
channel: Channel;
|
|
13
|
+
/** 临时邮箱地址 */
|
|
4
14
|
email: string;
|
|
15
|
+
/** 认证令牌,部分渠道在获取邮件时需要此令牌 */
|
|
5
16
|
token?: string;
|
|
17
|
+
/** 邮箱过期时间(ISO 8601 字符串或 Unix 时间戳) */
|
|
6
18
|
expiresAt?: string | number;
|
|
19
|
+
/** 邮箱创建时间(ISO 8601 字符串) */
|
|
7
20
|
createdAt?: string;
|
|
8
21
|
}
|
|
22
|
+
/**
|
|
23
|
+
* 标准化邮件附件
|
|
24
|
+
* 不同渠道的附件字段名不同,SDK 统一归一化为此结构
|
|
25
|
+
*/
|
|
26
|
+
export interface EmailAttachment {
|
|
27
|
+
/** 附件文件名 */
|
|
28
|
+
filename: string;
|
|
29
|
+
/** 附件大小(字节) */
|
|
30
|
+
size?: number;
|
|
31
|
+
/** MIME 类型,如 application/pdf */
|
|
32
|
+
contentType?: string;
|
|
33
|
+
/** 附件下载地址 */
|
|
34
|
+
url?: string;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* 标准化邮件格式 - 所有提供商返回统一结构
|
|
38
|
+
*/
|
|
9
39
|
export interface Email {
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
from
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
date?: string | number;
|
|
29
|
-
e_date?: number;
|
|
30
|
-
timestamp?: number;
|
|
31
|
-
received_at?: string;
|
|
32
|
-
created_at?: string;
|
|
33
|
-
createdAt?: string;
|
|
34
|
-
is_read?: number;
|
|
35
|
-
has_html?: boolean;
|
|
36
|
-
[key: string]: any;
|
|
40
|
+
/** 邮件唯一标识 */
|
|
41
|
+
id: string;
|
|
42
|
+
/** 发件人邮箱地址 */
|
|
43
|
+
from: string;
|
|
44
|
+
/** 收件人邮箱地址 */
|
|
45
|
+
to: string;
|
|
46
|
+
/** 邮件主题 */
|
|
47
|
+
subject: string;
|
|
48
|
+
/** 纯文本内容 */
|
|
49
|
+
text: string;
|
|
50
|
+
/** HTML 内容 */
|
|
51
|
+
html: string;
|
|
52
|
+
/** ISO 8601 格式的日期字符串 */
|
|
53
|
+
date: string;
|
|
54
|
+
/** 是否已读 */
|
|
55
|
+
isRead: boolean;
|
|
56
|
+
/** 附件列表 */
|
|
57
|
+
attachments: EmailAttachment[];
|
|
37
58
|
}
|
|
59
|
+
/**
|
|
60
|
+
* 获取邮件列表的返回结果
|
|
61
|
+
* success 为 false 时表示请求失败(重试耗尽),emails 为空数组
|
|
62
|
+
*/
|
|
38
63
|
export interface GetEmailsResult {
|
|
64
|
+
/** 所使用的渠道 */
|
|
39
65
|
channel: Channel;
|
|
66
|
+
/** 查询的邮箱地址 */
|
|
40
67
|
email: string;
|
|
68
|
+
/** 邮件列表,失败时为空数组 */
|
|
41
69
|
emails: Email[];
|
|
70
|
+
/** 请求是否成功,false 表示重试耗尽后仍失败 */
|
|
42
71
|
success: boolean;
|
|
43
72
|
}
|
|
73
|
+
/**
|
|
74
|
+
* 重试配置
|
|
75
|
+
* SDK 内部对网络错误、超时、5xx 服务端错误自动重试
|
|
76
|
+
* 4xx 客户端错误(如参数错误)不会重试
|
|
77
|
+
*
|
|
78
|
+
* @example
|
|
79
|
+
* ```ts
|
|
80
|
+
* // 自定义重试策略:最多重试 3 次,首次延迟 2 秒
|
|
81
|
+
* const email = await generateEmail({
|
|
82
|
+
* channel: 'mail-tm',
|
|
83
|
+
* retry: { maxRetries: 3, initialDelay: 2000 },
|
|
84
|
+
* });
|
|
85
|
+
* ```
|
|
86
|
+
*/
|
|
87
|
+
export interface RetryConfig {
|
|
88
|
+
/** 最大重试次数(不含首次请求),默认 2 */
|
|
89
|
+
maxRetries?: number;
|
|
90
|
+
/** 初始重试延迟(毫秒),采用指数退避策略,默认 1000 */
|
|
91
|
+
initialDelay?: number;
|
|
92
|
+
/** 最大重试延迟上限(毫秒),默认 5000 */
|
|
93
|
+
maxDelay?: number;
|
|
94
|
+
/** 单次请求超时时间(毫秒),默认 15000 */
|
|
95
|
+
timeout?: number;
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* 创建临时邮箱的选项
|
|
99
|
+
*
|
|
100
|
+
* @example
|
|
101
|
+
* ```ts
|
|
102
|
+
* // 使用指定渠道创建邮箱
|
|
103
|
+
* const email = await generateEmail({ channel: 'mail-tm' });
|
|
104
|
+
*
|
|
105
|
+
* // 随机选择渠道
|
|
106
|
+
* const email = await generateEmail();
|
|
107
|
+
* ```
|
|
108
|
+
*/
|
|
44
109
|
export interface GenerateEmailOptions {
|
|
110
|
+
/** 指定渠道,不传则随机选择 */
|
|
45
111
|
channel?: Channel;
|
|
112
|
+
/** 邮箱有效时长 */
|
|
46
113
|
duration?: number;
|
|
114
|
+
/** 指定邮箱域名 */
|
|
47
115
|
domain?: string | null;
|
|
116
|
+
/** 重试配置,不传则使用默认值(最多重试 2 次) */
|
|
117
|
+
retry?: RetryConfig;
|
|
48
118
|
}
|
|
119
|
+
/**
|
|
120
|
+
* 获取邮件列表的选项
|
|
121
|
+
*
|
|
122
|
+
* @example
|
|
123
|
+
* ```ts
|
|
124
|
+
* const result = await getEmails({
|
|
125
|
+
* channel: emailInfo.channel,
|
|
126
|
+
* email: emailInfo.email,
|
|
127
|
+
* token: emailInfo.token,
|
|
128
|
+
* });
|
|
129
|
+
* if (result.success && result.emails.length > 0) {
|
|
130
|
+
* console.log('收到邮件:', result.emails);
|
|
131
|
+
* }
|
|
132
|
+
* ```
|
|
133
|
+
*/
|
|
49
134
|
export interface GetEmailsOptions {
|
|
135
|
+
/** 渠道标识,必填 */
|
|
50
136
|
channel: Channel;
|
|
137
|
+
/** 邮箱地址,必填 */
|
|
51
138
|
email: string;
|
|
139
|
+
/** 认证令牌 */
|
|
52
140
|
token?: string;
|
|
141
|
+
/** 重试配置,不传则使用默认值(最多重试 2 次) */
|
|
142
|
+
retry?: RetryConfig;
|
|
53
143
|
}
|
package/dist/types.js
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICog5pSv5oyB55qE5Li05pe26YKu566x5rig6YGT5qCH6K+GXG4gKiDmr4/kuKrmuKDpgZPlr7nlupTkuIDkuKrnrKzkuInmlrnkuLTml7bpgq7nrrHmnI3liqHllYZcbiAqL1xuZXhwb3J0IHR5cGUgQ2hhbm5lbCA9ICd0ZW1wbWFpbCcgfCAnbGluc2hpLWVtYWlsJyB8ICd0ZW1wbWFpbC1sb2wnIHwgJ2NoYXRncHQtb3JnLXVrJyB8ICd0ZW1wbWFpbC1sYScgfCAndGVtcC1tYWlsLWlvJyB8ICdhd2FtYWlsJyB8ICdtYWlsLXRtJyB8ICdkcm9wbWFpbCc7XG5cbi8qKlxuICog5Yib5bu65Li05pe26YKu566x5ZCO6L+U5Zue55qE6YKu566x5L+h5oGvXG4gKiDljIXlkKvpgq7nrrHlnLDlnYDjgIHorqTor4Hku6TniYzlkoznlJ/lkb3lkajmnJ/kv6Hmga9cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFbWFpbEluZm8ge1xuICAvKiog5Yib5bu66K+l6YKu566x5omA5L2/55So55qE5rig6YGTICovXG4gIGNoYW5uZWw6IENoYW5uZWw7XG4gIC8qKiDkuLTml7bpgq7nrrHlnLDlnYAgKi9cbiAgZW1haWw6IHN0cmluZztcbiAgLyoqIOiupOivgeS7pOeJjO+8jOmDqOWIhua4oOmBk+WcqOiOt+WPlumCruS7tuaXtumcgOimgeatpOS7pOeJjCAqL1xuICB0b2tlbj86IHN0cmluZztcbiAgLyoqIOmCrueusei/h+acn+aXtumXtO+8iElTTyA4NjAxIOWtl+espuS4suaIliBVbml4IOaXtumXtOaIs++8iSAqL1xuICBleHBpcmVzQXQ/OiBzdHJpbmcgfCBudW1iZXI7XG4gIC8qKiDpgq7nrrHliJvlu7rml7bpl7TvvIhJU08gODYwMSDlrZfnrKbkuLLvvIkgKi9cbiAgY3JlYXRlZEF0Pzogc3RyaW5nO1xufVxuXG4vKipcbiAqIOagh+WHhuWMlumCruS7tumZhOS7tlxuICog5LiN5ZCM5rig6YGT55qE6ZmE5Lu25a2X5q615ZCN5LiN5ZCM77yMU0RLIOe7n+S4gOW9kuS4gOWMluS4uuatpOe7k+aehFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEVtYWlsQXR0YWNobWVudCB7XG4gIC8qKiDpmYTku7bmlofku7blkI0gKi9cbiAgZmlsZW5hbWU6IHN0cmluZztcbiAgLyoqIOmZhOS7tuWkp+Wwj++8iOWtl+iKgu+8iSAqL1xuICBzaXplPzogbnVtYmVyO1xuICAvKiogTUlNRSDnsbvlnovvvIzlpoIgYXBwbGljYXRpb24vcGRmICovXG4gIGNvbnRlbnRUeXBlPzogc3RyaW5nO1xuICAvKiog6ZmE5Lu25LiL6L295Zyw5Z2AICovXG4gIHVybD86IHN0cmluZztcbn1cblxuLyoqXG4gKiDmoIflh4bljJbpgq7ku7bmoLzlvI8gLSDmiYDmnInmj5DkvpvllYbov5Tlm57nu5/kuIDnu5PmnoRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFbWFpbCB7XG4gIC8qKiDpgq7ku7bllK/kuIDmoIfor4YgKi9cbiAgaWQ6IHN0cmluZztcbiAgLyoqIOWPkeS7tuS6uumCrueuseWcsOWdgCAqL1xuICBmcm9tOiBzdHJpbmc7XG4gIC8qKiDmlLbku7bkurrpgq7nrrHlnLDlnYAgKi9cbiAgdG86IHN0cmluZztcbiAgLyoqIOmCruS7tuS4u+mimCAqL1xuICBzdWJqZWN0OiBzdHJpbmc7XG4gIC8qKiDnuq/mlofmnKzlhoXlrrkgKi9cbiAgdGV4dDogc3RyaW5nO1xuICAvKiogSFRNTCDlhoXlrrkgKi9cbiAgaHRtbDogc3RyaW5nO1xuICAvKiogSVNPIDg2MDEg5qC85byP55qE5pel5pyf5a2X56ym5LiyICovXG4gIGRhdGU6IHN0cmluZztcbiAgLyoqIOaYr+WQpuW3suivuyAqL1xuICBpc1JlYWQ6IGJvb2xlYW47XG4gIC8qKiDpmYTku7bliJfooaggKi9cbiAgYXR0YWNobWVudHM6IEVtYWlsQXR0YWNobWVudFtdO1xufVxuXG4vKipcbiAqIOiOt+WPlumCruS7tuWIl+ihqOeahOi/lOWbnue7k+aenFxuICogc3VjY2VzcyDkuLogZmFsc2Ug5pe26KGo56S66K+35rGC5aSx6LSl77yI6YeN6K+V6ICX5bC977yJ77yMZW1haWxzIOS4uuepuuaVsOe7hFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEdldEVtYWlsc1Jlc3VsdCB7XG4gIC8qKiDmiYDkvb/nlKjnmoTmuKDpgZMgKi9cbiAgY2hhbm5lbDogQ2hhbm5lbDtcbiAgLyoqIOafpeivoueahOmCrueuseWcsOWdgCAqL1xuICBlbWFpbDogc3RyaW5nO1xuICAvKiog6YKu5Lu25YiX6KGo77yM5aSx6LSl5pe25Li656m65pWw57uEICovXG4gIGVtYWlsczogRW1haWxbXTtcbiAgLyoqIOivt+axguaYr+WQpuaIkOWKn++8jGZhbHNlIOihqOekuumHjeivleiAl+WwveWQjuS7jeWksei0pSAqL1xuICBzdWNjZXNzOiBib29sZWFuO1xufVxuXG4vKipcbiAqIOmHjeivlemFjee9rlxuICogU0RLIOWGhemDqOWvuee9kee7nOmUmeivr+OAgei2heaXtuOAgTV4eCDmnI3liqHnq6/plJnor6/oh6rliqjph43or5VcbiAqIDR4eCDlrqLmiLfnq6/plJnor6/vvIjlpoLlj4LmlbDplJnor6/vvInkuI3kvJrph43or5VcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIC8vIOiHquWumuS5iemHjeivleetlueVpe+8muacgOWkmumHjeivlSAzIOasoe+8jOmmluasoeW7tui/nyAyIOenklxuICogY29uc3QgZW1haWwgPSBhd2FpdCBnZW5lcmF0ZUVtYWlsKHtcbiAqICAgY2hhbm5lbDogJ21haWwtdG0nLFxuICogICByZXRyeTogeyBtYXhSZXRyaWVzOiAzLCBpbml0aWFsRGVsYXk6IDIwMDAgfSxcbiAqIH0pO1xuICogYGBgXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmV0cnlDb25maWcge1xuICAvKiog5pyA5aSn6YeN6K+V5qyh5pWw77yI5LiN5ZCr6aaW5qyh6K+35rGC77yJ77yM6buY6K6kIDIgKi9cbiAgbWF4UmV0cmllcz86IG51bWJlcjtcbiAgLyoqIOWIneWni+mHjeivleW7tui/n++8iOavq+enku+8ie+8jOmHh+eUqOaMh+aVsOmAgOmBv+etlueVpe+8jOm7mOiupCAxMDAwICovXG4gIGluaXRpYWxEZWxheT86IG51bWJlcjtcbiAgLyoqIOacgOWkp+mHjeivleW7tui/n+S4iumZkO+8iOavq+enku+8ie+8jOm7mOiupCA1MDAwICovXG4gIG1heERlbGF5PzogbnVtYmVyO1xuICAvKiog5Y2V5qyh6K+35rGC6LaF5pe25pe26Ze077yI5q+r56eS77yJ77yM6buY6K6kIDE1MDAwICovXG4gIHRpbWVvdXQ/OiBudW1iZXI7XG59XG5cbi8qKlxuICog5Yib5bu65Li05pe26YKu566x55qE6YCJ6aG5XG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHRzXG4gKiAvLyDkvb/nlKjmjIflrprmuKDpgZPliJvlu7rpgq7nrrFcbiAqIGNvbnN0IGVtYWlsID0gYXdhaXQgZ2VuZXJhdGVFbWFpbCh7IGNoYW5uZWw6ICdtYWlsLXRtJyB9KTtcbiAqXG4gKiAvLyDpmo/mnLrpgInmi6nmuKDpgZNcbiAqIGNvbnN0IGVtYWlsID0gYXdhaXQgZ2VuZXJhdGVFbWFpbCgpO1xuICogYGBgXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgR2VuZXJhdGVFbWFpbE9wdGlvbnMge1xuICAvKiog5oyH5a6a5rig6YGT77yM5LiN5Lyg5YiZ6ZqP5py66YCJ5oupICovXG4gIGNoYW5uZWw/OiBDaGFubmVsO1xuICAvKiog6YKu566x5pyJ5pWI5pe26ZW/ICovXG4gIGR1cmF0aW9uPzogbnVtYmVyO1xuICAvKiog5oyH5a6a6YKu566x5Z+f5ZCNICovXG4gIGRvbWFpbj86IHN0cmluZyB8IG51bGw7XG4gIC8qKiDph43or5XphY3nva7vvIzkuI3kvKDliJnkvb/nlKjpu5jorqTlgLzvvIjmnIDlpJrph43or5UgMiDmrKHvvIkgKi9cbiAgcmV0cnk/OiBSZXRyeUNvbmZpZztcbn1cblxuLyoqXG4gKiDojrflj5bpgq7ku7bliJfooajnmoTpgInpoblcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGdldEVtYWlscyh7XG4gKiAgIGNoYW5uZWw6IGVtYWlsSW5mby5jaGFubmVsLFxuICogICBlbWFpbDogZW1haWxJbmZvLmVtYWlsLFxuICogICB0b2tlbjogZW1haWxJbmZvLnRva2VuLFxuICogfSk7XG4gKiBpZiAocmVzdWx0LnN1Y2Nlc3MgJiYgcmVzdWx0LmVtYWlscy5sZW5ndGggPiAwKSB7XG4gKiAgIGNvbnNvbGUubG9nKCfmlLbliLDpgq7ku7Y6JywgcmVzdWx0LmVtYWlscyk7XG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBHZXRFbWFpbHNPcHRpb25zIHtcbiAgLyoqIOa4oOmBk+agh+ivhu+8jOW/heWhqyAqL1xuICBjaGFubmVsOiBDaGFubmVsO1xuICAvKiog6YKu566x5Zyw5Z2A77yM5b+F5aGrICovXG4gIGVtYWlsOiBzdHJpbmc7XG4gIC8qKiDorqTor4Hku6TniYwgKi9cbiAgdG9rZW4/OiBzdHJpbmc7XG4gIC8qKiDph43or5XphY3nva7vvIzkuI3kvKDliJnkvb/nlKjpu5jorqTlgLzvvIjmnIDlpJrph43or5UgMiDmrKHvvIkgKi9cbiAgcmV0cnk/OiBSZXRyeUNvbmZpZztcbn1cbiJdfQ==
|
package/package.json
CHANGED
package/src/index.ts
CHANGED
|
@@ -2,49 +2,117 @@ import * as tempmail from './providers/tempmail';
|
|
|
2
2
|
import * as linshiEmail from './providers/linshi-email';
|
|
3
3
|
import * as tempmailLol from './providers/tempmail-lol';
|
|
4
4
|
import * as chatgptOrgUk from './providers/chatgpt-org-uk';
|
|
5
|
-
import
|
|
5
|
+
import * as tempmailLa from './providers/tempmail-la';
|
|
6
|
+
import * as tempMailIO from './providers/temp-mail-io';
|
|
7
|
+
import * as awamail from './providers/awamail';
|
|
8
|
+
import * as mailTm from './providers/mail-tm';
|
|
9
|
+
import * as dropmail from './providers/dropmail';
|
|
10
|
+
import { Channel, EmailInfo, Email, EmailAttachment, GetEmailsResult, GenerateEmailOptions, GetEmailsOptions } from './types';
|
|
11
|
+
import { withRetry, RetryOptions } from './retry';
|
|
12
|
+
import { logger } from './logger';
|
|
6
13
|
|
|
7
|
-
export { Channel, EmailInfo, Email, GetEmailsResult, GenerateEmailOptions, GetEmailsOptions } from './types';
|
|
14
|
+
export { Channel, EmailInfo, Email, EmailAttachment, GetEmailsResult, GenerateEmailOptions, GetEmailsOptions } from './types';
|
|
15
|
+
export { normalizeEmail } from './normalize';
|
|
16
|
+
export { withRetry, fetchWithTimeout, RetryOptions } from './retry';
|
|
17
|
+
export { LogLevel, LogHandler, setLogLevel, getLogLevel, setLogger, logger } from './logger';
|
|
8
18
|
|
|
19
|
+
/** 渠道名称到 provider 实现的映射表 */
|
|
9
20
|
const providers = {
|
|
10
21
|
'tempmail': tempmail,
|
|
11
22
|
'linshi-email': linshiEmail,
|
|
12
23
|
'tempmail-lol': tempmailLol,
|
|
13
24
|
'chatgpt-org-uk': chatgptOrgUk,
|
|
25
|
+
'tempmail-la': tempmailLa,
|
|
26
|
+
'temp-mail-io': tempMailIO,
|
|
27
|
+
'awamail': awamail,
|
|
28
|
+
'mail-tm': mailTm,
|
|
29
|
+
'dropmail': dropmail,
|
|
14
30
|
};
|
|
15
31
|
|
|
16
|
-
|
|
32
|
+
/** 所有支持的渠道列表,用于随机选择和遍历 */
|
|
33
|
+
const allChannels: Channel[] = ['tempmail', 'linshi-email', 'tempmail-lol', 'chatgpt-org-uk', 'tempmail-la', 'temp-mail-io', 'awamail', 'mail-tm', 'dropmail'];
|
|
17
34
|
|
|
35
|
+
/**
|
|
36
|
+
* 渠道信息,包含渠道标识、显示名称和对应网站
|
|
37
|
+
*/
|
|
18
38
|
export interface ChannelInfo {
|
|
39
|
+
/** 渠道标识 */
|
|
19
40
|
channel: Channel;
|
|
41
|
+
/** 渠道显示名称 */
|
|
20
42
|
name: string;
|
|
43
|
+
/** 对应的临时邮箱服务网站 */
|
|
21
44
|
website: string;
|
|
22
45
|
}
|
|
23
46
|
|
|
47
|
+
/** 渠道信息映射表 */
|
|
24
48
|
const channelInfoMap: Record<Channel, ChannelInfo> = {
|
|
25
49
|
'tempmail': { channel: 'tempmail', name: 'TempMail', website: 'tempmail.ing' },
|
|
26
50
|
'linshi-email': { channel: 'linshi-email', name: '临时邮箱', website: 'linshi-email.com' },
|
|
27
51
|
'tempmail-lol': { channel: 'tempmail-lol', name: 'TempMail LOL', website: 'tempmail.lol' },
|
|
28
52
|
'chatgpt-org-uk': { channel: 'chatgpt-org-uk', name: 'ChatGPT Mail', website: 'mail.chatgpt.org.uk' },
|
|
53
|
+
'tempmail-la': { channel: 'tempmail-la', name: 'TempMail LA', website: 'tempmail.la' },
|
|
54
|
+
'temp-mail-io': { channel: 'temp-mail-io', name: 'Temp Mail IO', website: 'temp-mail.io' },
|
|
55
|
+
'awamail': { channel: 'awamail', name: 'AwaMail', website: 'awamail.com' },
|
|
56
|
+
'mail-tm': { channel: 'mail-tm', name: 'Mail.tm', website: 'mail.tm' },
|
|
57
|
+
'dropmail': { channel: 'dropmail', name: 'DropMail', website: 'dropmail.me' },
|
|
29
58
|
};
|
|
30
59
|
|
|
31
60
|
/**
|
|
32
61
|
* 获取所有支持的渠道列表
|
|
62
|
+
*
|
|
63
|
+
* @returns 所有渠道的信息数组
|
|
64
|
+
*
|
|
65
|
+
* @example
|
|
66
|
+
* ```ts
|
|
67
|
+
* const channels = listChannels();
|
|
68
|
+
* channels.forEach(ch => console.log(`${ch.name} (${ch.website})`));
|
|
69
|
+
* ```
|
|
33
70
|
*/
|
|
34
71
|
export function listChannels(): ChannelInfo[] {
|
|
35
72
|
return allChannels.map(ch => channelInfoMap[ch]);
|
|
36
73
|
}
|
|
37
74
|
|
|
38
75
|
/**
|
|
39
|
-
*
|
|
76
|
+
* 获取指定渠道的详细信息
|
|
77
|
+
*
|
|
78
|
+
* @param channel - 渠道标识
|
|
79
|
+
* @returns 渠道信息,不存在时返回 undefined
|
|
40
80
|
*/
|
|
41
81
|
export function getChannelInfo(channel: Channel): ChannelInfo | undefined {
|
|
42
82
|
return channelInfoMap[channel];
|
|
43
83
|
}
|
|
44
84
|
|
|
85
|
+
/**
|
|
86
|
+
* 创建临时邮箱
|
|
87
|
+
*
|
|
88
|
+
* 错误处理策略:
|
|
89
|
+
* - 网络错误、超时、服务端 5xx 错误 → 自动重试(默认 2 次,指数退避)
|
|
90
|
+
* - 4xx 客户端错误、参数错误 → 直接抛出异常
|
|
91
|
+
*
|
|
92
|
+
* @param options - 创建选项,可指定渠道、有效时长、域名等
|
|
93
|
+
* @returns 邮箱信息,包含地址、令牌等
|
|
94
|
+
* @throws 重试耗尽后仍失败时抛出异常
|
|
95
|
+
*
|
|
96
|
+
* @example
|
|
97
|
+
* ```ts
|
|
98
|
+
* const emailInfo = await generateEmail({ channel: 'temp-mail-io' });
|
|
99
|
+
* console.log(emailInfo.email); // 临时邮箱地址
|
|
100
|
+
* ```
|
|
101
|
+
*/
|
|
45
102
|
export async function generateEmail(options: GenerateEmailOptions = {}): Promise<EmailInfo> {
|
|
46
103
|
const channel = options.channel || allChannels[Math.floor(Math.random() * allChannels.length)];
|
|
47
|
-
|
|
104
|
+
|
|
105
|
+
logger.info(`创建临时邮箱, 渠道: ${channel}`);
|
|
106
|
+
const result = await withRetry(() => generateEmailOnce(channel, options), options.retry);
|
|
107
|
+
logger.info(`邮箱创建成功: ${result.email}`);
|
|
108
|
+
return result;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* 单次创建邮箱(不含重试逻辑)
|
|
113
|
+
* 根据渠道类型分发到对应的 provider 实现
|
|
114
|
+
*/
|
|
115
|
+
async function generateEmailOnce(channel: Channel, options: GenerateEmailOptions): Promise<EmailInfo> {
|
|
48
116
|
switch (channel) {
|
|
49
117
|
case 'tempmail':
|
|
50
118
|
return tempmail.generateEmail(options.duration || 30);
|
|
@@ -54,14 +122,50 @@ export async function generateEmail(options: GenerateEmailOptions = {}): Promise
|
|
|
54
122
|
return tempmailLol.generateEmail(options.domain || null);
|
|
55
123
|
case 'chatgpt-org-uk':
|
|
56
124
|
return chatgptOrgUk.generateEmail();
|
|
125
|
+
case 'tempmail-la':
|
|
126
|
+
return tempmailLa.generateEmail();
|
|
127
|
+
case 'temp-mail-io':
|
|
128
|
+
return tempMailIO.generateEmail();
|
|
129
|
+
case 'awamail':
|
|
130
|
+
return awamail.generateEmail();
|
|
131
|
+
case 'mail-tm':
|
|
132
|
+
return mailTm.generateEmail();
|
|
133
|
+
case 'dropmail':
|
|
134
|
+
return dropmail.generateEmail();
|
|
57
135
|
default:
|
|
58
136
|
throw new Error(`Unknown channel: ${channel}`);
|
|
59
137
|
}
|
|
60
138
|
}
|
|
61
139
|
|
|
140
|
+
/**
|
|
141
|
+
* 获取邮件列表
|
|
142
|
+
*
|
|
143
|
+
* 错误处理策略:
|
|
144
|
+
* - 网络错误、超时、服务端 5xx 错误 → 自动重试(默认 2 次)
|
|
145
|
+
* - 重试耗尽后返回 { success: false, emails: [] },不抛异常
|
|
146
|
+
* - 参数校验错误(缺少 channel / token)直接抛出
|
|
147
|
+
*
|
|
148
|
+
* 这种设计让调用方在轮询场景下不会因网络波动而中断整个流程,
|
|
149
|
+
* 只需检查 success 字段即可判断本次请求是否成功。
|
|
150
|
+
*
|
|
151
|
+
* @param options - 获取选项,包含渠道、邮箱地址、令牌
|
|
152
|
+
* @returns 邮件结果,包含 success 标记和邮件列表
|
|
153
|
+
*
|
|
154
|
+
* @example
|
|
155
|
+
* ```ts
|
|
156
|
+
* const result = await getEmails({
|
|
157
|
+
* channel: emailInfo.channel,
|
|
158
|
+
* email: emailInfo.email,
|
|
159
|
+
* token: emailInfo.token,
|
|
160
|
+
* });
|
|
161
|
+
* if (result.success && result.emails.length > 0) {
|
|
162
|
+
* console.log('收到邮件:', result.emails[0].subject);
|
|
163
|
+
* }
|
|
164
|
+
* ```
|
|
165
|
+
*/
|
|
62
166
|
export async function getEmails(options: GetEmailsOptions): Promise<GetEmailsResult> {
|
|
63
167
|
const { channel, email, token } = options;
|
|
64
|
-
|
|
168
|
+
|
|
65
169
|
if (!channel) {
|
|
66
170
|
throw new Error('Channel is required');
|
|
67
171
|
}
|
|
@@ -69,44 +173,93 @@ export async function getEmails(options: GetEmailsOptions): Promise<GetEmailsRes
|
|
|
69
173
|
throw new Error('Email is required');
|
|
70
174
|
}
|
|
71
175
|
|
|
72
|
-
|
|
176
|
+
logger.debug(`获取邮件, 渠道: ${channel}, 邮箱: ${email}`);
|
|
177
|
+
try {
|
|
178
|
+
const emails = await withRetry(() => getEmailsOnce(channel, email, token), options.retry);
|
|
179
|
+
if (emails.length > 0) {
|
|
180
|
+
logger.info(`获取到 ${emails.length} 封邮件, 渠道: ${channel}`);
|
|
181
|
+
} else {
|
|
182
|
+
logger.debug(`暂无邮件, 渠道: ${channel}`);
|
|
183
|
+
}
|
|
184
|
+
return { channel, email, emails, success: true };
|
|
185
|
+
} catch (err: any) {
|
|
186
|
+
/*
|
|
187
|
+
* 重试耗尽后仍然失败 → 返回空结果而非抛异常
|
|
188
|
+
* 这样调用方在轮询场景下不会因为一次网络波动而中断整个流程
|
|
189
|
+
*/
|
|
190
|
+
logger.error(`获取邮件失败, 渠道: ${channel}, 错误: ${err.message || err}`);
|
|
191
|
+
return { channel, email, emails: [], success: false };
|
|
192
|
+
}
|
|
193
|
+
}
|
|
73
194
|
|
|
195
|
+
/**
|
|
196
|
+
* 单次获取邮件(不含重试逻辑)
|
|
197
|
+
* 根据渠道类型分发到对应的 provider 实现,并校验必需的 token 参数
|
|
198
|
+
*/
|
|
199
|
+
async function getEmailsOnce(channel: Channel, email: string, token?: string): Promise<Email[]> {
|
|
74
200
|
switch (channel) {
|
|
75
201
|
case 'tempmail':
|
|
76
|
-
|
|
77
|
-
break;
|
|
202
|
+
return tempmail.getEmails(email);
|
|
78
203
|
case 'linshi-email':
|
|
79
|
-
|
|
80
|
-
break;
|
|
204
|
+
return linshiEmail.getEmails(email);
|
|
81
205
|
case 'tempmail-lol':
|
|
82
|
-
if (!token)
|
|
83
|
-
|
|
84
|
-
}
|
|
85
|
-
emails = await tempmailLol.getEmails(token);
|
|
86
|
-
break;
|
|
206
|
+
if (!token) throw new Error('Token is required for tempmail-lol channel');
|
|
207
|
+
return tempmailLol.getEmails(token, email);
|
|
87
208
|
case 'chatgpt-org-uk':
|
|
88
|
-
|
|
89
|
-
|
|
209
|
+
return chatgptOrgUk.getEmails(email);
|
|
210
|
+
case 'tempmail-la':
|
|
211
|
+
return tempmailLa.getEmails(email);
|
|
212
|
+
case 'temp-mail-io':
|
|
213
|
+
return tempMailIO.getEmails(email);
|
|
214
|
+
case 'awamail':
|
|
215
|
+
if (!token) throw new Error('Token is required for awamail channel');
|
|
216
|
+
return awamail.getEmails(token, email);
|
|
217
|
+
case 'mail-tm':
|
|
218
|
+
if (!token) throw new Error('Token is required for mail-tm channel');
|
|
219
|
+
return mailTm.getEmails(token, email);
|
|
220
|
+
case 'dropmail':
|
|
221
|
+
if (!token) throw new Error('Token is required for dropmail channel');
|
|
222
|
+
return dropmail.getEmails(token, email);
|
|
90
223
|
default:
|
|
91
224
|
throw new Error(`Unknown channel: ${channel}`);
|
|
92
225
|
}
|
|
93
|
-
|
|
94
|
-
return {
|
|
95
|
-
channel,
|
|
96
|
-
email,
|
|
97
|
-
emails,
|
|
98
|
-
success: true,
|
|
99
|
-
};
|
|
100
226
|
}
|
|
101
227
|
|
|
228
|
+
/**
|
|
229
|
+
* 临时邮箱客户端
|
|
230
|
+
* 封装了邮箱创建和邮件获取的完整流程,自动管理邮箱信息和认证令牌
|
|
231
|
+
*
|
|
232
|
+
* @example
|
|
233
|
+
* ```ts
|
|
234
|
+
* const client = new TempEmailClient();
|
|
235
|
+
* const emailInfo = await client.generate({ channel: 'mail-tm' });
|
|
236
|
+
* console.log('邮箱:', emailInfo.email);
|
|
237
|
+
*
|
|
238
|
+
* // 轮询获取邮件
|
|
239
|
+
* const result = await client.getEmails();
|
|
240
|
+
* if (result.success) {
|
|
241
|
+
* console.log('邮件数:', result.emails.length);
|
|
242
|
+
* }
|
|
243
|
+
* ```
|
|
244
|
+
*/
|
|
102
245
|
export class TempEmailClient {
|
|
103
246
|
private emailInfo: EmailInfo | null = null;
|
|
104
247
|
|
|
248
|
+
/**
|
|
249
|
+
* 创建临时邮箱并缓存邮箱信息
|
|
250
|
+
* 后续调用 getEmails() 时自动使用此邮箱的渠道、地址和令牌
|
|
251
|
+
*/
|
|
105
252
|
async generate(options: GenerateEmailOptions = {}): Promise<EmailInfo> {
|
|
106
253
|
this.emailInfo = await generateEmail(options);
|
|
107
254
|
return this.emailInfo;
|
|
108
255
|
}
|
|
109
256
|
|
|
257
|
+
/**
|
|
258
|
+
* 获取当前邮箱的邮件列表
|
|
259
|
+
* 必须先调用 generate() 创建邮箱
|
|
260
|
+
*
|
|
261
|
+
* @throws 未调用 generate() 时抛出异常
|
|
262
|
+
*/
|
|
110
263
|
async getEmails(): Promise<GetEmailsResult> {
|
|
111
264
|
if (!this.emailInfo) {
|
|
112
265
|
throw new Error('No email generated. Call generate() first.');
|
|
@@ -119,6 +272,10 @@ export class TempEmailClient {
|
|
|
119
272
|
});
|
|
120
273
|
}
|
|
121
274
|
|
|
275
|
+
/**
|
|
276
|
+
* 获取当前缓存的邮箱信息
|
|
277
|
+
* 未调用 generate() 时返回 null
|
|
278
|
+
*/
|
|
122
279
|
getEmailInfo(): EmailInfo | null {
|
|
123
280
|
return this.emailInfo;
|
|
124
281
|
}
|