Compare commits

...

99 Commits

Author SHA1 Message Date
Dream Hunter
6ae3b0d85e feat: update docs (#574) 2025-01-24 17:36:59 +08:00
Dream Hunter
01e6cb1075 feat: |worker| health_check add JWT_SECRET and DOMAINS (#573) 2025-01-24 15:00:50 +08:00
Dream Hunter
814f6fada2 feat: |UI| admin worker config page add overflow: auto (#572) 2025-01-22 23:34:49 +08:00
Dream Hunter
31901aacc5 feat: update docs (#571) 2025-01-22 23:25:40 +08:00
Dream Hunter
fb9b9f6ae4 feat: update CHANGE LOG (#570) 2025-01-22 23:19:53 +08:00
Dream Hunter
095951ab45 feat: update docs (#569) 2025-01-22 23:14:38 +08:00
Dream Hunter
37614ce6fa feat: footer support html (#567) 2025-01-21 10:24:13 +08:00
Dream Hunter
3f81fbee6d feat: announcement support html (#566)
* feat: announcement support html

* feat: update dependencies
2025-01-20 13:53:40 +08:00
Dream Hunter
cf13236e7b fix: telegram mail page use iframe show email (#564) 2025-01-18 14:59:08 +08:00
Dream Hunter
36e9c611e6 feat: |Worker| add REMOVE_ALL_ATTACHMENT and REMOVE_EXCEED_SIZE_ATTAC… (#563)
feat: |Worker| add REMOVE_ALL_ATTACHMENT and REMOVE_EXCEED_SIZE_ATTACHMENT
2025-01-18 14:43:09 +08:00
Dream Hunter
047200c1c2 feat: |Worker| add REMOVE_ALL_ATTACHMENT and REMOVE_EXCEED_SIZE_ATTAC… (#562)
feat: |Worker| add REMOVE_ALL_ATTACHMENT and REMOVE_EXCEED_SIZE_ATTACHMENT
2025-01-18 14:12:01 +08:00
Dream Hunter
a22add0e14 fix: telegram mail page use iframe show email (#561) 2025-01-18 13:52:09 +08:00
Dream Hunter
7b1c4cc72a fix: mail-parser-wasm parsedEmailContext cache (#560) 2025-01-18 13:26:09 +08:00
刘志聪
3870727a08 fix: rpc headers covert & typo (#559)
Co-authored-by: liuzhicong <liuzhicong@dhgate.com>
2025-01-16 00:20:02 +08:00
Dream Hunter
2bb033964c feat: update doc (#557) 2025-01-11 18:56:36 +08:00
Dream Hunter
9db5a00b35 feat: v0.8.5 && update dependencies && fix deprecated warning for `… (#556)
feat: v0.8.5 && update dependencies && fix `deprecated` warning for `mail-parser-wasm-worker`
2025-01-11 18:46:46 +08:00
Dream Hunter
e161eb5d14 fix: telegram query email do not pass parsedEmailContext (#555) 2025-01-11 18:14:34 +08:00
Dream Hunter
b604f56d56 feat: |Github Action| Deploy Backend add DEBUG_MODE for logging && BA… (#554)
feat: |Github Action| Deploy Backend add DEBUG_MODE for logging && BACKEND_USE_MAIL_WASM_PARSER to enable mail-parser-wasm-worker
2025-01-11 18:04:53 +08:00
Dream Hunter
52caf811f5 feat: add JUNK_MAIL_CHECK_LIST for check exits and passed item && add ParsedEmailContext to cache the parsed Email (#553)
* feat: Junk mail only check JUNK_MAIL_FORCE_PASS_LIST

* feat: add `JUNK_MAIL_CHECK_LIST` for check exits and passed item && add `ParsedEmailContext` to cache the parsed Email
2025-01-11 17:42:20 +08:00
Dream Hunter
ee3884914b Update CHANGELOG.md 2025-01-09 22:50:22 +08:00
Dream Hunter
844fc52bbc feat: |UI| add configAutoRefreshInterval && autoRefresh useStorage (#549)
* feat: |UI| add configAutoRefreshInterval && autoRefresh useStorage

* Update MailBox.vue

* Update MailBox.vue
2025-01-09 22:49:25 +08:00
Dream Hunter
b87b49f09d Update CHANGELOG.md 2025-01-08 20:04:11 +08:00
刘志聪
5bfa588f70 feat: trigger another worker (#547) 2025-01-08 20:02:48 +08:00
Dream Hunter
92620cdedb feat: add DISABLE_ANONYMOUS_USER_CREATE_EMAIL which only allow logi… (#545)
feat: add `DISABLE_ANONYMOUS_USER_CREATE_EMAIL` which only allow login user create email address
2025-01-05 18:51:48 +08:00
Dream Hunter
e9748be9fe Update vite.config.js (#544) 2025-01-05 02:05:40 +08:00
Dream Hunter
479322c430 feat: |Telegram Bot| add new command to clean invalid jwts (#543) 2025-01-05 01:52:07 +08:00
Dream Hunter
934e58e23b fix: |UI| admin mails unknown page call wrong api (#542) 2025-01-05 01:14:36 +08:00
Dream Hunter
c964d77a59 feat: |UI| add JUNK_MAIL_FORCE_PASS_LIST (#539) 2024-12-30 18:38:33 +08:00
Dream Hunter
8a03d3e57f feat: |UI| admin portal user oauth config support delete (#538) 2024-12-30 18:08:17 +08:00
Dream Hunter
6caba7c863 feat: add docs (#537) 2024-12-28 13:52:08 +08:00
Dream Hunter
43e5bdc764 feat: update dependencies (#536) 2024-12-28 00:32:07 +08:00
Dream Hunter
7bec0daba4 feat: update docs (#534) 2024-12-27 10:47:14 +08:00
Dream Hunter
13e5adef17 feat: update docs (#533) 2024-12-26 22:23:41 +08:00
Dream Hunter
440238133e feat: |Github Action| add upstream sync and auto deploy frontend&&bac… (#528)
feat: |Github Action| add upstream sync and auto deploy frontend&&backend
2024-12-23 22:55:10 +08:00
Dream Hunter
4a881e2d2b feat: upgrade dependencies (#527) 2024-12-23 21:10:45 +08:00
Dream Hunter
b0bf7a5f13 feat: add NO_LIMIT_SEND_ADDRESS_LIST_KEY in admin account settings page (#525) 2024-12-22 15:52:53 +08:00
Dream Hunter
a9bb8785ba feat: support send mail from admin portal(no balance limit) (#524) 2024-12-22 15:40:26 +08:00
Dream Hunter
0b48baff6d fix: frontend github actions cannot use branch param to deploy (#520)
* Update frontend_deploy.yaml

* Update frontend_deploy.yaml
2024-12-19 17:58:54 +08:00
Dream Hunter
e0b5e80efd feat: |doc| update doc (#510) 2024-12-04 00:56:52 +08:00
Dream Hunter
b0e36ac2aa feat: |doc| update Telegram Bot doc (#509) 2024-12-04 00:33:47 +08:00
Dream Hunter
51db19c85b feat: |UI| add tip for multiple tag (#508) 2024-12-04 00:29:01 +08:00
Dream Hunter
e52b010aa4 feat: |doc| update doc (#507) 2024-12-03 22:04:46 +08:00
Dream Hunter
8f6793402c feat: |UI| add forward in mail page (#502) 2024-11-30 15:53:48 +08:00
Dream Hunter
e86c530116 feat: |UI| hide ID for user (#501) 2024-11-30 15:11:37 +08:00
Dream Hunter
0308f518da feat: upgrade dependencies && |doc| update ui install worker doc (#494) 2024-11-22 14:42:35 +08:00
Dream Hunter
3c2a8ed056 feat: remove service workbox html cache (#486) 2024-11-15 01:39:14 +08:00
Dream Hunter
5f45ec7c14 feat: remove service workbox (#485) 2024-11-15 01:22:43 +08:00
Dream Hunter
1b7ebc98c5 feat: support transfer address from user to user (#484)
* feat: support transfer address from user to user

* feat: remove service worker
2024-11-15 01:10:25 +08:00
Dream Hunter
c102004f4d feat: |UI| show local datetime string and add useUTCDate option (#483) 2024-11-15 00:04:17 +08:00
Dream Hunter
3c81e05a2f feat: |UI| random fake name support MAX_ADDRESS_LEN (#482) 2024-11-14 23:58:42 +08:00
Dream Hunter
5ff2ceb5e8 feat: pages support Cloudflare Zero Trust (#477) 2024-11-11 23:55:49 +08:00
Dream Hunter
6c82efb738 feat: docs: ui_install worker update (#476) 2024-11-08 13:09:28 +08:00
Dream Hunter
e99acdcc6e fix: CI (#471) 2024-11-07 01:11:00 +08:00
Dream Hunter
8f30505706 feat: v0.7.6 (#470) 2024-11-07 01:00:26 +08:00
Dream Hunter
ddfa2c5d03 feat: add ENABLE_CHECK_JUNK_MAIL (#469) 2024-11-07 00:58:15 +08:00
Dream Hunter
49b3f10838 feat: upgrade dependencies && add ci build telegram-frontend.zip (#467) 2024-11-06 23:42:39 +08:00
Dream Hunter
cc9ac67319 feat: upgrade dependencies (#448) 2024-09-27 22:30:37 +08:00
Dream Hunter
7cc2a2b576 feat: doc: add mail id and url in webhook (#444) 2024-09-09 22:49:53 +08:00
Dream Hunter
393c5902c3 feat: add mail id and url in webhook (#443) 2024-09-09 22:29:18 +08:00
Dream Hunter
5ece49a576 feat: telegram Set manually to avoid implicit call in (#442) 2024-09-09 20:59:12 +08:00
dependabot[bot]
de80857e2c build(deps): bump twisted from 24.3.0 to 24.7.0 in /smtp_proxy_server (#385)
Bumps [twisted](https://github.com/twisted/twisted) from 24.3.0 to 24.7.0.
- [Release notes](https://github.com/twisted/twisted/releases)
- [Changelog](https://github.com/twisted/twisted/blob/trunk/NEWS.rst)
- [Commits](https://github.com/twisted/twisted/compare/twisted-24.3.0...twisted-24.7.0)

---
updated-dependencies:
- dependency-name: twisted
  dependency-type: direct:production
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2024-08-27 01:40:45 +08:00
Dream Hunter
a57a42b2a1 fix: name check bug (#434) 2024-08-25 16:39:55 +08:00
Dream Hunter
a24cc1f642 fix: bugs && release v0.7.4 (#432) 2024-08-24 15:07:07 +08:00
Dream Hunter
4c6fd3c2af feat: UI add min-width for table page (#428) 2024-08-19 22:53:13 +08:00
Dream Hunter
1cf38c1768 feat: UI: add WorkerConfig && release v0.7.3 (#421) 2024-08-18 14:58:57 +08:00
Dream Hunter
b5b59acdb3 feat: add Oauth2 Login (#420) 2024-08-18 14:39:50 +08:00
Dream Hunter
6d4783e1cd fix: UI admin page show modal when no need password (#419) 2024-08-17 23:54:03 +08:00
Dream Hunter
34e3e1b439 fix: UI admin page show modal when no need password (#418) 2024-08-17 23:14:35 +08:00
Dream Hunter
56104cd23a fix: UI tab active icon wrong position (#416) 2024-08-17 01:46:40 +08:00
Dream Hunter
3664028e06 feat: add ADDRESS_CHECK_REGEX (#415) 2024-08-17 00:11:28 +08:00
Dream Hunter
9888f98d74 feat: update dependencies (#411) 2024-08-15 01:05:05 +08:00
Dream Hunter
ac5605f17f release v0.7.2 doc (#410) 2024-08-15 01:02:15 +08:00
Dream Hunter
a9719cb3ec release v0.7.2 (#409) 2024-08-15 00:56:15 +08:00
Dream Hunter
5f4978645b release v0.7.2 (#408) 2024-08-15 00:52:18 +08:00
Dream Hunter
621476cb79 feat: update webhook to support global webhook (#407) 2024-08-15 00:23:31 +08:00
Dream Hunter
c969c4b082 fix: DISABLE_ADMIN_PASSWORD_CHECK still show admin password modal (#406) 2024-08-14 22:52:45 +08:00
Dream Hunter
d90f54345d feat: add ADDRESS_REGEX (#401) 2024-08-13 23:21:19 +08:00
Dream Hunter
797b8bb019 fix: NO_LIMIT_SEND_ROLE no access token (#400) 2024-08-13 01:38:20 +08:00
Dream Hunter
7e5d142924 fix: NO_LIMIT_SEND_ROLE when user settings not call (#396) 2024-08-11 23:45:24 +08:00
Dream Hunter
c6d0307eac Release v0.7.1 2024-08-11 22:46:40 +08:00
Dream Hunter
ac31042e69 feat: add EMAIL_KV_BLACK_LIST (#394) 2024-08-11 20:34:10 +08:00
Dream Hunter
c733d3bf4d fix: get user role before all requests (#393) 2024-08-11 19:29:49 +08:00
Dream Hunter
bf1243f4c4 release: v0.7.0 (#387) 2024-08-11 00:21:15 +08:00
Dream Hunter
15063b2e97 feat: add DISABLE_ADMIN_PASSWORD_CHECK (#386) 2024-08-11 00:10:16 +08:00
Dream Hunter
fc07f1cd87 feat: add passkey (#384) 2024-08-10 23:56:05 +08:00
Dream Hunter
9246550cc5 feat: add NO_LIMIT_SEND_ROLE (#373) 2024-08-04 21:02:11 +08:00
Dream Hunter
979b6eae1a feat: add SHOW_GITHUB config (#372) 2024-08-04 14:36:24 +08:00
Dream Hunter
10da337a9c feat: add SHOW_GITHUB config (#371) 2024-08-04 14:34:35 +08:00
Dream Hunter
9c5e8857af feat: add loading when process mails (#367) 2024-07-27 23:14:18 +08:00
Dream Hunter
84b4baa99e feat: add .github/workflows/pr_agent.yml (#366) 2024-07-27 23:06:54 +08:00
Dream Hunter
b57d46244a feat: add loading when process mails (#364) 2024-07-27 22:30:38 +08:00
Dream Hunter
5faae8796d feat: add ADMIN_USER_ROLE for user access admin panel (#363) 2024-07-27 22:04:18 +08:00
666-eth
a0805bc0ce Docs: Update new-address-api.md (#360) 2024-07-23 13:47:37 +08:00
Dream Hunter
d0ccc3ded1 v0.6.1 2024-07-22 13:09:42 +08:00
Dream Hunter
163d9451f7 feat: worker: newAddress if domain is not set, use the first domain (#358) 2024-07-22 13:05:50 +08:00
Dream Hunter
60dda7e3fe feat: add ANNOUNCEMENT (#357) 2024-07-22 13:01:38 +08:00
Dream Hunter
384eb9b041 fix: imap proxy do not support password && cleanup days translate (#356) 2024-07-19 22:40:53 +08:00
tqjason
38816cbf0f Add new workflow action and Fix cleanup bug (#355)
* Create frontend_pagefunction_deploy.yaml

* Update frontend_pagefunction_deploy.yaml

* Update cleanup_api.ts

* Update common.ts

* Update cleanup_api.ts

* Update common.ts
2024-07-19 22:34:01 +08:00
Dream Hunter
d7d1ba6b64 feat: wrangler d1 execute dev add --remote (#352) 2024-07-15 12:04:14 +08:00
136 changed files with 8977 additions and 5387 deletions

View File

@@ -0,0 +1,44 @@
diff --git a/worker/src/common.ts b/worker/src/common.ts
index bd9bcc9..e7e2748 100644
--- a/worker/src/common.ts
+++ b/worker/src/common.ts
@@ -273,23 +273,23 @@ export const commonParseMail = async (parsedEmailContext: ParsedEmailContext): P
}
const raw_mail = parsedEmailContext.rawEmail;
// TODO: WASM parse email
- // try {
- // const { parse_message_wrapper } = await import('mail-parser-wasm-worker');
+ try {
+ const { parse_message_wrapper } = await import('mail-parser-wasm-worker');
- // const parsedEmail = parse_message_wrapper(raw_mail);
- // parsedEmailContext.parsedEmail = {
- // sender: parsedEmail.sender || "",
- // subject: parsedEmail.subject || "",
- // text: parsedEmail.text || "",
- // headers: parsedEmail.headers?.map(
- // (header) => ({ key: header.key, value: header.value })
- // ) || [],
- // html: parsedEmail.body_html || "",
- // };
- // return parsedEmailContext.parsedEmail;
- // } catch (e) {
- // console.error("Failed use mail-parser-wasm-worker to parse email", e);
- // }
+ const parsedEmail = parse_message_wrapper(raw_mail);
+ parsedEmailContext.parsedEmail = {
+ sender: parsedEmail.sender || "",
+ subject: parsedEmail.subject || "",
+ text: parsedEmail.text || "",
+ headers: parsedEmail.headers?.map(
+ (header) => ({ key: header.key, value: header.value })
+ ) || [],
+ html: parsedEmail.body_html || "",
+ };
+ return parsedEmailContext.parsedEmail;
+ } catch (e) {
+ console.error("Failed use mail-parser-wasm-worker to parse email", e);
+ }
try {
const { default: PostalMime } = await import('postal-mime');
const parsedEmail = await PostalMime.parse(raw_mail);

View File

@@ -1,6 +1,9 @@
name: Deploy Backend Production name: Deploy Backend
on: on:
workflow_run:
workflows: [Upstream Sync]
types: [completed]
push: push:
tags: tags:
- "*" - "*"
@@ -29,14 +32,28 @@ jobs:
- name: Deploy Backend for ${{ github.ref_name }} - name: Deploy Backend for ${{ github.ref_name }}
run: | run: |
export debug_mode=${{ secrets.DEBUG_MODE }}
export use_mail_wasm_parser=${{ secrets.BACKEND_USE_MAIL_WASM_PARSER }}
cd worker/ cd worker/
echo '${{ secrets.BACKEND_TOML }}' > wrangler.toml echo '${{ secrets.BACKEND_TOML }}' > wrangler.toml
pnpm install --no-frozen-lockfile pnpm install --no-frozen-lockfile
output=$(pnpm run deploy 2>&1)
if [ $? -ne 0 ]; then if [ -n "$use_mail_wasm_parser" ]; then
code=$? echo "Using mail-parser-wasm-worker"
echo "Command failed with exit code $code" pnpm add mail-parser-wasm-worker
exit $code git apply ../.github/config/mail-parser-wasm-worker.patch
echo "Applied mail-parser-wasm-worker patch"
fi
if [ -n "$debug_mode" ]; then
pnpm run deploy
else
output=$(pnpm run deploy 2>&1)
if [ $? -ne 0 ]; then
code=$?
echo "Command failed with exit code $code"
exit $code
fi
fi fi
echo "Deployed for tag ${{ github.ref_name }}" echo "Deployed for tag ${{ github.ref_name }}"
env: env:

View File

@@ -1,9 +1,10 @@
name: Deploy Frontend name: Deploy Frontend
on: on:
workflow_run:
workflows: [Upstream Sync]
types: [completed]
push: push:
paths:
- "frontend/**"
tags: tags:
- "*" - "*"
workflow_dispatch: workflow_dispatch:
@@ -38,12 +39,12 @@ jobs:
export frontend_branch=${{ secrets.FRONTEND_BRANCH }} export frontend_branch=${{ secrets.FRONTEND_BRANCH }}
if [ -n "$frontend_branch" ]; then if [ -n "$frontend_branch" ]; then
echo "Deploying branch $frontend_branch" echo "Deploying branch $frontend_branch"
pnpm run deploy:actions --project-name=$project_name pnpm run deploy:actions --project-name=$project_name --branch $frontend_branch
else else
echo "Deploying branch prodcution" echo "Deploying branch production"
pnpm run deploy --project-name=$project_name pnpm run deploy --project-name=$project_name
fi fi
echo "Deploying prodcution for ${{ github.ref_name }}" echo "Deploying production for ${{ github.ref_name }}"
echo "Deployed for tag ${{ github.ref_name }}" echo "Deployed for tag ${{ github.ref_name }}"
export tg_mini_app_project_name=${{ secrets.TG_FRONTEND_NAME }} export tg_mini_app_project_name=${{ secrets.TG_FRONTEND_NAME }}
@@ -51,9 +52,9 @@ jobs:
echo "Deploying telegram mini app $tg_mini_app_project_name" echo "Deploying telegram mini app $tg_mini_app_project_name"
if [ -n "$frontend_branch" ]; then if [ -n "$frontend_branch" ]; then
echo "Deploying telegram mini app branch $frontend_branch" echo "Deploying telegram mini app branch $frontend_branch"
pnpm run deploy:actions:telegram --project-name=$tg_mini_app_project_name pnpm run deploy:actions:telegram --project-name=$tg_mini_app_project_name --branch $frontend_branch
else else
echo "Deploying telegram mini app branch prodcution" echo "Deploying telegram mini app branch production"
pnpm run deploy:telegram --project-name=$tg_mini_app_project_name pnpm run deploy:telegram --project-name=$tg_mini_app_project_name
fi fi
echo "Deployed telegram mini app for ${{ github.ref_name }}" echo "Deployed telegram mini app for ${{ github.ref_name }}"

View File

@@ -0,0 +1,39 @@
name: Deploy Frontend with page function
on:
workflow_dispatch:
jobs:
deploy:
runs-on: ubuntu-latest
permissions:
contents: write
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Install Node.js
uses: actions/setup-node@v4
with:
node-version: 18
- uses: pnpm/action-setup@v3
name: Install pnpm
id: pnpm-install
with:
version: 8
run_install: false
- name: Deploy Frontend for ${{ github.ref_name }}
run: |
cd frontend/
pnpm install --no-frozen-lockfile
pnpm build:pages
cd ../pages/
echo '${{ secrets.PAGE_TOML }}' > wrangler.toml
pnpm install --no-frozen-lockfile
pnpm run deploy
echo "Deploying prodcution for ${{ github.ref_name }}"
env:
CLOUDFLARE_ACCOUNT_ID: ${{ secrets.CLOUDFLARE_ACCOUNT_ID }}
CLOUDFLARE_API_TOKEN: ${{ secrets.CLOUDFLARE_API_TOKEN }}

25
.github/workflows/pr_agent.yml vendored Normal file
View File

@@ -0,0 +1,25 @@
name: Codium PR Agent
on:
pull_request:
types: [opened, reopened, ready_for_review]
jobs:
pr_agent_job:
if: ${{ github.event.sender.type != 'Bot' }}
runs-on: ubuntu-latest
permissions:
issues: write
pull-requests: write
name: Run pr agent on every pull request, respond to user comments
steps:
- name: PR Agent action step
id: pragent
uses: Codium-ai/pr-agent@main
env:
PR_REVIEWER.REQUIRE_TESTS_REVIEW: "false"
OPENAI_KEY: ${{ secrets.OPENAI_KEY }}
OPENAI_API_BASE: ${{ secrets.OPENAI_API_BASE }}
CONFIG.MODEL: "gpt-4o"
CONFIG.MODEL_TURBO: "gpt-4o"
OPENAI.API_BASE: ${{ secrets.OPENAI_API_BASE }}
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

25
.github/workflows/sync.yaml vendored Normal file
View File

@@ -0,0 +1,25 @@
name: Upstream Sync
on:
schedule:
- cron: "0 0 * * 1"
workflow_dispatch:
jobs:
sync_latest_from_upstream:
name: Sync latest commits from upstream repo
runs-on: ubuntu-latest
if: ${{ github.event.repository.fork }}
steps:
- uses: actions/checkout@v4
- name: Sync upstream changes
id: sync
uses: aormsby/Fork-Sync-With-Upstream-action@v3.4
with:
upstream_sync_repo: dreamhunter2333/cloudflare_temp_email
upstream_sync_branch: main
target_sync_branch: main
target_repo_token: ${{ secrets.GITHUB_TOKEN }} # automatically generated, no need to set
test_mode: false

View File

@@ -30,7 +30,13 @@ jobs:
run: cd frontend && pnpm install --no-frozen-lockfile && pnpm build:release run: cd frontend && pnpm install --no-frozen-lockfile && pnpm build:release
- name: Zip Frontend dist - name: Zip Frontend dist
run: cd frontend/dist/ && zip -r frontend.zip * run: cd frontend/dist/ && zip -r frontend.zip * && mv frontend.zip ../
- name: Build Telegram Frontend
run: cd frontend && pnpm install --no-frozen-lockfile && pnpm build:telegram:release
- name: Zip Telegram Frontend dist
run: cd frontend/dist/ && zip -r telegram-frontend.zip * && mv telegram-frontend.zip ../
- name: cp wrangler.toml - name: cp wrangler.toml
run: cd worker && cp wrangler.toml.template wrangler.toml run: cd worker && cp wrangler.toml.template wrangler.toml
@@ -42,5 +48,6 @@ jobs:
uses: softprops/action-gh-release@v2 uses: softprops/action-gh-release@v2
with: with:
files: | files: |
frontend/dist/frontend.zip frontend/frontend.zip
frontend/telegram-frontend.zip
worker/dist/worker.js worker/dist/worker.js

8
.vscode/extensions.json vendored Normal file
View File

@@ -0,0 +1,8 @@
{
"recommendations": [
"ms-python.vscode-pylance",
"1yib.rust-bundle",
"rust-lang.rust-analyzer",
"vue.volar"
]
}

View File

@@ -1,6 +1,125 @@
<!-- markdownlint-disable-file MD004 MD024 MD034 MD036 --> <!-- markdownlint-disable-file MD004 MD024 MD034 MD036 -->
# CHANGE LOG # CHANGE LOG
## main(v0.8.6)
- feat: |UI| 公告支持 html 格式
- feat: |UI| `COPYRIGHT` 支持 html 格式
- feat: |Doc| 优化部署文档,补充了 `Github Actions 部署文档`,增加了 `Worker 变量说明`
## v0.8.5
- feat: |mail-parser-wasm-worker| 修复 `initSync` 函数调用时的 `deprecated` 参数警告
- feat: rpc headers covert & typo (#559)
- fix: telegram mail page use iframe show email (#561)
- feat: |Worker| 增加 `REMOVE_ALL_ATTACHMENT``REMOVE_EXCEED_SIZE_ATTACHMENT` 用于移除邮件附件,由于是解析邮件的一些信息会丢失,比如图片等.
## v0.8.4
- fix: |UI| 修复 admin portal 无收件人邮箱删除调用api 错误
- feat: |Telegram Bot| 增加 telegram bot 清理无效地址凭证命令
- feat: 增加 worker 配置 `DISABLE_ANONYMOUS_USER_CREATE_EMAIL` 禁用匿名用户创建邮箱地址,只允许登录用户创建邮箱地址
- feat: 增加 worker 配置 `ENABLE_ANOTHER_WORKER``ANOTHER_WORKER_LIST` ,用于调用其他 worker 的 rpc 接口 (#547)
- feat: |UI| 自动刷新配置保存到浏览器,可配置刷新间隔
- feat: 垃圾邮件检测增加存在时才检查的列表 `JUNK_MAIL_CHECK_LIST` 配置
- feat: | Worker | 增加 `ParsedEmailContext` 类用于缓存解析后的邮件内容,减少解析次数
- feat: |Github Action| Worker 部署增加 `DEBUG_MODE` 输出日志, `BACKEND_USE_MAIL_WASM_PARSER` 配置是否使用 wasm 解析邮件
## v0.8.3
- feat: |Github Action| 增加自动更新并部署功能
- feat: |UI| admin 用户设置,支持 oauth2 配置的删除
- feat: 增加垃圾邮件检测必须通过的列表 `JUNK_MAIL_FORCE_PASS_LIST` 配置
## v0.8.2
- fix: |Doc| 修复文档中的一些错误
- fix: |Github Action| 修复 frontend 部署分支错误的问题
- feat: admin 发送邮件功能
- feat: admin 后台,账号配置页面添加无限发送邮件的地址列表
## v0.8.1
- feat: |Doc| 更新 UI 安装的文档
- feat: |UI| 对用户隐藏邮箱账号的 ID
- feat: |UI| 增加邮件详情页的 `转发` 按钮
## v0.8.0
- feat: |UI| 随机生成地址时不超过最大长度
- feat: |UI| 邮件时间显示浏览器时区,可在设置中切换显示为 UTC 时间
- feat: 支持转移邮件到其他用户
## v0.7.6
### Breaking Changes
UI 部署 worker 需要点击 Settings -> Runtime, 修改 Compatibility flags, 增加 `nodejs_compat`
![worker-runtime](vitepress-docs/docs/public/ui_install/worker-runtime.png)
### Changes
- feat: 支持提前设置 bot info, 降低 telegram 回调延迟 (#441)
- feat: 增加 telegram mini app 的 build 压缩包
- feat: 增加是否启用垃圾邮件检查 `ENABLE_CHECK_JUNK_MAIL` 配置
## v0.7.5
- fix: 修复 `name` 的校验检查
## v0.7.4
- feat: UI 列表页面增加最小宽度
- fix: 修复 `name` 的校验检查
- fix: 修复 `DEFAULT_DOMAINS` 配置为空不生效的问题
## v0.7.3
- feat: worker 增加 `ADDRESS_CHECK_REGEX`, address name 的正则表达式, 只用于检查,符合条件将通过检查
- fix: UI 修复登录页面 tab 激活图标错位
- fix: UI 修复 admin 页面刷新弹框输入密码的问题
- feat: support `Oath2` 登录, 可以通过 `Github` `Authentik` 等第三方登录, 详情查看 [OAuth2 第三方登录](https://temp-mail-docs.awsl.uk/zh/guide/feature/user-oauth2.html)
## v0.7.2
### Breaking Changes
`webhook` 的结构增加了 `enabled` 字段,已经配置了的需要重新在页面开启并保存。
### Changes
- fix: worker 增加 `NO_LIMIT_SEND_ROLE` 配置, 加载失败的问题
- feat: worker 增加 `# ADDRESS_REGEX = "[^a-z.0-9]"` 配置, 替换非法符号的正则表达式,如果不设置,默认为 [^a-z0-9], 需谨慎使用, 有些符号可能导致无法收件
- feat: worker 优化 webhook 逻辑, 支持 admin 配置全局 webhook, 添加 `message pusher` 集成示例
## v0.7.1
- fix: 修复用户角色加载失败的问题
- feat: admin 账号设置增加来源邮件地址黑名单配置
## v0.7.0
### Breaking Changes
DB changes: 增加用户 `passkey` 表, 需要执行 `db/2024-08-10-patch.sql` 更新 `D1` 数据库
### Changes
- Docs: Update new-address-api.md (#360)
- feat: worker 增加 `ADMIN_USER_ROLE` 配置, 用于配置管理员用户角色,此角色的用户可访问 admin 管理页面 (#363)
- feat: worker 增加 `DISABLE_SHOW_GITHUB` 配置, 用于配置是否显示 github 链接
- feat: worker 增加 `NO_LIMIT_SEND_ROLE` 配置, 用于配置可以无限发送邮件的角色
- feat: 用户增加 `passkey` 登录方式, 用于用户登录, 无需输入密码
- feat: worker 增加 `DISABLE_ADMIN_PASSWORD_CHECK` 配置, 用于配置是否禁用 admin 控制台密码检查, 若你的网站只可私人访问,可通过此禁用检查
## v0.6.1
- pages github actions && 修复清理邮件天数为 0 不生效 by @tqjason (#355)
- fix: imap proxy server 不支持 密码 by @dreamhunter2333 (#356)
- worker 新增 `ANNOUNCEMENT` 配置, 用于配置公告信息 by @dreamhunter2333 (#357)
- fix: telegram bot 新建地址默认选择第一个域名 by @dreamhunter2333 (#358)
## v0.6.0 ## v0.6.0
### Breaking Changes ### Breaking Changes
@@ -313,7 +432,7 @@ The `mails` table will be discarded, and the `raw` text of the new `mail` will b
```bash ```bash
git checkout v0.2.0 git checkout v0.2.0
cd worker cd worker
wrangler d1 execute dev --file=../db/2024-04-09-patch.sql wrangler d1 execute dev --file=../db/2024-04-09-patch.sql --remote
pnpm run deploy pnpm run deploy
cd ../frontend cd ../frontend
pnpm run deploy pnpm run deploy

View File

@@ -31,9 +31,9 @@
## [查看部署文档](https://temp-mail-docs.awsl.uk) ## [查看部署文档](https://temp-mail-docs.awsl.uk)
[![Deploy to Cloudflare Workers](https://deploy.workers.cloudflare.com/button)](https://deploy.workers.cloudflare.com/?url=https://github.com/dreamhunter2333/cloudflare_temp_email) [![Deploy to Cloudflare Workers](https://deploy.workers.cloudflare.com/button)](https://temp-mail-docs.awsl.uk/zh/guide/actions/github-action.html)
[Github Action 部署文档](https://temp-mail-docs.awsl.uk/zh/guide/github-action.html) [Github Action 部署文档](https://temp-mail-docs.awsl.uk/zh/guide/actions/github-action.html)
[English Docs](https://temp-mail-docs.awsl.uk/en/) [English Docs](https://temp-mail-docs.awsl.uk/en/)

14
db/2024-08-10-patch.sql Normal file
View File

@@ -0,0 +1,14 @@
CREATE TABLE IF NOT EXISTS user_passkeys (
id INTEGER PRIMARY KEY,
user_id INTEGER NOT NULL,
passkey_name TEXT NOT NULL,
passkey_id TEXT NOT NULL,
passkey TEXT NOT NULL,
counter INTEGER DEFAULT 0,
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP
);
CREATE INDEX IF NOT EXISTS idx_user_passkeys_user_id ON user_passkeys(user_id);
CREATE UNIQUE INDEX IF NOT EXISTS idx_user_passkeys_user_id_passkey_id ON user_passkeys(user_id, passkey_id);

View File

@@ -10,7 +10,7 @@ CREATE TABLE IF NOT EXISTS raw_mails (
CREATE INDEX IF NOT EXISTS idx_raw_mails_address ON raw_mails(address); CREATE INDEX IF NOT EXISTS idx_raw_mails_address ON raw_mails(address);
CREATE TABLE IF NOT EXISTS address ( CREATE TABLE IF NOT EXISTS address (
id INTEGER PRIMARY KEY, id INTEGER PRIMARY KEY AUTOINCREMENT,
name TEXT UNIQUE, name TEXT UNIQUE,
created_at DATETIME DEFAULT CURRENT_TIMESTAMP, created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP updated_at DATETIME DEFAULT CURRENT_TIMESTAMP
@@ -88,3 +88,18 @@ CREATE TABLE IF NOT EXISTS user_roles (
); );
CREATE INDEX IF NOT EXISTS idx_user_roles_user_id ON user_roles(user_id); CREATE INDEX IF NOT EXISTS idx_user_roles_user_id ON user_roles(user_id);
CREATE TABLE IF NOT EXISTS user_passkeys (
id INTEGER PRIMARY KEY,
user_id INTEGER NOT NULL,
passkey_name TEXT NOT NULL,
passkey_id TEXT NOT NULL,
passkey TEXT NOT NULL,
counter INTEGER DEFAULT 0,
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP
);
CREATE INDEX IF NOT EXISTS idx_user_passkeys_user_id ON user_passkeys(user_id);
CREATE UNIQUE INDEX IF NOT EXISTS idx_user_passkeys_user_id_passkey_id ON user_passkeys(user_id, passkey_id);

View File

@@ -1,3 +0,0 @@
{
"recommendations": ["Vue.volar", "Vue.vscode-typescript-vue-plugin"]
}

View File

@@ -1,6 +1,6 @@
{ {
"name": "cloudflare_temp_email", "name": "cloudflare_temp_email",
"version": "0.6.0", "version": "0.8.6",
"private": true, "private": true,
"type": "module", "type": "module",
"scripts": { "scripts": {
@@ -8,7 +8,9 @@
"build": "vite build -m prod --emptyOutDir", "build": "vite build -m prod --emptyOutDir",
"build:release": "vite build -m example --emptyOutDir", "build:release": "vite build -m example --emptyOutDir",
"build:pages": "vite build -m pages --emptyOutDir", "build:pages": "vite build -m pages --emptyOutDir",
"build:pages:nopwa": "VITE_PWA_DISABLED=true vite build -m pages --emptyOutDir",
"build:telegram": "VITE_IS_TELEGRAM=true vite build -m prod --emptyOutDir", "build:telegram": "VITE_IS_TELEGRAM=true vite build -m prod --emptyOutDir",
"build:telegram:release": "VITE_IS_TELEGRAM=true vite build -m example --emptyOutDir",
"preview": "vite preview", "preview": "vite preview",
"deploy:telegram": "npm run build:telegram && wrangler pages deploy ./dist --branch production", "deploy:telegram": "npm run build:telegram && wrangler pages deploy ./dist --branch production",
"deploy:actions:telegram": "npm run build:telegram && wrangler pages deploy ./dist", "deploy:actions:telegram": "npm run build:telegram && wrangler pages deploy ./dist",
@@ -17,32 +19,34 @@
"deploy:actions": "npm run build && wrangler pages deploy ./dist" "deploy:actions": "npm run build && wrangler pages deploy ./dist"
}, },
"dependencies": { "dependencies": {
"@unhead/vue": "^1.9.15", "@simplewebauthn/browser": "10.0.0",
"@vicons/material": "^0.12.0", "@unhead/vue": "^1.11.18",
"@vueuse/core": "^10.11.0", "@vueuse/core": "^12.5.0",
"@wangeditor/editor": "^5.1.23", "@wangeditor/editor": "^5.1.23",
"@wangeditor/editor-for-vue": "^5.1.12", "@wangeditor/editor-for-vue": "^5.1.12",
"axios": "^1.7.2", "axios": "^1.7.9",
"jszip": "^3.10.1", "jszip": "^3.10.1",
"mail-parser-wasm": "^0.1.8", "mail-parser-wasm": "^0.2.1",
"naive-ui": "^2.38.2", "naive-ui": "^2.41.0",
"postal-mime": "^2.2.5", "postal-mime": "^2.4.1",
"vooks": "^0.2.12", "vooks": "^0.2.12",
"vue": "^3.4.31", "vue": "^3.5.13",
"vue-clipboard3": "^2.0.0", "vue-clipboard3": "^2.0.0",
"vue-i18n": "^9.13.1", "vue-i18n": "^11.0.1",
"vue-router": "^4.4.0" "vue-router": "^4.5.0"
}, },
"devDependencies": { "devDependencies": {
"@vicons/fa": "^0.12.0", "@vicons/fa": "^0.13.0",
"@vitejs/plugin-vue": "^5.0.5", "@vicons/material": "^0.13.0",
"unplugin-auto-import": "^0.17.6", "@vitejs/plugin-vue": "^5.2.1",
"unplugin-vue-components": "^0.27.2", "unplugin-auto-import": "^19.0.0",
"vite": "^5.3.3", "unplugin-vue-components": "^28.0.0",
"vite-plugin-pwa": "^0.19.8", "vite": "^6.0.11",
"vite-plugin-top-level-await": "^1.4.1", "vite-plugin-pwa": "^0.21.1",
"vite-plugin-wasm": "^3.3.0", "vite-plugin-top-level-await": "^1.4.4",
"workbox-window": "^7.1.0", "vite-plugin-wasm": "^3.4.1",
"wrangler": "^3.63.1" "workbox-build": "^7.3.0",
"workbox-window": "^7.3.0",
"wrangler": "^3.104.0"
} }
} }

4752
frontend/pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@@ -6,7 +6,7 @@ import { useGlobalState } from './store'
import { useIsMobile } from './utils/composables' import { useIsMobile } from './utils/composables'
import Header from './views/Header.vue'; import Header from './views/Header.vue';
import Footer from './views/Footer.vue'; import Footer from './views/Footer.vue';
import { api } from './api'
const { const {
isDark, loading, useSideMargin, telegramApp, isTelegram isDark, loading, useSideMargin, telegramApp, isTelegram
@@ -19,6 +19,13 @@ const showSideMargin = computed(() => !isMobile.value && useSideMargin.value);
onMounted(async () => { onMounted(async () => {
try {
await api.getUserSettings();
} catch (error) {
console.error(error);
}
const token = import.meta.env.VITE_CF_WEB_ANALY_TOKEN; const token = import.meta.env.VITE_CF_WEB_ANALY_TOKEN;
const exist = document.querySelector('script[src="https://static.cloudflareinsights.com/beacon.min.js"]') !== null const exist = document.querySelector('script[src="https://static.cloudflareinsights.com/beacon.min.js"]') !== null
@@ -54,24 +61,26 @@ onMounted(async () => {
<n-config-provider :locale="localeConfig" :theme="theme"> <n-config-provider :locale="localeConfig" :theme="theme">
<n-global-style /> <n-global-style />
<n-spin description="loading..." :show="loading"> <n-spin description="loading..." :show="loading">
<n-message-provider> <n-notification-provider container-style="margin-top: 60px;">
<n-grid x-gap="12" :cols="12"> <n-message-provider container-style="margin-top: 20px;">
<n-gi v-if="showSideMargin" span="1"></n-gi> <n-grid x-gap="12" :cols="12">
<n-gi :span="!showSideMargin ? 12 : 10"> <n-gi v-if="showSideMargin" span="1"></n-gi>
<div class="main"> <n-gi :span="!showSideMargin ? 12 : 10">
<n-space vertical> <div class="main">
<n-layout style="min-height: 80vh;"> <n-space vertical>
<Header /> <n-layout style="min-height: 80vh;">
<router-view></router-view> <Header />
</n-layout> <router-view></router-view>
<Footer /> </n-layout>
</n-space> <Footer />
</div> </n-space>
</n-gi> </div>
<n-gi v-if="showSideMargin" span="1"></n-gi> </n-gi>
</n-grid> <n-gi v-if="showSideMargin" span="1"></n-gi>
<n-back-top /> </n-grid>
</n-message-provider> <n-back-top />
</n-message-provider>
</n-notification-provider>
</n-spin> </n-spin>
</n-config-provider> </n-config-provider>
</template> </template>

View File

@@ -1,10 +1,11 @@
import { useGlobalState } from '../store' import { useGlobalState } from '../store'
import { h } from 'vue'
import axios from 'axios' import axios from 'axios'
const API_BASE = import.meta.env.VITE_API_BASE || ""; const API_BASE = import.meta.env.VITE_API_BASE || "";
const { const {
loading, auth, jwt, settings, openSettings, loading, auth, jwt, settings, openSettings,
userOpenSettings, userSettings, userOpenSettings, userSettings, announcement,
showAuth, adminAuth, showAdminAuth, userJwt showAuth, adminAuth, showAdminAuth, userJwt
} = useGlobalState(); } = useGlobalState();
@@ -22,6 +23,7 @@ const apiFetch = async (path, options = {}) => {
data: options.body || null, data: options.body || null,
headers: { headers: {
'x-user-token': userJwt.value, 'x-user-token': userJwt.value,
'x-user-access-token': userSettings.value.access_token,
'x-custom-auth': auth.value, 'x-custom-auth': auth.value,
'x-admin-auth': adminAuth.value, 'x-admin-auth': adminAuth.value,
'Authorization': `Bearer ${jwt.value}`, 'Authorization': `Bearer ${jwt.value}`,
@@ -51,11 +53,15 @@ const apiFetch = async (path, options = {}) => {
} }
} }
const getOpenSettings = async (message) => { const getOpenSettings = async (message, notification) => {
try { try {
const res = await api.fetch("/open_api/settings"); const res = await api.fetch("/open_api/settings");
const domainLabels = res["domainLabels"] || []; const domainLabels = res["domainLabels"] || [];
if (res["domains"]?.length < 1) {
message.error("No domains found, please check your worker settings");
}
Object.assign(openSettings.value, { Object.assign(openSettings.value, {
...res,
title: res["title"] || "", title: res["title"] || "",
prefix: res["prefix"] || "", prefix: res["prefix"] || "",
minAddressLen: res["minAddressLen"] || 1, minAddressLen: res["minAddressLen"] || 1,
@@ -70,6 +76,7 @@ const getOpenSettings = async (message) => {
}), }),
adminContact: res["adminContact"] || "", adminContact: res["adminContact"] || "",
enableUserCreateEmail: res["enableUserCreateEmail"] || false, enableUserCreateEmail: res["enableUserCreateEmail"] || false,
disableAnonymousUserCreateEmail: res["disableAnonymousUserCreateEmail"] || false,
enableUserDeleteEmail: res["enableUserDeleteEmail"] || false, enableUserDeleteEmail: res["enableUserDeleteEmail"] || false,
enableAutoReply: res["enableAutoReply"] || false, enableAutoReply: res["enableAutoReply"] || false,
enableIndexAbout: res["enableIndexAbout"] || false, enableIndexAbout: res["enableIndexAbout"] || false,
@@ -81,8 +88,20 @@ const getOpenSettings = async (message) => {
if (openSettings.value.needAuth) { if (openSettings.value.needAuth) {
showAuth.value = true; showAuth.value = true;
} }
if (openSettings.value.announcement && openSettings.value.announcement != announcement.value) {
announcement.value = openSettings.value.announcement;
notification.info({
content: () => {
return h("div", {
innerHTML: announcement.value
});
}
});
}
} catch (error) { } catch (error) {
message.error(error.message || "error"); message.error(error.message || "error");
} finally {
openSettings.value.fetched = true;
} }
} }
@@ -109,6 +128,8 @@ const getUserOpenSettings = async (message) => {
Object.assign(userOpenSettings.value, res); Object.assign(userOpenSettings.value, res);
} catch (error) { } catch (error) {
message.error(error.message || "fetch settings failed"); message.error(error.message || "fetch settings failed");
} finally {
userOpenSettings.value.fetched = true;
} }
} }
@@ -118,7 +139,7 @@ const getUserSettings = async (message) => {
const res = await api.fetch("/user_api/settings") const res = await api.fetch("/user_api/settings")
Object.assign(userSettings.value, res) Object.assign(userSettings.value, res)
} catch (error) { } catch (error) {
message.error(error.message || "error"); message?.error(error.message || "error");
} finally { } finally {
userSettings.value.fetched = true; userSettings.value.fetched = true;
} }

View File

@@ -3,9 +3,10 @@ import { watch, onMounted, ref, onBeforeUnmount } from "vue";
import { useMessage } from 'naive-ui' import { useMessage } from 'naive-ui'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { useGlobalState } from '../store' import { useGlobalState } from '../store'
import { CloudDownloadRound, ReplyFilled } from '@vicons/material' import { CloudDownloadRound, ReplyFilled, ForwardFilled } from '@vicons/material'
import { useIsMobile } from '../utils/composables' import { useIsMobile } from '../utils/composables'
import { processItem, getDownloadEmlUrl } from '../utils/email-parser' import { processItem, getDownloadEmlUrl } from '../utils/email-parser'
import { utcToLocalDate } from '../utils';
const message = useMessage() const message = useMessage()
const isMobile = useIsMobile() const isMobile = useIsMobile()
@@ -14,46 +15,45 @@ const props = defineProps({
enableUserDeleteEmail: { enableUserDeleteEmail: {
type: Boolean, type: Boolean,
default: false, default: false,
requried: false required: false
}, },
showEMailTo: { showEMailTo: {
type: Boolean, type: Boolean,
default: true, default: true,
requried: false required: false
}, },
fetchMailData: { fetchMailData: {
type: Function, type: Function,
default: () => { }, default: () => { },
requried: true required: true
}, },
deleteMail: { deleteMail: {
type: Function, type: Function,
default: () => { }, default: () => { },
requried: false required: false
}, },
showReply: { showReply: {
type: Boolean, type: Boolean,
default: false, default: false,
requried: false required: false
}, },
showSaveS3: { showSaveS3: {
type: Boolean, type: Boolean,
default: false, default: false,
requried: false required: false
}, },
saveToS3: { saveToS3: {
type: Function, type: Function,
default: (mail_id, filename, blob) => { }, default: (mail_id, filename, blob) => { },
requried: false required: false
}, },
}) })
const { const {
isDark, mailboxSplitSize, indexTab, loading, isDark, mailboxSplitSize, indexTab, loading, useUTCDate, autoRefresh, configAutoRefreshInterval,
useIframeShowMail, sendMailModel, preferShowTextMail useIframeShowMail, sendMailModel, preferShowTextMail
} = useGlobalState() } = useGlobalState()
const autoRefresh = ref(false) const autoRefreshInterval = ref(configAutoRefreshInterval.value)
const autoRefreshInterval = ref(30)
const data = ref([]) const data = ref([])
const timer = ref(null) const timer = ref(null)
@@ -85,6 +85,7 @@ const { t } = useI18n({
delete: 'Delete', delete: 'Delete',
deleteMailTip: 'Are you sure you want to delete mail?', deleteMailTip: 'Are you sure you want to delete mail?',
reply: 'Reply', reply: 'Reply',
forwardMail: 'Forward',
showTextMail: 'Show Text Mail', showTextMail: 'Show Text Mail',
showHtmlMail: 'Show Html Mail', showHtmlMail: 'Show Html Mail',
saveToS3: 'Save to S3', saveToS3: 'Save to S3',
@@ -104,6 +105,7 @@ const { t } = useI18n({
delete: '删除', delete: '删除',
deleteMailTip: '确定要删除邮件吗?', deleteMailTip: '确定要删除邮件吗?',
reply: '回复', reply: '回复',
forwardMail: '转发',
showTextMail: '显示纯文本邮件', showTextMail: '显示纯文本邮件',
showHtmlMail: '显示HTML邮件', showHtmlMail: '显示HTML邮件',
saveToS3: '保存到S3', saveToS3: '保存到S3',
@@ -116,14 +118,16 @@ const { t } = useI18n({
}); });
const setupAutoRefresh = async (autoRefresh) => { const setupAutoRefresh = async (autoRefresh) => {
// auto refresh every 30 seconds // auto refresh every configAutoRefreshInterval seconds
autoRefreshInterval.value = 30; autoRefreshInterval.value = configAutoRefreshInterval.value;
if (autoRefresh) { if (autoRefresh) {
clearInterval(timer.value);
timer.value = setInterval(async () => { timer.value = setInterval(async () => {
if (loading.value) return;
autoRefreshInterval.value--; autoRefreshInterval.value--;
if (autoRefreshInterval.value <= 0) { if (autoRefreshInterval.value <= 0) {
autoRefreshInterval.value = 30; autoRefreshInterval.value = configAutoRefreshInterval.value;
await refresh(); await backFirstPageAndRefresh();
} }
}, 1000) }, 1000)
} else { } else {
@@ -134,7 +138,7 @@ const setupAutoRefresh = async (autoRefresh) => {
watch(autoRefresh, async (autoRefresh, old) => { watch(autoRefresh, async (autoRefresh, old) => {
setupAutoRefresh(autoRefresh) setupAutoRefresh(autoRefresh)
}) }, { immediate: true })
watch([page, pageSize], async ([page, pageSize], [oldPage, oldPageSize]) => { watch([page, pageSize], async ([page, pageSize], [oldPage, oldPageSize]) => {
if (page !== oldPage || pageSize !== oldPageSize) { if (page !== oldPage || pageSize !== oldPageSize) {
@@ -147,6 +151,7 @@ const refresh = async () => {
const { results, count: totalCount } = await props.fetchMailData( const { results, count: totalCount } = await props.fetchMailData(
pageSize.value, (page.value - 1) * pageSize.value pageSize.value, (page.value - 1) * pageSize.value
); );
loading.value = true;
data.value = await Promise.all(results.map(async (item) => { data.value = await Promise.all(results.map(async (item) => {
item.checked = false; item.checked = false;
return await processItem(item); return await processItem(item);
@@ -161,9 +166,16 @@ const refresh = async () => {
} catch (error) { } catch (error) {
message.error(error.message || "error"); message.error(error.message || "error");
console.error(error); console.error(error);
} finally {
loading.value = false;
} }
}; };
const backFirstPageAndRefresh = async () =>{
page.value = 1;
await refresh();
}
const clickRow = async (row) => { const clickRow = async (row) => {
if (multiActionMode.value) { if (multiActionMode.value) {
row.checked = !row.checked; row.checked = !row.checked;
@@ -211,6 +223,15 @@ const replyMail = async () => {
indexTab.value = 'sendmail'; indexTab.value = 'sendmail';
}; };
const forwardMail = async () => {
Object.assign(sendMailModel.value, {
subject: `${t('forwardMail')}: ${curMail.value.subject}`,
contentType: curMail.value.message ? 'html' : 'text',
content: curMail.value.message || curMail.value.text,
});
indexTab.value = 'sendmail';
};
const onSpiltSizeChange = (size) => { const onSpiltSizeChange = (size) => {
mailboxSplitSize.value = size; mailboxSplitSize.value = size;
} }
@@ -351,7 +372,7 @@ onBeforeUnmount(() => {
{{ t('autoRefresh') }} {{ t('autoRefresh') }}
</template> </template>
</n-switch> </n-switch>
<n-button @click="refresh" type="primary" tertiary> <n-button @click="backFirstPageAndRefresh" type="primary" tertiary>
{{ t('refresh') }} {{ t('refresh') }}
</n-button> </n-button>
</n-space> </n-space>
@@ -372,7 +393,7 @@ onBeforeUnmount(() => {
ID: {{ row.id }} ID: {{ row.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${row.created_at} UTC` }} {{ utcToLocalDate(row.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ row.source }} FROM: {{ row.source }}
@@ -394,7 +415,7 @@ onBeforeUnmount(() => {
ID: {{ curMail.id }} ID: {{ curMail.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${curMail.created_at} UTC` }} {{ utcToLocalDate(curMail.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ curMail.source }} FROM: {{ curMail.source }}
@@ -425,6 +446,12 @@ onBeforeUnmount(() => {
</template> </template>
{{ t('reply') }} {{ t('reply') }}
</n-button> </n-button>
<n-button v-if="showReply" size="small" tertiary type="info" @click="forwardMail">
<template #icon>
<n-icon :component="ForwardFilled" />
</template>
{{ t('forwardMail') }}
</n-button>
<n-button size="small" tertiary type="info" @click="showTextMail = !showTextMail"> <n-button size="small" tertiary type="info" @click="showTextMail = !showTextMail">
{{ showTextMail ? t('showHtmlMail') : t('showTextMail') }} {{ showTextMail ? t('showHtmlMail') : t('showTextMail') }}
</n-button> </n-button>
@@ -455,7 +482,7 @@ onBeforeUnmount(() => {
{{ t('autoRefresh') }} {{ t('autoRefresh') }}
</template> </template>
</n-switch> </n-switch>
<n-button @click="refresh" tertiary size="small" type="primary"> <n-button @click="backFirstPageAndRefresh" tertiary size="small" type="primary">
{{ t('refresh') }} {{ t('refresh') }}
</n-button> </n-button>
</n-space> </n-space>
@@ -468,7 +495,7 @@ onBeforeUnmount(() => {
ID: {{ row.id }} ID: {{ row.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${row.created_at} UTC` }} {{ utcToLocalDate(row.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ row.source }} FROM: {{ row.source }}
@@ -490,7 +517,7 @@ onBeforeUnmount(() => {
ID: {{ curMail.id }} ID: {{ curMail.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${curMail.created_at} UTC` }} {{ utcToLocalDate(curMail.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ curMail.source }} FROM: {{ curMail.source }}
@@ -519,6 +546,12 @@ onBeforeUnmount(() => {
</template> </template>
{{ t('reply') }} {{ t('reply') }}
</n-button> </n-button>
<n-button v-if="showReply" size="small" tertiary type="info" @click="forwardMail">
<template #icon>
<n-icon :component="ForwardFilled" />
</template>
{{ t('forwardMail') }}
</n-button>
<n-button size="small" tertiary type="info" @click="showTextMail = !showTextMail"> <n-button size="small" tertiary type="info" @click="showTextMail = !showTextMail">
{{ showTextMail ? t('showHtmlMail') : t('showTextMail') }} {{ showTextMail ? t('showHtmlMail') : t('showTextMail') }}
</n-button> </n-button>

View File

@@ -4,6 +4,7 @@ import { useMessage } from 'naive-ui'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { useGlobalState } from '../store' import { useGlobalState } from '../store'
import { useIsMobile } from '../utils/composables' import { useIsMobile } from '../utils/composables'
import { utcToLocalDate } from '../utils';
const message = useMessage() const message = useMessage()
const isMobile = useIsMobile() const isMobile = useIsMobile()
@@ -12,7 +13,7 @@ const props = defineProps({
enableUserDeleteEmail: { enableUserDeleteEmail: {
type: Boolean, type: Boolean,
default: false, default: false,
requried: false required: false
}, },
showEMailFrom: { showEMailFrom: {
type: Boolean, type: Boolean,
@@ -21,16 +22,16 @@ const props = defineProps({
fetchMailData: { fetchMailData: {
type: Function, type: Function,
default: () => { }, default: () => { },
requried: true required: true
}, },
deleteMail: { deleteMail: {
type: Function, type: Function,
default: () => { }, default: () => { },
requried: false required: false
}, },
}) })
const { isDark, mailboxSplitSize, loading } = useGlobalState() const { isDark, mailboxSplitSize, loading, useUTCDate } = useGlobalState()
const data = ref([]) const data = ref([])
const count = ref(0) const count = ref(0)
@@ -251,7 +252,7 @@ onMounted(async () => {
ID: {{ row.id }} ID: {{ row.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${row.created_at} UTC` }} {{ utcToLocalDate(row.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag v-if="showEMailFrom" type="info"> <n-tag v-if="showEMailFrom" type="info">
FROM: {{ row.address }} FROM: {{ row.address }}
@@ -273,7 +274,7 @@ onMounted(async () => {
ID: {{ curMail.id }} ID: {{ curMail.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${curMail.created_at} UTC` }} {{ utcToLocalDate(curMail.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ curMail.address }} FROM: {{ curMail.address }}
@@ -320,7 +321,7 @@ onMounted(async () => {
ID: {{ row.id }} ID: {{ row.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${row.created_at} UTC` }} {{ utcToLocalDate(row.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag v-if="showEMailFrom" type="info"> <n-tag v-if="showEMailFrom" type="info">
FROM: {{ row.address }} FROM: {{ row.address }}
@@ -342,7 +343,7 @@ onMounted(async () => {
ID: {{ curMail.id }} ID: {{ curMail.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
{{ `${curMail.created_at} UTC` }} {{ utcToLocalDate(curMail.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ curMail.address }} FROM: {{ curMail.address }}

View File

@@ -0,0 +1,179 @@
<script setup lang="ts">
import { onMounted, ref } from 'vue'
import { useI18n } from 'vue-i18n'
const props = defineProps({
fetchData: {
type: Function,
default: () => { },
required: true
},
saveSettings: {
type: Function,
default: (webhookSettings: WebhookSettings) => { },
required: true
},
testSettings: {
type: Function,
default: (webhookSettings: WebhookSettings) => { },
required: true
},
})
// @ts-ignore
const message = useMessage()
const { t } = useI18n({
messages: {
en: {
successTip: 'Success',
test: 'Test',
save: 'Save',
notEnabled: 'Webhook is not enabled for you',
urlMissing: 'URL is required',
enable: 'Enable',
messagePusherDemo: 'Fill with Message Pusher Demo',
messagePusherDoc: 'Message Pusher Doc',
fillInDemoTip: 'Please modify the URL and other settings to your own',
},
zh: {
successTip: '成功',
test: '测试',
save: '保存',
notEnabled: 'Webhook 未开启,请联系管理员开启',
urlMissing: 'URL 不能为空',
enable: '启用',
messagePusherDemo: '填入MessagePusher示例',
messagePusherDoc: 'MessagePusher文档',
fillInDemoTip: '请修改URL和其他设置为您自己的配置',
}
}
});
class WebhookSettings {
enabled: boolean = false
url: string = ''
method: string = 'POST'
headers: string = JSON.stringify({}, null, 2)
body: string = JSON.stringify({}, null, 2)
}
const messagePusherDocLink = "https://github.com/songquanpeng/message-pusher";
const messagePusherDemo = {
enabled: true,
url: 'https://msgpusher.com/push/username',
method: 'POST',
headers: JSON.stringify({
'Content-Type': 'application/json',
}, null, 2),
body: JSON.stringify({
"token": "token",
"title": "${subject}",
"description": "${subject}",
"content": "*${subject}*\n\nFrom: ${from}\nTo: ${to}\n\n${parsedText}\n"
}, null, 2),
} as WebhookSettings;
const fillMessagePuhserDemo = () => {
Object.assign(webhookSettings.value, messagePusherDemo)
message.success(t('fillInDemoTip'))
}
const webhookSettings = ref<WebhookSettings>(new WebhookSettings())
const enableWebhook = ref(false)
const fetchData = async () => {
try {
const res = await props.fetchData()
Object.assign(webhookSettings.value, res)
enableWebhook.value = true
} catch (error) {
message.error((error as Error).message || "error");
}
}
const saveSettings = async () => {
if (!webhookSettings.value.url) {
message.error(t('urlMissing'))
return
}
try {
await props.saveSettings(webhookSettings.value)
message.success(t('successTip'))
} catch (error) {
message.error((error as Error).message || "error");
}
}
const testSettings = async () => {
if (!webhookSettings.value.url) {
message.error(t('urlMissing'))
return
}
try {
await props.testSettings(webhookSettings.value)
message.success(t('successTip'))
} catch (error) {
message.error((error as Error).message || "error");
}
}
onMounted(async () => {
await fetchData();
})
</script>
<template>
<div class="center">
<n-card :bordered="false" embedded v-if="enableWebhook" style="max-width: 800px; overflow: auto;">
<n-flex justify="end">
<n-button tag="a" :href="messagePusherDocLink" target="_blank" secondary>
{{ t('messagePusherDoc') }}
</n-button>
<n-button @click="fillMessagePuhserDemo" secondary>
{{ t('messagePusherDemo') }}
</n-button>
<n-button v-if="webhookSettings.enabled" @click="testSettings" secondary>
{{ t('test') }}
</n-button>
<n-button @click="saveSettings" type="primary">
{{ t('save') }}
</n-button>
</n-flex>
<n-form-item-row :label="t('enable')">
<n-switch v-model:value="webhookSettings.enabled" :round="false" />
</n-form-item-row>
<div v-if="webhookSettings.enabled">
<n-form-item-row label="URL">
<n-input v-model:value="webhookSettings.url" />
</n-form-item-row>
<n-form-item-row label="METHOD">
<n-select v-model:value="webhookSettings.method" tag :options='[
{ label: "POST", value: "POST" }
]' />
</n-form-item-row>
<n-form-item-row label="HEADERS">
<n-input v-model:value="webhookSettings.headers" type="textarea" :autosize="{ minRows: 3 }" />
</n-form-item-row>
<n-form-item-row label="BODY">
<n-input v-model:value="webhookSettings.body" type="textarea" :autosize="{ minRows: 3 }" />
</n-form-item-row>
</div>
</n-card>
<n-result v-else status="404" :title="t('notEnabled')" />
</div>
</template>
<style scoped>
.center {
display: flex;
text-align: left;
place-items: center;
justify-content: center;
}
.n-button {
margin-top: 10px;
}
</style>

View File

@@ -0,0 +1,8 @@
const COMMOM_MAIL = [
"gmail.com", "163.com", "126.com", "qq.com", "outlook.com", "hotmail.com",
"icloud.com", "yahoo.com", "foxmail.com"
]
export default {
COMMOM_MAIL
}

View File

@@ -2,10 +2,8 @@ import { createApp } from 'vue'
import App from './App.vue' import App from './App.vue'
import { createI18n } from 'vue-i18n' import { createI18n } from 'vue-i18n'
import router from './router' import router from './router'
import { registerSW } from 'virtual:pwa-register'
import { createHead } from '@unhead/vue' import { createHead } from '@unhead/vue'
registerSW({ immediate: true })
const i18n = createI18n({ const i18n = createI18n({
legacy: false, // you must set `false`, to use Composition API legacy: false, // you must set `false`, to use Composition API
locale: 'zh', // set locale locale: 'zh', // set locale

View File

@@ -0,0 +1,15 @@
export type UserOauth2Settings = {
name: string;
clientID: string;
clientSecret: string;
authorizationURL: string;
accessTokenURL: string;
accessTokenFormat?: string;
userInfoURL: string;
redirectURL: string;
logoutURL?: string;
userEmailKey: string;
scope: string;
enableMailAllowList?: boolean | undefined;
mailAllowList?: string[] | undefined;
}

View File

@@ -1,7 +1,7 @@
import { createRouter, createWebHistory } from 'vue-router' import { createRouter, createWebHistory } from 'vue-router'
import Index from '../views/Index.vue' import Index from '../views/Index.vue'
import User from '../views/User.vue' import User from '../views/User.vue'
import { useGlobalState } from '../store' import UserOauth2Callback from '../views/user/UserOauth2Callback.vue'
const router = createRouter({ const router = createRouter({
history: createWebHistory(), history: createWebHistory(),
@@ -16,6 +16,11 @@ const router = createRouter({
alias: "/:lang/user", alias: "/:lang/user",
component: User component: User
}, },
{
path: '/user/oauth2/callback',
alias: "/:lang/user/oauth2/callback",
component: UserOauth2Callback
},
{ {
path: '/admin', path: '/admin',
alias: "/:lang/admin", alias: "/:lang/admin",

View File

@@ -1,17 +1,25 @@
import { ref } from "vue"; import { computed, ref } from "vue";
import { createGlobalState, useStorage, useDark, useToggle } from '@vueuse/core' import {
createGlobalState, useStorage, useDark, useToggle,
useLocalStorage, useSessionStorage
} from '@vueuse/core'
export const useGlobalState = createGlobalState( export const useGlobalState = createGlobalState(
() => { () => {
const isDark = useDark() const isDark = useDark()
const toggleDark = useToggle(isDark) const toggleDark = useToggle(isDark)
const loading = ref(false); const loading = ref(false);
const announcement = useLocalStorage('announcement', '');
const openSettings = ref({ const openSettings = ref({
fetched: false,
title: '', title: '',
announcement: '',
prefix: '', prefix: '',
addressRegex: '',
needAuth: false, needAuth: false,
adminContact: '', adminContact: '',
enableUserCreateEmail: false, enableUserCreateEmail: false,
disableAnonymousUserCreateEmail: false,
enableUserDeleteEmail: false, enableUserDeleteEmail: false,
enableAutoReply: false, enableAutoReply: false,
enableIndexAbout: false, enableIndexAbout: false,
@@ -23,6 +31,8 @@ export const useGlobalState = createGlobalState(
cfTurnstileSiteKey: '', cfTurnstileSiteKey: '',
enableWebhook: false, enableWebhook: false,
isS3Enabled: false, isS3Enabled: false,
showGithub: true,
disableAdminPasswordCheck: false,
}) })
const settings = ref({ const settings = ref({
fetched: false, fetched: false,
@@ -36,7 +46,7 @@ export const useGlobalState = createGlobalState(
name: '', name: '',
} }
}); });
const sendMailModel = useStorage('sendMailModel', { const sendMailModel = useSessionStorage('sendMailModel', {
fromName: "", fromName: "",
toName: "", toName: "",
toMail: "", toMail: "",
@@ -50,20 +60,26 @@ export const useGlobalState = createGlobalState(
const auth = useStorage('auth', ''); const auth = useStorage('auth', '');
const adminAuth = useStorage('adminAuth', ''); const adminAuth = useStorage('adminAuth', '');
const jwt = useStorage('jwt', ''); const jwt = useStorage('jwt', '');
const adminTab = ref("account"); const adminTab = useSessionStorage('adminTab', "account");
const adminMailTabAddress = ref(""); const adminMailTabAddress = ref("");
const adminSendBoxTabAddress = ref(""); const adminSendBoxTabAddress = ref("");
const mailboxSplitSize = useStorage('mailboxSplitSize', 0.25); const mailboxSplitSize = useStorage('mailboxSplitSize', 0.25);
const useIframeShowMail = useStorage('useIframeShowMail', false); const useIframeShowMail = useStorage('useIframeShowMail', false);
const preferShowTextMail = useStorage('preferShowTextMail', false); const preferShowTextMail = useStorage('preferShowTextMail', false);
const userJwt = useStorage('userJwt', ''); const userJwt = useStorage('userJwt', '');
const userTab = useStorage('userTab', 'user_settings'); const userTab = useSessionStorage('userTab', 'user_settings');
const indexTab = useStorage('indexTab', 'mailbox'); const indexTab = useSessionStorage('indexTab', 'mailbox');
const globalTabplacement = useStorage('globalTabplacement', 'top'); const globalTabplacement = useStorage('globalTabplacement', 'top');
const useSideMargin = useStorage('useSideMargin', true); const useSideMargin = useStorage('useSideMargin', true);
const useUTCDate = useStorage('useUTCDate', false);
const autoRefresh = useStorage('autoRefresh', false);
const configAutoRefreshInterval = useStorage("configAutoRefreshInterval", 60);
const userOpenSettings = ref({ const userOpenSettings = ref({
fetched: false,
enable: false, enable: false,
enableMailVerify: false, enableMailVerify: false,
/** @type {{ clientID: string, name: string }[]} */
oauth2ClientIDs: [],
}); });
const userSettings = ref({ const userSettings = ref({
/** @type {boolean} */ /** @type {boolean} */
@@ -72,17 +88,29 @@ export const useGlobalState = createGlobalState(
user_email: '', user_email: '',
/** @type {number} */ /** @type {number} */
user_id: 0, user_id: 0,
/** @type {boolean} */
is_admin: false,
/** @type {string | null} */
access_token: null,
/** @type {null | {domains: string[] | undefined | null, role: string, prefix: string | undefined | null}} */ /** @type {null | {domains: string[] | undefined | null, role: string, prefix: string | undefined | null}} */
user_role: null, user_role: null,
}); });
const showAdminPage = computed(() =>
!!adminAuth.value
|| userSettings.value.is_admin
|| openSettings.value.disableAdminPasswordCheck
);
const telegramApp = ref(window.Telegram?.WebApp || {}); const telegramApp = ref(window.Telegram?.WebApp || {});
const isTelegram = ref(!!window.Telegram?.WebApp?.initData); const isTelegram = ref(!!window.Telegram?.WebApp?.initData);
const userOauth2SessionState = useSessionStorage('userOauth2SessionState', '');
const userOauth2SessionClientID = useSessionStorage('userOauth2SessionClientID', '');
return { return {
isDark, isDark,
toggleDark, toggleDark,
loading, loading,
settings, settings,
sendMailModel, sendMailModel,
announcement,
openSettings, openSettings,
showAuth, showAuth,
showAddressCredential, showAddressCredential,
@@ -103,8 +131,14 @@ export const useGlobalState = createGlobalState(
userSettings, userSettings,
globalTabplacement, globalTabplacement,
useSideMargin, useSideMargin,
useUTCDate,
autoRefresh,
configAutoRefreshInterval,
telegramApp, telegramApp,
isTelegram, isTelegram,
showAdminPage,
userOauth2SessionState,
userOauth2SessionClientID,
} }
}, },
) )

View File

@@ -11,3 +11,17 @@ export const getRouterPathWithLang = (path: string, lang: string) => {
} }
return `/${lang}${path}`; return `/${lang}${path}`;
} }
export const utcToLocalDate = (utcDate: string, useUTCDate: boolean) => {
const utcDateString = `${utcDate} UTC`;
if (useUTCDate) {
return utcDateString;
}
try {
const date = new Date(utcDateString);
return date.toLocaleString();
} catch (e) {
console.error(e);
}
return utcDateString;
}

View File

@@ -1,8 +1,9 @@
<script setup> <script setup>
import { onMounted } from 'vue'; import { computed, onMounted, ref } from 'vue';
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { useGlobalState } from '../store' import { useGlobalState } from '../store'
import { api } from '../api'
import SenderAccess from './admin/SenderAccess.vue' import SenderAccess from './admin/SenderAccess.vue'
import Statistics from "./admin/Statistics.vue" import Statistics from "./admin/Statistics.vue"
@@ -12,6 +13,7 @@ import CreateAccount from './admin/CreateAccount.vue';
import AccountSettings from './admin/AccountSettings.vue'; import AccountSettings from './admin/AccountSettings.vue';
import UserManagement from './admin/UserManagement.vue'; import UserManagement from './admin/UserManagement.vue';
import UserSettings from './admin/UserSettings.vue'; import UserSettings from './admin/UserSettings.vue';
import UserOauth2Settings from './admin/UserOauth2Settings.vue';
import Mails from './admin/Mails.vue'; import Mails from './admin/Mails.vue';
import MailsUnknow from './admin/MailsUnknow.vue'; import MailsUnknow from './admin/MailsUnknow.vue';
import About from './common/About.vue'; import About from './common/About.vue';
@@ -19,14 +21,19 @@ import Maintenance from './admin/Maintenance.vue';
import Appearance from './common/Appearance.vue'; import Appearance from './common/Appearance.vue';
import Telegram from './admin/Telegram.vue'; import Telegram from './admin/Telegram.vue';
import Webhook from './admin/Webhook.vue'; import Webhook from './admin/Webhook.vue';
import MailWebhook from './admin/MailWebhook.vue';
import WorkerConfig from './admin/WorkerConfig.vue';
import SendMail from './admin/SendMail.vue';
const { const {
adminAuth, showAdminAuth, adminTab, loading, globalTabplacement adminAuth, showAdminAuth, adminTab, loading,
globalTabplacement, showAdminPage, userSettings
} = useGlobalState() } = useGlobalState()
const message = useMessage() const message = useMessage()
const authFunc = async () => { const authFunc = async () => {
try { try {
adminAuth.value = tmpAdminAuth.value;
location.reload() location.reload()
} catch (error) { } catch (error) {
message.error(error.message || "error"); message.error(error.message || "error");
@@ -39,70 +46,94 @@ const { t } = useI18n({
accessHeader: 'Admin Password', accessHeader: 'Admin Password',
accessTip: 'Please enter the admin password', accessTip: 'Please enter the admin password',
mails: 'Emails', mails: 'Emails',
sendMail: 'Send Mail',
qucickSetup: 'Quick Setup',
account: 'Account', account: 'Account',
account_create: 'Create Account', account_create: 'Create Account',
account_settings: 'Account Settings', account_settings: 'Account Settings',
user: 'User', user: 'User',
user_management: 'User Management', user_management: 'User Management',
user_settings: 'User Settings', user_settings: 'User Settings',
userOauth2Settings: 'Oauth2 Settings',
unknow: 'Mails with unknow receiver', unknow: 'Mails with unknow receiver',
senderAccess: 'Sender Access Control', senderAccess: 'Sender Access Control',
sendBox: 'Send Box', sendBox: 'Send Box',
telegram: 'Telegram Bot', telegram: 'Telegram Bot',
webhook: 'Webhook', webhookSettings: 'Webhook Settings',
statistics: 'Statistics', statistics: 'Statistics',
maintenance: 'Maintenance', maintenance: 'Maintenance',
workerconfig: 'Worker Config',
appearance: 'Appearance', appearance: 'Appearance',
about: 'About', about: 'About',
ok: 'OK', ok: 'OK',
mailWebhook: 'Mail Webhook',
}, },
zh: { zh: {
accessHeader: 'Admin 密码', accessHeader: 'Admin 密码',
accessTip: '请输入 Admin 密码', accessTip: '请输入 Admin 密码',
mails: '邮件', mails: '邮件',
sendMail: '发送邮件',
qucickSetup: '快速设置',
account: '账号', account: '账号',
account_create: '创建账号', account_create: '创建账号',
account_settings: '账号设置', account_settings: '账号设置',
user: '用户', user: '用户',
user_management: '用户管理', user_management: '用户管理',
user_settings: '用户设置', user_settings: '用户设置',
userOauth2Settings: 'Oauth2 设置',
unknow: '无收件人邮件', unknow: '无收件人邮件',
senderAccess: '发件权限控制', senderAccess: '发件权限控制',
sendBox: '发件箱', sendBox: '发件箱',
telegram: '电报机器人', telegram: '电报机器人',
webhook: 'Webhook', webhookSettings: 'Webhook 设置',
statistics: '统计', statistics: '统计',
maintenance: '维护', maintenance: '维护',
workerconfig: 'Worker 配置',
appearance: '外观', appearance: '外观',
about: '关于', about: '关于',
ok: '确定', ok: '确定',
mailWebhook: '邮件 Webhook',
} }
} }
}); });
const showAdminPasswordModal = computed(() => !showAdminPage.value || showAdminAuth.value)
const tmpAdminAuth = ref('')
onMounted(async () => { onMounted(async () => {
if (!adminAuth.value) { // make sure user_id is fetched
showAdminAuth.value = true; if (!userSettings.value.user_id) await api.getUserSettings(message);
return;
}
}) })
</script> </script>
<template> <template>
<div> <div>
<n-modal v-model:show="showAdminAuth" :closable="false" :closeOnEsc="false" :maskClosable="false" preset="dialog" <n-modal v-model:show="showAdminPasswordModal" :closable="false" :closeOnEsc="false" :maskClosable="false"
:title="t('accessHeader')"> preset="dialog" :title="t('accessHeader')">
<p>{{ t('accessTip') }}</p> <p>{{ t('accessTip') }}</p>
<n-input v-model:value="adminAuth" type="textarea" :autosize="{ minRows: 3 }" /> <n-input v-model:value="tmpAdminAuth" type="password" show-password-on="click" />
<template #action> <template #action>
<n-button @click="authFunc" type="primary" :loading="loading"> <n-button @click="authFunc" type="primary" :loading="loading">
{{ t('ok') }} {{ t('ok') }}
</n-button> </n-button>
</template> </template>
</n-modal> </n-modal>
<n-tabs type="card" v-model:value="adminTab" :placement="globalTabplacement"> <n-tabs v-if="showAdminPage" type="card" v-model:value="adminTab" :placement="globalTabplacement">
<n-tab-pane name="qucickSetup" :tab="t('qucickSetup')">
<n-tabs type="bar" justify-content="center" animated>
<n-tab-pane name="account_settings" :tab="t('account_settings')">
<AccountSettings />
</n-tab-pane>
<n-tab-pane name="user_settings" :tab="t('user_settings')">
<UserSettings />
</n-tab-pane>
<n-tab-pane name="workerconfig" :tab="t('workerconfig')">
<WorkerConfig />
</n-tab-pane>
</n-tabs>
</n-tab-pane>
<n-tab-pane name="account" :tab="t('account')"> <n-tab-pane name="account" :tab="t('account')">
<n-tabs type="bar" animated> <n-tabs type="bar" justify-content="center" animated>
<n-tab-pane name="account" :tab="t('account')"> <n-tab-pane name="account" :tab="t('account')">
<Account /> <Account />
</n-tab-pane> </n-tab-pane>
@@ -115,34 +146,43 @@ onMounted(async () => {
<n-tab-pane name="senderAccess" :tab="t('senderAccess')"> <n-tab-pane name="senderAccess" :tab="t('senderAccess')">
<SenderAccess /> <SenderAccess />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="webhook" :tab="t('webhook')"> <n-tab-pane name="webhook" :tab="t('webhookSettings')">
<Webhook /> <Webhook />
</n-tab-pane> </n-tab-pane>
</n-tabs> </n-tabs>
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="user" :tab="t('user')"> <n-tab-pane name="user" :tab="t('user')">
<n-tabs type="bar" animated> <n-tabs type="bar" justify-content="center" animated>
<n-tab-pane name="user_management" :tab="t('user_management')"> <n-tab-pane name="user_management" :tab="t('user_management')">
<UserManagement /> <UserManagement />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="user_settings" :tab="t('user_settings')"> <n-tab-pane name="user_settings" :tab="t('user_settings')">
<UserSettings /> <UserSettings />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="userOauth2Settings" :tab="t('userOauth2Settings')">
<UserOauth2Settings />
</n-tab-pane>
</n-tabs> </n-tabs>
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="mails" :tab="t('mails')"> <n-tab-pane name="mails" :tab="t('mails')">
<n-tabs type="bar" animated> <n-tabs type="bar" justify-content="center" animated>
<n-tab-pane name="mails" :tab="t('mails')"> <n-tab-pane name="mails" :tab="t('mails')">
<Mails /> <Mails />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="unknow" :tab="t('unknow')"> <n-tab-pane name="unknow" :tab="t('unknow')">
<MailsUnknow /> <MailsUnknow />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="sendBox" :tab="t('sendBox')">
<SendBox />
</n-tab-pane>
<n-tab-pane name="sendMail" :tab="t('sendMail')">
<SendMail />
</n-tab-pane>
<n-tab-pane name="mailWebhook" :tab="t('mailWebhook')">
<MailWebhook />
</n-tab-pane>
</n-tabs> </n-tabs>
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="sendBox" :tab="t('sendBox')">
<SendBox />
</n-tab-pane>
<n-tab-pane name="telegram" :tab="t('telegram')"> <n-tab-pane name="telegram" :tab="t('telegram')">
<Telegram /> <Telegram />
</n-tab-pane> </n-tab-pane>
@@ -150,7 +190,14 @@ onMounted(async () => {
<Statistics /> <Statistics />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="maintenance" :tab="t('maintenance')"> <n-tab-pane name="maintenance" :tab="t('maintenance')">
<Maintenance /> <n-tabs type="bar" justify-content="center" animated>
<n-tab-pane name="workerconfig" :tab="t('workerconfig')">
<WorkerConfig />
</n-tab-pane>
<n-tab-pane name="maintenance" :tab="t('maintenance')">
<Maintenance />
</n-tab-pane>
</n-tabs>
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="appearance" :tab="t('appearance')"> <n-tab-pane name="appearance" :tab="t('appearance')">
<Appearance /> <Appearance />

View File

@@ -21,9 +21,14 @@ const { t } = useI18n({
<div> <div>
<n-divider class="footer-divider" /> <n-divider class="footer-divider" />
<div style="text-align: center; padding: 20px"> <div style="text-align: center; padding: 20px">
<n-text depth="3"> <n-space justify="center">
{{ t('copyright') }} © 2023-{{ new Date().getFullYear() }} {{ openSettings.copyright }} <n-text depth="3">
</n-text> {{ t('copyright') }} © 2023-{{ new Date().getFullYear() }}
</n-text>
<n-text depth="3">
<div v-html="openSettings.copyright"></div>
</n-text>
</n-space>
</div> </div>
</div> </div>
</template> </template>

View File

@@ -15,10 +15,11 @@ import { api } from '../api'
import { getRouterPathWithLang } from '../utils' import { getRouterPathWithLang } from '../utils'
const message = useMessage() const message = useMessage()
const notification = useNotification()
const { const {
toggleDark, isDark, isTelegram, toggleDark, isDark, isTelegram, showAdminPage,
showAuth, adminAuth, auth, loading, openSettings showAuth, auth, loading, openSettings, userSettings
} = useGlobalState() } = useGlobalState()
const route = useRoute() const route = useRoute()
const router = useRouter() const router = useRouter()
@@ -134,7 +135,7 @@ const menuOptions = computed(() => [
icon: () => h(NIcon, { component: AdminPanelSettingsFilled }), icon: () => h(NIcon, { component: AdminPanelSettingsFilled }),
} }
), ),
show: !!adminAuth.value, show: showAdminPage.value,
key: "admin" key: "admin"
}, },
{ {
@@ -192,6 +193,7 @@ const menuOptions = computed(() => [
icon: () => h(NIcon, { component: GithubAlt }) icon: () => h(NIcon, { component: GithubAlt })
} }
), ),
show: openSettings.value?.showGithub,
key: "github" key: "github"
} }
]); ]);
@@ -222,7 +224,9 @@ const logoClick = async () => {
} }
onMounted(async () => { onMounted(async () => {
await api.getOpenSettings(message); await api.getOpenSettings(message, notification);
// make sure user_id is fetched
if (!userSettings.value.user_id) await api.getUserSettings(message);
}); });
</script> </script>
@@ -257,7 +261,7 @@ onMounted(async () => {
<n-modal v-model:show="showAuth" :closable="false" :closeOnEsc="false" :maskClosable="false" preset="dialog" <n-modal v-model:show="showAuth" :closable="false" :closeOnEsc="false" :maskClosable="false" preset="dialog"
:title="t('accessHeader')"> :title="t('accessHeader')">
<p>{{ t('accessTip') }}</p> <p>{{ t('accessTip') }}</p>
<n-input v-model:value="auth" type="textarea" :autosize="{ minRows: 3 }" /> <n-input v-model:value="auth" type="password" show-password-on="click" />
<template #action> <template #action>
<n-button :loading="loading" @click="authFunc" type="primary"> <n-button :loading="loading" @click="authFunc" type="primary">
{{ t('ok') }} {{ t('ok') }}

View File

@@ -1,6 +1,7 @@
<script setup> <script setup>
import { defineAsyncComponent } from 'vue' import { defineAsyncComponent, onMounted, watch } from 'vue'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { useRoute } from 'vue-router'
import { useGlobalState } from '../store' import { useGlobalState } from '../store'
import { api } from '../api' import { api } from '../api'
@@ -17,6 +18,7 @@ import About from './common/About.vue';
const SendMail = defineAsyncComponent(() => import('./index/SendMail.vue')); const SendMail = defineAsyncComponent(() => import('./index/SendMail.vue'));
const { settings, openSettings, indexTab, globalTabplacement } = useGlobalState() const { settings, openSettings, indexTab, globalTabplacement } = useGlobalState()
const message = useMessage() const message = useMessage()
const route = useRoute()
const { t } = useI18n({ const { t } = useI18n({
messages: { messages: {
@@ -29,6 +31,8 @@ const { t } = useI18n({
about: 'About', about: 'About',
s3Attachment: 'S3 Attachment', s3Attachment: 'S3 Attachment',
saveToS3Success: 'save to s3 success', saveToS3Success: 'save to s3 success',
webhookSettings: 'Webhook Settings',
query: 'Query',
}, },
zh: { zh: {
mailbox: '收件箱', mailbox: '收件箱',
@@ -39,11 +43,18 @@ const { t } = useI18n({
about: '关于', about: '关于',
s3Attachment: 'S3附件', s3Attachment: 'S3附件',
saveToS3Success: '保存到s3成功', saveToS3Success: '保存到s3成功',
webhookSettings: 'Webhook 设置',
query: '查询',
} }
} }
}); });
const fetchMailData = async (limit, offset) => { const fetchMailData = async (limit, offset) => {
if (mailIdQuery.value > 0) {
const singleMail = await api.fetch(`/api/mail/${mailIdQuery.value}`);
if (singleMail) return { results: [singleMail], count: 1 };
return { results: [], count: 0 };
}
return await api.fetch(`/api/mails?limit=${limit}&offset=${offset}`); return await api.fetch(`/api/mails?limit=${limit}&offset=${offset}`);
}; };
@@ -78,6 +89,30 @@ const saveToS3 = async (mail_id, filename, blob) => {
message.error(error.message || "save to s3 error"); message.error(error.message || "save to s3 error");
} }
} }
const mailBoxKey = ref("")
const mailIdQuery = ref("")
const showMailIdQuery = ref(false)
const queryMail = () => {
mailBoxKey.value = Date.now();
}
watch(route, () => {
if (!route.query.mail_id) {
showMailIdQuery.value = false;
mailIdQuery.value = "";
queryMail();
}
})
onMounted(() => {
if (route.query.mail_id) {
showMailIdQuery.value = true;
mailIdQuery.value = route.query.mail_id;
queryMail();
}
})
</script> </script>
<template> <template>
@@ -85,9 +120,17 @@ const saveToS3 = async (mail_id, filename, blob) => {
<AddressBar /> <AddressBar />
<n-tabs v-if="settings.address" type="card" v-model:value="indexTab" :placement="globalTabplacement"> <n-tabs v-if="settings.address" type="card" v-model:value="indexTab" :placement="globalTabplacement">
<n-tab-pane name="mailbox" :tab="t('mailbox')"> <n-tab-pane name="mailbox" :tab="t('mailbox')">
<MailBox :showEMailTo="false" :showReply="true" :showSaveS3="openSettings.isS3Enabled" :saveToS3="saveToS3" <div v-if="showMailIdQuery" style="margin-bottom: 10px;">
:enableUserDeleteEmail="openSettings.enableUserDeleteEmail" :fetchMailData="fetchMailData" <n-input-group>
:deleteMail="deleteMail" /> <n-input v-model:value="mailIdQuery" />
<n-button @click="queryMail" type="primary" tertiary>
{{ t('query') }}
</n-button>
</n-input-group>
</div>
<MailBox :key="mailBoxKey" :showEMailTo="false" :showReply="true" :showSaveS3="openSettings.isS3Enabled"
:saveToS3="saveToS3" :enableUserDeleteEmail="openSettings.enableUserDeleteEmail"
:fetchMailData="fetchMailData" :deleteMail="deleteMail" />
</n-tab-pane> </n-tab-pane>
<n-tab-pane name="sendbox" :tab="t('sendbox')"> <n-tab-pane name="sendbox" :tab="t('sendbox')">
<SendBox :fetchMailData="fetchSenboxData" :enableUserDeleteEmail="openSettings.enableUserDeleteEmail" <SendBox :fetchMailData="fetchSenboxData" :enableUserDeleteEmail="openSettings.enableUserDeleteEmail"
@@ -102,7 +145,7 @@ const saveToS3 = async (mail_id, filename, blob) => {
<n-tab-pane v-if="openSettings.enableAutoReply" name="auto_reply" :tab="t('auto_reply')"> <n-tab-pane v-if="openSettings.enableAutoReply" name="auto_reply" :tab="t('auto_reply')">
<AutoReply /> <AutoReply />
</n-tab-pane> </n-tab-pane>
<n-tab-pane v-if="openSettings.enableWebhook" name="webhook" :tab="t('webhook')"> <n-tab-pane v-if="openSettings.enableWebhook" name="webhook" :tab="t('webhookSettings')">
<Webhook /> <Webhook />
</n-tab-pane> </n-tab-pane>
<n-tab-pane v-if="openSettings.isS3Enabled" name="s3_attachment" :tab="t('s3Attachment')"> <n-tab-pane v-if="openSettings.isS3Enabled" name="s3_attachment" :tab="t('s3Attachment')">

View File

@@ -9,8 +9,8 @@ import { NButton, NMenu } from 'naive-ui';
import { MenuFilled } from '@vicons/material' import { MenuFilled } from '@vicons/material'
const { const {
adminAuth, showAdminAuth, loading, loading, adminTab,
adminTab, adminMailTabAddress, adminSendBoxTabAddress adminMailTabAddress, adminSendBoxTabAddress
} = useGlobalState() } = useGlobalState()
const message = useMessage() const message = useMessage()
@@ -252,10 +252,6 @@ watch([page, pageSize], async () => {
}) })
onMounted(async () => { onMounted(async () => {
if (!adminAuth.value) {
showAdminAuth.value = true;
return;
}
await fetchData() await fetchData()
}) })
</script> </script>
@@ -290,15 +286,17 @@ onMounted(async () => {
{{ t('query') }} {{ t('query') }}
</n-button> </n-button>
</n-input-group> </n-input-group>
<div style="display: inline-block;"> <div style="overflow: auto;">
<n-pagination v-model:page="page" v-model:page-size="pageSize" :item-count="count" <div style="display: inline-block;">
:page-sizes="[20, 50, 100]" show-size-picker> <n-pagination v-model:page="page" v-model:page-size="pageSize" :item-count="count"
<template #prefix="{ itemCount }"> :page-sizes="[20, 50, 100]" show-size-picker>
{{ t('itemCount') }}: {{ itemCount }} <template #prefix="{ itemCount }">
</template> {{ t('itemCount') }}: {{ itemCount }}
</n-pagination> </template>
</n-pagination>
</div>
<n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
<n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
</template> </template>
@@ -307,4 +305,8 @@ onMounted(async () => {
margin-top: 10px; margin-top: 10px;
margin-bottom: 10px; margin-bottom: 10px;
} }
.n-data-table {
min-width: 1000px;
}
</style> </style>

View File

@@ -11,27 +11,35 @@ const message = useMessage()
const { t } = useI18n({ const { t } = useI18n({
messages: { messages: {
en: { en: {
tip: 'You can manually input the following multiple select input and enter',
save: 'Save', save: 'Save',
successTip: 'Save Success', successTip: 'Save Success',
address_block_list: 'Address Block Keywords for Users(Admin can skip)', address_block_list: 'Address Block Keywords for Users(Admin can skip)',
address_block_list_placeholder: 'Please enter the keywords you want to block', address_block_list_placeholder: 'Please enter the keywords you want to block',
send_address_block_list: 'Address Block Keywords for send email', send_address_block_list: 'Address Block Keywords for send email',
noLimitSendAddressList: 'No Balance Limit Send Address List',
verified_address_list: 'Verified Address List(Can send email by cf internal api)', verified_address_list: 'Verified Address List(Can send email by cf internal api)',
fromBlockList: 'Block Keywords for receive email',
}, },
zh: { zh: {
tip: '您可以手动输入以下多选输入框, 回车增加',
save: '保存', save: '保存',
successTip: '保存成功', successTip: '保存成功',
address_block_list: '邮件地址屏蔽关键词(管理员可跳过检查)', address_block_list: '邮件地址屏蔽关键词(管理员可跳过检查)',
address_block_list_placeholder: '请输入您想要屏蔽的关键词', address_block_list_placeholder: '请输入您想要屏蔽的关键词',
send_address_block_list: '发送邮件地址屏蔽关键词', send_address_block_list: '发送邮件地址屏蔽关键词',
noLimitSendAddressList: '无余额限制发送地址列表',
verified_address_list: '已验证地址列表(可通过 cf 内部 api 发送邮件)', verified_address_list: '已验证地址列表(可通过 cf 内部 api 发送邮件)',
fromBlockList: '接收邮件地址屏蔽关键词',
} }
} }
}); });
const addressBlockList = ref([]) const addressBlockList = ref([])
const sendAddressBlockList = ref([]) const sendAddressBlockList = ref([])
const noLimitSendAddressList = ref([])
const verifiedAddressList = ref([]) const verifiedAddressList = ref([])
const fromBlockList = ref([])
const fetchData = async () => { const fetchData = async () => {
try { try {
@@ -39,6 +47,8 @@ const fetchData = async () => {
addressBlockList.value = res.blockList || [] addressBlockList.value = res.blockList || []
sendAddressBlockList.value = res.sendBlockList || [] sendAddressBlockList.value = res.sendBlockList || []
verifiedAddressList.value = res.verifiedAddressList || [] verifiedAddressList.value = res.verifiedAddressList || []
fromBlockList.value = res.fromBlockList || []
noLimitSendAddressList.value = res.noLimitSendAddressList || []
} catch (error) { } catch (error) {
message.error(error.message || "error"); message.error(error.message || "error");
} }
@@ -51,7 +61,9 @@ const save = async () => {
body: JSON.stringify({ body: JSON.stringify({
blockList: addressBlockList.value || [], blockList: addressBlockList.value || [],
sendBlockList: sendAddressBlockList.value || [], sendBlockList: sendAddressBlockList.value || [],
verifiedAddressList: verifiedAddressList.value || [] verifiedAddressList: verifiedAddressList.value || [],
fromBlockList: fromBlockList.value || [],
noLimitSendAddressList: noLimitSendAddressList.value || [],
}) })
}) })
message.success(t('successTip')) message.success(t('successTip'))
@@ -69,6 +81,9 @@ onMounted(async () => {
<template> <template>
<div class="center"> <div class="center">
<n-card :bordered="false" embedded style="max-width: 600px;"> <n-card :bordered="false" embedded style="max-width: 600px;">
<n-alert :show-icon="false" type="warning" style="margin-bottom: 10px;">
{{ t("tip") }}
</n-alert>
<n-form-item-row :label="t('address_block_list')"> <n-form-item-row :label="t('address_block_list')">
<n-select v-model:value="addressBlockList" filterable multiple tag <n-select v-model:value="addressBlockList" filterable multiple tag
:placeholder="t('address_block_list_placeholder')" /> :placeholder="t('address_block_list_placeholder')" />
@@ -77,10 +92,17 @@ onMounted(async () => {
<n-select v-model:value="sendAddressBlockList" filterable multiple tag <n-select v-model:value="sendAddressBlockList" filterable multiple tag
:placeholder="t('address_block_list_placeholder')" /> :placeholder="t('address_block_list_placeholder')" />
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('noLimitSendAddressList')">
<n-select v-model:value="noLimitSendAddressList" filterable multiple tag
:placeholder="t('noLimitSendAddressList')" />
</n-form-item-row>
<n-form-item-row :label="t('verified_address_list')"> <n-form-item-row :label="t('verified_address_list')">
<n-select v-model:value="verifiedAddressList" filterable multiple tag <n-select v-model:value="verifiedAddressList" filterable multiple tag
:placeholder="t('verified_address_list')" /> :placeholder="t('verified_address_list')" />
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('fromBlockList')">
<n-select v-model:value="fromBlockList" filterable multiple tag :placeholder="t('fromBlockList')" />
</n-form-item-row>
<n-button @click="save" type="primary" block :loading="loading"> <n-button @click="save" type="primary" block :loading="loading">
{{ t('save') }} {{ t('save') }}
</n-button> </n-button>

View File

@@ -0,0 +1,30 @@
<script setup lang="ts">
// @ts-ignore
import { api } from '../../api'
// @ts-ignore
import WebhookComponent from '../../components/WebhookComponent.vue'
const fetchData = async () => {
return await api.fetch(`/admin/mail_webhook/settings`)
}
const saveSettings = async (webhookSettings: any) => {
await api.fetch(`/admin/mail_webhook/settings`, {
method: 'POST',
body: JSON.stringify(webhookSettings),
})
}
const testSettings = async (webhookSettings: any) => {
await api.fetch(`/admin/mail_webhook/test`, {
method: 'POST',
body: JSON.stringify(webhookSettings),
})
}
</script>
<template>
<WebhookComponent :fetchData="fetchData" :saveSettings="saveSettings" :testSettings="testSettings" />
</template>

View File

@@ -6,10 +6,7 @@ import { useGlobalState } from '../../store'
import { api } from '../../api' import { api } from '../../api'
import MailBox from '../../components/MailBox.vue'; import MailBox from '../../components/MailBox.vue';
const { const { adminMailTabAddress } = useGlobalState()
adminAuth, showAdminAuth,
adminMailTabAddress
} = useGlobalState()
const { t } = useI18n({ const { t } = useI18n({
messages: { messages: {
@@ -48,13 +45,6 @@ const fetchMailData = async (limit, offset) => {
const deleteMail = async (curMailId) => { const deleteMail = async (curMailId) => {
await api.fetch(`/admin/mails/${curMailId}`, { method: 'DELETE' }); await api.fetch(`/admin/mails/${curMailId}`, { method: 'DELETE' });
}; };
onMounted(async () => {
if (!adminAuth.value) {
showAdminAuth.value = true;
return;
}
})
</script> </script>
<template> <template>

View File

@@ -1,12 +1,7 @@
<script setup> <script setup>
import { onMounted } from 'vue';
import { useGlobalState } from '../../store'
import { api } from '../../api' import { api } from '../../api'
import MailBox from '../../components/MailBox.vue'; import MailBox from '../../components/MailBox.vue';
const { adminAuth, showAdminAuth } = useGlobalState()
const fetchMailUnknowData = async (limit, offset) => { const fetchMailUnknowData = async (limit, offset) => {
return await api.fetch( return await api.fetch(
`/admin/mails_unknow` `/admin/mails_unknow`
@@ -16,19 +11,12 @@ const fetchMailUnknowData = async (limit, offset) => {
} }
const deleteMail = async (curMailId) => { const deleteMail = async (curMailId) => {
await api.fetch(`/api/mails/${curMailId}`, { method: 'DELETE' }); await api.fetch(`/admin/mails/${curMailId}`, { method: 'DELETE' });
}; };
onMounted(async () => {
if (!adminAuth.value) {
showAdminAuth.value = true;
return;
}
})
</script> </script>
<template> <template>
<div v-if="adminAuth" style="margin-top: 10px;"> <div style="margin-top: 10px;">
<MailBox :enableUserDeleteEmail="true" :fetchMailData="fetchMailUnknowData" :deleteMail="deleteMail" /> <MailBox :enableUserDeleteEmail="true" :fetchMailData="fetchMailUnknowData" :deleteMail="deleteMail" />
</div> </div>
</template> </template>

View File

@@ -1,12 +1,10 @@
<script setup> <script setup>
import { ref, h, onMounted, watch } from 'vue'; import { ref, onMounted } from 'vue';
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { CleaningServicesFilled } from '@vicons/material' import { CleaningServicesFilled } from '@vicons/material'
import { useGlobalState } from '../../store'
import { api } from '../../api' import { api } from '../../api'
const { adminAuth, showAdminAuth } = useGlobalState()
const message = useMessage() const message = useMessage()
const cleanupModel = ref({ const cleanupModel = ref({
enableMailsAutoCleanup: false, enableMailsAutoCleanup: false,
@@ -22,10 +20,10 @@ const cleanupModel = ref({
const { t } = useI18n({ const { t } = useI18n({
messages: { messages: {
en: { en: {
tip: 'Please input the cleanup days', tip: 'Please input the days',
mailBoxLabel: 'Clean up days for mailbox', mailBoxLabel: 'Cleanup the inbox before n days',
mailUnknowLabel: "Clean up days for unknow receiver", mailUnknowLabel: "Cleanup the unknow mail before n days",
sendBoxLabel: "Clean up days for sendbox", sendBoxLabel: "Cleanup the sendbox before n days",
cleanupNow: "Cleanup now", cleanupNow: "Cleanup now",
autoCleanup: "Auto cleanup", autoCleanup: "Auto cleanup",
cleanupSuccess: "Cleanup success", cleanupSuccess: "Cleanup success",
@@ -33,10 +31,10 @@ const { t } = useI18n({
cronTip: "Enable cron cleanup, need to configure [crons] in worker, please refer to the document", cronTip: "Enable cron cleanup, need to configure [crons] in worker, please refer to the document",
}, },
zh: { zh: {
tip: '请输入清理天数', tip: '请输入天数',
mailBoxLabel: '收件箱清理天数', mailBoxLabel: '清理 n 天前的收件箱',
mailUnknowLabel: "无收件人邮件清理天数", mailUnknowLabel: "清理 n 天前的无收件人邮件",
sendBoxLabel: "发件箱清理天数", sendBoxLabel: "清理 n 天前的发件箱",
autoCleanup: "自动清理", autoCleanup: "自动清理",
cleanupSuccess: "清理成功", cleanupSuccess: "清理成功",
cleanupNow: "立即清理", cleanupNow: "立即清理",
@@ -80,10 +78,6 @@ const save = async () => {
} }
onMounted(async () => { onMounted(async () => {
if (!adminAuth.value) {
showAdminAuth.value = true;
return;
}
await fetchData(); await fetchData();
}) })
</script> </script>

View File

@@ -0,0 +1,199 @@
<script setup>
import '@wangeditor/editor/dist/css/style.css'
import { Editor, Toolbar } from '@wangeditor/editor-for-vue'
import { useI18n } from 'vue-i18n'
import { onBeforeUnmount, ref, shallowRef } from 'vue'
import { useSessionStorage } from '@vueuse/core'
import { api } from '../../api'
const message = useMessage()
const isPreview = ref(false)
const editorRef = shallowRef()
const sendMailModel = useSessionStorage('sendMailByAdminModel', {
fromName: "",
fromMail: "",
toName: "",
toMail: "",
subject: "",
contentType: 'text',
content: "",
});
const { t } = useI18n({
locale: 'zh',
messages: {
en: {
successSend: 'Please check your sendbox. If failed, please try again later.',
fromName: 'Your Name and Address, leave Name blank to use email address',
toName: 'Recipient Name and Address, leave Name blank to use email address',
subject: 'Subject',
options: 'Options',
edit: 'Edit',
preview: 'Preview',
content: 'Content',
send: 'Send',
text: 'Text',
html: 'HTML',
'rich text': 'Rich Text',
tooLarge: 'Too large file, please upload file less than 1MB.',
},
zh: {
successSend: '请查看您的发件箱, 如果失败, 请检查稍后重试。',
fromName: '你的名称和地址,名称不填写则使用邮箱地址',
toName: '收件人名称和地址,名称不填写则使用邮箱地址',
subject: '主题',
options: '选项',
edit: '编辑',
preview: '预览',
content: '内容',
send: '发送',
text: '文本',
html: 'HTML',
'rich text': '富文本',
tooLarge: '文件过大, 请上传小于1MB的文件。',
}
}
});
const contentTypes = [
{ label: t('text'), value: 'text' },
{ label: t('html'), value: 'html' },
{ label: t('rich text'), value: 'rich' },
]
const send = async () => {
try {
await api.fetch(`/admin/send_mail`,
{
method: 'POST',
body:
JSON.stringify({
from_name: sendMailModel.value.fromName,
from_mail: sendMailModel.value.fromMail,
to_name: sendMailModel.value.toName,
to_mail: sendMailModel.value.toMail,
subject: sendMailModel.value.subject,
is_html: sendMailModel.value.contentType != 'text',
content: sendMailModel.value.content,
})
})
sendMailModel.value = {
fromName: "",
fromMail: "",
toName: "",
toMail: "",
subject: "",
contentType: 'text',
content: "",
}
} catch (error) {
message.error(error.message || "error");
} finally {
message.success(t("successSend"));
}
}
const toolbarConfig = {
excludeKeys: ["uploadVideo"]
}
const editorConfig = {
MENU_CONF: {
'uploadImage': {
async customUpload() {
message.error(t('tooLarge'))
},
maxFileSize: 1 * 1024 * 1024,
base64LimitSize: 1 * 1024 * 1024,
}
}
}
onBeforeUnmount(() => {
const editor = editorRef.value
if (editor == null) return
editor.destroy()
})
const handleCreated = (editor) => {
editorRef.value = editor;
}
</script>
<template>
<div class="center">
<n-card :bordered="false" embedded>
<n-flex justify="end">
<n-button type="primary" @click="send">{{ t('send') }}</n-button>
</n-flex>
<div class="left">
<n-form :model="sendMailModel">
<n-form-item :label="t('fromName')" label-placement="top">
<n-input-group>
<n-input v-model:value="sendMailModel.fromName" />
<n-input v-model:value="sendMailModel.fromMail" />
</n-input-group>
</n-form-item>
<n-form-item :label="t('toName')" label-placement="top">
<n-input-group>
<n-input v-model:value="sendMailModel.toName" />
<n-input v-model:value="sendMailModel.toMail" />
</n-input-group>
</n-form-item>
<n-form-item :label="t('subject')" label-placement="top">
<n-input v-model:value="sendMailModel.subject" />
</n-form-item>
<n-form-item :label="t('options')" label-placement="top">
<n-radio-group v-model:value="sendMailModel.contentType">
<n-radio-button v-for="option in contentTypes" :key="option.value" :value="option.value"
:label="option.label" />
</n-radio-group>
<n-button v-if="sendMailModel.contentType != 'text'" @click="isPreview = !isPreview"
style="margin-left: 10px;">
{{ isPreview ? t('edit') : t('preview') }}
</n-button>
</n-form-item>
<n-form-item :label="t('content')" label-placement="top">
<n-card :bordered="false" embedded v-if="isPreview">
<div v-html="sendMailModel.content" />
</n-card>
<div v-else-if="sendMailModel.contentType == 'rich'" style="border: 1px solid #ccc">
<Toolbar style="border-bottom: 1px solid #ccc" :defaultConfig="toolbarConfig"
:editor="editorRef" mode="default" />
<Editor style="height: 500px; overflow-y: hidden;" v-model="sendMailModel.content"
:defaultConfig="editorConfig" mode="default" @onCreated="handleCreated" />
</div>
<n-input v-else type="textarea" v-model:value="sendMailModel.content" :autosize="{
minRows: 3
}" />
</n-form-item>
</n-form>
</div>
</n-card>
</div>
</template>
<style scoped>
.n-card {
max-width: 800px;
}
.n-button {
text-align: left;
margin-right: 10px;
}
.center {
display: flex;
text-align: center;
place-items: center;
justify-content: center;
}
.left {
text-align: left;
place-items: left;
justify-content: left;
}
</style>

View File

@@ -198,15 +198,17 @@ onMounted(async () => {
{{ t('query') }} {{ t('query') }}
</n-button> </n-button>
</n-input-group> </n-input-group>
<div style="display: inline-block;"> <div style="overflow: auto;">
<n-pagination v-model:page="page" v-model:page-size="pageSize" :item-count="count" :page-sizes="[20, 50, 100]" <div style="display: inline-block;">
show-size-picker> <n-pagination v-model:page="page" v-model:page-size="pageSize" :item-count="count" :page-sizes="[20, 50, 100]"
<template #prefix="{ itemCount }"> show-size-picker>
{{ t('itemCount') }}: {{ itemCount }} <template #prefix="{ itemCount }">
</template> {{ t('itemCount') }}: {{ itemCount }}
</n-pagination> </template>
</n-pagination>
</div>
<n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
<n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
</template> </template>
@@ -215,4 +217,8 @@ onMounted(async () => {
margin-top: 10px; margin-top: 10px;
margin-bottom: 10px; margin-bottom: 10px;
} }
.n-data-table {
min-width: 700px;
}
</style> </style>

View File

@@ -4,10 +4,8 @@ import { useI18n } from 'vue-i18n'
import { User, UserCheck, MailBulk } from '@vicons/fa' import { User, UserCheck, MailBulk } from '@vicons/fa'
import { SendOutlined } from '@vicons/material' import { SendOutlined } from '@vicons/material'
import { useGlobalState } from '../../store'
import { api } from '../../api' import { api } from '../../api'
const { adminAuth } = useGlobalState()
const message = useMessage() const message = useMessage()
const { t } = useI18n({ const { t } = useI18n({
@@ -60,9 +58,6 @@ const fetchStatistics = async () => {
} }
onMounted(async () => { onMounted(async () => {
if (!adminAuth.value) {
return;
}
await fetchStatistics() await fetchStatistics()
}) })
</script> </script>

View File

@@ -17,7 +17,7 @@ const { t } = useI18n({
status: 'Check Status', status: 'Check Status',
enableTelegramAllowList: 'Enable Telegram Allow List(Manually input user ID)', enableTelegramAllowList: 'Enable Telegram Allow List(Manually input user ID)',
enable: 'Enable', enable: 'Enable',
telegramAllowList: 'Telegram Allow List', telegramAllowList: 'Telegram Allow List(Manually input telegram user ID)',
save: 'Save', save: 'Save',
miniAppUrl: 'Telegram Mini App URL', miniAppUrl: 'Telegram Mini App URL',
enableGlobalMailPush: 'Enable Global Mail Push(Manually input telegram user ID)', enableGlobalMailPush: 'Enable Global Mail Push(Manually input telegram user ID)',
@@ -27,12 +27,12 @@ const { t } = useI18n({
init: '初始化', init: '初始化',
successTip: '成功', successTip: '成功',
status: '查看状态', status: '查看状态',
enableTelegramAllowList: '启用 Telegram 白名单(手动输入用户 ID)', enableTelegramAllowList: '启用 Telegram 白名单(手动输入用户 ID, 回车增加)',
enable: '启用', enable: '启用',
telegramAllowList: 'Telegram 白名单', telegramAllowList: 'Telegram 白名单(手动输入用户 ID, 回车增加)',
save: '保存', save: '保存',
miniAppUrl: '电报小程序 URL(请输入你部署的电报小程序网页地址)', miniAppUrl: '电报小程序 URL(请输入你部署的电报小程序网页地址)',
enableGlobalMailPush: '启用全局邮件推送(手动输入邮箱管理员的 telegram 用户 ID)', enableGlobalMailPush: '启用全局邮件推送(手动输入邮箱管理员的 telegram 用户 ID, 回车增加)',
globalMailPushList: '全局邮件推送用户列表', globalMailPushList: '全局邮件推送用户列表',
} }
} }

View File

@@ -368,21 +368,23 @@ onMounted(async () => {
{{ t('query') }} {{ t('query') }}
</n-button> </n-button>
</n-input-group> </n-input-group>
<div style="display: inline-block;"> <div style="overflow: auto;">
<n-pagination v-model:page="page" v-model:page-size="pageSize" :item-count="count" <div style="display: inline-block;">
:page-sizes="[20, 50, 100]" show-size-picker> <n-pagination v-model:page="page" v-model:page-size="pageSize" :item-count="count"
<template #prefix="{ itemCount }"> :page-sizes="[20, 50, 100]" show-size-picker>
{{ t('itemCount') }}: {{ itemCount }} <template #prefix="{ itemCount }">
</template> {{ t('itemCount') }}: {{ itemCount }}
<template #suffix> </template>
<n-button @click="showCreateUser = true" size="small" tertiary type="primary" <template #suffix>
style="margin-left: 10px"> <n-button @click="showCreateUser = true" size="small" tertiary type="primary"
{{ t('createUser') }} style="margin-left: 10px">
</n-button> {{ t('createUser') }}
</template> </n-button>
</n-pagination> </template>
</n-pagination>
</div>
<n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
<n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
</template> </template>
@@ -391,4 +393,8 @@ onMounted(async () => {
margin-top: 10px; margin-top: 10px;
margin-bottom: 10px; margin-bottom: 10px;
} }
.n-data-table {
min-width: 800px;
}
</style> </style>

View File

@@ -0,0 +1,266 @@
<script setup lang="ts">
import { onMounted, ref } from 'vue';
import { useI18n } from 'vue-i18n'
// @ts-ignore
import { useGlobalState } from '../../store'
// @ts-ignore
import { api } from '../../api'
import constant from '../../constant'
import { UserOauth2Settings } from '../../models';
const { loading } = useGlobalState()
// @ts-ignore
const message = useMessage()
const { t } = useI18n({
messages: {
en: {
save: 'Save',
delete: 'Delete',
successTip: 'Save Success',
enable: 'Enable',
enableMailAllowList: 'Enable Mail Address Allow List(Manually enterable)',
mailAllowList: 'Mail Address Allow List',
addOauth2: 'Add Oauth2',
name: 'Name',
oauth2Type: 'Oauth2 Type',
tip: 'Third-party login will automatically use the user\'s email to register an account (the same email will be regarded as the same account), this account is the same as the registered account, and you can also set the password through the forget password',
},
zh: {
save: '保存',
delete: '删除',
successTip: '保存成功',
enable: '启用',
enableMailAllowList: '启用邮件地址白名单(可手动输入, 回车增加)',
mailAllowList: '邮件地址白名单',
addOauth2: '添加 Oauth2',
name: '名称',
oauth2Type: 'Oauth2 类型',
tip: '第三方登录会自动使用用户邮箱注册账号(邮箱相同将视为同一账号), 此账号和注册的账号相同, 也可以通过忘记密码设置密码',
}
}
});
const mailAllowOptions = constant.COMMOM_MAIL.map((item) => {
return { label: item, value: item }
})
const userOauth2Settings = ref([] as UserOauth2Settings[])
const showAddOauth2 = ref(false)
const newOauth2Name = ref('')
const newOauth2Type = ref('custom')
const fetchData = async () => {
try {
const res = await api.fetch(`/admin/user_oauth2_settings`)
Object.assign(userOauth2Settings.value, res)
} catch (error) {
message.error((error as Error).message || "error");
}
}
const save = async () => {
try {
await api.fetch(`/admin/user_oauth2_settings`, {
method: 'POST',
body: JSON.stringify(userOauth2Settings.value)
})
message.success(t('successTip'))
} catch (error) {
message.error((error as Error).message || "error");
}
}
const addNewOauth2 = () => {
const authorizationURL = () => {
switch (newOauth2Type.value) {
case 'github':
return 'https://github.com/login/oauth/authorize'
case 'authentik':
return 'https://youdomain/application/o/authorize/'
default:
return ''
}
}
const accessTokenURL = () => {
switch (newOauth2Type.value) {
case 'github':
return 'https://github.com/login/oauth/access_token'
case 'authentik':
return 'https://youdomain/application/o/token/'
default:
return ''
}
}
const accessTokenFormat = () => {
switch (newOauth2Type.value) {
case 'github':
return 'json'
case 'authentik':
return 'urlencoded'
default:
return ''
}
}
const userInfoURL = () => {
switch (newOauth2Type.value) {
case 'github':
return 'https://api.github.com/user'
case 'authentik':
return 'https://youdomain/application/o/userinfo/'
default:
return ''
}
}
const userEmailKey = () => {
switch (newOauth2Type.value) {
case 'github':
return 'email'
case 'authentik':
return 'email'
default:
return ''
}
}
const scope = () => {
switch (newOauth2Type.value) {
case 'github':
return 'user:email'
case 'authentik':
return 'email openid'
default:
return ''
}
}
userOauth2Settings.value.push({
name: newOauth2Name.value,
clientID: '',
clientSecret: '',
authorizationURL: authorizationURL(),
accessTokenURL: accessTokenURL(),
accessTokenFormat: accessTokenFormat(),
userInfoURL: userInfoURL(),
userEmailKey: userEmailKey(),
redirectURL: `${window.location.origin}/user/oauth2/callback`,
logoutURL: '',
scope: scope(),
enableMailAllowList: false,
mailAllowList: constant.COMMOM_MAIL
} as UserOauth2Settings)
newOauth2Name.value = ''
showAddOauth2.value = false
}
const accessTokenFormatOptions = [
{ label: 'json', value: 'json' },
{ label: 'urlencoded', value: 'urlencoded' },
]
onMounted(async () => {
await fetchData();
})
</script>
<template>
<div class="center">
<n-modal v-model:show="showAddOauth2" preset="dialog" :title="t('addOauth2')">
<n-form>
<n-form-item-row :label="t('name')" required>
<n-input v-model:value="newOauth2Name" />
</n-form-item-row>
<n-form-item-row :label="t('oauth2Type')" required>
<n-radio-group v-model:value="newOauth2Type">
<n-radio-button value="github" label="Github" />
<n-radio-button value="authentik" label="Authentik" />
<n-radio-button value="custom" label="Custom" />
</n-radio-group>
</n-form-item-row>
</n-form>
<template #action>
<n-button :loading="loading" @click="addNewOauth2" size="small" tertiary type="primary">
{{ t('addOauth2') }}
</n-button>
</template>
</n-modal>
<n-card :bordered="false" embedded style="max-width: 600px;">
<n-alert :show-icon="false" type="warning" closable style="margin-bottom: 10px;">
{{ t("tip") }}
</n-alert>
<n-flex justify="end">
<n-button @click="showAddOauth2 = true" secondary :loading="loading">
{{ t('addOauth2') }}
</n-button>
<n-button @click="save" type="primary" :loading="loading">
{{ t('save') }}
</n-button>
</n-flex>
<n-divider />
<n-collapse default-expanded-names="1" accordion :trigger-areas="['main', 'arrow']">
<n-collapse-item v-for="(item, index) in userOauth2Settings" :key="index" :title="item.name">
<template #header-extra>
<n-popconfirm @positive-click="userOauth2Settings.splice(index, 1)">
<template #trigger>
<n-button tertiary type="error">
{{ t('delete') }}
</n-button>
</template>
{{ t('delete') }}
</n-popconfirm>
</template>
<n-form :model="item">
<n-form-item-row :label="t('name')" required>
<n-input v-model:value="item.name" />
</n-form-item-row>
<n-form-item-row label="Client ID" required>
<n-input v-model:value="item.clientID" />
</n-form-item-row>
<n-form-item-row label="Client Secret" required>
<n-input v-model:value="item.clientSecret" type="password" show-password-on="click" />
</n-form-item-row>
<n-form-item-row label="Authorization URL" required>
<n-input v-model:value="item.authorizationURL" />
</n-form-item-row>
<n-form-item-row label="Access Token URL" required>
<n-input v-model:value="item.accessTokenURL" />
</n-form-item-row>
<n-form-item-row label="Access Token accessTokenFormat" required>
<n-select v-model:value="item.accessTokenFormat" :options="accessTokenFormatOptions" />
</n-form-item-row>
<n-form-item-row label="User Info URL" required>
<n-input v-model:value="item.userInfoURL" />
</n-form-item-row>
<n-form-item-row label="User Email Key" required>
<n-input v-model:value="item.userEmailKey" />
</n-form-item-row>
<n-form-item-row label="Redirect URL" required>
<n-input v-model:value="item.redirectURL" />
</n-form-item-row>
<n-form-item-row label="Scope" required>
<n-input v-model:value="item.scope" />
</n-form-item-row>
<n-form-item-row :label="t('enableMailAllowList')">
<n-input-group>
<n-checkbox v-model:checked="item.enableMailAllowList" style="width: 20%;">
{{ t('enable') }}
</n-checkbox>
<n-select v-model:value="item.mailAllowList" v-if="item.enableMailAllowList" filterable
multiple tag style="width: 80%;" :options="mailAllowOptions"
:placeholder="t('mailAllowList')" />
</n-input-group>
</n-form-item-row>
</n-form>
</n-collapse-item>
</n-collapse>
</n-card>
</div>
</template>
<style scoped>
.center {
display: flex;
text-align: left;
place-items: center;
justify-content: center;
}
</style>

View File

@@ -28,7 +28,7 @@ const { t } = useI18n({
enableUserRegister: "允许用户注册", enableUserRegister: "允许用户注册",
enableMailVerify: '启用邮件验证(发送地址必须是系统中能有余额且能正常发送邮件的地址)', enableMailVerify: '启用邮件验证(发送地址必须是系统中能有余额且能正常发送邮件的地址)',
verifyMailSender: '验证邮件发送地址', verifyMailSender: '验证邮件发送地址',
enableMailAllowList: '启用邮件地址白名单(可手动输入)', enableMailAllowList: '启用邮件地址白名单(可手动输入, 回车增加)',
mailAllowList: '邮件地址白名单', mailAllowList: '邮件地址白名单',
maxAddressCount: '可绑定最大邮箱地址数量', maxAddressCount: '可绑定最大邮箱地址数量',
} }
@@ -92,8 +92,8 @@ onMounted(async () => {
<n-checkbox v-model:checked="userSettings.enableMailVerify" style="width: 20%;"> <n-checkbox v-model:checked="userSettings.enableMailVerify" style="width: 20%;">
{{ t('enable') }} {{ t('enable') }}
</n-checkbox> </n-checkbox>
<n-input v-model:value="userSettings.verifyMailSender" style="width: 80%;" <n-input v-model:value="userSettings.verifyMailSender" v-if="userSettings.enableMailVerify"
:placeholder="t('verifyMailSender')" /> style="width: 80%;" :placeholder="t('verifyMailSender')" />
</n-input-group> </n-input-group>
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('enableMailAllowList')"> <n-form-item-row :label="t('enableMailAllowList')">
@@ -101,8 +101,9 @@ onMounted(async () => {
<n-checkbox v-model:checked="userSettings.enableMailAllowList" style="width: 20%;"> <n-checkbox v-model:checked="userSettings.enableMailAllowList" style="width: 20%;">
{{ t('enable') }} {{ t('enable') }}
</n-checkbox> </n-checkbox>
<n-select v-model:value="userSettings.mailAllowList" filterable multiple tag style="width: 80%;" <n-select v-model:value="userSettings.mailAllowList" v-if="userSettings.enableMailAllowList"
:options="mailAllowOptions" :placeholder="t('mailAllowList')" /> filterable multiple tag style="width: 80%;" :options="mailAllowOptions"
:placeholder="t('mailAllowList')" />
</n-input-group> </n-input-group>
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('maxAddressCount')"> <n-form-item-row :label="t('maxAddressCount')">

View File

@@ -13,13 +13,15 @@ const { t } = useI18n({
messages: { messages: {
en: { en: {
successTip: 'Success', successTip: 'Success',
webhookAllowList: 'Webhook Allow List(Enter the address that is allowed to use webhook)', webhookAllowList: 'Webhook Allow List(Enter the address that is allowed to use webhook and enter)',
save: 'Save', save: 'Save',
notEnabled: 'Webhook is not enabled',
}, },
zh: { zh: {
successTip: '成功', successTip: '成功',
webhookAllowList: 'Webhook 白名单(请输入允许使用webhook 的地址)', webhookAllowList: 'Webhook 白名单(请输入允许使用webhook 的地址, 回车增加)',
save: '保存', save: '保存',
notEnabled: 'Webhook 未开启',
} }
} }
}); });
@@ -33,13 +35,16 @@ class WebhookSettings {
} }
const webhookSettings = ref(new WebhookSettings([])) const webhookSettings = ref(new WebhookSettings([]))
const webhookEnabled = ref(false)
const errorInfo = ref('')
const getSettings = async () => { const getSettings = async () => {
try { try {
const res = await api.fetch(`/admin/webhook/settings`) const res = await api.fetch(`/admin/webhook/settings`)
Object.assign(webhookSettings.value, res) Object.assign(webhookSettings.value, res)
webhookEnabled.value = true
} catch (error) { } catch (error) {
message.error((error as Error).message || "error"); errorInfo.value = (error as Error).message || "error";
} }
} }
@@ -62,7 +67,7 @@ onMounted(async () => {
<template> <template>
<div class="center"> <div class="center">
<n-card :bordered="false" embedded style="max-width: 800px; overflow: auto;"> <n-card v-if="webhookEnabled" :bordered="false" embedded style="max-width: 800px; overflow: auto;">
<n-form-item-row :label="t('webhookAllowList')"> <n-form-item-row :label="t('webhookAllowList')">
<n-select v-model:value="webhookSettings.allowList" filterable multiple tag <n-select v-model:value="webhookSettings.allowList" filterable multiple tag
:placeholder="t('webhookAllowList')" /> :placeholder="t('webhookAllowList')" />
@@ -71,6 +76,7 @@ onMounted(async () => {
{{ t('save') }} {{ t('save') }}
</n-button> </n-button>
</n-card> </n-card>
<n-result v-else status="404" :title="t('notEnabled')" :description="errorInfo" />
</div> </div>
</template> </template>

View File

@@ -0,0 +1,42 @@
<script setup>
import { onMounted, ref } from 'vue';
import { useI18n } from 'vue-i18n'
import { useGlobalState } from '../../store'
import { api } from '../../api'
const { loading } = useGlobalState()
const message = useMessage()
const settings = ref({})
const fetchData = async () => {
try {
const res = await api.fetch(`/admin/worker/configs`)
Object.assign(settings.value, res)
} catch (error) {
message.error(error.message || "error");
}
}
onMounted(async () => {
await fetchData();
})
</script>
<template>
<div class="center">
<n-card :bordered="false" embedded style="max-width: 600px; overflow: auto;">
<pre>{{ JSON.stringify(settings, null, 2) }}</pre>
</n-card>
</div>
</template>
<style scoped>
.center {
display: flex;
text-align: left;
place-items: center;
justify-content: center;
}
</style>

View File

@@ -1,10 +1,13 @@
<script setup> <script setup>
import { GithubAlt, Discord, Telegram } from '@vicons/fa' import { GithubAlt, Discord, Telegram } from '@vicons/fa'
import { useGlobalState } from '../../store'
const { announcement } = useGlobalState()
</script> </script>
<template> <template>
<div class="center"> <div class="center">
<n-card :bordered="false" embedded> <n-card :bordered="false" embedded>
<div v-html="announcement"></div>
<n-button tag="a" target="_blank" href="https://github.com/dreamhunter2333/cloudflare_temp_email"> <n-button tag="a" target="_blank" href="https://github.com/dreamhunter2333/cloudflare_temp_email">
<template #icon> <template #icon>
<n-icon :component="GithubAlt" /> <n-icon :component="GithubAlt" />

View File

@@ -5,8 +5,8 @@ import { useIsMobile } from '../../utils/composables'
import { useGlobalState } from '../../store' import { useGlobalState } from '../../store'
const { const {
mailboxSplitSize, useIframeShowMail, preferShowTextMail, mailboxSplitSize, useIframeShowMail, preferShowTextMail, configAutoRefreshInterval,
globalTabplacement, useSideMargin globalTabplacement, useSideMargin, useUTCDate
} = useGlobalState() } = useGlobalState()
const isMobile = useIsMobile() const isMobile = useIsMobile()
@@ -22,6 +22,8 @@ const { t } = useI18n({
top: 'top', top: 'top',
right: 'right', right: 'right',
bottom: 'bottom', bottom: 'bottom',
useUTCDate: 'Use UTC Date',
autoRefreshInterval: 'Auto Refresh Interval(Sec)',
}, },
zh: { zh: {
mailboxSplitSize: '邮箱界面分栏大小', mailboxSplitSize: '邮箱界面分栏大小',
@@ -33,6 +35,8 @@ const { t } = useI18n({
top: '顶部', top: '顶部',
right: '右侧', right: '右侧',
bottom: '底部', bottom: '底部',
useUTCDate: '使用 UTC 时间',
autoRefreshInterval: '自动刷新间隔(秒)',
} }
} }
}); });
@@ -48,12 +52,20 @@ const { t } = useI18n({
0.75: '0.75' 0.75: '0.75'
}" /> }" />
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('autoRefreshInterval')">
<n-slider v-model:value="configAutoRefreshInterval" :min="30" :max="300" :step="1" :marks="{
60: '60', 120: '120', 180: '180', 240: '240'
}" />
</n-form-item-row>
<n-form-item-row :label="t('preferShowTextMail')"> <n-form-item-row :label="t('preferShowTextMail')">
<n-switch v-model:value="preferShowTextMail" :round="false" /> <n-switch v-model:value="preferShowTextMail" :round="false" />
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('useIframeShowMail')"> <n-form-item-row :label="t('useIframeShowMail')">
<n-switch v-model:value="useIframeShowMail" :round="false" /> <n-switch v-model:value="useIframeShowMail" :round="false" />
</n-form-item-row> </n-form-item-row>
<n-form-item-row :label="t('useUTCDate')">
<n-switch v-model:value="useUTCDate" :round="false" />
</n-form-item-row>
<n-form-item-row v-if="!isMobile" :label="t('useSideMargin')"> <n-form-item-row v-if="!isMobile" :label="t('useSideMargin')">
<n-switch v-model:value="useSideMargin" :round="false" /> <n-switch v-model:value="useSideMargin" :round="false" />
</n-form-item-row> </n-form-item-row>

View File

@@ -15,7 +15,7 @@ const props = defineProps({
bindUserAddress: { bindUserAddress: {
type: Function, type: Function,
default: async () => { await api.bindUserAddress(); }, default: async () => { await api.bindUserAddress(); },
requried: true required: true
}, },
newAddressPath: { newAddressPath: {
type: Function, type: Function,
@@ -29,11 +29,12 @@ const props = defineProps({
}), }),
}); });
}, },
requried: true required: true
}, },
}) })
const message = useMessage() const message = useMessage()
const notification = useNotification()
const router = useRouter() const router = useRouter()
const { const {
@@ -72,7 +73,7 @@ const { locale, t } = useI18n({
login: 'Login', login: 'Login',
pleaseGetNewEmail: 'Please login or click "Get New Email" button to get a new email address', pleaseGetNewEmail: 'Please login or click "Get New Email" button to get a new email address',
getNewEmail: 'Create New Email', getNewEmail: 'Create New Email',
getNewEmailTip1: 'Please input the email you want to use. only allow a-z and 0-9', getNewEmailTip1: 'Please input the email you want to use. only allow: ',
getNewEmailTip2: 'Levaing it blank will generate a random email address.', getNewEmailTip2: 'Levaing it blank will generate a random email address.',
getNewEmailTip3: 'You can choose a domain from the dropdown list.', getNewEmailTip3: 'You can choose a domain from the dropdown list.',
credential: 'Email Address Credential', credential: 'Email Address Credential',
@@ -87,7 +88,7 @@ const { locale, t } = useI18n({
login: '登录', login: '登录',
pleaseGetNewEmail: '请"登录"或点击 "注册新邮箱" 按钮来获取一个新的邮箱地址', pleaseGetNewEmail: '请"登录"或点击 "注册新邮箱" 按钮来获取一个新的邮箱地址',
getNewEmail: '创建新邮箱', getNewEmail: '创建新邮箱',
getNewEmailTip1: '请输入你想要使用的邮箱地址, 只允许 a-z, 0-9', getNewEmailTip1: '请输入你想要使用的邮箱地址, 只允许: ',
getNewEmailTip2: '留空将会生成一个随机的邮箱地址。', getNewEmailTip2: '留空将会生成一个随机的邮箱地址。',
getNewEmailTip3: '你可以从下拉列表中选择一个域名。', getNewEmailTip3: '你可以从下拉列表中选择一个域名。',
credential: '邮箱地址凭据', credential: '邮箱地址凭据',
@@ -101,6 +102,18 @@ const { locale, t } = useI18n({
} }
}); });
const addressRegex = computed(() => {
try {
if (openSettings.value.addressRegex) {
return new RegExp(openSettings.value.addressRegex, 'g');
}
} catch (error) {
console.error(error);
message.error(`Invalid addressRegex: ${openSettings.value.addressRegex}`);
}
return /[^a-z0-9]/g;
});
const generateNameLoading = ref(false); const generateNameLoading = ref(false);
const generateName = async () => { const generateName = async () => {
try { try {
@@ -110,8 +123,12 @@ const generateName = async () => {
.split('@')[0] .split('@')[0]
.replace(/\s+/g, '.') .replace(/\s+/g, '.')
.replace(/\.{2,}/g, '.') .replace(/\.{2,}/g, '.')
.replace(/[^a-z0-9]/g, '') .replace(addressRegex.value, '')
.toLowerCase(); .toLowerCase();
// support maxAddressLen
if (emailName.value.length > openSettings.value.maxAddressLen) {
emailName.value = emailName.value.slice(0, openSettings.value.maxAddressLen);
}
} catch (error) { } catch (error) {
message.error(error.message || "error"); message.error(error.message || "error");
} finally { } finally {
@@ -168,9 +185,18 @@ const domainsOptions = computed(() => {
}); });
}); });
const showNewAddressTab = computed(() => {
if (openSettings.value.disableAnonymousUserCreateEmail
&& !userSettings.value.user_email
) {
return false;
}
return openSettings.value.enableUserCreateEmail;
});
onMounted(async () => { onMounted(async () => {
if (!openSettings.value.domains || openSettings.value.domains.length === 0) { if (!openSettings.value.domains || openSettings.value.domains.length === 0) {
await api.getOpenSettings(); await api.getOpenSettings(message, notification);
} }
emailDomain.value = domainsOptions.value ? domainsOptions.value[0]?.value : ""; emailDomain.value = domainsOptions.value ? domainsOptions.value[0]?.value : "";
}); });
@@ -181,7 +207,7 @@ onMounted(async () => {
<n-alert v-if="userSettings.user_email" :show-icon="false" :bordered="false" closable> <n-alert v-if="userSettings.user_email" :show-icon="false" :bordered="false" closable>
<span>{{ t('bindUserInfo') }}</span> <span>{{ t('bindUserInfo') }}</span>
</n-alert> </n-alert>
<n-tabs v-model:value="tabValue" size="large" justify-content="space-evenly"> <n-tabs v-if="openSettings.fetched" v-model:value="tabValue" size="large" justify-content="space-evenly">
<n-tab-pane name="signin" :tab="t('login')"> <n-tab-pane name="signin" :tab="t('login')">
<n-form> <n-form>
<n-form-item-row :label="t('credential')" required> <n-form-item-row :label="t('credential')" required>
@@ -193,8 +219,7 @@ onMounted(async () => {
</template> </template>
{{ t('login') }} {{ t('login') }}
</n-button> </n-button>
<n-button v-if="openSettings.enableUserCreateEmail" @click="tabValue = 'register'" block secondary <n-button v-if="showNewAddressTab" @click="tabValue = 'register'" block secondary strong>
strong>
<template #icon> <template #icon>
<n-icon :component="NewLabelOutlined" /> <n-icon :component="NewLabelOutlined" />
</template> </template>
@@ -202,11 +227,11 @@ onMounted(async () => {
</n-button> </n-button>
</n-form> </n-form>
</n-tab-pane> </n-tab-pane>
<n-tab-pane v-if="openSettings.enableUserCreateEmail" name="register" :tab="t('getNewEmail')"> <n-tab-pane v-if="showNewAddressTab" name="register" :tab="t('getNewEmail')">
<n-spin :show="generateNameLoading"> <n-spin :show="generateNameLoading">
<n-form> <n-form>
<span> <span>
<p>{{ t("getNewEmailTip1") }}</p> <p>{{ t("getNewEmailTip1") + addressRegex.source }}</p>
<p>{{ t("getNewEmailTip2") }}</p> <p>{{ t("getNewEmailTip2") }}</p>
<p>{{ t("getNewEmailTip3") }}</p> <p>{{ t("getNewEmailTip3") }}</p>
</span> </span>

View File

@@ -13,7 +13,7 @@ const isPreview = ref(false)
const editorRef = shallowRef() const editorRef = shallowRef()
const { settings, sendMailModel, indexTab } = useGlobalState() const { settings, sendMailModel, indexTab, userSettings } = useGlobalState()
const { t } = useI18n({ const { t } = useI18n({
locale: 'zh', locale: 'zh',
@@ -136,6 +136,8 @@ const handleCreated = (editor) => {
} }
onMounted(async () => { onMounted(async () => {
// make sure user_id is fetched
if (!userSettings.value.user_id) await api.getUserSettings(message);
await api.getSettings(); await api.getSettings();
}) })
</script> </script>
@@ -149,16 +151,15 @@ onMounted(async () => {
<n-button type="primary" tertiary @click="requestAccess" size="small">{{ t('requestAccess') <n-button type="primary" tertiary @click="requestAccess" size="small">{{ t('requestAccess')
}}</n-button> }}</n-button>
</n-alert> </n-alert>
<br />
<AdminContact /> <AdminContact />
</div> </div>
<div v-else> <div v-else>
<n-alert type="info" :show-icon="false" :bordered="false"> <n-alert type="info" :show-icon="false" :bordered="false" closable>
{{ t('send_balance') }}: {{ settings.send_balance }} {{ t('send_balance') }}: {{ settings.send_balance }}
</n-alert> </n-alert>
<div class="right"> <n-flex justify="end">
<n-button type="primary" @click="send">{{ t('send') }}</n-button> <n-button type="primary" @click="send">{{ t('send') }}</n-button>
</div> </n-flex>
<div class="left"> <div class="left">
<n-form :model="sendMailModel"> <n-form :model="sendMailModel">
<n-form-item :label="t('fromName')" label-placement="top"> <n-form-item :label="t('fromName')" label-placement="top">
@@ -230,9 +231,7 @@ onMounted(async () => {
justify-content: left; justify-content: left;
} }
.right { .n-alert {
text-align: right; margin-bottom: 10px;
place-items: right;
justify-content: right;
} }
</style> </style>

View File

@@ -1,129 +1,28 @@
<script setup lang="ts"> <script setup lang="ts">
import { onMounted, ref } from 'vue'
import { useI18n } from 'vue-i18n'
// @ts-ignore
import { useGlobalState } from '../../store'
// @ts-ignore // @ts-ignore
import { api } from '../../api' import { api } from '../../api'
const { settings } = useGlobalState()
// @ts-ignore // @ts-ignore
const message = useMessage() import WebhookComponent from '../../components/WebhookComponent.vue'
const { t } = useI18n({
messages: {
en: {
successTip: 'Success',
test: 'Test',
save: 'Save',
notEnabled: 'Webhook is not enabled for you',
urlMissing: 'URL is required',
},
zh: {
successTip: '成功',
test: '测试',
save: '保存',
notEnabled: 'Webhook 未开启,请联系管理员开启',
urlMissing: 'URL 不能为空',
}
}
});
class WebhookSettings {
url: string = ''
method: string = 'POST'
headers: string = JSON.stringify({}, null, 2)
body: string = JSON.stringify({}, null, 2)
}
const webhookSettings = ref<WebhookSettings>(new WebhookSettings())
const enableWebhook = ref(false)
const fetchData = async () => { const fetchData = async () => {
try { return await api.fetch(`/api/webhook/settings`)
const res = await api.fetch(`/api/webhook/settings`)
Object.assign(webhookSettings.value, res)
enableWebhook.value = true
} catch (error) {
message.error((error as Error).message || "error");
}
} }
const saveSettings = async () => { const saveSettings = async (webhookSettings: any) => {
if (!webhookSettings.value.url) { await api.fetch(`/api/webhook/settings`, {
message.error(t('urlMissing')) method: 'POST',
return body: JSON.stringify(webhookSettings),
} })
try {
await api.fetch(`/api/webhook/settings`, {
method: 'POST',
body: JSON.stringify(webhookSettings.value),
})
message.success(t('successTip'))
} catch (error) {
message.error((error as Error).message || "error");
}
} }
const testSettings = async () => { const testSettings = async (webhookSettings: any) => {
if (!webhookSettings.value.url) { await api.fetch(`/api/webhook/test`, {
message.error(t('urlMissing')) method: 'POST',
return body: JSON.stringify(webhookSettings),
} })
try {
await api.fetch(`/api/webhook/test`, {
method: 'POST',
body: JSON.stringify(webhookSettings.value),
})
message.success(t('successTip'))
} catch (error) {
message.error((error as Error).message || "error");
}
} }
onMounted(async () => {
await fetchData();
})
</script> </script>
<template> <template>
<div class="center" v-if="settings.address"> <WebhookComponent :fetchData="fetchData" :saveSettings="saveSettings" :testSettings="testSettings" />
<n-card :bordered="false" embedded v-if="enableWebhook" style="max-width: 800px; overflow: auto;">
<n-form-item-row label="URL">
<n-input v-model:value="webhookSettings.url" />
</n-form-item-row>
<n-form-item-row label="METHOD">
<n-select v-model:value="webhookSettings.method" tag :options='[
{ label: "POST", value: "POST" }
]' />
</n-form-item-row>
<n-form-item-row label="HEADERS">
<n-input v-model:value="webhookSettings.headers" type="textarea" :autosize="{ minRows: 3 }" />
</n-form-item-row>
<n-form-item-row label="BODY">
<n-input v-model:value="webhookSettings.body" type="textarea" :autosize="{ minRows: 3 }" />
</n-form-item-row>
<n-button @click="testSettings" secondary block strong>
{{ t('test') }}
</n-button>
<n-button @click="saveSettings" type="primary" block>
{{ t('save') }}
</n-button>
</n-card>
<n-result v-else status="404" :title="t('notEnabled')" />
</div>
</template> </template>
<style scoped>
.center {
display: flex;
text-align: left;
place-items: center;
justify-content: center;
}
.n-button {
margin-top: 10px;
}
</style>

View File

@@ -5,8 +5,9 @@ import { useGlobalState } from '../../store'
import { api } from '../../api' import { api } from '../../api'
import { onMounted, watch } from 'vue'; import { onMounted, watch } from 'vue';
import { processItem } from '../../utils/email-parser' import { processItem } from '../../utils/email-parser'
import { utcToLocalDate } from '../../utils';
const { telegramApp } = useGlobalState() const { telegramApp, loading, useUTCDate } = useGlobalState()
const route = useRoute() const route = useRoute()
const curMail = ref({}); const curMail = ref({});
@@ -26,12 +27,16 @@ const fetchMailData = async () => {
mailId: route.query.mail_id mailId: route.query.mail_id
}) })
}); });
loading.value = true;
return await processItem(res); return await processItem(res);
} }
catch (error) { catch (error) {
console.error(error); console.error(error);
return {}; return {};
} }
finally {
loading.value = false;
}
}; };
onMounted(async () => { onMounted(async () => {
@@ -41,12 +46,12 @@ onMounted(async () => {
<template> <template>
<div class="center"> <div class="center">
<n-card :bordered="false" embedded v-if="curMail.message" style="max-width: 800px; overflow: auto;"> <n-card :bordered="false" embedded v-if="curMail.message" style="max-width: 800px; height: 100%;">
<n-tag type="info"> <n-tag type="info">
ID: {{ curMail.id }} ID: {{ curMail.id }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
Date: {{ curMail.created_at }} Date: {{ utcToLocalDate(curMail.created_at, useUTCDate) }}
</n-tag> </n-tag>
<n-tag type="info"> <n-tag type="info">
FROM: {{ curMail.source }} FROM: {{ curMail.source }}
@@ -54,7 +59,8 @@ onMounted(async () => {
<n-tag v-if="showEMailTo" type="info"> <n-tag v-if="showEMailTo" type="info">
TO: {{ curMail.address }} TO: {{ curMail.address }}
</n-tag> </n-tag>
<div v-html="curMail.message" style="margin-top: 10px;"></div> <iframe :srcdoc="curMail.message" style="margin-top: 10px;width: 100%; height: 100%;">
</iframe>
</n-card> </n-card>
</div> </div>
</template> </template>
@@ -66,5 +72,6 @@ onMounted(async () => {
text-align: left; text-align: left;
place-items: center; place-items: center;
justify-content: center; justify-content: center;
height: 80vh;
} }
</style> </style>

View File

@@ -20,9 +20,12 @@ const { locale, t } = useI18n({
mail_count: 'Mail Count', mail_count: 'Mail Count',
send_count: 'Send Count', send_count: 'Send Count',
actions: 'Actions', actions: 'Actions',
changeMailAddress: 'Change Mail Address', changeMailAddress: 'Change Address',
unbindAddress: 'Unbind Address', unbindAddress: 'Unbind Address',
unbindAddressTip: 'Before unbinding, please switch to this email address and save the email address credential.', unbindAddressTip: 'Before unbinding, please switch to this email address and save the email address credential.',
transferAddress: 'Transfer Address',
targetUserEmail: 'Target User Email',
transferAddressTip: 'Transfer address to another user will remove the address from your account and transfer it to another user. Are you sure to transfer the address?'
}, },
zh: { zh: {
success: '成功', success: '成功',
@@ -30,14 +33,21 @@ const { locale, t } = useI18n({
mail_count: '邮件数量', mail_count: '邮件数量',
send_count: '发送数量', send_count: '发送数量',
actions: '操作', actions: '操作',
changeMailAddress: '切换邮箱地址', changeMailAddress: '切换地址',
unbindAddress: '解绑地址', unbindAddress: '解绑地址',
unbindAddressTip: '解绑前请切换到此邮箱地址并保存邮箱地址凭证。', unbindAddressTip: '解绑前请切换到此邮箱地址并保存邮箱地址凭证。',
transferAddress: '转移地址',
targetUserEmail: '目标用户邮箱',
transferAddressTip: '转移地址到其他用户将会从你的账户中移除此地址并转移给其他用户。确定要转移地址吗?'
} }
} }
}); });
const data = ref([]) const data = ref([])
const showTranferAddress = ref(false)
const currentAddress = ref("")
const currentAddressId = ref(0)
const targetUserEmail = ref('')
const changeMailAddress = async (address_id) => { const changeMailAddress = async (address_id) => {
try { try {
@@ -70,6 +80,35 @@ const unbindAddress = async (address_id) => {
} }
} }
const transferAddress = async () => {
if (!targetUserEmail.value) {
message.error("targetUserEmail is required");
return;
}
if (!currentAddressId.value) {
message.error("currentAddressId is required");
return;
}
try {
const res = await api.fetch(`/user_api/transfer_address`, {
method: 'POST',
body: JSON.stringify({
address_id: currentAddressId.value,
target_user_email: targetUserEmail.value
})
});
message.success(t('transferAddress') + " " + t('success'));
await fetchData();
showTranferAddress.value = false;
currentAddressId.value = 0;
currentAddress.value = "";
targetUserEmail.value = "";
} catch (error) {
console.log(error)
message.error(error.message || "error");
}
}
const fetchData = async () => { const fetchData = async () => {
try { try {
const { results, count: addressCount } = await api.fetch( const { results, count: addressCount } = await api.fetch(
@@ -86,10 +125,6 @@ const fetchData = async () => {
} }
const columns = [ const columns = [
{
title: "ID",
key: "id"
},
{ {
title: t('name'), title: t('name'),
key: "name" key: "name"
@@ -138,6 +173,18 @@ const columns = [
default: () => `${t('changeMailAddress')}?` default: () => `${t('changeMailAddress')}?`
} }
), ),
h(NButton,
{
tertiary: true,
type: "primary",
onClick: () => {
currentAddressId.value = row.id;
currentAddress.value = row.name;
showTranferAddress.value = true;
}
},
{ default: () => t('transferAddress') }
),
h(NPopconfirm, h(NPopconfirm,
{ {
onPositiveClick: () => unbindAddress(row.id) onPositiveClick: () => unbindAddress(row.id)
@@ -164,7 +211,25 @@ onMounted(async () => {
</script> </script>
<template> <template>
<div> <n-modal v-model:show="showTranferAddress" preset="dialog" :title="t('transferAddress')">
<span>
<p>{{ t("transferAddressTip") }}</p>
<p>{{ t('transferAddress') + ": " + currentAddress }}</p>
<n-input v-model:value="targetUserEmail" :placeholder="t('targetUserEmail')" />
</span>
<template #action>
<n-button :loading="loading" @click="transferAddress" size="small" tertiary type="error">
{{ t('transferAddress') }}
</n-button>
</template>
</n-modal>
<div style="overflow: auto;">
<n-data-table :columns="columns" :data="data" :bordered="false" embedded /> <n-data-table :columns="columns" :data="data" :bordered="false" embedded />
</div> </div>
</template> </template>
<style scoped>
.n-data-table {
min-width: 700px;
}
</style>

View File

@@ -31,7 +31,8 @@ const { t } = useI18n({
onMounted(async () => { onMounted(async () => {
await api.getUserOpenSettings(message); await api.getUserOpenSettings(message);
await api.getUserSettings(message); // make sure user_id is fetched
if (!userSettings.value.user_id) await api.getUserSettings(message);
}); });
</script> </script>

View File

@@ -1,18 +1,21 @@
<script setup> <script setup>
import { useMessage } from 'naive-ui' import { useMessage } from 'naive-ui'
import { useRouter } from 'vue-router' import { onMounted, ref } from "vue";
import { computed, onMounted, ref } from "vue";
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { KeyFilled } from '@vicons/material'
import { api } from '../../api'; import { api } from '../../api';
import { useGlobalState } from '../../store' import { useGlobalState } from '../../store'
import { hashPassword } from '../../utils'; import { hashPassword } from '../../utils';
import { startAuthentication } from '@simplewebauthn/browser';
import Turnstile from '../../components/Turnstile.vue'; import Turnstile from '../../components/Turnstile.vue';
const { userJwt, userTab, userOpenSettings, openSettings } = useGlobalState() const {
userJwt, userOpenSettings, openSettings,
userOauth2SessionState, userOauth2SessionClientID
} = useGlobalState()
const message = useMessage(); const message = useMessage();
const router = useRouter();
const { t } = useI18n({ const { t } = useI18n({
messages: { messages: {
@@ -33,6 +36,8 @@ const { t } = useI18n({
pleaseInputCode: 'Please input code', pleaseInputCode: 'Please input code',
pleaseCompleteTurnstile: 'Please complete turnstile', pleaseCompleteTurnstile: 'Please complete turnstile',
pleaseLogin: 'Please login', pleaseLogin: 'Please login',
loginWithPasskey: 'Login with Passkey',
loginWith: 'Login with {provider}',
}, },
zh: { zh: {
login: '登录', login: '登录',
@@ -51,6 +56,8 @@ const { t } = useI18n({
pleaseInputCode: '请输入验证码', pleaseInputCode: '请输入验证码',
pleaseCompleteTurnstile: '请完成人机验证', pleaseCompleteTurnstile: '请完成人机验证',
pleaseLogin: '请登录', pleaseLogin: '请登录',
loginWithPasskey: '使用 Passkey 登录',
loginWith: '使用 {provider} 登录',
} }
} }
}); });
@@ -156,6 +163,45 @@ const emailSignup = async () => {
} }
}; };
const passkeyLogin = async () => {
try {
const options = await api.fetch(`/user_api/passkey/authenticate_request`, {
method: 'POST',
body: JSON.stringify({
domain: location.hostname,
})
})
const credential = await startAuthentication(options)
// Send the result to the server and return the promise.
const res = await api.fetch(`/user_api/passkey/authenticate_response`, {
method: 'POST',
body: JSON.stringify({
origin: location.origin,
domain: location.hostname,
credential
})
})
userJwt.value = res.jwt;
location.reload();
} catch (e) {
console.error(e)
message.error(e.message)
}
};
const oauth2Login = async (clientID) => {
try {
userOauth2SessionClientID.value = clientID;
userOauth2SessionState.value = Math.random().toString(36).substring(2);
const res = await api.fetch(`/user_api/oauth2/login_url?clientID=${clientID}&state=${userOauth2SessionState.value}`);
// redirect to oauth2 login page
location.href = res.url;
} catch (error) {
message.error(error.message || "login failed");
}
};
onMounted(async () => { onMounted(async () => {
}); });
@@ -163,7 +209,7 @@ onMounted(async () => {
<template> <template>
<div class="center"> <div class="center">
<n-tabs v-model:value="tabValue" size="large" justify-content="space-evenly"> <n-tabs v-model:value="tabValue" size="large" v-if="userOpenSettings.fetched" justify-content="space-evenly">
<n-tab-pane name="signin" :tab="t('login')"> <n-tab-pane name="signin" :tab="t('login')">
<n-form> <n-form>
<n-form-item-row :label="t('email')" required> <n-form-item-row :label="t('email')" required>
@@ -178,6 +224,17 @@ onMounted(async () => {
<n-button @click="showModal = true" type="info" quaternary size="tiny"> <n-button @click="showModal = true" type="info" quaternary size="tiny">
{{ t('forgotPassword') }} {{ t('forgotPassword') }}
</n-button> </n-button>
<n-divider />
<n-button @click="passkeyLogin" type="primary" block secondary strong>
<template #icon>
<n-icon :component="KeyFilled" />
</template>
{{ t('loginWithPasskey') }}
</n-button>
<n-button @click="oauth2Login(item.clientID)" v-for="item in userOpenSettings.oauth2ClientIDs"
:key="item.clientID" block secondary strong>
{{ t('loginWith', { provider: item.name }) }}
</n-button>
</n-form> </n-form>
</n-tab-pane> </n-tab-pane>
<n-tab-pane v-if="userOpenSettings.enable" name="signup" :tab="t('register')"> <n-tab-pane v-if="userOpenSettings.enable" name="signup" :tab="t('register')">
@@ -244,4 +301,8 @@ onMounted(async () => {
place-items: center; place-items: center;
justify-content: center; justify-content: center;
} }
.n-button {
margin-top: 10px;
}
</style> </style>

View File

@@ -0,0 +1,65 @@
<script setup>
import { ref, onMounted } from 'vue';
import { useI18n } from 'vue-i18n'
import { useRoute, useRouter } from 'vue-router';
import { useGlobalState } from '../../store'
import { api } from '../../api';
const {
userJwt, userOauth2SessionState, userOauth2SessionClientID
} = useGlobalState()
const message = useMessage();
const route = useRoute()
const router = useRouter()
const errorInfo = ref('')
const { t } = useI18n({
messages: {
en: {
logging: 'Logging in...',
stateNotMatch: 'state not match',
},
zh: {
logging: '登录中...',
stateNotMatch: 'state 不匹配',
}
}
});
onMounted(async () => {
const state = route.query.state;
if (state != userOauth2SessionState.value) {
console.error('state not match');
message.error(t('stateNotMatch'));
return;
}
const code = route.query.code;
if (!code) {
console.error('code not found');
message.error('code not found');
return;
}
try {
const res = await api.fetch(`/user_api/oauth2/callback`, {
method: 'POST',
body: JSON.stringify({
code: code,
clientID: userOauth2SessionClientID.value
})
});
userJwt.value = res.jwt;
router.push('/user');
} catch (error) {
console.error(error);
message.error(error.message || 'error');
}
});
</script>
<template>
<n-card :bordered="false" embedded>
<n-result status="info" :title="t('logging')" :description="errorInfo">
</n-result>
</n-card>
</template>

View File

@@ -1,16 +1,22 @@
<script setup> <script setup>
import { onMounted, ref } from 'vue' import { ref } from 'vue'
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { useRouter } from 'vue-router' import { startRegistration } from '@simplewebauthn/browser';
import { NButton, NPopconfirm } from 'naive-ui'
import { useGlobalState } from '../../store' import { useGlobalState } from '../../store'
import { api } from '../../api' import { api } from '../../api'
const { userJwt, userSettings, } = useGlobalState() const { userJwt, userSettings, } = useGlobalState()
const router = useRouter()
const message = useMessage() const message = useMessage()
const showLogout = ref(false) const showLogout = ref(false)
const showCreatePasskey = ref(false)
const passkeyName = ref('')
const showPasskeyList = ref(false)
const showRenamePasskey = ref(false)
const currentPasskeyId = ref(null)
const currentPasskeyName = ref('')
const { t } = useI18n({ const { t } = useI18n({
messages: { messages: {
@@ -18,11 +24,35 @@ const { t } = useI18n({
logout: 'Logout', logout: 'Logout',
logoutConfirm: 'Are you sure you want to logout?', logoutConfirm: 'Are you sure you want to logout?',
passordTip: 'The server will only receive the hash value of the password, and will not receive the plaintext password, so it cannot view or retrieve your password. If the administrator enables email verification, you can reset the password in incognito mode', passordTip: 'The server will only receive the hash value of the password, and will not receive the plaintext password, so it cannot view or retrieve your password. If the administrator enables email verification, you can reset the password in incognito mode',
createPasskey: 'Create Passkey',
showPasskeyList: 'Show Passkey List',
passkeyCreated: 'Passkey created successfully',
passkeyNamePlaceholder: 'Please enter the passkey name or leave it empty to generate a random one',
renamePasskey: 'Rename Passkey',
deletePasskey: 'Delete Passkey',
passkey_name: 'Passkey Name',
created_at: 'Created At',
updated_at: 'Updated At',
actions: 'Actions',
renamePasskey: 'Rename Passkey',
renamePasskeyNamePlaceholder: 'Please enter the new passkey name',
}, },
zh: { zh: {
logout: '退出登录', logout: '退出登录',
logoutConfirm: '确定要退出登录吗?', logoutConfirm: '确定要退出登录吗?',
passordTip: '服务器只会接收到密码的哈希值,不会接收到明文密码,因此无法查看或者找回您的密码, 如果管理员启用了邮件验证您可以在无痕模式重置密码', passordTip: '服务器只会接收到密码的哈希值,不会接收到明文密码,因此无法查看或者找回您的密码, 如果管理员启用了邮件验证您可以在无痕模式重置密码',
createPasskey: '创建 Passkey',
showPasskeyList: '查看 Passkey 列表',
passkeyCreated: 'Passkey 创建成功',
passkeyNamePlaceholder: '请输入 Passkey 名称或者留空自动生成',
renamePasskey: '重命名 Passkey',
deletePasskey: '删除 Passkey',
passkey_name: 'Passkey 名称',
created_at: '创建时间',
updated_at: '更新时间',
actions: '操作',
renamePasskey: '重命名 Passkey',
renamePasskeyNamePlaceholder: '请输入新的 Passkey 名称',
} }
} }
}); });
@@ -33,17 +63,144 @@ const logout = async () => {
location.reload() location.reload()
} }
const fetchData = async () => { const createPasskey = async () => {
try {
const options = await api.fetch(`/user_api/passkey/register_request`, {
method: 'POST',
body: JSON.stringify({
domain: location.hostname,
})
})
const credential = await startRegistration(options)
// Send the result to the server and return the promise.
await api.fetch(`/user_api/passkey/register_response`, {
method: 'POST',
body: JSON.stringify({
origin: location.origin,
passkey_name: passkeyName.value || (
(window.navigator.userAgentData?.platform || "Unknown")
+ ": " + Math.random().toString(36).substring(7)
),
credential
})
})
message.success(t('passkeyCreated'));
} catch (e) {
console.error(e)
message.error(e.message)
} finally {
passkeyName.value = ''
showCreatePasskey.value = false
}
} }
onMounted(async () => { const passkeyColumns = [
await fetchData() {
}) title: "Passkey ID",
key: "passkey_id"
},
{
title: t('passkey_name'),
key: "passkey_name"
},
{
title: t('created_at'),
key: "created_at"
},
{
title: t('updated_at'),
key: "updated_at"
},
{
title: t('actions'),
key: 'actions',
render(row) {
return h('div', [
[
h(NButton,
{
tertiary: true,
type: "primary",
onClick: () => {
showRenamePasskey.value = true;
currentPasskeyId.value = row.passkey_id;
}
},
{ default: () => t('renamePasskey') }
),
h(NPopconfirm,
{
onPositiveClick: async () => {
try {
await api.fetch(`/user_api/passkey/${row.passkey_id}`, {
method: 'DELETE'
})
await fetchPasskeyList()
} catch (e) {
console.error(e)
message.error(e.message)
}
}
},
{
trigger: () => h(NButton,
{
tertiary: true,
type: "error",
},
{ default: () => t('deletePasskey') }
),
default: () => `${t('deletePasskey')}?`
}
),
]
])
}
}
]
const passkeyData = ref([])
const fetchPasskeyList = async () => {
try {
const data = await api.fetch(`/user_api/passkey`)
passkeyData.value = data
} catch (e) {
console.error(e)
message.error(e.message)
}
}
const renamePasskey = async () => {
try {
await api.fetch(`/user_api/passkey/rename`, {
method: 'POST',
body: JSON.stringify({
passkey_name: currentPasskeyName.value,
passkey_id: currentPasskeyId.value
})
})
await fetchPasskeyList()
} catch (e) {
console.error(e)
message.error(e.message)
} finally {
currentPasskeyName.value = ''
showRenamePasskey.value = false
}
}
</script> </script>
<template> <template>
<div class="center" v-if="userSettings.user_email"> <div class="center" v-if="userSettings.user_email">
<n-card :bordered="false" embedded> <n-card :bordered="false" embedded>
<n-button @click="showPasskeyList = true; fetchPasskeyList();" secondary block strong>
{{ t('showPasskeyList') }}
</n-button>
<n-button @click="showCreatePasskey = true" type="primary" secondary block strong>
{{ t('createPasskey') }}
</n-button>
<n-alert :show-icon="false" :bordered="false"> <n-alert :show-icon="false" :bordered="false">
<span> <span>
{{ t('passordTip') }} {{ t('passordTip') }}
@@ -53,6 +210,25 @@ onMounted(async () => {
{{ t('logout') }} {{ t('logout') }}
</n-button> </n-button>
</n-card> </n-card>
<n-modal v-model:show="showCreatePasskey" preset="dialog" :title="t('createPasskey')">
<n-input v-model:value="passkeyName" :placeholder="t('passkeyNamePlaceholder')" />
<template #action>
<n-button :loading="loading" @click="createPasskey" size="small" tertiary type="primary">
{{ t('createPasskey') }}
</n-button>
</template>
</n-modal>
<n-modal v-model:show="showRenamePasskey" preset="dialog" :title="t('renamePasskey')">
<n-input v-model:value="currentPasskeyName" :placeholder="t('renamePasskeyNamePlaceholder')" />
<template #action>
<n-button :loading="loading" @click="renamePasskey" size="small" tertiary type="primary">
{{ t('renamePasskey') }}
</n-button>
</template>
</n-modal>
<n-modal v-model:show="showPasskeyList" preset="card" :title="t('showPasskeyList')">
<n-data-table :columns="passkeyColumns" :data="passkeyData" :bordered="false" embedded />
</n-modal>
<n-modal v-model:show="showLogout" preset="dialog" :title="t('logout')"> <n-modal v-model:show="showLogout" preset="dialog" :title="t('logout')">
<p>{{ t('logoutConfirm') }}</p> <p>{{ t('logoutConfirm') }}</p>
<template #action> <template #action>
@@ -78,5 +254,6 @@ onMounted(async () => {
.n-button { .n-button {
margin-top: 10px; margin-top: 10px;
margin-bottom: 10px;
} }
</style> </style>

View File

@@ -24,6 +24,7 @@ export default defineConfig({
{ {
'naive-ui': [ 'naive-ui': [
'useMessage', 'useMessage',
'useNotification',
'NButton', 'NButton',
'NPopconfirm', 'NPopconfirm',
'NIcon', 'NIcon',
@@ -35,13 +36,16 @@ export default defineConfig({
resolvers: [NaiveUiResolver()] resolvers: [NaiveUiResolver()]
}), }),
VitePWA({ VitePWA({
registerType: 'autoUpdate', registerType: null,
devOptions: { devOptions: {
enabled: true enabled: false
}, },
workbox: { workbox: {
disableDevLogs: true, disableDevLogs: true,
globPatterns: ['**/*.{js,css,html,ico,png,svg}'], globPatterns: [],
runtimeCaching: [],
navigateFallback: null,
cleanupOutdatedCaches: true,
}, },
manifest: { manifest: {
name: 'Temp Email', name: 'Temp Email',
@@ -65,5 +69,10 @@ export default defineConfig({
}, },
define: { define: {
'import.meta.env.PACKAGE_VERSION': JSON.stringify(process.env.npm_package_version), 'import.meta.env.PACKAGE_VERSION': JSON.stringify(process.env.npm_package_version),
},
esbuild: {
supported: {
'top-level-await': true
},
} }
}) })

View File

@@ -1,6 +1,6 @@
[package] [package]
name = "mail-parser-wasm" name = "mail-parser-wasm"
version = "0.1.8" version = "0.2.1"
edition = "2021" edition = "2021"
description = "A simple mail parser for wasm" description = "A simple mail parser for wasm"
license = "MIT" license = "MIT"
@@ -9,5 +9,5 @@ license = "MIT"
crate-type = ["cdylib"] crate-type = ["cdylib"]
[dependencies] [dependencies]
mail-parser = "0.9.3" mail-parser = "0.9.4"
wasm-bindgen = "0.2.92" wasm-bindgen = "0.2.99"

View File

@@ -35,10 +35,31 @@ impl AttachmentResult {
} }
} }
#[derive(Clone)]
#[wasm_bindgen]
pub struct MessageHeader {
key: String,
value: String,
}
#[wasm_bindgen]
impl MessageHeader {
#[wasm_bindgen(getter)]
pub fn key(&self) -> String {
self.key.clone()
}
#[wasm_bindgen(getter)]
pub fn value(&self) -> String {
self.value.clone()
}
}
#[wasm_bindgen] #[wasm_bindgen]
pub struct MessageResult { pub struct MessageResult {
sender: String, sender: String,
subject: String, subject: String,
headers: Vec<MessageHeader>,
body_html: String, body_html: String,
text: String, text: String,
attachments: Vec<AttachmentResult>, attachments: Vec<AttachmentResult>,
@@ -56,6 +77,11 @@ impl MessageResult {
self.subject.clone() self.subject.clone()
} }
#[wasm_bindgen(getter)]
pub fn headers(&self) -> Vec<MessageHeader> {
self.headers.clone()
}
#[wasm_bindgen(getter)] #[wasm_bindgen(getter)]
pub fn body_html(&self) -> String { pub fn body_html(&self) -> String {
self.body_html.clone() self.body_html.clone()
@@ -119,6 +145,7 @@ pub fn parse_message(raw_message: &str) -> MessageResult {
return MessageResult { return MessageResult {
sender: String::new(), sender: String::new(),
subject: String::new(), subject: String::new(),
headers: Vec::new(),
body_html: String::new(), body_html: String::new(),
text: String::new(), text: String::new(),
attachments: Vec::new(), attachments: Vec::new(),
@@ -146,6 +173,14 @@ pub fn parse_message(raw_message: &str) -> MessageResult {
.subject() .subject()
.map(|subject| subject.to_owned()) .map(|subject| subject.to_owned())
.unwrap_or(String::new()), .unwrap_or(String::new()),
headers: message
.headers()
.iter()
.map(|header| MessageHeader {
key: header.name().to_owned(),
value: header.value().as_text().unwrap_or("").to_owned(),
})
.collect(),
body_html: message body_html: message
.body_html(0) .body_html(0)
.map(|html| html.into_owned()) .map(|html| html.into_owned())

View File

@@ -1,12 +1,12 @@
import initAsync, { initSync, parse_message } from './mail_parser_wasm'; import initAsync, { initSync, parse_message } from './mail_parser_wasm';
import MODULE from './mail_parser_wasm_bg.wasm'; import MODULE from './mail_parser_wasm_bg.wasm';
initSync(MODULE); initSync({ module: MODULE });
export { initAsync, MODULE }; export { initAsync, MODULE };
export * from './mail_parser_wasm'; export * from './mail_parser_wasm';
export const parse_message_wrapper = (raw_message) => { export const parse_message_wrapper = (raw_message) => {
initSync(MODULE); initSync({ module: MODULE });
return parse_message(raw_message); return parse_message(raw_message);
} }

View File

@@ -7,7 +7,7 @@
"url": "https://github.com/dreamhunter2333/cloudflare_temp_email", "url": "https://github.com/dreamhunter2333/cloudflare_temp_email",
"directory": "mail-parser-wasm" "directory": "mail-parser-wasm"
}, },
"version": "0.1.8", "version": "0.2.1",
"license": "MIT", "license": "MIT",
"files": [ "files": [
"mail_parser_wasm_bg.wasm", "mail_parser_wasm_bg.wasm",

View File

@@ -1,6 +1,6 @@
{ {
"name": "temp-email-pages", "name": "temp-email-pages",
"version": "1.0.0", "version": "0.8.6",
"description": "", "description": "",
"main": "index.js", "main": "index.js",
"scripts": { "scripts": {
@@ -11,6 +11,6 @@
"author": "", "author": "",
"license": "ISC", "license": "ISC",
"devDependencies": { "devDependencies": {
"wrangler": "^3.62.0" "wrangler": "^3.104.0"
} }
} }

15
scripts/update-dependencies.sh Executable file
View File

@@ -0,0 +1,15 @@
cd frontend/
pnpm up
cd ..
cd worker/
pnpm up
cd ..
cd pages/
pnpm up
cd ..
cd vitepress-docs/
pnpm up
cd ..

View File

@@ -13,6 +13,7 @@ class Settings(BaseSettings):
proxy_url: str = "http://localhost:8787" proxy_url: str = "http://localhost:8787"
port: int = 8025 port: int = 8025
imap_port: int = 11143 imap_port: int = 11143
basic_password: str = ""
class Config: class Config:
env_file = ".env" env_file = ".env"

View File

@@ -121,6 +121,7 @@ class SimpleMailbox:
f"{settings.proxy_url}/api/mails?limit={limit}&offset={start - 1}", f"{settings.proxy_url}/api/mails?limit={limit}&offset={start - 1}",
headers={ headers={
"Authorization": f"Bearer {self.password}", "Authorization": f"Bearer {self.password}",
"x-custom-auth": f"{settings.basic_password}",
"Content-Type": "application/json" "Content-Type": "application/json"
} }
) )
@@ -147,6 +148,7 @@ class SimpleMailbox:
f"{settings.proxy_url}/api/sendbox?limit={limit}&offset={start - 1}", f"{settings.proxy_url}/api/sendbox?limit={limit}&offset={start - 1}",
headers={ headers={
"Authorization": f"Bearer {self.password}", "Authorization": f"Bearer {self.password}",
"x-custom-auth": f"{settings.basic_password}",
"Content-Type": "application/json" "Content-Type": "application/json"
} }
) )

View File

@@ -1,5 +1,5 @@
aiosmtpd==1.4.6 aiosmtpd==1.4.6
pydantic-settings==2.2.1 pydantic-settings==2.2.1
requests==2.32.0 requests==2.32.0
twisted==24.3.0 twisted==24.7.0
httpx==0.27.0 httpx==0.27.0

View File

@@ -96,7 +96,7 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] {
}, },
{ {
text: '通过命令行部署', text: '通过命令行部署',
collapsed: true, collapsed: false,
items: [ items: [
{ text: '命令行部署准备', link: 'cli/pre-requisite' }, { text: '命令行部署准备', link: 'cli/pre-requisite' },
{ text: 'D1 数据库', link: 'cli/d1' }, { text: 'D1 数据库', link: 'cli/d1' },
@@ -108,7 +108,7 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] {
}, },
{ {
text: '通过用户界面部署', text: '通过用户界面部署',
collapsed: true, collapsed: false,
items: [ items: [
{ text: 'D1 数据库', link: 'ui/d1' }, { text: 'D1 数据库', link: 'ui/d1' },
{ text: 'Cloudflare workers 后端', link: 'ui/worker' }, { text: 'Cloudflare workers 后端', link: 'ui/worker' },
@@ -121,12 +121,24 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] {
text: '通过 Github Actions 部署', text: '通过 Github Actions 部署',
collapsed: true, collapsed: true,
items: [ items: [
{ text: '通过 Github Actions 部署', link: 'github-action' }, { text: 'D1 数据库', link: 'actions/d1' },
{ text: 'Github Actions 配置', link: 'actions/github-action' },
{ text: '配置邮件转发', link: 'email-routing.md' },
{ text: '配置发送邮件', link: 'config-send-mail' },
{ text: '自动更新配置', link: 'actions/auto-update' },
]
},
{
text: '通用',
collapsed: false,
items: [
{ text: 'worker变量说明', link: 'worker-vars' },
{ text: '常见问题', link: 'common-issues' },
] ]
}, },
{ {
text: '附加功能', text: '附加功能',
collapsed: true, collapsed: false,
items: [ items: [
{ text: '配置 SMTP IMAP 代理服务', link: 'feature/config-smtp-proxy' }, { text: '配置 SMTP IMAP 代理服务', link: 'feature/config-smtp-proxy' },
{ text: '发送邮件 API', link: 'feature/send-mail-api' }, { text: '发送邮件 API', link: 'feature/send-mail-api' },
@@ -135,12 +147,15 @@ function sidebarGuide(): DefaultTheme.SidebarItem[] {
{ text: '配置 Telegram Bot', link: 'feature/telegram' }, { text: '配置 Telegram Bot', link: 'feature/telegram' },
{ text: '配置 S3 附件', link: 'feature/s3-attachment' }, { text: '配置 S3 附件', link: 'feature/s3-attachment' },
{ text: '配置 worker 使用 wasm 解析邮件', link: 'feature/mail_parser_wasm_worker' }, { text: '配置 worker 使用 wasm 解析邮件', link: 'feature/mail_parser_wasm_worker' },
{ text: '配置 webhook', link: 'feature/webhook' },
{ text: '新建邮箱地址 API', link: 'feature/new-address-api' }, { text: '新建邮箱地址 API', link: 'feature/new-address-api' },
{ text: 'Oauth2 第三方登录', link: 'feature/user-oauth2' },
{ text: '配置其他worker增强', link: 'feature/another-worker-enhanced' },
] ]
}, },
{ {
text: '功能简介', text: '功能简介',
collapsed: true, collapsed: false,
items: [ items: [
{ text: 'Admin 控制台', link: 'feature/admin' }, { text: 'Admin 控制台', link: 'feature/admin' },
{ text: 'Admin 用户管理', link: 'feature/admin-user-management' }, { text: 'Admin 用户管理', link: 'feature/admin-user-management' },

View File

@@ -34,10 +34,10 @@ git clone https://github.com/dreamhunter2333/cloudflare_temp_email.git
```bash ```bash
# create a database, and copy the output to wrangler.toml in the next step # create a database, and copy the output to wrangler.toml in the next step
wrangler d1 create dev wrangler d1 create dev
wrangler d1 execute dev --file=db/schema.sql wrangler d1 execute dev --file=db/schema.sql --remote
# schema update, if you have initialized the database before this date, you can execute this command to update # schema update, if you have initialized the database before this date, you can execute this command to update
# wrangler d1 execute dev --file=db/2024-01-13-patch.sql # wrangler d1 execute dev --file=db/2024-01-13-patch.sql --remote
# wrangler d1 execute dev --file=db/2024-04-03-patch.sql # wrangler d1 execute dev --file=db/2024-04-03-patch.sql --remote
# create a namespace, and copy the output to wrangler.toml in the next step # create a namespace, and copy the output to wrangler.toml in the next step
wrangler kv:namespace create DEV wrangler kv:namespace create DEV
``` ```
@@ -61,8 +61,8 @@ pnpm run deploy
```toml ```toml
name = "cloudflare_temp_email" name = "cloudflare_temp_email"
main = "src/worker.ts" main = "src/worker.ts"
compatibility_date = "2023-08-14" compatibility_date = "2024-09-23"
node_compat = true compatibility_flags = [ "nodejs_compat" ]
# enable cron if you want set auto clean up # enable cron if you want set auto clean up
# [triggers] # [triggers]
@@ -79,10 +79,17 @@ PREFIX = "tmp" # The mailbox name prefix to be processed
# (min, max) length of the adderss, if not set, the default is (1, 30) # (min, max) length of the adderss, if not set, the default is (1, 30)
# MIN_ADDRESS_LEN = 1 # MIN_ADDRESS_LEN = 1
# MAX_ADDRESS_LEN = 30 # MAX_ADDRESS_LEN = 30
# ANNOUNCEMENT = "Custom Announcement"
# address check REGEX, if not set, will not check
# ADDRESS_CHECK_REGEX = "^(?!.*admin).*"
# address name replace REGEX, if not set, the default is [^a-z0-9]
# ADDRESS_REGEX = "[^a-z0-9]"
# If you want your site to be private, uncomment below and change your password # If you want your site to be private, uncomment below and change your password
# PASSWORDS = ["123", "456"] # PASSWORDS = ["123", "456"]
# admin console password, if not configured, access to the console is not allowed # admin console password, if not configured, access to the console is not allowed
# ADMIN_PASSWORDS = ["123", "456"] # ADMIN_PASSWORDS = ["123", "456"]
# warning: no password or user check for admin portal
# DISABLE_ADMIN_PASSWORD_CHECK = false
# admin contact information. If not configured, it will not be displayed. Any string can be configured. # admin contact information. If not configured, it will not be displayed. Any string can be configured.
# ADMIN_CONTACT = "xx@xx.xxx" # ADMIN_CONTACT = "xx@xx.xxx"
DEFAULT_DOMAINS = ["xxx.xxx1" , "xxx.xxx2"] # domain name for no role users DEFAULT_DOMAINS = ["xxx.xxx1" , "xxx.xxx2"] # domain name for no role users
@@ -90,6 +97,7 @@ DOMAINS = ["xxx.xxx1" , "xxx.xxx2"] # all your domain name
# For chinese domain name, you can use DOMAIN_LABELS to show chinese domain name # For chinese domain name, you can use DOMAIN_LABELS to show chinese domain name
# DOMAIN_LABELS = ["中文.xxx", "xxx.xxx2"] # DOMAIN_LABELS = ["中文.xxx", "xxx.xxx2"]
# USER_DEFAULT_ROLE = "vip" # default role for new users(only when enable mail verification) # USER_DEFAULT_ROLE = "vip" # default role for new users(only when enable mail verification)
# ADMIN_USER_ROLE = "admin" # the role which can access admin panel
# User roles configuration, if domains is empty will use default_domains, if prefix is null will use default prefix, if prefix is empty string will not use prefix # User roles configuration, if domains is empty will use default_domains, if prefix is null will use default prefix, if prefix is empty string will not use prefix
# USER_ROLES = [ # USER_ROLES = [
# { domains = ["xxx.xxx1" , "xxx.xxx2"], role = "vip", prefix = "vip" }, # { domains = ["xxx.xxx1" , "xxx.xxx2"], role = "vip", prefix = "vip" },
@@ -99,6 +107,8 @@ JWT_SECRET = "xxx" # Key used to generate jwt
BLACK_LIST = "" # Blacklist, used to filter senders, comma separated BLACK_LIST = "" # Blacklist, used to filter senders, comma separated
# Allow users to create email addresses # Allow users to create email addresses
ENABLE_USER_CREATE_EMAIL = true ENABLE_USER_CREATE_EMAIL = true
# Disable anonymous user create email, if set true, users can only create email addresses after logging in
# DISABLE_ANONYMOUS_USER_CREATE_EMAIL = true
# Allow users to delete messages # Allow users to delete messages
ENABLE_USER_DELETE_EMAIL = true ENABLE_USER_DELETE_EMAIL = true
# Allow automatic replies to emails # Allow automatic replies to emails
@@ -107,15 +117,31 @@ ENABLE_AUTO_REPLY = false
# ENABLE_WEBHOOK = true # ENABLE_WEBHOOK = true
# Footer text # Footer text
# COPYRIGHT = "Dream Hunter" # COPYRIGHT = "Dream Hunter"
# DISABLE_SHOW_GITHUB = true # Disable Show GitHub link
# default send balance, if not set, it will be 0 # default send balance, if not set, it will be 0
# DEFAULT_SEND_BALANCE = 1 # DEFAULT_SEND_BALANCE = 1
# NO_LIMIT_SEND_ROLE = "vip" # the role which can send emails without limit
# Turnstile verification configuration # Turnstile verification configuration
# CF_TURNSTILE_SITE_KEY = "" # CF_TURNSTILE_SITE_KEY = ""
# CF_TURNSTILE_SECRET_KEY = "" # CF_TURNSTILE_SECRET_KEY = ""
# telegram bot # telegram bot
# TG_MAX_ACCOUNTS = 5 # TG_MAX_ADDRESS = 5
# telegram bot info, predefined bot info can reduce latency of the webhook
# TG_BOT_INFO = "{}"
# global forward address list, if set, all emails will be forwarded to these addresses # global forward address list, if set, all emails will be forwarded to these addresses
# FORWARD_ADDRESS_LIST = ["xxx@xxx.com"] # FORWARD_ADDRESS_LIST = ["xxx@xxx.com"]
# Frontend URL
# FRONTEND_URL = "https://xxxx.xxx"
# Enable check junk mail
# ENABLE_CHECK_JUNK_MAIL = false
# junk mail check list, if status exists and status is not pass, will be marked as junk mail
# JUNK_MAIL_CHECK_LIST = = ["spf", "dkim", "dmarc"]
# junk mail force check pass list, if no status or status is not pass, will be marked as junk mail
# JUNK_MAIL_FORCE_PASS_LIST = ["spf", "dkim", "dmarc"]
# remove attachment if size exceed 2MB, mail maybe mising some information due to parsing
# REMOVE_EXCEED_SIZE_ATTACHMENT = true
# remove all attachment, mail maybe mising some information due to parsing
# REMOVE_ALL_ATTACHMENT = true
[[d1_databases]] [[d1_databases]]
binding = "DB" binding = "DB"

View File

@@ -15,14 +15,17 @@ hero:
- theme: alt - theme: alt
text: 通过用户界面部署 text: 通过用户界面部署
link: /zh/guide/quick-start link: /zh/guide/quick-start
- theme: alt
text: 通过 Github Actions 部署
link: /zh/guide/quick-start
features: features:
- title: 免费托管在 CloudFlare无需服务器 - title: 仅需域名即可私有部署, 免费托管在 CloudFlare无需服务器
details: Cloudflare D1 数据库Cloudflare Pages 前端Cloudflare Workers 后端, Cloudflare Email Routing details: 支持 password 登录邮箱, 用户注册,使用访问密码可作为私人站点,支持附件功能。
- title: 仅需域名即可私有部署
details: 支持 password 登录邮箱,使用访问密码可作为私人站点,支持附件功能
- title: 使用 rust wasm 解析邮件 - title: 使用 rust wasm 解析邮件
details: 使用 rust wasm 解析邮件支持邮件各种RFC标准支持附件, 速度极快 details: 使用 rust wasm 解析邮件支持邮件各种RFC标准支持附件, 速度极快
- title: 支持 Telegram Bot 和 Webhook
details: 邮件可转发到 Telegram 或者 webhook, Telegram Bot 支持绑定邮箱,查看邮件, Telegram 小程序
- title: 支持发送邮件(UI/API/SMTP) - title: 支持发送邮件(UI/API/SMTP)
details: 支持通过域名邮箱发送 txt 或者 html 邮件,支持 DKIM 签名, UI/API/SMTP 发送邮件 details: 支持通过域名邮箱发送 txt 或者 html 邮件,支持 DKIM 签名, UI/API/SMTP 发送邮件
--- ---

Binary file not shown.

After

Width:  |  Height:  |  Size: 21 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 30 KiB

After

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 20 KiB

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 98 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 94 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 121 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 231 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 46 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 24 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

View File

@@ -0,0 +1,10 @@
# Github Actions 部署如何配置自动更新
::: warning 注意
有问题请通过 `Github Issues` 反馈,感谢。
自动更新不会执行 D1 数据库的 sql 文件,当数据库 schema 变动时,需要手动执行。
:::
1. 打开仓库的 `Actions` 页面,找到 `Upstream Sync`,点击 `enable workflow` 启用 `workflow`
2. 如果 `Upstream Sync` 运行失败,到仓库主页点击 `Sync` 手动同步即可
3. 修改 `Upstream Sync``schedule` 配置可自定义更新间隔,参考 [cron 表达式](https://crontab.guru/)

View File

@@ -0,0 +1,3 @@
# 初始化/更新 D1 数据库
参考 [命令行更新 d1](/zh/guide/cli/d1) 或者 [用户界面更新 d1](/zh/guide/ui/d1)

View File

@@ -0,0 +1,31 @@
# 通过 Github Actions 部署
::: warning 注意
目前只支持 worker 和 pages 的部署。
有问题请通过 `Github Issues` 反馈,感谢。
:::
## 部署步骤
1. 在 GitHub fork 本仓库
2. 打开仓库的 `Actions` 页面,找到 `Deploy Backend Production``Deploy Frontend`,点击 `enable workflow` 启用 `workflow`
3. 然后在仓库页面 `Settings` -> `Secrets and variables` -> `Actions` -> `Repository secrets`, 添加以下 `secrets`:
- `CLOUDFLARE_ACCOUNT_ID`: Cloudflare 账户 ID, [参考文档](https://developers.cloudflare.com/workers/wrangler/ci-cd/#cloudflare-account-id)
- `CLOUDFLARE_API_TOKEN`: Cloudflare API Token, [参考文档](https://developers.cloudflare.com/workers/wrangler/ci-cd/#api-token)
- `BACKEND_TOML`: 后端配置文件,[参考此处](/zh/guide/cli/worker.html#修改-wrangler-toml-配置文件)
- `FRONTEND_ENV`: 前端配置文件,请复制 `frontend/.env.example` 的内容,[并参考此处修改](/zh/guide/cli/pages.html)
- `FRONTEND_NAME`: 你在 Cloudflare Pages 创建的项目名称,可通过 [用户界面](https://temp-mail-docs.awsl.uk/zh/guide/ui/pages.html) 或者 [命令行](https://temp-mail-docs.awsl.uk/zh/guide/cli/pages.html) 创建
- `FRONTEND_BRANCH`: (可选) pages 部署的分支,可不配置,默认 `production`
- `TG_FRONTEND_NAME`: (可选) 你在 Cloudflare Pages 创建的项目名称,同 `FRONTEND_NAME`,如果需要 Telegram Mini App 功能,请填写
- `DEBUG_MODE`: (可选) 是否开启调试模式,配置为 `true` 开启, 默认 worker 部署日志不会输出到 Github Actions 页面,开启后会输出
- `BACKEND_USE_MAIL_WASM_PARSER`: (可选) 是否使用 wasm 解析邮件,配置为 `true` 开启, 功能参考 [配置 worker 使用 wasm 解析邮件](/zh/guide/feature/mail_parser_wasm_worker)
4. 打开仓库的 `Actions` 页面,找到 `Deploy Backend Production``Deploy Frontend`,点击 `Run workflow` 选择分支手动部署
## 如何配置自动更新
1. 打开仓库的 `Actions` 页面,找到 `Upstream Sync`,点击 `enable workflow` 启用 `workflow`
2. 如果 `Upstream Sync` 运行失败,到仓库主页点击 `Sync` 手动同步即可

View File

@@ -9,7 +9,7 @@ cd worker
cp wrangler.toml.template wrangler.toml cp wrangler.toml.template wrangler.toml
# 创建 D1 并执行 schema.sql # 创建 D1 并执行 schema.sql
wrangler d1 create dev wrangler d1 create dev
wrangler d1 execute dev --file=../db/schema.sql wrangler d1 execute dev --file=../db/schema.sql --remote
``` ```
创建完成后,我们在 cloudflare 的控制台可以看到 D1 数据库 创建完成后,我们在 cloudflare 的控制台可以看到 D1 数据库
@@ -25,6 +25,6 @@ wrangler d1 execute dev --file=../db/schema.sql
```bash ```bash
cd worker cd worker
wrangler d1 execute dev --file=../db/2024-01-13-patch.sql wrangler d1 execute dev --file=../db/2024-01-13-patch.sql --remote
wrangler d1 execute dev --file=../db/2024-04-03-patch.sql wrangler d1 execute dev --file=../db/2024-04-03-patch.sql --remote
``` ```

View File

@@ -41,6 +41,7 @@ pnpm run deploy
```bash ```bash
cd frontend cd frontend
pnpm install pnpm install
# 如果你要启用 Cloudflare Zero Trust, 需要使用 pnpm build:pages:nopwa 来禁用缓存
pnpm build:pages pnpm build:pages
cd ../pages cd ../pages
pnpm run deploy pnpm run deploy

View File

@@ -22,15 +22,19 @@ wrangler kv:namespace create DEV
## 修改 `wrangler.toml` 配置文件 ## 修改 `wrangler.toml` 配置文件
> [!NOTE] 注意
> 更多变量的配置请查看 [worker变量说明](/zh/guide/worker-vars)
```toml ```toml
name = "cloudflare_temp_email" name = "cloudflare_temp_email"
main = "src/worker.ts" main = "src/worker.ts"
compatibility_date = "2023-12-01" compatibility_date = "2024-09-23"
compatibility_flags = [ "nodejs_compat" ]
# 如果你想使用自定义域名,你需要添加 routes 配置 # 如果你想使用自定义域名,你需要添加 routes 配置
# routes = [ # routes = [
# { pattern = "temp-email-api.xxxxx.xyz", custom_domain = true }, # { pattern = "temp-email-api.xxxxx.xyz", custom_domain = true },
# ] # ]
node_compat = true
# 如果你想要使用定时任务清理邮件,取消下面的注释,并修改 cron 表达式 # 如果你想要使用定时任务清理邮件,取消下面的注释,并修改 cron 表达式
# [triggers] # [triggers]
@@ -42,50 +46,20 @@ node_compat = true
# ] # ]
[vars] [vars]
# TITLE = "Custom Title" # 自定义网站标题 # 邮箱名称前缀,不需要后缀可配置为空字符串或者不配置
PREFIX = "tmp" # 要处理的邮箱名称前缀,不需要后缀可配置为空字符串 PREFIX = "tmp"
# (min, max) adderss的长度如果不设置默认为(1, 30) # 用于临时邮箱的所有域名, 支持多个域名
# MIN_ADDRESS_LEN = 1 DOMAINS = ["xxx.xxx1" , "xxx.xxx2"]
# MAX_ADDRESS_LEN = 30 # 用于生成 jwt 的密钥, jwt 用于给用户登录以及鉴权
# 如果你想要你的网站私有,取消下面的注释,并修改密码 JWT_SECRET = "xxx"
# PASSWORDS = ["123", "456"]
# admin 控制台密码, 不配置则不允许访问控制台 # admin 控制台密码, 不配置则不允许访问控制台
# ADMIN_PASSWORDS = ["123", "456"] # ADMIN_PASSWORDS = ["123", "456"]
# admin 联系方式,不配置则不显示,可配置任意字符串
# ADMIN_CONTACT = "xx@xx.xxx"
# DEFAULT_DOMAINS = ["xxx.xxx1" , "xxx.xxx2"] # 默认用户可用的域名(未登录或未分配角色的用户)
DOMAINS = ["xxx.xxx1" , "xxx.xxx2"] # 你的域名, 支持多个域名
# 对于中文域名,可以使用 DOMAIN_LABELS 显示域名的中文展示名称
# DOMAIN_LABELS = ["中文.xxx", "xxx.xxx2"]
# 新用户默认角色, 仅在启用邮件验证时有效
# USER_DEFAULT_ROLE = "vip"
# 用户角色配置, 如果 domains 为空将使用 default_domains
# 如果 prefix 为 null 将使用默认前缀, 如果 prefix 为空字符串将不使用前缀
# USER_ROLES = [
# { domains = ["xxx.xxx1" , "xxx.xxx2"], role = "vip", prefix = "vip" },
# { domains = ["xxx.xxx1" , "xxx.xxx2"], role = "admin", prefix = "" },
# ]
JWT_SECRET = "xxx" # 用于生成 jwt 的密钥, jwt 用于给用户登录以及鉴权
BLACK_LIST = "" # 黑名单,用于过滤发件人,逗号分隔
# 是否允许用户创建邮件, 不配置则不允许 # 是否允许用户创建邮件, 不配置则不允许
ENABLE_USER_CREATE_EMAIL = true ENABLE_USER_CREATE_EMAIL = true
# 允许用户删除邮件, 不配置则不允许 # 允许用户删除邮件, 不配置则不允许
ENABLE_USER_DELETE_EMAIL = true ENABLE_USER_DELETE_EMAIL = true
# 允许自动回复邮件
ENABLE_AUTO_REPLY = false
# 是否启用 webhook
# ENABLE_WEBHOOK = true
# 前端界面页脚文本
# COPYRIGHT = "Dream Hunter"
# 默认发送邮件余额,如果不设置,将为 0
# DEFAULT_SEND_BALANCE = 1
# Turnstile 人机验证配置
# CF_TURNSTILE_SITE_KEY = ""
# CF_TURNSTILE_SECRET_KEY = ""
# telegram bot 最多绑定邮箱数量
# TG_MAX_ACCOUNTS = 5
# 全局转发地址列表,如果不配置则不启用,启用后所有邮件都会转发到列表中的地址
# FORWARD_ADDRESS_LIST = ["xxx@xxx.com"]
# D1 数据库的名称和 ID 可以在 cloudflare 控制台查看 # D1 数据库的名称和 ID 可以在 cloudflare 控制台查看
[[d1_databases]] [[d1_databases]]
@@ -105,6 +79,11 @@ database_id = "xxx" # D1 数据库 ID
# namespace_id = "1001" # namespace_id = "1001"
# # 10 requests per minute # # 10 requests per minute
# simple = { limit = 10, period = 60 } # simple = { limit = 10, period = 60 }
# 绑定其他 worker 处理邮件,例如通过 auth-inbox ai 能力解析验证码或激活链接
# [[services]]
# binding = "AUTH_INBOX"
# service = "auth-inbox"
``` ```
## Telegram Bot 配置 ## Telegram Bot 配置

View File

@@ -0,0 +1,41 @@
# 常见问题
> [!NOTE] 注意
> 如果你的问题没有在这里找到解决方案,请到 `Github Issues` 中搜索或者提问, 或者到 Telegram 群组中提问。
## 通用
| 问题 | 解决方案 |
| ---------------------------------------------- | ------------------------------------------------------------------------------- |
| 使用 Cloudflare Workers 给已认证的邮箱发送邮件 | 使用 cf 的 API 进行发送,只支持绑定到 CF 上的收件地址,即 CF EMAIL 转发目的地址 |
| 绑定多个域名 | 每个域名都需要设置 email 转发到 worker |
## worker 相关
| 问题 | 解决方案 |
| ------------------------------------------------------------------ | --------------------------------------------------------------------------- |
| `Uncaught Error: No such module "path". imported from "worker.js"` | [参考](/zh/guide/ui/worker) |
| `No such module "node:stream". imported from "worker.js"` | [参考](/zh/guide/ui/worker) |
| `二级域名无法发送邮件` | [参考](https://github.com/dreamhunter2333/cloudflare_temp_email/issues/515) |
| `Failed to send verify code: No balance` | admin 后台设置无限制邮件或者发件权限页面增加额度 |
| `Github OAuth无法获取到邮箱 400 Failed to get user email` | 需要 github 用户设置公开邮箱 |
| `Cannot read properties of undefined (reading 'map')` | worker 变量没有设置成功 |
## pages 相关
| 问题 | 解决方案 |
| --------------- | ---------------------------------------- |
| `network error` | 使用无痕模式或者清空浏览器缓存DNS 缓存 |
## telegram bot
| 问题 | 解决方案 |
| -------------------------------------------------------------- | -------------------------------------------------- |
| `Telgram Bot获取邮件失败400Bad Request:BUTTON_URL_INVALID` | tg mini app 的 URL 填写错误,需要填写 pages 的 URL |
| `Telegram bot bind error: bind adress count reach the limit` | 需要设置 worker 变量 `TG_MAX_ADDRESS` |
## Github Actions
| 问题 | 解决方案 |
| ------------------------------------------ | --------------------------------------------------------------------------------- |
| Github Action部署后cf里始终是preview分支 | 到 cf pages 页面的设置中确认 前端的分支 和 Github Action 的 前端部署分支 是否相同 |

View File

@@ -11,9 +11,16 @@ admin 后台 账号配置 `已验证地址列表(可通过 cf 内部 api 发送
`API KEYS` 页面创建 `api key` `API KEYS` 页面创建 `api key`
使用 cli 或者直接添加到 `wrangler.toml``vars`,或者在 cloudflare worker 页面的变量中添加 `RESEND_TOKEN` 然后执行下面的命令,将 `RESEND_TOKEN` 添加到 secrets 中
> [!NOTE]
> 如果你觉得麻烦,也可以直接明文放在 `wrangler.toml` 中 `[vars]` 下面,但是不推荐这样做
如果你是通过 UI 部署的,可以在 Cloudflare 的 UI 界面中添加到 `Variables and Secrets` 下面
```bash ```bash
# 切换到 worker 目录
cd worker
wrangler secret put RESEND_TOKEN wrangler secret put RESEND_TOKEN
``` ```

View File

@@ -1,9 +1,9 @@
# Cloudflare Email Routing # Cloudflare Email Routing
1. 配置对应域名的 `电子邮件 DNS 记录`, 如果是多个域名,需要配置多个域名的 `电子邮件 DNS 记录` 1. 在 CF 控制台网页的对应域名的 `Email Routing` 下,配置 `电子邮件 DNS 记录`, 如果是多个域名,需要配置多个域名的 `电子邮件 DNS 记录`
2. 在将电子邮件地址绑定到您的 Worker 之前,您需要启用电子邮件路由并拥有至少一个经过验证的电子邮件地址。 2. 在将电子邮件地址绑定到您的 Worker 之前,您需要启用电子邮件路由并拥有至少一个经过验证的电子邮件地址(目标地址)
3. 配置每个域名的 `Cloudflare Email Routing` catch-all 发送到 `worker` 3. 配置每个域名的 `Email Routing` 的路由规则中的 `Catch-all 地址` 发送到 `worker`
![email](/readme_assets/email.png) ![email](/readme_assets/email.png)

View File

@@ -1,7 +1,5 @@
# Admin 用户相关 # Admin 用户相关
默认不允许用户注册,可通过
## 用户管理页面 ## 用户管理页面
![admin-user-management](/feature/admin-user-management.png) ![admin-user-management](/feature/admin-user-management.png)

View File

@@ -1,7 +1,15 @@
# Admin 控制台 # Admin 控制台
部署前端应用之后,访问 `/admin` 路径即可进入管理控制台。 > [!NOTE]
> 需要配置 `ADMIN_PASSWORDS` 或者 `ADMIN_USER_ROLE` 才可以访问 admin 控制台
> admin 角色配置, 如果用户角色等于 ADMIN_USER_ROLE 则可以访问 admin 控制台
需要在后端配置 `admin 控制台密码`, 不配置则不允许访问控制台。 部署前端应用之后,点击 左上角 logo 5 次 或者访问 `/admin` 路径即可进入管理控制台。
需要在后端配置 `ADMIN_PASSWORDS` 或者当前用户角色为 `ADMIN_USER_ROLE`, 则不允许访问控制台。
![admin](/feature/admin.png) ![admin](/feature/admin.png)
## 如果你的网站只可私人访问,可通过此禁用检查
`DISABLE_ADMIN_PASSWORD_CHECK = true`

View File

@@ -0,0 +1,144 @@
# 通过其他 worker 增强
> 临时邮箱的核心能力在邮件的管理,通过其他 worker 可以增强临时邮箱的功能,例如通过 auth-inbox ai 能力解析验证码或激活链接
> 该功能仅触发其他 worker ,在 webhook 后执行
> [!NOTE]
> 如果要使用 worker 增强,请提前创建可以 rpc 调用的 worker具体下文详述
> 参考:
> - https://developers.cloudflare.com/workers/runtime-apis/bindings/service-bindings/rpc/
> - https://developers.cloudflare.com/workers/runtime-apis/rpc/
> - auth-inbox 项目https://github.com/TooonyChen/AuthInbox
## 创建其他 worker以 auth-inbox 项目ai解析验证码为例子
### worker 改造为继承 WorkerEntrypoint
一个简单,作为被调用方,提供 rpc 方法调用的worker代码如下rpcEmail 方法为样例)
(使用已经修改好的项目 https://github.com/oneisall8955/AuthInbox-fork
src/index.ts 文件
```js
import { WorkerEntrypoint } from "cloudflare:workers";
interface Env {
DB: D1Database;
// ...
}
export default class extends WorkerEntrypoint<Env> {
async fetch(request: Request): Promise<Response> {
console.log("原本fetch接口入参是request,env,ctx");
console.log("修改为WorkerEntrypoint风格后只有一个入参request获取环境变量和上下文有小改动");
// 环境变量及上下文改动详见:
// https://developers.cloudflare.com/workers/runtime-apis/bindings/service-bindings/rpc/#bindings-env
// https://developers.cloudflare.com/workers/runtime-apis/bindings/service-bindings/rpc/#lifecycle-methods-ctx
const env: Env = this.env;
const ctx: ExecutionContext = this.ctx;
console.log("后续逻辑不变");
return new Response('ok', { status: 200 });
}
// 主要功能
async email(message: ForwardableEmailMessage): Promise<void> {
console.log("原本fetch接口入参是message,env,ctx");
console.log("修改为WorkerEntrypoint风格后只有一个入参message获取环境变量和上下文和fetch方法一样");
const env: Env = this.env;
const ctx: ExecutionContext = this.ctx;
console.log("接受email routing请求后后续逻辑不变");
}
// 暴露rpc接口处理来自其他worker的邮件请求
async rpcEmail(requestBody: string): Promise<void> {
console.log(`接受其他worker临时邮件服务cloudflare_temp_email的请求request body: ${requestBody}`);
// requestBody json 格式,由临时邮件服务发送,格式如下
// type RPCEmailMessage = {
// from: string | undefined | null,
// to: string | undefined | null,
// rawEmail: string | undefined | null,
// headers: Map<string, string>,
// }
// ... todo ...
}
}
```
### 部署其他 worker
修改好或者使用 以auth-inbox 为例,部署到 cloudflare worker 上,详见 https://github.com/TooonyChen/AuthInbox ,或者使用已经修改好的项目 https://github.com/oneisall8955/AuthInbox-fork
## 配置临时邮件服务,使用指定其他 worker 增强
## 绑定服务
### 通过 wrangler.toml 配置
```toml
[[services]]
binding = "AUTH_INBOX"
service = "auth-inbox"
```
这里的 `binding = "AUTH_INBOX"` 可以自定义,可以是任何字符串,`service = "auth-inbox"` 是部署好的提供rpc接口调用的worker名称。
### 用户界面配置
在设置-绑定,添加绑定,选择绑定服务。
变量名称填写自定义的名称,可以任意字符串 ,例如 `AUTH_INBOX`
服务绑定选择上一步创建好的服务,例如 `auth-inbox`
![another-worker-enhanced-01.png](/feature/another-worker-enhanced-01.png)
![another-worker-enhanced-02.png](/feature/another-worker-enhanced-02.png)
## 环境变量配置
### 通过 wrangler.toml 配置
```toml
ENABLE_ANOTHER_WORKER = true
ANOTHER_WORKER_LIST ="""
[
{
"binding":"AUTH_INBOX",
"method":"rpcEmail",
"keywords":[
"","","","","","","","","","","","","","","","","","",
"account","activation","verify","verification","activate","confirmation","email","code","validate","registration","login","code","expire","confirm"
]
}
]
"""
```
环境变量解释:
- ENABLE_ANOTHER_WORKER = true默认为falsetrue则开启其他 worker 处理邮件
- ANOTHER_WORKER_LIST 是一个JOSN数组每个对象包3个字段
- binding: *必填必须与services部分指定的 binding = "XXX" 保持一致*,例子中为 AUTH_INBOX
- method: 可选,默认 rpcEmail指的是调用这个 worker 的哪一个 rpc 方法处理
- keywords: 关键词数组,忽略大小写。用于过滤,如果*解析后邮件文本*匹配到这些关键词,触发这个 worker并且调用这个 worker 的 `method` 方法
### 用户界面配置
在设置-环境变量,添加环境变量
- ENABLE_ANOTHER_WORKER = true
- ANOTHER_WORKER_LIST 为上面提及的JSON数组字符串不再复述详细介绍看上文
```json
[
{
"binding":"AUTH_INBOX",
"method":"rpcEmail",
"keywords":[
"验证码","激活码","激活链接","确认链接","验证邮箱","确认邮件","账号激活","邮件验证","账户确认","安全码","认证码","安全验证","登陆码","确认码","启用账户","激活账户","账号验证","注册确认",
"account","activation","verify","verification","activate","confirmation","email","code","validate","registration","login","code","expire","confirm"
]
}
]
```
![another-worker-enhanced-03.png](/feature/another-worker-enhanced-03.png)
## 测试
发送一个邮件到临时邮箱观察worker日志到或者到 auth-inbox 提供的面板上查看验证码
![another-worker-enhanced-04.png](/feature/another-worker-enhanced-04.png)

View File

@@ -32,6 +32,7 @@ export const commonParseMail = async (raw_mail: string | undefined | null): Prom
sender: parsedEmail.sender || "", sender: parsedEmail.sender || "",
subject: parsedEmail.subject || "", subject: parsedEmail.subject || "",
text: parsedEmail.text || "", text: parsedEmail.text || "",
headers: parsedEmail.headers || [],
html: parsedEmail.body_html || "", html: parsedEmail.body_html || "",
}; };
} catch (e) { } catch (e) {

View File

@@ -23,3 +23,70 @@ res = requests.post(
# 返回值 {"jwt": "<Jwt>"} # 返回值 {"jwt": "<Jwt>"}
print(res.json()) print(res.json())
``` ```
# 批量创建随机用户名邮箱地址 API 示例
## 通过 admin API 批量新建邮箱地址
这是一个 `python` 的例子,使用 `requests` 库发送邮件。
```python
import requests
import random
import string
from concurrent.futures import ThreadPoolExecutor, as_completed
def generate_random_name():
# 生成5位英文字符
letters1 = ''.join(random.choices(string.ascii_lowercase, k=5))
# 生成1-3个数字
numbers = ''.join(random.choices(string.digits, k=random.randint(1, 3)))
# 生成1-3个英文字符
letters2 = ''.join(random.choices(string.ascii_lowercase, k=random.randint(1, 3)))
# 组合成最终名称
return letters1 + numbers + letters2
def fetch_email_data(name):
try:
res = requests.post(
"https://<worker 域名>",
json={
"enablePrefix": True,
"name": name,
"domain": "<邮箱域名>",
},
headers={
'x-admin-auth': "<你的网站admin密码>",
"Content-Type": "application/json"
}
)
if res.status_code == 200:
response_data = res.json()
email = response_data.get("address", "无地址")
jwt = response_data.get("jwt", "无jwt")
return f"{email}----{jwt}\n"
else:
print(f"请求失败,状态码: {res.status_code}")
return None
except requests.RequestException as e:
print(f"请求出现错误: {e}")
return None
def generate_and_save_emails(num_emails):
with ThreadPoolExecutor(max_workers=30) as executor, open('email.txt', 'a') as file:
futures = [executor.submit(fetch_email_data, generate_random_name()) for _ in range(num_emails)]
for future in as_completed(futures):
result = future.result()
if result:
file.write(result)
# 生成10个邮箱并追加到现有文件
generate_and_save_emails(10)
```

View File

@@ -1,7 +1,9 @@
# 配置子域名邮箱 # 配置子域名邮箱
::: warning ::: warning 注意
子域名邮箱发送邮件可能无法发送邮件,建议使用主域名邮箱发送邮件,子域名邮箱仅用于接收邮件。 子域名邮箱发送邮件可能无法发送邮件,建议使用主域名邮箱发送邮件,子域名邮箱仅用于接收邮件。
mail channel 已不被支持,下面参考中仅限收件部分。
::: :::
参考 参考

View File

@@ -1,15 +1,30 @@
# 配置 Telegram Bot # 配置 Telegram Bot
试用地址:[@cf_temp_mail_bot](https://t.me/cf_temp_mail_bot)
::: warning 注意
worker 默认的 `worker.dev` 域名的证书是不被 telegram 支持的,配置 Telegram Bot 请使用自定义域名
:::
> [!NOTE]
> 如果要使用 Telegram Bot, 请先绑定 `KV`
>
> 如果不需要 Telegram Bot, 可跳过此步骤
>
> 如果你想 Telegram 的解析邮件能力更强,参考 [配置 worker 使用 wasm 解析邮件](/zh/guide/feature/mail_parser_wasm_worker)
## Telegram Bot 配置 ## Telegram Bot 配置
> [!NOTE]
> 如果不需要 Telegram Bot, 可跳过此步骤
请先创建一个 Telegram Bot然后获取 `token`,然后执行下面的命令,将 `token` 添加到 secrets 中 请先创建一个 Telegram Bot然后获取 `token`,然后执行下面的命令,将 `token` 添加到 secrets 中
你也可以在 Cloudflare 的 UI 界面中添加 `secrets` > [!NOTE]
> 如果你觉得麻烦,也可以直接明文放在 `wrangler.toml` 中 `[vars]` 下面,但是不推荐这样做
如果你是通过 UI 部署的,可以在 Cloudflare 的 UI 界面中添加到 `Variables and Secrets` 下面
```bash ```bash
# 切换到 worker 目录
cd worker
pnpm wrangler secret put TELEGRAM_BOT_TOKEN pnpm wrangler secret put TELEGRAM_BOT_TOKEN
``` ```
@@ -23,6 +38,21 @@ pnpm wrangler secret put TELEGRAM_BOT_TOKEN
## Mini App ## Mini App
可以通过命令行部署,或者 UI 界面部署
### UI 部署
其他步骤参考 [UI 部署](/zh/guide/cli/pages) 中的 `前后端分离部署`
> [!NOTE]
> 从这里下载 zip, [telegram-frontend.zip](https://github.com/dreamhunter2333/cloudflare_temp_email/releases/latest/download/telegram-frontend.zip)
>
> 修改压缩包里面的 index-xxx.js 文件 xx 是随机的字符串
>
> 搜索 `https://temp-email-api.xxx.xxx` 替换成你worker 的域名然后部署新的zip文件
### 命令行部署
```bash ```bash
cd frontend cd frontend
pnpm install pnpm install
@@ -31,8 +61,6 @@ cp .env.example .env.prod
pnpm run deploy:telegram --project-name=<你的项目名称> pnpm run deploy:telegram --project-name=<你的项目名称>
``` ```
部署完成后,请在 admin 后台的 `设置` -> `电报小程序` 页面 `电报小程序 URL` - 部署完成后,请在 admin 后台的 `设置` -> `电报小程序` 页面 `电报小程序 URL` 中填写网页 URL
- 请在 `@BotFather` 处执行 `/setmenubutton`,然后输入你的网页地址,设置左下角的 `Open App` 按钮。
请在 `@BotFather` 处执行 `/setmenubutton`,然后输入你的网页地址,设置左下角的 `Open App` 按钮。 - 请在 `@BotFather` 处执行 `/newapp` 新建 app 来注册 mini app。
你也可以在 `@BotFather` 处执行 `/newapp` 新建 app 来获得 mini app 的链接

View File

@@ -0,0 +1,26 @@
# OAuth2 第三方登录
> [!WARNING]
> 第三方登录会自动使用用户邮箱注册账号(邮箱相同将视为同一账号)
>
> 此账号和注册的账号相同, 也可以通过忘记密码设置密码
## 在第三方平台注册 OAuth2
### GitHub
- 请先创建一个 OAuth App然后获取 `Client ID``Client Secret`
参考 [Creating an OAuth App](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/creating-an-oauth-app)
### Authentik
- [Authentik OAuth2 Provider](https://docs.goauthentik.io/docs/providers/oauth2/)
## Admin 后台配置 OAuth2
![oauth2](/feature/oauth2.png)
## 测试用户登录页面
![oauth2 login](/feature/oauth2-login.png)

View File

@@ -0,0 +1,44 @@
# 配置 webhook
> [!NOTE]
> 如果要使用 webhook请先绑定 `KV` 并且 `worker` 变量配置 `ENABLE_WEBHOOK = true`
>
> 如果你想 webhook 的解析邮件能力更强,参考 [配置 worker 使用 wasm 解析邮件](/zh/guide/feature/mail_parser_wasm_worker)
## 前提条件
你需要自建一个 `webhook 服务` 或者 使用 `第三方平台`,这个服务需要能够接收 `POST` 请求,并且能够解析 `json` 数据。
本项目使用了 [songquanpeng/message-pusher](https://github.com/songquanpeng/message-pusher) 示例作为 webhook 服务。
- 可以使用 [msgpusher.com](https://msgpusher.com) 提供的服务
- 也可以自建 `message-pusher` 服务,参考 [songquanpeng/message-pusher](https://github.com/songquanpeng/message-pusher)
## admin 配置全局 webhook
![telegram](/feature/admin-mail-webhook.png)
## admin 允许邮箱使用 webhook
![telegram](/feature/admin-webhook-settings.png)
## 某个邮箱配置 webhook
![telegram](/feature/address-webhook.png)
## webhook 数据格式
要获取 url 需要配置 worker 的 `FRONTEND_URL` 为你的前端地址,或者你可以通过 `id` 自己拼接 url = `${FRONTEND_URL}?mail_id=${id}`
```json
{
"id": "${id}",
"url": "${url}",
"from": "${from}",
"to": "${to}",
"subject": "${subject}",
"raw": "${raw}",
"parsedText": "${parsedText}",
"parsedHtml": "${parsedHtml}",
}
```

View File

@@ -1,23 +0,0 @@
# 通过 Github Actions 部署
::: warning
有问题请通过 `Github Issues` 反馈,感谢。
:::
[![Deploy to Cloudflare Workers](https://deploy.workers.cloudflare.com/button)](https://deploy.workers.cloudflare.com/?url=https://github.com/dreamhunter2333/cloudflare_temp_email)
1. 点击按钮 fork 本仓库 或者直接 fork 本仓库
2. 打开仓库的 `Actions` 页面,找到 `Deploy Backend Production``Deploy Frontend`,点击 `enable workflow` 启用 `workflow`
3. 然后在仓库页面 `Settings` -> `Secrets and variables` -> `Actions` -> `Repository secrets`, 添加以下 `secrets`:
- `CLOUDFLARE_ACCOUNT_ID`: Cloudflare 账户 ID, [参考文档](https://developers.cloudflare.com/workers/wrangler/ci-cd/#cloudflare-account-id)
- `CLOUDFLARE_API_TOKEN`: Cloudflare API Token, [参考文档](https://developers.cloudflare.com/workers/wrangler/ci-cd/#api-token)
- `BACKEND_TOML`: 后端配置文件,[参考此处](/zh/guide/cli/worker.html#修改-wrangler-toml-配置文件)
- `FRONTEND_ENV`: 前端配置文件,请复制 `frontend/.env.example` 的内容,[并参考此处修改](/zh/guide/cli/pages.html)
- `FRONTEND_NAME`: 你在 Cloudflare Pages 创建的项目名称,可通过 [用户界面](https://temp-mail-docs.awsl.uk/zh/guide/ui/pages.html) 或者 [命令行](https://temp-mail-docs.awsl.uk/zh/guide/cli/pages.html) 创建
- `FRONTEND_BRANCH`: (可选) pages 部署的分支,可不配置,默认 `production`
- `TG_FRONTEND_NAME`: (可选) 你在 Cloudflare Pages 创建的项目名称,同 `FRONTEND_NAME`,如果需要 Telegram Mini App 功能,请填写
1. 打开仓库的 `Actions` 页面,找到 `Deploy Backend Production``Deploy Frontend`,点击 `Run workflow` 选择分支手动部署

Some files were not shown because too many files have changed in this diff Show More