Compare commits

..

117 Commits

Author SHA1 Message Date
xiaojunnuo
c725cee044 v1.37.1 2025-09-29 20:35:30 +08:00
xiaojunnuo
367ef4ecb2 build: prepare to build 2025-09-29 20:32:54 +08:00
xiaojunnuo
c3a64facd5 chore 2025-09-29 20:32:31 +08:00
xiaojunnuo
2671781e1b fix: 修复某些情况下cname申请证书报错主域名不一致的bug 2025-09-29 18:58:19 +08:00
xiaojunnuo
9291fa68aa perf: dns解析支持阿里esa 2025-09-28 23:29:56 +08:00
xiaojunnuo
6ebb3659f4 perf: cname主域名校验提示优化,显示不一致的两方便于排查问题 2025-09-28 16:18:39 +08:00
xiaojunnuo
109696e965 fix: 修复版本比较bug 2025-09-28 12:49:53 +08:00
xiaojunnuo
b86bbd370c chore: deploy 1.37.0 2025-09-28 12:44:02 +08:00
xiaojunnuo
1575a4fb1a chore: deploy 1.37.0 2025-09-28 12:42:54 +08:00
xiaojunnuo
e2f500be90 chore: deploy 1.37.0 2025-09-28 12:37:22 +08:00
xiaojunnuo
284b00a826 chore: 2025-09-28 12:21:42 +08:00
xiaojunnuo
66180e19b5 build: publish 2025-09-28 12:18:46 +08:00
xiaojunnuo
1531462d22 build: trigger build image 2025-09-28 12:18:30 +08:00
xiaojunnuo
e17cd1f298 v1.37.0 2025-09-28 12:17:05 +08:00
xiaojunnuo
13092e9f80 build: prepare to build 2025-09-28 12:14:35 +08:00
xiaojunnuo
8133b8b9dd build: prepare to build 2025-09-28 12:07:54 +08:00
xiaojunnuo
e25aafac6d chore: 2025-09-28 12:07:51 +08:00
xiaojunnuo
037c7beb1b build: prepare to build 2025-09-28 12:06:22 +08:00
xiaojunnuo
faac4dfc30 chore: 2025-09-28 12:06:13 +08:00
xiaojunnuo
469a088a4d build: prepare to build 2025-09-28 11:30:24 +08:00
xiaojunnuo
9c854f727f chore: 2025-09-28 11:29:57 +08:00
xiaojunnuo
8f6e5bd24b feat: dist打包前检查 2025-09-28 11:02:25 +08:00
xiaojunnuo
992f91cf4c Merge branch 'refs/heads/v2' into v2-dev 2025-09-28 10:42:17 +08:00
xiaojunnuo
0c61d4c978 feat: @certd/ui-server module import报错的问题 2025-09-28 09:58:22 +08:00
xiaojunnuo
72d32edf9a build: publish 2025-09-27 09:41:54 +08:00
xiaojunnuo
dde39def9e build: trigger build image 2025-09-27 09:41:37 +08:00
xiaojunnuo
6b43007c44 v1.36.25 2025-09-27 09:40:13 +08:00
xiaojunnuo
876558cf77 build: prepare to build 2025-09-27 09:38:14 +08:00
xiaojunnuo
b35a146edf chore: 2025-09-27 09:37:30 +08:00
xiaojunnuo
86cf6a9908 chore: 2025-09-27 09:30:22 +08:00
xiaojunnuo
b0f7288ac0 build: prepare to build 2025-09-27 09:28:26 +08:00
xiaojunnuo
32fcc1a8fb chore: 2025-09-27 09:27:21 +08:00
xiaojunnuo
eb4d125eaf fix: 固定midwayjs版本,修复ui-server import 错误的bug 2025-09-27 09:26:17 +08:00
xiaojunnuo
87e5cced3c chore: 2025-09-27 09:03:29 +08:00
xiaojunnuo
bcd9ee2d48 build: publish 2025-09-27 08:36:18 +08:00
xiaojunnuo
edf3d87458 build: trigger build image 2025-09-27 08:36:02 +08:00
xiaojunnuo
0c0c353ecc v1.36.24 2025-09-27 08:34:35 +08:00
xiaojunnuo
aaa4c8f899 build: prepare to build 2025-09-27 08:32:23 +08:00
xiaojunnuo
57e3565c11 chore: 2025-09-27 08:32:17 +08:00
xiaojunnuo
fbcf72d762 build: prepare to build 2025-09-27 08:31:29 +08:00
xiaojunnuo
ca8daa836e Merge remote-tracking branch 'origin/v2-dev' into v2-dev 2025-09-27 08:29:44 +08:00
xiaojunnuo
c2ccdbec9d fix: 修复 ui-server 加载失败问题 2025-09-27 08:29:22 +08:00
xiaojunnuo
69aee36e75 chore: 修复 ui-server 加载失败问题 2025-09-27 08:29:16 +08:00
xiaojunnuo
063f5c3b55 fix: 修复 ui-server 加载失败问题 2025-09-27 08:26:18 +08:00
xiaojunnuo
6d1b8ca65e chore: fix ui-server 加载失败问题 2025-09-27 08:24:39 +08:00
xiaojunnuo
03899d4d9c perf: 重置管理员密码同时会关闭验证码,防止验证码失效之后无法登录 2025-09-27 01:47:53 +08:00
xiaojunnuo
2b84af977d build: publish 2025-09-27 01:45:35 +08:00
xiaojunnuo
e15b180322 build: trigger build image 2025-09-27 01:45:19 +08:00
xiaojunnuo
f070030f6b v1.36.23 2025-09-27 01:43:28 +08:00
xiaojunnuo
330ac66b38 build: prepare to build 2025-09-27 01:39:53 +08:00
xiaojunnuo
12a9e650af chore: sql 2025-09-27 01:39:30 +08:00
xiaojunnuo
1e5ccd811e build: prepare to build 2025-09-27 01:35:54 +08:00
xiaojunnuo
2902ee6ad5 chore: sql 2025-09-27 01:35:39 +08:00
xiaojunnuo
90ce4fec2c chore: sql 2025-09-27 01:24:34 +08:00
xiaojunnuo
a7ab26d08d chore: 2025-09-27 01:19:32 +08:00
xiaojunnuo
dcc396afb7 perf: 动态加载验证码script 2025-09-27 00:43:20 +08:00
xiaojunnuo
3f1722d54d fix: 授权页面,id列位置不在第一列的bug 2025-09-27 00:17:29 +08:00
xiaojunnuo
c79658afbb chore: 2025-09-26 01:32:26 +08:00
xiaojunnuo
6f84ebb323 chore: 2025-09-26 01:27:55 +08:00
xiaojunnuo
54c8d62243 perf: 开启子域名托管之后cname记录支持重置 2025-09-26 01:21:24 +08:00
xiaojunnuo
83e6476408 perf: 验证码支持测试,登录验证码需要测试通过后才能开启 2025-09-26 01:21:01 +08:00
xiaojunnuo
03f317ffdb perf: 支持腾讯云验证码 2025-09-26 01:20:25 +08:00
xiaojunnuo
3f67c7c74a docs: 增加子域名托管下的cname记录问题说明 2025-09-25 22:38:13 +08:00
xiaojunnuo
b8b4660563 chore: 1 2025-09-24 23:55:43 +08:00
xiaojunnuo
3d42bfd479 perf: 手动上传证书优化,增加到期前报错提醒 2025-09-24 14:14:19 +08:00
xiaojunnuo
2ae193092d build: publish 2025-09-24 01:50:34 +08:00
xiaojunnuo
7e1d52ff00 build: trigger build image 2025-09-24 01:50:19 +08:00
xiaojunnuo
c98f43b984 v1.36.22 2025-09-24 01:48:55 +08:00
xiaojunnuo
e93f128a7a build: prepare to build 2025-09-24 01:46:48 +08:00
xiaojunnuo
71d8e7edd2 perf: 优化连接失败的报错提示 2025-09-24 01:40:11 +08:00
xiaojunnuo
48f4298a8d chore: 新网已支持 2025-09-24 00:55:31 +08:00
xiaojunnuo
1c15beadc7 perf: 登录失败时清除验证码状态 2025-09-24 00:06:00 +08:00
xiaojunnuo
2c1600ddfb chore: 新网dns完善 2025-09-23 23:27:36 +08:00
xiaojunnuo
298f7d9d52 chore: 新网dns完善 2025-09-23 23:24:36 +08:00
xiaojunnuo
105f0bfde2 chore: 2025-09-23 00:56:08 +08:00
xiaojunnuo
cf3a78e114 perf: dns支持新网域名解析 2025-09-22 23:30:28 +08:00
xiaojunnuo
9cc5f0f889 perf: 公共cname支持权限校验 2025-09-22 23:29:55 +08:00
xiaojunnuo
e30db9ee77 docs: 2025-09-21 17:33:44 +08:00
xiaojunnuo
235be757f8 Merge branch 'v2' into v2-dev 2025-09-19 18:03:15 +08:00
Zero Clover
e31d26a887 perf: add preferred chain for google trust service (#539) @ZeroClover 2025-09-19 17:36:29 +08:00
xiaojunnuo
2293ba02ea docs: 宝塔动态IP白名单 2025-09-19 15:31:59 +08:00
xiaojunnuo
7188997dd1 perf: 7001绑定::地址 2025-09-18 10:05:07 +08:00
xiaojunnuo
31cfb09468 fix: 选择授权对话框编辑时,名称字段排在最后的bug 2025-09-17 16:15:31 +08:00
xiaojunnuo
b76f2e2008 fix: 修复旧版本升级上来报错eab授权的bug 2025-09-17 13:25:08 +08:00
xiaojunnuo
4b90972341 perf: gcore flush plugin ssl_id改为必填项 2025-09-16 10:26:16 +08:00
xiaojunnuo
f4ff34224c Merge remote-tracking branch 'origin/v2-dev' into v2-dev 2025-09-16 09:31:54 +08:00
xiaojunnuo
877c9c4ff9 perf: 增加自签名证书提示 2025-09-16 09:31:02 +08:00
xiaojunnuo
ac0b7291dd build: publish 2025-09-15 21:10:22 +08:00
xiaojunnuo
491ef6085a build: trigger build image 2025-09-15 21:09:51 +08:00
xiaojunnuo
3cedef4974 v1.36.21 2025-09-15 21:08:26 +08:00
xiaojunnuo
22ab04bd2b build: prepare to build 2025-09-15 20:59:51 +08:00
xiaojunnuo
e5a080aebe fix: 修复导入插件对话框无法打开的bug,修复插件编辑页面打开多个代码编辑器消失的bug 2025-09-15 18:03:55 +08:00
xiaojunnuo
c560cc5add fix: 修复ssl.com报EMAILADDRESS数量不对的bug 2025-09-14 23:01:18 +08:00
xiaojunnuo
0d27bc323b chore: build new version 2025-09-14 02:29:41 +08:00
xiaojunnuo
c71d3cef18 chore: 升级fast-crud 2025-09-14 02:29:22 +08:00
xiaojunnuo
4e2d8daa3a chore: 2025-09-14 02:16:55 +08:00
xiaojunnuo
d0f51da0af chore: 2025-09-14 01:51:16 +08:00
xiaojunnuo
aeb73bca27 chore: 2025-09-14 01:40:55 +08:00
xiaojunnuo
f239b03291 chore: 2025-09-14 01:39:52 +08:00
xiaojunnuo
297c2965f4 Merge branch 'v2-dev' into v2 2025-09-14 01:04:20 +08:00
xiaojunnuo
daddf4d98e build: publish 2025-09-14 01:01:39 +08:00
xiaojunnuo
e05f9bfebf build: trigger build image 2025-09-14 01:01:18 +08:00
xiaojunnuo
ef46aeae6f v1.36.20 2025-09-14 00:59:40 +08:00
xiaojunnuo
7edb3fd856 build: prepare to build 2025-09-14 00:56:22 +08:00
xiaojunnuo
43b79778ea fix: 修复授权类型和名称字段排到最后的bug 2025-09-14 00:47:30 +08:00
xiaojunnuo
37f1f53b56 chore: 数据库同步 2025-09-14 00:40:38 +08:00
xiaojunnuo
67bd1cdcd9 chore: 2025-09-14 00:22:17 +08:00
xiaojunnuo
506385e5a2 fix: 修复证书手动托管时新上传的证书无效的bug 2025-09-13 23:59:16 +08:00
ahe
2d4586b1c4 perf: 证书到期剩余天数进度条根据实际证书有效期计算 (#528) nicheng-he
* Create FUNDING.yml

* Update FUNDING.yml

* Update README.md

* Update README.md

* Update README.md

* Update README.md

* Update README_en.md

* 证书到期剩余天数进度条根据实际证书时间计算

---------

Co-authored-by: greper <xiaojunnuo@qq.com>
2025-09-13 23:40:06 +08:00
xiaojunnuo
1476b9cb9c chore: 设置tab页签标题中英文优化 2025-09-13 23:33:18 +08:00
greper
768bdc2cc2 Update README_en.md 2025-09-08 22:27:59 +08:00
greper
a0a093e260 Update README.md 2025-09-08 22:27:28 +08:00
greper
0b2a7fdc15 Update README.md 2025-09-08 22:27:01 +08:00
greper
f1876e20f8 Update README.md 2025-09-08 22:25:52 +08:00
greper
7d6a6e53f7 Update README.md 2025-09-08 22:23:43 +08:00
greper
6b765a1f77 Update FUNDING.yml 2025-09-08 22:14:57 +08:00
greper
3b3c93dd53 Create FUNDING.yml 2025-09-08 22:06:57 +08:00
184 changed files with 3383 additions and 924 deletions

5
.github/FUNDING.yml vendored Normal file
View File

@@ -0,0 +1,5 @@
# These are supported funding model platforms
github: greper
buy_me_a_coffee: greper
custom: ['https://afdian.com/a/greper']

View File

@@ -3,7 +3,7 @@ on:
push: push:
branches: ['v2-dev'] branches: ['v2-dev']
paths: paths:
- "build.trigger" - "trigger/build.trigger"
# schedule: # schedule:
# - # 国际时间 19:17 执行北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间 # - # 国际时间 19:17 执行北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
@@ -68,7 +68,7 @@ jobs:
registry: ghcr.io registry: ghcr.io
username: ${{ github.actor }} username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }} password: ${{ secrets.GITHUB_TOKEN }}
- name: Login to Docker Hub - name: Login to Docker Hub
uses: docker/login-action@v3 uses: docker/login-action@v3
with: with:
@@ -82,34 +82,4 @@ jobs:
push: true push: true
context: ./packages/ui/ context: ./packages/ui/
tags: | tags: |
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}} registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}
greper/certd:latest
greper/certd:${{steps.get_certd_version.outputs.result}}
ghcr.io/${{ github.repository }}:latest
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}
- name: Build armv7
uses: docker/build-push-action@v6
with:
platforms: linux/arm/v7
push: true
context: ./packages/ui/
tags: |
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}-armv7
greper/certd:armv7
greper/certd:${{steps.get_certd_version.outputs.result}}-armv7
ghcr.io/${{ github.repository }}:armv7
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}-armv7
# - name: Build agent
# uses: docker/build-push-action@v6
# with:
# platforms: linux/amd64,linux/arm64
# push: true
# context: ./packages/ui/agent/
# tags: |
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:latest
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:${{steps.get_certd_version.outputs.result}}
# greper/certd-agent:latest
# greper/certd-agent:${{steps.get_certd_version.outputs.result}}

View File

@@ -3,12 +3,13 @@ on:
push: push:
branches: ['v2-dev'] branches: ['v2-dev']
paths: paths:
- "deploy.trigger" - "trigger/deploy.trigger"
workflow_run: workflow_run:
workflows: [ "build-image" ] workflows: [ "build-image" ]
types: types:
- completed - completed
# schedule: # schedule:
# - # 国际时间 19:17 执行北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间 # - # 国际时间 19:17 执行北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
# - cron: '17 19 * * *' # - cron: '17 19 * * *'
@@ -54,14 +55,3 @@ jobs:
retry-count: 3 retry-count: 3
retry-delay: 5000 retry-delay: 5000
- name: deploy-certd-doc
uses: tyrrrz/action-http-request@master
with:
url: http://flow-openapi.aliyun.com/pipeline/webhook/IiSxLDp9aOhgDUxJPytv
method: POST
body: |
{}
headers: |
Content-Type: application/json
retry-count: 3
retry-delay: 5000

View File

@@ -1,9 +1,13 @@
name: build-image-for-test name: build-image-for-release
on: on:
push: push:
branches: ['v2-dev'] branches: ['v2-dev']
paths: paths:
- "build-dev.trigger" - "trigger/release.trigger"
# workflow_run:
# workflows: [ "deploy-demo" ]
# types:
# - completed
# schedule: # schedule:
# - # 国际时间 19:17 执行北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间 # - # 国际时间 19:17 执行北京时间3:17 ↙↙↙ 改成你想要每天自动执行的时间
@@ -20,7 +24,7 @@ jobs:
uses: actions/checkout@v4 uses: actions/checkout@v4
with: with:
fetch-depth: 0 fetch-depth: 0
ref: v2-dev lfs: true
- name: get_certd_version - name: get_certd_version
id: get_certd_version id: get_certd_version
@@ -75,17 +79,19 @@ jobs:
username: ${{ secrets.dockerhub_username }} username: ${{ secrets.dockerhub_username }}
password: ${{ secrets.dockerhub_password }} password: ${{ secrets.dockerhub_password }}
# - name: Build default platforms - name: Build default platforms
# uses: docker/build-push-action@v6 uses: docker/build-push-action@v6
# with: with:
# platforms: linux/amd64,linux/arm64 platforms: linux/amd64,linux/arm64
# push: true push: true
# context: ./packages/ui/ context: ./packages/ui/
# tags: | tags: |
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-dev:latest registry.cn-shenzhen.aliyuncs.com/handsfree/certd:latest
# greper/certd-dev:latest registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}
# ghcr.io/${{ github.repository }}:dev-latest greper/certd:latest
greper/certd:${{steps.get_certd_version.outputs.result}}
ghcr.io/${{ github.repository }}:latest
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}
- name: Build armv7 - name: Build armv7
uses: docker/build-push-action@v6 uses: docker/build-push-action@v6
with: with:
@@ -96,4 +102,30 @@ jobs:
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7 registry.cn-shenzhen.aliyuncs.com/handsfree/certd:armv7
registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}-armv7 registry.cn-shenzhen.aliyuncs.com/handsfree/certd:${{steps.get_certd_version.outputs.result}}-armv7
greper/certd:armv7 greper/certd:armv7
greper/certd:${{steps.get_certd_version.outputs.result}}-armv7 greper/certd:${{steps.get_certd_version.outputs.result}}-armv7
ghcr.io/${{ github.repository }}:armv7
ghcr.io/${{ github.repository }}:${{steps.get_certd_version.outputs.result}}-armv7
# - name: Build agent
# uses: docker/build-push-action@v6
# with:
# platforms: linux/amd64,linux/arm64
# push: true
# context: ./packages/ui/agent/
# tags: |
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:latest
# registry.cn-shenzhen.aliyuncs.com/handsfree/certd-agent:${{steps.get_certd_version.outputs.result}}
# greper/certd-agent:latest
# greper/certd-agent:${{steps.get_certd_version.outputs.result}}
- name: deploy-certd-doc
uses: tyrrrz/action-http-request@master
with:
url: http://flow-openapi.aliyun.com/pipeline/webhook/IiSxLDp9aOhgDUxJPytv
method: POST
body: |
{}
headers: |
Content-Type: application/json
retry-count: 3
retry-delay: 5000

2
.gitignore vendored
View File

@@ -1,6 +1,5 @@
./packages/core/lego ./packages/core/lego
# IntelliJ project files # IntelliJ project files
.vscode/
node_modules/ node_modules/
npm-debug.log npm-debug.log
yarn-error.log yarn-error.log
@@ -30,5 +29,4 @@ test/**/*.js
/packages/ui/certd-server/data/db.sqlite /packages/ui/certd-server/data/db.sqlite
/packages/ui/certd-server/data/keys.yaml /packages/ui/certd-server/data/keys.yaml
/packages/pro/ /packages/pro/
test.js test.js

43
.vscode/launch.json vendored Normal file
View File

@@ -0,0 +1,43 @@
{
// 使用 IntelliSense 了解相关属性。
// 悬停以查看现有属性的描述。
// 欲了解更多信息,请访问: https://go.microsoft.com/fwlink/?linkid=830387
"version": "0.2.0",
"configurations": [
{
"name": "client",
"type": "node",
"request": "launch",
"cwd": "${workspaceFolder}/packages/ui/certd-client",
"runtimeExecutable": "npm",
"runtimeArgs": ["run", "dev"],
"console": "integratedTerminal",
"internalConsoleOptions": "neverOpen"
},
{
"name": "server",
"type": "node",
"request": "launch",
"cwd": "${workspaceFolder}/packages/ui/certd-server",
"runtimeExecutable": "npm",
"runtimeArgs": ["run", "dev"],
"console": "integratedTerminal",
"internalConsoleOptions": "neverOpen"
},
{
"name": "server-local-plus",
"type": "node",
"request": "launch",
"cwd": "${workspaceFolder}/packages/ui/certd-server",
"runtimeExecutable": "npm",
"runtimeArgs": ["run", "dev-localplus"],
"console": "integratedTerminal",
"internalConsoleOptions": "neverOpen",
"env": {
"plus_use_prod": "false",
"PLUS_SERVER_BASE_URL": "http://127.0.0.1:11007"
}
}
]
}

4
.vscode/settings.json vendored Normal file
View File

@@ -0,0 +1,4 @@
{
"eslint.debug": false,
"eslint.format.enable": true
}

52
.vscode/tasks.json vendored Normal file
View File

@@ -0,0 +1,52 @@
{
"version": "2.0.0",
"tasks": [
{
"label": "启动Client",
"type": "shell",
"command": "npm",
"args": ["run", "dev"],
"options": {
"cwd": "${workspaceFolder}/packages/ui/certd-client"
},
"group": {
"kind": "build",
"isDefault": true
},
"presentation": {
"echo": true,
"reveal": "always",
"focus": false,
"panel": "shared"
}
},
{
"label": "启动Server",
"type": "shell",
"command": "npm",
"args": ["run", "dev"],
"options": {
"cwd": "${workspaceFolder}/packages/ui/certd-server"
},
"group": {
"kind": "build",
"isDefault": true
},
"presentation": {
"echo": true,
"reveal": "always",
"focus": false,
"panel": "shared"
}
},
{
"label": "同时启动Client和Server",
"dependsOn": ["启动Client", "启动Server"],
"group": {
"kind": "build",
"isDefault": true
},
"problemMatcher": []
}
]
}

View File

@@ -3,6 +3,101 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
### Bug Fixes
* 修复版本比较bug ([109696e](https://github.com/certd/certd/commit/109696e965d68c50c8627ffd40203edd1d2daea5))
* 修复某些情况下cname申请证书报错主域名不一致的bug ([2671781](https://github.com/certd/certd/commit/2671781e1bb0838981728d85eacf0e1a25a0fa48))
### Performance Improvements
* cname主域名校验提示优化显示不一致的两方便于排查问题 ([6ebb365](https://github.com/certd/certd/commit/6ebb3659f42155e4e8da600c493fb5227cd08137))
* dns解析支持阿里esa ([9291fa6](https://github.com/certd/certd/commit/9291fa68aa7a88a05c2f888bf3048df36a8fbde3))
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
### Features
* @certd/ui-server module import报错的问题 ([0c61d4c](https://github.com/certd/certd/commit/0c61d4c9788677c83c567db5381b9e257ec90bba))
* dist打包前检查 ([8f6e5bd](https://github.com/certd/certd/commit/8f6e5bd24b3b65fbfcba36c08f532a3abad2d606))
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
### Bug Fixes
* 固定midwayjs版本修复ui-server import 错误的bug ([eb4d125](https://github.com/certd/certd/commit/eb4d125eaf4a41e88c752d0c68993829589f8f27))
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
### Bug Fixes
* 修复 ui-server 加载失败问题 ([c2ccdbe](https://github.com/certd/certd/commit/c2ccdbec9dd08bca4688eeb2f34d0105eec43ba1))
* 修复 ui-server 加载失败问题 ([063f5c3](https://github.com/certd/certd/commit/063f5c3b55e47df22543a64f02e039e84f92cd14))
### Performance Improvements
* 重置管理员密码同时会关闭验证码,防止验证码失效之后无法登录 ([03899d4](https://github.com/certd/certd/commit/03899d4d9c76fc2077dacc53ab88e2c9ca41af7c))
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
### Bug Fixes
* 授权页面id列位置不在第一列的bug ([3f1722d](https://github.com/certd/certd/commit/3f1722d54debcb4849dc14521a2da0d9b304b69f))
### Performance Improvements
* 动态加载验证码script ([dcc396a](https://github.com/certd/certd/commit/dcc396afb7a23aeb8af57c01014b09af5f033e61))
* 开启子域名托管之后cname记录支持重置 ([54c8d62](https://github.com/certd/certd/commit/54c8d622437761d350db0f17e07f7517f1911211))
* 手动上传证书优化,增加到期前报错提醒 ([3d42bfd](https://github.com/certd/certd/commit/3d42bfd479eaacc4a49c401224815a6e2a0204b0))
* 验证码支持测试,登录验证码需要测试通过后才能开启 ([83e6476](https://github.com/certd/certd/commit/83e6476408090b741fabb1b542fb458d9a8b4134))
* 支持腾讯云验证码 ([03f317f](https://github.com/certd/certd/commit/03f317ffdb6595ce70e8a2302b05f390c52110c8))
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
### Bug Fixes
* 修复旧版本升级上来报错eab授权的bug ([b76f2e2](https://github.com/certd/certd/commit/b76f2e2008a7fefac4c91179c45c56c7a7a84b71))
* 选择授权对话框编辑时名称字段排在最后的bug ([31cfb09](https://github.com/certd/certd/commit/31cfb09468bda3272f5f63af65ff3e9272220b39))
### Performance Improvements
* 7001绑定::地址 ([7188997](https://github.com/certd/certd/commit/7188997dd1979f1c10fa29b30221015e0bd5fe9e))
* 登录失败时清除验证码状态 ([1c15bea](https://github.com/certd/certd/commit/1c15beadc7fe8a7c6ec1903b7e722ca2f52e05b3))
* 公共cname支持权限校验 ([9cc5f0f](https://github.com/certd/certd/commit/9cc5f0f889d4362ff36e7a1f0e448e02d32ecee7))
* 优化连接失败的报错提示 ([71d8e7e](https://github.com/certd/certd/commit/71d8e7edd23ad63fdc01a92766b52ede5074fe7c))
* 增加自签名证书提示 ([877c9c4](https://github.com/certd/certd/commit/877c9c4ff99f81d289f67afd96f440c0796b03ea))
* add preferred chain for google trust service ([#539](https://github.com/certd/certd/issues/539)) @ZeroClover ([e31d26a](https://github.com/certd/certd/commit/e31d26a8871c6088d9f8c0f580746ff2a810ae0c))
* dns支持新网域名解析 ([cf3a78e](https://github.com/certd/certd/commit/cf3a78e1145ff0505c87fbc485d9e731b1aa88a8))
* gcore flush plugin ssl_id改为必填项 ([4b90972](https://github.com/certd/certd/commit/4b909723411c57505aa13b07d8699fb9ac77c937))
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
### Bug Fixes
* 修复导入插件对话框无法打开的bug修复插件编辑页面打开多个代码编辑器消失的bug ([e5a080a](https://github.com/certd/certd/commit/e5a080aebe0d2f3e3c0f86bf863f75069c1bf7ab))
* 修复ssl.com报EMAILADDRESS数量不对的bug ([c560cc5](https://github.com/certd/certd/commit/c560cc5adda6e15bf3a8865d874042550a6c2688))
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Bug Fixes
* 修复商业版退出登录后丢失站点个性化设置的bug ([d75dd05](https://github.com/certd/certd/commit/d75dd058d65c85f80c49e1fa7a910e6c6f08e824))
* 修复授权类型和名称字段排到最后的bug ([43b7977](https://github.com/certd/certd/commit/43b79778ea9034065f6a15af3296274315597c6b))
* 修复证书监控某些情况下报 options.lookup不能为null的bug ([d2ecfe5](https://github.com/certd/certd/commit/d2ecfe5491b2639eb30b5cae293af6062d58bb9f))
* 修复证书手动托管时新上传的证书无效的bug ([506385e](https://github.com/certd/certd/commit/506385e5a2600887fe30854e0713583caaa2e689))
* 修复secret patch 类型多了type的bug ([d04f383](https://github.com/certd/certd/commit/d04f3831611011a90ec0594724b9694490d5edd0))
### Performance Improvements
* 登录支持极验验证码 ([370db62](https://github.com/certd/certd/commit/370db62bf0aece241859244927beabba32d6a257))
* 登录注册、找回密码都支持极验验证码和图片验证码 ([7bdde68](https://github.com/certd/certd/commit/7bdde68ecea29fe2c570fd3cb082139db6c93d93))
* 优化加量包展示效果 ([3c65f37](https://github.com/certd/certd/commit/3c65f37d84177ba107d4a6462648af12d2fc4b7a))
* 证书到期剩余天数进度条根据实际证书有效期计算 ([#528](https://github.com/certd/certd/issues/528)) nicheng-he ([2d4586b](https://github.com/certd/certd/commit/2d4586b1c42c39f97d2a95b9453cca4bc8bfbe61))
* add preferred chain option ([#519](https://github.com/certd/certd/issues/519)) @ZeroClover ([902359f](https://github.com/certd/certd/commit/902359f24ed12eee4f9b65178f1d6a60378351d2))
* ssh配置增加脚本类型设置bash还是sh ([ae41c60](https://github.com/certd/certd/commit/ae41c6038b27c9476e64a2402a8daf247c38a5b6))
* start.sh增加sudo ([b7271d7](https://github.com/certd/certd/commit/b7271d7a464773a1bf87d7d1f24d933ba0f86915))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Bug Fixes ### Bug Fixes

View File

@@ -152,7 +152,7 @@ https://certd.handfree.work/
## 八、捐赠 ## 八、捐赠
************************ ************************
支持开源,为爱发电,我已入驻爱发电 支持开源,为爱发电,我已入驻爱发电
https://afdian.com/a/greper https://afdian.com/a/greper
发电权益: 发电权益:
@@ -171,6 +171,7 @@ https://afdian.com/a/greper
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署、宝塔、1Panel等大部分插件 | 群晖 | | 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署、宝塔、1Panel等大部分插件 | 群晖 |
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、钉钉、飞书、anpush、server酱等 | | 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、钉钉、飞书、anpush、server酱等 |
************************
************************ ************************

View File

@@ -134,6 +134,8 @@ You can also add the author as a friend.
| QR Code | <img height="230" src="./docs/guide/contact/images/me.png"> | | QR Code | <img height="230" src="./docs/guide/contact/images/me.png"> |
## 8. Donation ## 8. Donation
************************
[![Sponsor](https://img.shields.io/badge/Sponsor-%E2%9D%A4-red)](https://github.com/sponsors/greper)
************************ ************************
Support open-source projects and contribute with love. I've joined Afdian. Support open-source projects and contribute with love. I've joined Afdian.
https://afdian.com/a/greper https://afdian.com/a/greper

View File

@@ -1 +0,0 @@
2

View File

@@ -1 +0,0 @@
00:34

View File

@@ -1 +0,0 @@
5

View File

@@ -119,6 +119,7 @@ export default defineConfig({
{text: "邮箱配置", link: "/guide/use/email/index.md"}, {text: "邮箱配置", link: "/guide/use/email/index.md"},
{text: "IPv6支持", link: "/guide/use/setting/ipv6.md"}, {text: "IPv6支持", link: "/guide/use/setting/ipv6.md"},
{text: "ESXi", link: "/guide/use/ESXi/index.md"}, {text: "ESXi", link: "/guide/use/ESXi/index.md"},
{text: "宝塔动态IP白名单", link: "/guide/use/baota/white_list.md"},
{text: "子域名托管", link: "/guide/use/cert/subdomain.md"}, {text: "子域名托管", link: "/guide/use/cert/subdomain.md"},
] ]
}, },

View File

@@ -3,6 +3,89 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
### Features
* @certd/ui-server module import报错的问题 ([0c61d4c](https://github.com/certd/certd/commit/0c61d4c9788677c83c567db5381b9e257ec90bba))
* dist打包前检查 ([8f6e5bd](https://github.com/certd/certd/commit/8f6e5bd24b3b65fbfcba36c08f532a3abad2d606))
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
### Bug Fixes
* 固定midwayjs版本修复ui-server import 错误的bug ([eb4d125](https://github.com/certd/certd/commit/eb4d125eaf4a41e88c752d0c68993829589f8f27))
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
### Bug Fixes
* 修复 ui-server 加载失败问题 ([c2ccdbe](https://github.com/certd/certd/commit/c2ccdbec9dd08bca4688eeb2f34d0105eec43ba1))
* 修复 ui-server 加载失败问题 ([063f5c3](https://github.com/certd/certd/commit/063f5c3b55e47df22543a64f02e039e84f92cd14))
### Performance Improvements
* 重置管理员密码同时会关闭验证码,防止验证码失效之后无法登录 ([03899d4](https://github.com/certd/certd/commit/03899d4d9c76fc2077dacc53ab88e2c9ca41af7c))
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
### Bug Fixes
* 授权页面id列位置不在第一列的bug ([3f1722d](https://github.com/certd/certd/commit/3f1722d54debcb4849dc14521a2da0d9b304b69f))
### Performance Improvements
* 动态加载验证码script ([dcc396a](https://github.com/certd/certd/commit/dcc396afb7a23aeb8af57c01014b09af5f033e61))
* 开启子域名托管之后cname记录支持重置 ([54c8d62](https://github.com/certd/certd/commit/54c8d622437761d350db0f17e07f7517f1911211))
* 手动上传证书优化,增加到期前报错提醒 ([3d42bfd](https://github.com/certd/certd/commit/3d42bfd479eaacc4a49c401224815a6e2a0204b0))
* 验证码支持测试,登录验证码需要测试通过后才能开启 ([83e6476](https://github.com/certd/certd/commit/83e6476408090b741fabb1b542fb458d9a8b4134))
* 支持腾讯云验证码 ([03f317f](https://github.com/certd/certd/commit/03f317ffdb6595ce70e8a2302b05f390c52110c8))
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
### Bug Fixes
* 修复旧版本升级上来报错eab授权的bug ([b76f2e2](https://github.com/certd/certd/commit/b76f2e2008a7fefac4c91179c45c56c7a7a84b71))
* 选择授权对话框编辑时名称字段排在最后的bug ([31cfb09](https://github.com/certd/certd/commit/31cfb09468bda3272f5f63af65ff3e9272220b39))
### Performance Improvements
* 7001绑定::地址 ([7188997](https://github.com/certd/certd/commit/7188997dd1979f1c10fa29b30221015e0bd5fe9e))
* 登录失败时清除验证码状态 ([1c15bea](https://github.com/certd/certd/commit/1c15beadc7fe8a7c6ec1903b7e722ca2f52e05b3))
* 公共cname支持权限校验 ([9cc5f0f](https://github.com/certd/certd/commit/9cc5f0f889d4362ff36e7a1f0e448e02d32ecee7))
* 优化连接失败的报错提示 ([71d8e7e](https://github.com/certd/certd/commit/71d8e7edd23ad63fdc01a92766b52ede5074fe7c))
* 增加自签名证书提示 ([877c9c4](https://github.com/certd/certd/commit/877c9c4ff99f81d289f67afd96f440c0796b03ea))
* add preferred chain for google trust service ([#539](https://github.com/certd/certd/issues/539)) @ZeroClover ([e31d26a](https://github.com/certd/certd/commit/e31d26a8871c6088d9f8c0f580746ff2a810ae0c))
* dns支持新网域名解析 ([cf3a78e](https://github.com/certd/certd/commit/cf3a78e1145ff0505c87fbc485d9e731b1aa88a8))
* gcore flush plugin ssl_id改为必填项 ([4b90972](https://github.com/certd/certd/commit/4b909723411c57505aa13b07d8699fb9ac77c937))
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
### Bug Fixes
* 修复导入插件对话框无法打开的bug修复插件编辑页面打开多个代码编辑器消失的bug ([e5a080a](https://github.com/certd/certd/commit/e5a080aebe0d2f3e3c0f86bf863f75069c1bf7ab))
* 修复ssl.com报EMAILADDRESS数量不对的bug ([c560cc5](https://github.com/certd/certd/commit/c560cc5adda6e15bf3a8865d874042550a6c2688))
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Bug Fixes
* 修复商业版退出登录后丢失站点个性化设置的bug ([d75dd05](https://github.com/certd/certd/commit/d75dd058d65c85f80c49e1fa7a910e6c6f08e824))
* 修复授权类型和名称字段排到最后的bug ([43b7977](https://github.com/certd/certd/commit/43b79778ea9034065f6a15af3296274315597c6b))
* 修复证书监控某些情况下报 options.lookup不能为null的bug ([d2ecfe5](https://github.com/certd/certd/commit/d2ecfe5491b2639eb30b5cae293af6062d58bb9f))
* 修复证书手动托管时新上传的证书无效的bug ([506385e](https://github.com/certd/certd/commit/506385e5a2600887fe30854e0713583caaa2e689))
* 修复secret patch 类型多了type的bug ([d04f383](https://github.com/certd/certd/commit/d04f3831611011a90ec0594724b9694490d5edd0))
### Performance Improvements
* 登录支持极验验证码 ([370db62](https://github.com/certd/certd/commit/370db62bf0aece241859244927beabba32d6a257))
* 登录注册、找回密码都支持极验验证码和图片验证码 ([7bdde68](https://github.com/certd/certd/commit/7bdde68ecea29fe2c570fd3cb082139db6c93d93))
* 优化加量包展示效果 ([3c65f37](https://github.com/certd/certd/commit/3c65f37d84177ba107d4a6462648af12d2fc4b7a))
* 证书到期剩余天数进度条根据实际证书有效期计算 ([#528](https://github.com/certd/certd/issues/528)) nicheng-he ([2d4586b](https://github.com/certd/certd/commit/2d4586b1c42c39f97d2a95b9453cca4bc8bfbe61))
* add preferred chain option ([#519](https://github.com/certd/certd/issues/519)) @ZeroClover ([902359f](https://github.com/certd/certd/commit/902359f24ed12eee4f9b65178f1d6a60378351d2))
* ssh配置增加脚本类型设置bash还是sh ([ae41c60](https://github.com/certd/certd/commit/ae41c6038b27c9476e64a2402a8daf247c38a5b6))
* start.sh增加sudo ([b7271d7](https://github.com/certd/certd/commit/b7271d7a464773a1bf87d7d1f24d933ba0f86915))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Bug Fixes ### Bug Fixes

View File

@@ -21,13 +21,13 @@
#### 2.2 容器编排方式部署 #### 2.2 容器编排方式部署
1. 打开`docker-compose.yaml`,整个内容复制下来 1. 打开`docker-compose.yaml`,整个内容复制下来
https://gitee.com/certd/certd/raw/v2/docker/run/docker-compose.yaml https://gitee.com/certd/certd/raw/v2/docker/run/docker-compose.yaml
然后到宝塔里面进到docker->容器编排->添加容器编排 然后到宝塔里面进到docker->容器编排->添加容器编排
![](./images/1.png) ![](./images/1.png)
点击确定,等待启动完成 点击确定,等待启动完成
![](./images/2.png) ![](./images/2.png)
> certd默认使用sqlite数据库另外支持`mysql`和`postgresql`数据库,[点我了解如何切换其他数据库](../database) > certd默认使用sqlite数据库另外支持`mysql`和`postgresql`数据库,[点我了解如何切换其他数据库](../database)
@@ -35,16 +35,16 @@
## 二、访问应用 ## 二、访问应用
http://ip:7001 http://ip:7001
https://ip:7002 https://ip:7002
默认账号密码 默认账号密码
admin/123456 admin/123456
登录后请及时修改密码 登录后请及时修改密码
## 三、如何升级 ## 三、如何升级
宝塔升级certd非常简单 宝塔升级certd非常简单
打开容器页面: `docker`->`容器编排`->`左侧选择Certd`->`更新镜像` 打开容器页面: `docker`->`容器编排`->`左侧选择Certd`->`更新镜像`
![img.png](./images/upgrade.png) ![img.png](./images/upgrade.png)
@@ -80,5 +80,8 @@ admin/123456
### 1. 无法访问Certd ### 1. 无法访问Certd
1. 确认服务器的安全规则,是否放开了对应端口 1. 确认服务器的安全规则,是否放开了对应端口
2. 确认宝塔防火墙是否放开对应端口 2. 确认宝塔防火墙是否放开对应端口
3. 尝试将Certd容器加入宝塔的`bridge`网络 3. 尝试将Certd容器加入宝塔的`bridge`网络
![](./images/network.png) ![](./images/network.png)
### 2. 动态IP无法加白名单问题
[Nginx代理解决方案](../../use/baota/white_list.md)

View File

@@ -17,4 +17,17 @@
解决方案:可以加多一个子域名,重新执行就可以规避次错误 解决方案:可以加多一个子域名,重新执行就可以规避次错误
``` ```
"detail": too many certificates (5) already issued for this exact set of idantifiers in the last 168hm0s "detail": too many certificates (5) already issued for this exact set of idantifiers in the last 168hm0s
``` ```
## ssl.com报错 CAA record does not include ssl.com which is required to issue the certificate
ssl.com申请证书要求必须设置CAA记录表示允许ssl.com为该域名颁发证书
请按如下格式添加CAA记录
| 示例 | 类型 | 域名前缀 | flag | tag | 值 |
|-------|-----| -- |-----------|--------|----------------------|
| 顶级域名 | CAA | @ | 0 | issue | "ssl.com" (注意有双引号) |
| 一级泛域名 | CAA | * | 0 | issue/issuewild | "ssl.com" |
| 固定子域名 | CAA | sub | 0 | issue |"ssl.com" |

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 36 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 27 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 58 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

View File

@@ -0,0 +1,98 @@
# 宝塔IP白名单与动态IP问题
调用宝塔接口需要添加IP白名单但当certd部署在动态IP环境下时IP白名单就不好添加
本章节提供两种解决方案:
1. 小范围网段放开(简单)
2. nginx代理
## 一、放开小范围网段
家庭网络IP虽然会变动但是只会在小范围变的。
你可以分析规律,将变动的部分,设置成网段即可
> 比如出现过: 100.25.1.5 100.25.1.8
>
> 那么你可以配置 100.25.1.1-100.25.1.255
> 如果出现过: 100.25.1.5 100.25.4.8
>
> 可以尝试配置 100.25.*.5
## 二、nginx代理方案
通过在宝塔中配置一个nginx反向代理代理宝塔自己的地址
然后在nginx中配置放开certd需要的接口缩小影响范围
让nginx来充当防火墙
架构图如下:
```
只要将127.0.0.1加入白名单即可
certd --------> nginx -------> 宝塔
拦截除更新证书之外的地址
```
### 1. 添加nginx反向代理
![](./images/white-1.png)
### 2. 域名和代理目标
![](./images/white-2.png)
### 3. 设置放开哪些接口
![](./images/white-3.png)
![img.png](images/white-4.png)
将如下脚本填入上方文本域中,保存
```nginx configuration
set $allow_access false;
# 检查请求的URI是否在白名单中
if ($request_uri ~* "^/(site\?action=get_site_types)") {
# 允许测试
set $allow_access true;
}
if ($request_uri ~* "^/(config\?action=SavePanelSSL)") {
# 允许部署到宝塔面板本身证书
set $allow_access true;
}
if ($request_uri ~* "^/(mod/docker/com/set_ssl|site\?action=SetSSL|ssl\?action=GetSiteDomain|mod/docker/com/get_site_list)") {
# 允许部署宝塔网站证书
set $allow_access true;
}
if ($request_uri ~* "^/(ssl?action=remove_cloud_cert|ssl\?action=get_cert_list)") {
# 允许删除宝塔过期证书
set $allow_access true;
}
if ($request_uri ~* "^/(datalist/get_data_list|site/set_site_ssl)") {
set $allow_access true;
}
# 如果不在白名单返回403禁止访问
if ($allow_access = false) {
return 405;
}
```
### 4. 接口IP白名单添加127.0.0.1
![img.png](images/white-5.png)
### 5. certd中宝塔授权配置改成新的这个域名地址
![img.png](images/white-6.png)
点击测试检查是否ok ,到这里就可以正常部署证书了
### 6. 安全加强将请求地址改成https
在宝塔中配置证书部署任务,选择刚才新建的这个网站,给他部署证书
勾选强制https
![img.png](images/white-safe-1.png)
更换443端口【可选】
![img.png](images/white-safe-2.png)
禁止http访问

View File

@@ -9,5 +9,5 @@
} }
}, },
"npmClient": "pnpm", "npmClient": "pnpm",
"version": "1.36.19" "version": "1.37.1"
} }

View File

@@ -18,7 +18,7 @@
"devb": "lerna run dev-build", "devb": "lerna run dev-build",
"i-all": "lerna link && lerna exec npm install ", "i-all": "lerna link && lerna exec npm install ",
"publish": "npm run prepublishOnly2 && lerna publish --force-publish=pro/plus-core --conventional-commits --create-release github && npm run afterpublishOnly && npm run commitAll", "publish": "npm run prepublishOnly2 && lerna publish --force-publish=pro/plus-core --conventional-commits --create-release github && npm run afterpublishOnly && npm run commitAll",
"afterpublishOnly": "npm run copylogs && time /t >build.trigger && git add ./build.trigger && git commit -m \"build: trigger build image\" && TIMEOUT /T 10 && git push", "afterpublishOnly": "npm run copylogs && time /t >trigger/build.trigger && git add ./trigger/build.trigger && git commit -m \"build: trigger build image\" && TIMEOUT /T 10 && git push",
"transform-sql": "cd ./packages/ui/certd-server/db/ && node --experimental-json-modules transform.js", "transform-sql": "cd ./packages/ui/certd-server/db/ && node --experimental-json-modules transform.js",
"commitAll": "git add . && git commit -m \"build: publish\" && git push && npm run commitPro", "commitAll": "git add . && git commit -m \"build: publish\" && git push && npm run commitPro",
"commitPro": "cd ./packages/pro/ && git add . && git commit -m \"build: publish\" && git push", "commitPro": "cd ./packages/pro/ && git add . && git commit -m \"build: publish\" && git push",
@@ -37,7 +37,6 @@
}, },
"license": "AGPL-3.0", "license": "AGPL-3.0",
"dependencies": { "dependencies": {
"@certd/ui-server": "link:packages/ui/certd-server",
"axios": "^1.7.7", "axios": "^1.7.7",
"copyfiles": "^2.4.1", "copyfiles": "^2.4.1",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",

View File

@@ -3,6 +3,38 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/publishlab/node-acme-client/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/acme-client
# [1.37.0](https://github.com/publishlab/node-acme-client/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/acme-client
## [1.36.25](https://github.com/publishlab/node-acme-client/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/acme-client
## [1.36.24](https://github.com/publishlab/node-acme-client/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/acme-client
## [1.36.23](https://github.com/publishlab/node-acme-client/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/acme-client
## [1.36.22](https://github.com/publishlab/node-acme-client/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/acme-client
## [1.36.21](https://github.com/publishlab/node-acme-client/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/acme-client
## [1.36.20](https://github.com/publishlab/node-acme-client/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/acme-client
## [1.36.19](https://github.com/publishlab/node-acme-client/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/publishlab/node-acme-client/compare/v1.36.18...v1.36.19) (2025-09-05)
### Performance Improvements ### Performance Improvements

View File

@@ -3,7 +3,7 @@
"description": "Simple and unopinionated ACME client", "description": "Simple and unopinionated ACME client",
"private": false, "private": false,
"author": "nmorsman", "author": "nmorsman",
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"module": "scr/index.js", "module": "scr/index.js",
"main": "src/index.js", "main": "src/index.js",
@@ -18,7 +18,7 @@
"types" "types"
], ],
"dependencies": { "dependencies": {
"@certd/basic": "^1.36.19", "@certd/basic": "^1.37.1",
"@peculiar/x509": "^1.11.0", "@peculiar/x509": "^1.11.0",
"asn1js": "^3.0.5", "asn1js": "^3.0.5",
"axios": "^1.7.2", "axios": "^1.7.2",
@@ -69,5 +69,5 @@
"bugs": { "bugs": {
"url": "https://github.com/publishlab/node-acme-client/issues" "url": "https://github.com/publishlab/node-acme-client/issues"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -502,7 +502,7 @@ class AcmeClient {
await verify[challenge.type](authz, challenge, keyAuthorization); await verify[challenge.type](authz, challenge, keyAuthorization);
}; };
log('Waiting for ACME challenge verification等待ACME挑战验证', this.backoffOpts); log('Waiting for ACME challenge verification等待ACME挑战验证');
return util.retry(verifyFn, this.backoffOpts); return util.retry(verifyFn, this.backoffOpts);
} }

View File

@@ -3,6 +3,42 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/basic
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/basic
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/basic
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/basic
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/basic
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
### Performance Improvements
* 优化连接失败的报错提示 ([71d8e7e](https://github.com/certd/certd/commit/71d8e7edd23ad63fdc01a92766b52ede5074fe7c))
* 增加自签名证书提示 ([877c9c4](https://github.com/certd/certd/commit/877c9c4ff99f81d289f67afd96f440c0796b03ea))
* dns支持新网域名解析 ([cf3a78e](https://github.com/certd/certd/commit/cf3a78e1145ff0505c87fbc485d9e731b1aa88a8))
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/basic
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/basic
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Bug Fixes ### Bug Fixes

View File

@@ -1 +1 @@
00:30 20:32

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/basic", "name": "@certd/basic",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"main": "./dist/index.js", "main": "./dist/index.js",
"module": "./dist/index.js", "module": "./dist/index.js",
@@ -45,5 +45,5 @@
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -22,12 +22,14 @@ import { sp } from "./util.sp.js";
import { hashUtils } from "./util.hash.js"; import { hashUtils } from "./util.hash.js";
import { promises } from "./util.promise.js"; import { promises } from "./util.promise.js";
import { fileUtils } from "./util.file.js"; import { fileUtils } from "./util.file.js";
import * as _ from "lodash-es";
import { cache } from "./util.cache.js"; import { cache } from "./util.cache.js";
import dayjs from "dayjs"; import dayjs from "dayjs";
import { domainUtils } from "./util.domain.js"; import { domainUtils } from "./util.domain.js";
export * from "./util.domain.js";
import { optionsUtils } from "./util.options.js"; import { optionsUtils } from "./util.options.js";
export * from "./util.options.js";
import { amountUtils } from "./util.amount.js"; import { amountUtils } from "./util.amount.js";
export * from "./util.amount.js";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
import * as id from "./util.id.js"; import * as id from "./util.id.js";
import { locker } from "./util.lock.js"; import { locker } from "./util.lock.js";
@@ -43,7 +45,6 @@ export const utils = {
hash: hashUtils, hash: hashUtils,
promises, promises,
file: fileUtils, file: fileUtils,
_,
mergeUtils, mergeUtils,
cache, cache,
nanoid, nanoid,

View File

@@ -17,10 +17,26 @@ function base64(data: string) {
function base64Decode(data: string) { function base64Decode(data: string) {
return Buffer.from(data, "base64").toString("utf8"); return Buffer.from(data, "base64").toString("utf8");
} }
function toHex(data: number | string) {
if (typeof data === "number") {
return data.toString(16);
}
return Buffer.from(data).toString("hex");
}
function hexToStr(data: string) {
return Buffer.from(data, "hex").toString("utf8");
}
function hexToNumber(data: string) {
return parseInt(data, 16);
}
export const hashUtils = { export const hashUtils = {
md5, md5,
sha256, sha256,
base64, base64,
base64Decode, base64Decode,
hmacSha256, hmacSha256,
toHex,
hexToStr,
hexToNumber,
}; };

View File

@@ -7,6 +7,13 @@ import * as https from "node:https";
import { merge } from "lodash-es"; import { merge } from "lodash-es";
import { safePromise } from "./util.promise.js"; import { safePromise } from "./util.promise.js";
import fs from "fs"; import fs from "fs";
const errorMap: Record<string, string> = {
"ssl3_get_record:wrong version number": "http协议错误服务端要求http协议请检查是否使用了https请求",
"getaddrinfo EAI_AGAIN": "无法解析域名请检查网络连接或dns配置更换docker-compose.yaml中dns配置",
"self-signed certificate": "目标站点为自签名证书,请勾选忽略证书校验",
};
export class HttpError extends Error { export class HttpError extends Error {
status?: number; status?: number;
statusText?: string; statusText?: string;
@@ -21,11 +28,12 @@ export class HttpError extends Error {
super(error.message || error.response?.statusText); super(error.message || error.response?.statusText);
const message = error?.message; const message = error?.message;
if (message && typeof message === "string") { if (message && typeof message === "string" && message.indexOf) {
if (message.indexOf && message.indexOf("ssl3_get_record:wrong version number") >= 0) { for (const key in errorMap) {
this.message = `${message}(http协议错误服务端要求http协议请检查是否使用了https请求)`; if (message.indexOf(key) > -1) {
} else if (message.indexOf("getaddrinfo EAI_AGAIN") >= 0) { this.message = `${this.message}(${errorMap[key]})`;
this.message = `${message}(无法解析域名请检查网络连接或dns配置更换docker-compose.yaml中dns配置)`; break;
}
} }
} }
@@ -199,10 +207,31 @@ export function createAxiosService({ logger }: { logger: ILogger }) {
case 505: case 505:
error.message = "HTTP版本不受支持"; error.message = "HTTP版本不受支持";
break; break;
case 302:
//重定向
return Promise.resolve(error.response);
default: default:
break; break;
} }
logger.error(`请求出错status:${error.response?.status},statusText:${error.response?.statusText},url:${error.config?.url},method:${error.config?.method}`);
const errorCode = error.code;
let errorMessage = null;
if (errorCode === "ECONNABORTED") {
errorMessage = "请求连接终止";
} else if (errorCode === "ETIMEDOUT") {
errorMessage = "请求连接超时";
} else if (errorCode === "ECONNRESET") {
errorMessage = "请求连接被重置";
} else if (errorCode === "ECONNREFUSED") {
errorMessage = "请求连接被服务端拒绝";
} else if (errorCode === "ENOTFOUND") {
errorMessage = "请求地址不存在";
}
if (errorMessage) {
error.message = errorMessage + "," + error.message;
}
logger.error(`请求出错status:${error.response?.status || error.code},statusText:${error.response?.statusText || error.code},url:${error.config?.url},method:${error.config?.method}`);
logger.error("返回数据:", JSON.stringify(error.response?.data)); logger.error("返回数据:", JSON.stringify(error.response?.data));
if (error.response?.data) { if (error.response?.data) {
const message = error.response.data.message || error.response.data.msg || error.response.data.error; const message = error.response.data.message || error.response.data.msg || error.response.data.error;

View File

@@ -3,6 +3,40 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/pipeline
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/pipeline
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/pipeline
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/pipeline
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
### Performance Improvements
* 手动上传证书优化,增加到期前报错提醒 ([3d42bfd](https://github.com/certd/certd/commit/3d42bfd479eaacc4a49c401224815a6e2a0204b0))
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/pipeline
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/pipeline
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/pipeline
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Bug Fixes ### Bug Fixes

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/pipeline", "name": "@certd/pipeline",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"main": "./dist/index.js", "main": "./dist/index.js",
"module": "./dist/index.js", "module": "./dist/index.js",
@@ -17,8 +17,8 @@
"pub": "npm publish" "pub": "npm publish"
}, },
"dependencies": { "dependencies": {
"@certd/basic": "^1.36.19", "@certd/basic": "^1.37.1",
"@certd/plus-core": "^1.36.19", "@certd/plus-core": "^1.37.1",
"dayjs": "^1.11.7", "dayjs": "^1.11.7",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",
"reflect-metadata": "^0.1.13" "reflect-metadata": "^0.1.13"
@@ -44,5 +44,5 @@
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -120,10 +120,9 @@ export class RunHistory {
delete e.stack; delete e.stack;
delete e.cause; delete e.cause;
if (runnable.runnableType === "step") { if (runnable.runnableType === "step") {
this._loggers[runnable.id].error(`[${runnable.runnableType}] [${runnable.title}]<id:${runnable.id}> `, e, stack, cause); this._loggers[runnable.id].error(stack, cause);
} else {
this._loggers[runnable.id].error(`[${runnable.runnableType}] [${runnable.title}]<id:${runnable.id}> `, e.message);
} }
this._loggers[runnable.id].error(`[${runnable.runnableType}] [${runnable.title}]<id:${runnable.id}> `, e.message);
} }
finally(runnable: Runnable) { finally(runnable: Runnable) {

View File

@@ -17,6 +17,7 @@ export type CnameRecord = {
cnameProvider: CnameProvider; cnameProvider: CnameProvider;
status: string; status: string;
commonDnsProvider?: any; commonDnsProvider?: any;
mainDomain?: string;
}; };
export type ICnameProxyService = { export type ICnameProxyService = {

View File

@@ -3,6 +3,38 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/lib-huawei
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/lib-huawei
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
**Note:** Version bump only for package @certd/lib-huawei **Note:** Version bump only for package @certd/lib-huawei

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/lib-huawei", "name": "@certd/lib-huawei",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"main": "./dist/bundle.js", "main": "./dist/bundle.js",
"module": "./dist/bundle.js", "module": "./dist/bundle.js",
"types": "./dist/d/index.d.ts", "types": "./dist/d/index.d.ts",
@@ -24,5 +24,5 @@
"prettier": "^2.8.8", "prettier": "^2.8.8",
"tslib": "^2.8.1" "tslib": "^2.8.1"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -3,6 +3,38 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/lib-iframe
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/lib-iframe
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
**Note:** Version bump only for package @certd/lib-iframe **Note:** Version bump only for package @certd/lib-iframe

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/lib-iframe", "name": "@certd/lib-iframe",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"main": "./dist/index.js", "main": "./dist/index.js",
"module": "./dist/index.js", "module": "./dist/index.js",
@@ -31,5 +31,5 @@
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -3,6 +3,38 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/jdcloud
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/jdcloud
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
**Note:** Version bump only for package @certd/jdcloud **Note:** Version bump only for package @certd/jdcloud

View File

@@ -1,6 +1,6 @@
{ {
"name": "@certd/jdcloud", "name": "@certd/jdcloud",
"version": "1.36.19", "version": "1.37.1",
"description": "jdcloud openApi sdk", "description": "jdcloud openApi sdk",
"main": "./dist/bundle.js", "main": "./dist/bundle.js",
"module": "./dist/bundle.js", "module": "./dist/bundle.js",
@@ -61,5 +61,5 @@
"fetch" "fetch"
] ]
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -3,6 +3,40 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/lib-k8s
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/lib-k8s
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/lib-k8s
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/lib-k8s
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/lib-k8s
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/lib-k8s
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/lib-k8s
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Bug Fixes
* 修复secret patch 类型多了type的bug ([d04f383](https://github.com/certd/certd/commit/d04f3831611011a90ec0594724b9694490d5edd0))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Bug Fixes ### Bug Fixes

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/lib-k8s", "name": "@certd/lib-k8s",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"main": "./dist/index.js", "main": "./dist/index.js",
"module": "./dist/index.js", "module": "./dist/index.js",
@@ -17,7 +17,7 @@
"pub": "npm publish" "pub": "npm publish"
}, },
"dependencies": { "dependencies": {
"@certd/basic": "^1.36.19", "@certd/basic": "^1.37.1",
"@kubernetes/client-node": "0.21.0" "@kubernetes/client-node": "0.21.0"
}, },
"devDependencies": { "devDependencies": {
@@ -32,5 +32,5 @@
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -3,6 +3,49 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/lib-server
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
### Features
* dist打包前检查 ([8f6e5bd](https://github.com/certd/certd/commit/8f6e5bd24b3b65fbfcba36c08f532a3abad2d606))
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
### Bug Fixes
* 固定midwayjs版本修复ui-server import 错误的bug ([eb4d125](https://github.com/certd/certd/commit/eb4d125eaf4a41e88c752d0c68993829589f8f27))
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
### Bug Fixes
* 修复 ui-server 加载失败问题 ([c2ccdbe](https://github.com/certd/certd/commit/c2ccdbec9dd08bca4688eeb2f34d0105eec43ba1))
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
### Performance Improvements
* 支持腾讯云验证码 ([03f317f](https://github.com/certd/certd/commit/03f317ffdb6595ce70e8a2302b05f390c52110c8))
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/lib-server
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/lib-server
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Performance Improvements
* 登录支持极验验证码 ([370db62](https://github.com/certd/certd/commit/370db62bf0aece241859244927beabba32d6a257))
* 登录注册、找回密码都支持极验验证码和图片验证码 ([7bdde68](https://github.com/certd/certd/commit/7bdde68ecea29fe2c570fd3cb082139db6c93d93))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
**Note:** Version bump only for package @certd/lib-server **Note:** Version bump only for package @certd/lib-server

View File

@@ -1,6 +1,6 @@
{ {
"name": "@certd/lib-server", "name": "@certd/lib-server",
"version": "1.36.19", "version": "1.37.1",
"description": "midway with flyway, sql upgrade way ", "description": "midway with flyway, sql upgrade way ",
"private": false, "private": false,
"type": "module", "type": "module",
@@ -27,18 +27,20 @@
], ],
"license": "AGPL", "license": "AGPL",
"dependencies": { "dependencies": {
"@certd/acme-client": "^1.36.19", "@certd/acme-client": "^1.37.1",
"@certd/basic": "^1.36.19", "@certd/basic": "^1.37.1",
"@certd/pipeline": "^1.36.19", "@certd/pipeline": "^1.37.1",
"@certd/plus-core": "^1.36.19", "@certd/plugin-lib": "^1.37.1",
"@midwayjs/cache": "~3.14.0", "@certd/plus-core": "^1.37.1",
"@midwayjs/core": "~3.20.3", "@midwayjs/cache": "3.14.0",
"@midwayjs/i18n": "~3.20.3", "@midwayjs/core": "3.20.11",
"@midwayjs/info": "~3.20.3", "@midwayjs/i18n": "3.20.13",
"@midwayjs/koa": "~3.20.3", "@midwayjs/info": "3.20.13",
"@midwayjs/logger": "~3.4.2", "@midwayjs/koa": "3.20.13",
"@midwayjs/typeorm": "~3.20.3", "@midwayjs/logger": "3.4.2",
"@midwayjs/upload": "^3.20.3", "@midwayjs/typeorm": "3.20.11",
"@midwayjs/upload": "3.20.13",
"@midwayjs/validate": "3.20.13",
"better-sqlite3": "^11.1.2", "better-sqlite3": "^11.1.2",
"cross-env": "^7.0.3", "cross-env": "^7.0.3",
"dayjs": "^1.11.7", "dayjs": "^1.11.7",
@@ -61,5 +63,5 @@
"typeorm": "^0.3.11", "typeorm": "^0.3.11",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -8,3 +8,5 @@ export * from './common-exception.js';
export * from './not-found-exception.js'; export * from './not-found-exception.js';
export * from './param-exception.js'; export * from './param-exception.js';
export * from './site-off-exception.js'; export * from './site-off-exception.js';
export * from './login-error-exception.js'
export * from './code-error-exception.js'

View File

@@ -1,3 +1,4 @@
export * from './service/plus-service.js'; export * from './service/plus-service.js';
export * from './service/file-service.js'; export * from './service/file-service.js';
export * from './service/encryptor.js'; export * from './service/encryptor.js';
export * from './service/ocr-service.js';

View File

@@ -0,0 +1,24 @@
import { Inject, Provide, Scope, ScopeEnum } from "@midwayjs/core";
import { PlusService } from "./plus-service.js";
import { IOcrService } from "@certd/plugin-lib";
/**
*/
@Provide("ocrService")
@Scope(ScopeEnum.Request, { allowDowngrade: true })
export class OcrService implements IOcrService {
@Inject()
plusService: PlusService;
async doOcrFromImage(opts: { image: string }): Promise<{ texts: string[] }> {
const res = await this.plusService.requestWithToken({
url: "/activation/certd/ocr",
method: "post",
data: {
image: opts.image
}
});
return res;
}
}

View File

@@ -4,7 +4,7 @@ import { cache, http, HttpRequestConfig, logger } from '@certd/basic';
import { SysInstallInfo, SysLicenseInfo, SysSettingsService } from '../../settings/index.js'; import { SysInstallInfo, SysLicenseInfo, SysSettingsService } from '../../settings/index.js';
import { merge } from 'lodash-es'; import { merge } from 'lodash-es';
@Provide() @Provide("plusService")
@Scope(ScopeEnum.Request, { allowDowngrade: true }) @Scope(ScopeEnum.Request, { allowDowngrade: true })
export class PlusService { export class PlusService {
@Inject() @Inject()

View File

@@ -1,6 +1,13 @@
import { HttpClient, ILogger, utils } from "@certd/basic"; import { HttpClient, ILogger, utils } from "@certd/basic";
import {upperFirst} from "lodash-es"; import {upperFirst} from "lodash-es";
import { FormItemProps, PluginRequestHandleReq, Registrable } from "@certd/pipeline"; import {
accessRegistry,
FormItemProps,
IAccessService,
IServiceGetter,
PluginRequestHandleReq,
Registrable
} from "@certd/pipeline";
export type AddonRequestHandleReqInput<T = any> = { export type AddonRequestHandleReqInput<T = any> = {
@@ -23,6 +30,7 @@ export type AddonDefine = Registrable & {
input?: { input?: {
[key: string]: AddonInputDefine; [key: string]: AddonInputDefine;
}; };
showTest?: boolean;
}; };
export type AddonInstanceConfig = { export type AddonInstanceConfig = {
@@ -47,6 +55,7 @@ export type AddonContext = {
http: HttpClient; http: HttpClient;
logger: ILogger; logger: ILogger;
utils: typeof utils; utils: typeof utils;
serviceGetter: IServiceGetter;
}; };
export abstract class BaseAddon implements IAddon { export abstract class BaseAddon implements IAddon {
@@ -57,8 +66,45 @@ export abstract class BaseAddon implements IAddon {
// eslint-disable-next-line @typescript-eslint/no-empty-function // eslint-disable-next-line @typescript-eslint/no-empty-function
async onInstance() {} async onInstance() {}
async getAccess<T = any>(accessId: string | number, isCommon = false) {
if (accessId == null) {
throw new Error("您还没有配置授权");
}
const accessService = await this.ctx.serviceGetter.get<IAccessService>("accessService")
let res: any = null;
if (isCommon) {
res = await accessService.getCommonById(accessId);
} else {
res = await accessService.getById(accessId);
}
if (res == null) {
throw new Error("授权不存在,可能已被删除,请前往任务配置里面重新选择授权");
}
// @ts-ignore
if (this.logger?.addSecret) {
// 隐藏加密信息,不在日志中输出
const type = res._type;
const plugin = accessRegistry.get(type);
const define = plugin.define;
// @ts-ignore
const input = define.input;
for (const key in input) {
if (input[key].encrypt && res[key] != null) {
// @ts-ignore
this.logger.addSecret(res[key]);
}
}
}
return res as T;
}
setCtx(ctx: AddonContext) { setCtx(ctx: AddonContext) {
this.ctx = ctx; this.ctx = ctx;
this.http = ctx.http; this.http = ctx.http;

View File

@@ -1,5 +1,3 @@
export * from './api/index.js' export * from './api/index.js'
export * from './entity/addon.js' export * from './entity/addon.js'
export * from './service/addon-service.js' export * from './service/addon-service.js'
export * from './service/addon-getter.js'
export * from './service/addon-sys-getter.js'

View File

@@ -1,18 +0,0 @@
import { IAddonGetter } from "../api/index.js";
export class AddonGetter implements IAddonGetter {
userId: number;
getter: <T>(id: any, userId?: number) => Promise<T>;
constructor(userId: number, getter: (id: any, userId: number) => Promise<any>) {
this.userId = userId;
this.getter = getter;
}
async getById<T = any>(id: any) {
return await this.getter<T>(id, this.userId);
}
async getCommonById<T = any>(id: any) {
return await this.getter<T>(id, 0);
}
}

View File

@@ -1,16 +1,15 @@
import { Provide, Scope, ScopeEnum } from "@midwayjs/core"; import { Provide, Scope, ScopeEnum } from "@midwayjs/core";
import { InjectEntityModel } from "@midwayjs/typeorm"; import { InjectEntityModel } from "@midwayjs/typeorm";
import { In, Repository } from "typeorm"; import { In, Repository } from "typeorm";
import { AddonDefine, BaseService, PageReq, PermissionException, ValidateException } from "../../../index.js"; import { AddonDefine, BaseService, PageReq, ValidateException } from "../../../index.js";
import { addonRegistry, newAddon } from "../api/index.js"; import { addonRegistry } from "../api/index.js";
import { AddonEntity } from "../entity/addon.js"; import { AddonEntity } from "../entity/addon.js";
import { http, logger, utils } from "@certd/basic";
/** /**
* Addon * Addon
*/ */
@Provide() @Provide()
@Scope(ScopeEnum.Request, {allowDowngrade: true}) @Scope(ScopeEnum.Request, { allowDowngrade: true })
export class AddonService extends BaseService<AddonEntity> { export class AddonService extends BaseService<AddonEntity> {
@InjectEntityModel(AddonEntity) @InjectEntityModel(AddonEntity)
repository: Repository<AddonEntity>; repository: Repository<AddonEntity>;
@@ -30,21 +29,21 @@ export class AddonService extends BaseService<AddonEntity> {
async add(param) { async add(param) {
let oldEntity = null; let oldEntity = null;
if (param._copyFrom){ if (param._copyFrom) {
oldEntity = await this.info(param._copyFrom); oldEntity = await this.info(param._copyFrom);
if (oldEntity == null) { if (oldEntity == null) {
throw new ValidateException('该Addon配置不存在,请确认是否已被删除'); throw new ValidateException("该Addon配置不存在,请确认是否已被删除");
} }
if (oldEntity.userId !== param.userId) { if (oldEntity.userId !== param.userId) {
throw new ValidateException('您无权查看该Addon配置'); throw new ValidateException("您无权查看该Addon配置");
} }
} }
if (!param.userId){ if (!param.userId) {
param.isSystem = true param.isSystem = true;
}else{ } else {
param.isSystem = false param.isSystem = false;
} }
delete param._copyFrom delete param._copyFrom;
return await super.add(param); return await super.add(param);
} }
@@ -56,7 +55,7 @@ export class AddonService extends BaseService<AddonEntity> {
async update(param) { async update(param) {
const oldEntity = await this.info(param.id); const oldEntity = await this.info(param.id);
if (oldEntity == null) { if (oldEntity == null) {
throw new ValidateException('该Addon配置不存在,请确认是否已被删除'); throw new ValidateException("该Addon配置不存在,请确认是否已被删除");
} }
return await super.update(param); return await super.update(param);
} }
@@ -64,63 +63,24 @@ export class AddonService extends BaseService<AddonEntity> {
async getSimpleInfo(id: number) { async getSimpleInfo(id: number) {
const entity = await this.info(id); const entity = await this.info(id);
if (entity == null) { if (entity == null) {
throw new ValidateException('该Addon配置不存在,请确认是否已被删除'); throw new ValidateException("该Addon配置不存在,请确认是否已被删除");
} }
return { return {
id: entity.id, id: entity.id,
name: entity.name, name: entity.name,
userId: entity.userId, userId: entity.userId,
addonType: entity.addonType, addonType: entity.addonType,
type: entity.type, type: entity.type
}; };
} }
async getAddonById(id: any, checkUserId: boolean, userId?: number): Promise<any> {
const ctx = {
http: http,
logger: logger,
utils: utils,
};
if (!id){
//使用图片验证码
return await newAddon("captcha", "image", {},ctx);
}
const entity = await this.info(id);
if (entity == null) {
//使用图片验证码
return await newAddon("captcha", "image", {},ctx);
}
if (checkUserId) {
if (userId == null) {
throw new ValidateException('userId不能为空');
}
if (userId !== entity.userId) {
throw new PermissionException('您对该Addon无访问权限');
}
}
const setting = JSON.parse(entity.setting ??"{}")
const input = {
id: entity.id,
...setting,
};
return await newAddon(entity.addonType, entity.type, input,ctx);
}
async getById(id: any, userId: number): Promise<any> {
return await this.getAddonById(id, true, userId);
}
getDefineList(addonType: string) { getDefineList(addonType: string) {
return addonRegistry.getDefineList(); return addonRegistry.getDefineList();
} }
getDefineByType(type: string,prefix?: string) { getDefineByType(type: string, prefix?: string) {
return addonRegistry.getDefine(type,prefix) as AddonDefine; return addonRegistry.getDefine(type, prefix) as AddonDefine;
} }
@@ -134,31 +94,30 @@ export class AddonService extends BaseService<AddonEntity> {
return await this.repository.find({ return await this.repository.find({
where: { where: {
id: In(ids), id: In(ids),
userId, userId
}, },
select: { select: {
id: true, id: true,
name: true, name: true,
addonType: true, addonType: true,
type: true, type: true,
userId:true, userId: true,
isSystem: true, isSystem: true
}, }
}); });
} }
async getDefault(userId: number, addonType: string): Promise<any> {
async getDefault(userId: number,addonType: string): Promise<any> {
const res = await this.repository.findOne({ const res = await this.repository.findOne({
where: { where: {
userId, userId,
addonType addonType
}, },
order: { order: {
isDefault: 'DESC', isDefault: "DESC"
}, }
}); });
if (!res) { if (!res) {
return null; return null;
@@ -174,16 +133,16 @@ export class AddonService extends BaseService<AddonEntity> {
type: res.type, type: res.type,
name: res.name, name: res.name,
userId: res.userId, userId: res.userId,
setting, setting
}; };
} }
async setDefault(id: number, userId: number,addonType:string) { async setDefault(id: number, userId: number, addonType: string) {
if (!id) { if (!id) {
throw new ValidateException('id不能为空'); throw new ValidateException("id不能为空");
} }
if (!userId) { if (!userId) {
throw new ValidateException('userId不能为空'); throw new ValidateException("userId不能为空");
} }
await this.repository.update( await this.repository.update(
{ {
@@ -191,7 +150,7 @@ export class AddonService extends BaseService<AddonEntity> {
addonType addonType
}, },
{ {
isDefault: false, isDefault: false
} }
); );
await this.repository.update( await this.repository.update(
@@ -201,22 +160,22 @@ export class AddonService extends BaseService<AddonEntity> {
addonType addonType
}, },
{ {
isDefault: true, isDefault: true
} }
); );
} }
async getOrCreateDefault(opts:{addonType:string,type:string, inputs: any, userId: any}) { async getOrCreateDefault(opts: { addonType: string, type: string, inputs: any, userId: any }) {
const {addonType,type,inputs,userId} = opts; const { addonType, type, inputs, userId } = opts;
const addonDefine = this.getDefineByType( type,addonType) const addonDefine = this.getDefineByType(type, addonType);
const defaultConfig = await this.getDefault(userId,addonType); const defaultConfig = await this.getDefault(userId, addonType);
if (defaultConfig) { if (defaultConfig) {
return defaultConfig; return defaultConfig;
} }
const setting = { const setting = {
...inputs, ...inputs
}; };
const res = await this.repository.save({ const res = await this.repository.save({
userId, userId,
@@ -224,7 +183,7 @@ export class AddonService extends BaseService<AddonEntity> {
type: type, type: type,
name: addonDefine.title, name: addonDefine.title,
setting: JSON.stringify(setting), setting: JSON.stringify(setting),
isDefault: true, isDefault: true
}); });
return this.buildAddonInstanceConfig(res); return this.buildAddonInstanceConfig(res);
} }

View File

@@ -1,17 +0,0 @@
import { IAccessService } from '@certd/pipeline';
import { AddonService } from './addon-service.js';
export class AddonSysGetter implements IAccessService {
addonService: AddonService;
constructor(addonService: AddonService) {
this.addonService = addonService;
}
async getById<T = any>(id: any) {
return await this.addonService.getById(id, 0);
}
async getCommonById<T = any>(id: any) {
return await this.addonService.getById(id, 0);
}
}

View File

@@ -3,6 +3,38 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
**Note:** Version bump only for package @certd/midway-flyway-js
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
**Note:** Version bump only for package @certd/midway-flyway-js
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
**Note:** Version bump only for package @certd/midway-flyway-js **Note:** Version bump only for package @certd/midway-flyway-js

View File

@@ -1,6 +1,6 @@
{ {
"name": "@certd/midway-flyway-js", "name": "@certd/midway-flyway-js",
"version": "1.36.19", "version": "1.37.1",
"description": "midway with flyway, sql upgrade way ", "description": "midway with flyway, sql upgrade way ",
"private": false, "private": false,
"type": "module", "type": "module",
@@ -25,9 +25,9 @@
], ],
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@midwayjs/core": "~3.20.3", "@midwayjs/core": "3.20.11",
"@midwayjs/logger": "~3.4.2", "@midwayjs/logger": "3.4.2",
"@midwayjs/typeorm": "~3.20.3", "@midwayjs/typeorm": "3.20.11",
"better-sqlite3": "^11.1.2" "better-sqlite3": "^11.1.2"
}, },
"devDependencies": { "devDependencies": {
@@ -46,5 +46,5 @@
"typeorm": "^0.3.11", "typeorm": "^0.3.11",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -3,6 +3,58 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
### Performance Improvements
* cname主域名校验提示优化显示不一致的两方便于排查问题 ([6ebb365](https://github.com/certd/certd/commit/6ebb3659f42155e4e8da600c493fb5227cd08137))
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/plugin-cert
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/plugin-cert
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/plugin-cert
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
### Performance Improvements
* 手动上传证书优化,增加到期前报错提醒 ([3d42bfd](https://github.com/certd/certd/commit/3d42bfd479eaacc4a49c401224815a6e2a0204b0))
* 支持腾讯云验证码 ([03f317f](https://github.com/certd/certd/commit/03f317ffdb6595ce70e8a2302b05f390c52110c8))
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
### Bug Fixes
* 修复旧版本升级上来报错eab授权的bug ([b76f2e2](https://github.com/certd/certd/commit/b76f2e2008a7fefac4c91179c45c56c7a7a84b71))
### Performance Improvements
* add preferred chain for google trust service ([#539](https://github.com/certd/certd/issues/539)) @ZeroClover ([e31d26a](https://github.com/certd/certd/commit/e31d26a8871c6088d9f8c0f580746ff2a810ae0c))
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
### Bug Fixes
* 修复ssl.com报EMAILADDRESS数量不对的bug ([c560cc5](https://github.com/certd/certd/commit/c560cc5adda6e15bf3a8865d874042550a6c2688))
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Bug Fixes
* 修复证书手动托管时新上传的证书无效的bug ([506385e](https://github.com/certd/certd/commit/506385e5a2600887fe30854e0713583caaa2e689))
### Performance Improvements
* 证书到期剩余天数进度条根据实际证书有效期计算 ([#528](https://github.com/certd/certd/issues/528)) nicheng-he ([2d4586b](https://github.com/certd/certd/commit/2d4586b1c42c39f97d2a95b9453cca4bc8bfbe61))
* add preferred chain option ([#519](https://github.com/certd/certd/issues/519)) @ZeroClover ([902359f](https://github.com/certd/certd/commit/902359f24ed12eee4f9b65178f1d6a60378351d2))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Performance Improvements ### Performance Improvements

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/plugin-cert", "name": "@certd/plugin-cert",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"main": "./dist/index.js", "main": "./dist/index.js",
"types": "./dist/index.d.ts", "types": "./dist/index.d.ts",
@@ -16,10 +16,10 @@
"pub": "npm publish" "pub": "npm publish"
}, },
"dependencies": { "dependencies": {
"@certd/acme-client": "^1.36.19", "@certd/acme-client": "^1.37.1",
"@certd/basic": "^1.36.19", "@certd/basic": "^1.37.1",
"@certd/pipeline": "^1.36.19", "@certd/pipeline": "^1.37.1",
"@certd/plugin-lib": "^1.36.19", "@certd/plugin-lib": "^1.37.1",
"@google-cloud/publicca": "^1.3.0", "@google-cloud/publicca": "^1.3.0",
"dayjs": "^1.11.7", "dayjs": "^1.11.7",
"jszip": "^3.10.1", "jszip": "^3.10.1",
@@ -43,5 +43,5 @@
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -1,5 +1,5 @@
import { HttpClient, ILogger, utils } from "@certd/basic"; import { HttpClient, ILogger, utils } from "@certd/basic";
import { IAccess, Registrable } from "@certd/pipeline"; import { IAccess, IServiceGetter, Registrable } from "@certd/pipeline";
export type DnsProviderDefine = Registrable & { export type DnsProviderDefine = Registrable & {
accessType: string; accessType: string;
@@ -25,6 +25,7 @@ export type DnsProviderContext = {
http: HttpClient; http: HttpClient;
utils: typeof utils; utils: typeof utils;
domainParser: IDomainParser; domainParser: IDomainParser;
serviceGetter: IServiceGetter;
}; };
export interface IDnsProvider<T = any> { export interface IDnsProvider<T = any> {

View File

@@ -2,3 +2,4 @@ export * from "./api.js";
export * from "./registry.js"; export * from "./registry.js";
export * from "./decorator.js"; export * from "./decorator.js";
export * from "./base.js"; export * from "./base.js";
export * from "./domain-parser.js";

View File

@@ -374,7 +374,7 @@ export class AcmeService {
commonName, commonName,
...csrInfo, ...csrInfo,
altNames, altNames,
emailAddress: email, // emailAddress: email,
}, },
privateKey privateKey
); );

View File

@@ -99,6 +99,7 @@ export abstract class CertApplyBaseConvertPlugin extends AbstractTaskPlugin {
const cert: CertInfo = certReader.toCertInfo(); const cert: CertInfo = certReader.toCertInfo();
this.cert = cert; this.cert = cert;
this._result.pipelineVars.certEffectiveTime = dayjs(certReader.detail.notBefore).valueOf();
this._result.pipelineVars.certExpiresTime = dayjs(certReader.detail.notAfter).valueOf(); this._result.pipelineVars.certExpiresTime = dayjs(certReader.detail.notAfter).valueOf();
if (!this._result.pipelinePrivateVars) { if (!this._result.pipelinePrivateVars) {
this._result.pipelinePrivateVars = {}; this._result.pipelinePrivateVars = {};

View File

@@ -35,6 +35,7 @@ export class CertReader {
detail: CertificateInfo; detail: CertificateInfo;
//毫秒时间戳 //毫秒时间戳
effective: number;
expires: number; expires: number;
constructor(certInfo: CertInfo) { constructor(certInfo: CertInfo) {
this.cert = certInfo; this.cert = certInfo;
@@ -52,8 +53,9 @@ export class CertReader {
} }
try { try {
const { detail, expires } = this.getCrtDetail(this.cert.crt); const { detail, effective, expires } = this.getCrtDetail(this.cert.crt);
this.detail = detail; this.detail = detail;
this.effective = effective.getTime();
this.expires = expires.getTime(); this.expires = expires.getTime();
} catch (e) { } catch (e) {
throw new Error("证书解析失败:" + e.message); throw new Error("证书解析失败:" + e.message);
@@ -102,8 +104,9 @@ export class CertReader {
static readCertDetail(crt: string) { static readCertDetail(crt: string) {
const detail = crypto.readCertificateInfo(crt.toString()); const detail = crypto.readCertificateInfo(crt.toString());
const effective = detail.notBefore;
const expires = detail.notAfter; const expires = detail.notAfter;
return { detail, expires }; return { detail, effective, expires };
} }
getAllDomains() { getAllDomains() {

View File

@@ -6,6 +6,7 @@ import dayjs from "dayjs";
export { CertReader }; export { CertReader };
export type { CertInfo }; export type { CertInfo };
@IsTaskPlugin({ @IsTaskPlugin({
name: "CertApplyUpload", name: "CertApplyUpload",
icon: "ph:certificate", icon: "ph:certificate",
@@ -62,6 +63,19 @@ export type { CertInfo };
}, },
}) })
export class CertApplyUploadPlugin extends CertApplyBaseConvertPlugin { export class CertApplyUploadPlugin extends CertApplyBaseConvertPlugin {
@TaskInput({
title: "过期前提醒",
value: 10,
component: {
name: "a-input-number",
vModel: "value",
},
required: true,
order: 100,
helper: "到期前多少天提醒",
})
renewDays!: number;
@TaskInput({ @TaskInput({
title: "手动上传证书", title: "手动上传证书",
component: { component: {
@@ -97,6 +111,7 @@ export class CertApplyUploadPlugin extends CertApplyBaseConvertPlugin {
this.userContext = this.ctx.userContext; this.userContext = this.ctx.userContext;
this.lastStatus = this.ctx.lastStatus as Step; this.lastStatus = this.ctx.lastStatus as Step;
} }
async onInit(): Promise<void> {} async onInit(): Promise<void> {}
async getCertFromStore() { async getCertFromStore() {
@@ -107,44 +122,54 @@ export class CertApplyUploadPlugin extends CertApplyBaseConvertPlugin {
} catch (e) { } catch (e) {
this.logger.warn("读取cert失败", e); this.logger.warn("读取cert失败", e);
} }
if (certReader == null) {
certReader = new CertReader(this.uploadCert);
}
if (!certReader.expires || certReader.expires < new Date().getTime()) {
throw new Error("证书已过期,停止部署,请重新上传证书");
}
return certReader; return certReader;
} }
async execute(): Promise<string | void> { private checkExpires(certReader: CertReader) {
const certReader = await this.getCertFromStore(); const renewDays = (this.renewDays ?? 10) * 24 * 60 * 60 * 1000;
const crtMd5 = this.ctx.utils.hash.md5(certReader.cert.crt); if (certReader.expires) {
if (certReader.expires < new Date().getTime()) {
const leftDays = dayjs(certReader.expires).diff(dayjs(), "day"); throw new Error("证书已过期,停止部署,请尽快上传新证书");
this.logger.info(`证书过期时间${dayjs(certReader.expires).format("YYYY-MM-DD HH:mm:ss")},剩余${leftDays}`); }
if (certReader.expires < new Date().getTime() + renewDays) {
if (!this.ctx.inputChanged) { throw new Error("证书即将已过期,停止部署,请尽快上传新证书");
this.logger.info("输入参数无变化"); }
const lastCrtMd5 = this.lastStatus?.status?.output?.certMd5; }
this.logger.info("证书MD5", crtMd5); }
this.logger.info("上次证书MD5", lastCrtMd5);
if (lastCrtMd5 === crtMd5) { async execute(): Promise<string | void> {
this.logger.info("证书无变化,跳过"); const oldCertReader = await this.getCertFromStore();
//输出证书MD5 if (oldCertReader) {
this.certMd5 = crtMd5; const leftDays = dayjs(oldCertReader.expires).diff(dayjs(), "day");
await this.output(certReader, false); this.logger.info(`证书过期时间${dayjs(oldCertReader.expires).format("YYYY-MM-DD HH:mm:ss")},剩余${leftDays}`);
return "skip"; this.checkExpires(oldCertReader);
if (!this.ctx.inputChanged) {
this.logger.info("输入参数无变化");
const lastCrtMd5 = this.lastStatus?.status?.output?.certMd5;
const newCrtMd5 = this.ctx.utils.hash.md5(this.uploadCert.crt);
this.logger.info("证书MD5", newCrtMd5);
this.logger.info("上次证书MD5", lastCrtMd5);
if (lastCrtMd5 === newCrtMd5) {
this.logger.info("证书无变化,跳过");
//输出证书MD5
this.certMd5 = newCrtMd5;
await this.output(oldCertReader, false);
return "skip";
}
this.logger.info("证书有变化,重新部署");
} else {
this.logger.info("输入参数有变化,重新部署");
} }
this.logger.info("证书有变化,重新部署");
} else {
this.logger.info("输入参数有变化,重新部署");
} }
const newCertReader = new CertReader(this.uploadCert);
this.clearLastStatus(); this.clearLastStatus();
//输出证书MD5 //输出证书MD5
this.certMd5 = crtMd5; this.certMd5 = this.ctx.utils.hash.md5(newCertReader.cert.crt);
await this.output(certReader, true); const newLeftDays = dayjs(newCertReader.expires).diff(dayjs(), "day");
this.logger.info(`新证书过期时间${dayjs(newCertReader.expires).format("YYYY-MM-DD HH:mm:ss")},剩余${newLeftDays}`);
this.checkExpires(newCertReader);
await this.output(newCertReader, true);
//必须output之后执行 //必须output之后执行
await this.emitCertApplySuccess(); await this.emitCertApplySuccess();

View File

@@ -38,6 +38,53 @@ export type DomainsVerifyPlanInput = {
[key: string]: DomainVerifyPlanInput; [key: string]: DomainVerifyPlanInput;
}; };
const preferredChainConfigs = {
letsencrypt: {
helper: "如无特殊需求保持默认即可",
options: [
{ value: "ISRG Root X1", label: "ISRG Root X1" },
{ value: "ISRG Root X2", label: "ISRG Root X2" },
],
},
google: {
helper: "GlobalSign 提供对老旧设备更好的兼容性,但证书链会变长",
options: [
{ value: "GTS Root R1", label: "GTS Root R1" },
{ value: "GlobalSign", label: "GlobalSign" },
],
},
} as const;
const preferredChainSupportedProviders = Object.keys(preferredChainConfigs);
const preferredChainMergeScript = (() => {
const configs = JSON.stringify(preferredChainConfigs);
const supportedProviders = JSON.stringify(preferredChainSupportedProviders);
const defaultProvider = JSON.stringify(preferredChainSupportedProviders[0]);
return `
const chainConfigs = ${configs};
const supportedProviders = ${supportedProviders};
const defaultProvider = ${defaultProvider};
const getConfig = (provider)=> chainConfigs[provider] || chainConfigs[defaultProvider];
return {
show: ctx.compute(({form})=> supportedProviders.includes(form.sslProvider)),
component: {
options: ctx.compute(({form})=> getConfig(form.sslProvider).options)
},
helper: ctx.compute(({form})=> getConfig(form.sslProvider).helper),
value: ctx.compute(({form})=>{
const { options } = getConfig(form.sslProvider);
const allowed = options.map(item=>item.value);
const current = form.preferredChain;
if(allowed.includes(current)){
return current;
}
return allowed[0];
})
};
`;
})();
@IsTaskPlugin({ @IsTaskPlugin({
name: "CertApply", name: "CertApply",
title: "证书申请JS版", title: "证书申请JS版",
@@ -92,7 +139,7 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
{ value: "sslcom", label: "SSL.com仅主域名和www免费", icon: "la:expeditedssl" }, { value: "sslcom", label: "SSL.com仅主域名和www免费", icon: "la:expeditedssl" },
], ],
}, },
helper: "Let's Encrypt申请最简单\nGoogle大厂光环兼容性好仅首次需要翻墙获取EAB授权\nZeroSSL需要EAB授权无需翻墙", helper: "Let's Encrypt申请最简单\nGoogle大厂光环兼容性好仅首次需要翻墙获取EAB授权\nZeroSSL需要EAB授权无需翻墙\nSSL.com仅主域名和www免费,必须设置CAA记录",
required: true, required: true,
}) })
sslProvider!: SSLProvider; sslProvider!: SSLProvider;
@@ -294,24 +341,14 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
@TaskInput({ @TaskInput({
title: "首选链", title: "首选链",
value: "ISRG Root X1",
component: { component: {
name: "a-select", name: "a-select",
vModel: "value", vModel: "value",
options: [ options: preferredChainConfigs.letsencrypt.options,
{ value: "ISRG Root X1", label: "ISRG Root X1" },
{ value: "ISRG Root X2", label: "ISRG Root X2" },
],
}, },
helper: "仅 Let's Encrypt 可选,默认为 ISRG Root X1", helper: preferredChainConfigs.letsencrypt.helper,
required: false, required: false,
mergeScript: ` mergeScript: preferredChainMergeScript,
return {
show: ctx.compute(({form})=>{
return form.sslProvider === 'letsencrypt'
})
}
`,
}) })
preferredChain!: string; preferredChain!: string;
@@ -375,7 +412,7 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
async onInit() { async onInit() {
let eab: EabAccess = null; let eab: EabAccess = null;
if (this.sslProvider !== "letsencrypt") { if (this.sslProvider && this.sslProvider !== "letsencrypt") {
if (this.sslProvider === "google" && this.googleAccessId) { if (this.sslProvider === "google" && this.googleAccessId) {
this.logger.info("当前正在使用 google服务账号授权获取EAB"); this.logger.info("当前正在使用 google服务账号授权获取EAB");
const googleAccess = await this.getAccess(this.googleAccessId); const googleAccess = await this.getAccess(this.googleAccessId);
@@ -487,6 +524,7 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
http: this.ctx.http, http: this.ctx.http,
utils, utils,
domainParser, domainParser,
serviceGetter: this.ctx.serviceGetter,
}; };
return await createDnsProvider({ return await createDnsProvider({
dnsProviderType, dnsProviderType,
@@ -503,7 +541,7 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
const mainDomain = await domainParser.parse(domain); const mainDomain = await domainParser.parse(domain);
const planSetting: DomainVerifyPlanInput = verifyPlanSetting[mainDomain]; const planSetting: DomainVerifyPlanInput = verifyPlanSetting[mainDomain];
if (planSetting == null) { if (planSetting == null) {
throw new Error(`没有找到域名(${domain})的校验计划`); throw new Error(`没有找到域名(${domain})的校验计划如果您在流水线创建之后设置了子域名托管需要重新编辑证书申请任务和重新校验cname记录的校验状态`);
} }
if (planSetting.type === "dns") { if (planSetting.type === "dns") {
plan[domain] = await this.createDnsDomainVerifyPlan(planSetting, domain, mainDomain); plan[domain] = await this.createDnsDomainVerifyPlan(planSetting, domain, mainDomain);
@@ -592,10 +630,20 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
if (cnameRecord == null) { if (cnameRecord == null) {
throw new Error(`请先配置${domain}的CNAME记录并通过校验`); throw new Error(`请先配置${domain}的CNAME记录并通过校验`);
} }
if (cnameRecord.status !== "valid") {
throw new Error(`CNAME记录${domain}的校验状态为${cnameRecord.status},请等待校验通过`);
}
// 主域名异常
if (cnameRecord.mainDomain && mainDomain && cnameRecord.mainDomain !== mainDomain) {
throw new Error(`CNAME记录${domain}的域名与配置的主域名不一致(${cnameRecord.mainDomain}${mainDomain}请确认是否在流水线创建之后修改了子域名托管您需要重新校验CNAME记录的校验状态`);
}
let dnsProvider = cnameRecord.commonDnsProvider; let dnsProvider = cnameRecord.commonDnsProvider;
if (cnameRecord.cnameProvider.id > 0) { if (cnameRecord.cnameProvider.id > 0) {
dnsProvider = await this.createDnsProvider(cnameRecord.cnameProvider.dnsProviderType, cnameRecord.cnameProvider.access); dnsProvider = await this.createDnsProvider(cnameRecord.cnameProvider.dnsProviderType, cnameRecord.cnameProvider.access);
} }
return { return {
type: "cname", type: "cname",
domain, domain,

View File

@@ -3,6 +3,42 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
### Performance Improvements
* dns解析支持阿里esa ([9291fa6](https://github.com/certd/certd/commit/9291fa68aa7a88a05c2f888bf3048df36a8fbde3))
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/plugin-lib
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/plugin-lib
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/plugin-lib
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
**Note:** Version bump only for package @certd/plugin-lib
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
**Note:** Version bump only for package @certd/plugin-lib
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
**Note:** Version bump only for package @certd/plugin-lib
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Performance Improvements
* ssh配置增加脚本类型设置bash还是sh ([ae41c60](https://github.com/certd/certd/commit/ae41c6038b27c9476e64a2402a8daf247c38a5b6))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Performance Improvements ### Performance Improvements

View File

@@ -1,7 +1,7 @@
{ {
"name": "@certd/plugin-lib", "name": "@certd/plugin-lib",
"private": false, "private": false,
"version": "1.36.19", "version": "1.37.1",
"type": "module", "type": "module",
"main": "./dist/index.js", "main": "./dist/index.js",
"types": "./dist/index.d.ts", "types": "./dist/index.d.ts",
@@ -21,8 +21,8 @@
"@alicloud/pop-core": "^1.7.10", "@alicloud/pop-core": "^1.7.10",
"@alicloud/tea-util": "^1.4.10", "@alicloud/tea-util": "^1.4.10",
"@aws-sdk/client-s3": "^3.787.0", "@aws-sdk/client-s3": "^3.787.0",
"@certd/basic": "^1.36.19", "@certd/basic": "^1.37.1",
"@certd/pipeline": "^1.36.19", "@certd/pipeline": "^1.37.1",
"@kubernetes/client-node": "0.21.0", "@kubernetes/client-node": "0.21.0",
"ali-oss": "^6.22.0", "ali-oss": "^6.22.0",
"basic-ftp": "^5.0.5", "basic-ftp": "^5.0.5",
@@ -53,5 +53,5 @@
"tslib": "^2.8.1", "tslib": "^2.8.1",
"typescript": "^5.4.2" "typescript": "^5.4.2"
}, },
"gitHead": "6d8981479517b5de9634e242c1ebf22e70527ec4" "gitHead": "e17cd1f298a72595f01e466b5e5c8971828c6bd9"
} }

View File

@@ -0,0 +1,45 @@
import { AccessInput, BaseAccess, IsAccess } from "@certd/pipeline";
@IsAccess({
name: "aliesa",
title: "阿里云ESA授权",
desc: "",
icon: "ant-design:aliyun-outlined",
order: 0,
})
export class AliesaAccess extends BaseAccess {
@AccessInput({
title: "阿里云授权",
component: {
name: "access-selector",
vModel: "modelValue",
type: "aliyun",
},
helper: "请选择阿里云授权",
required: true,
})
accessId = "";
@AccessInput({
title: "地区",
component: {
name: "a-select",
vModel: "value",
options: [
{
label: "杭州",
value: "cn-hangzhou",
},
{
label: "新加坡",
value: "ap-southeast-1",
},
],
},
helper: "请选择ESA地区",
required: true,
})
region = "";
}
new AliesaAccess();

View File

@@ -1,2 +1,3 @@
export * from "./aliyun-access.js"; export * from "./aliyun-access.js";
export * from "./alioss-access.js"; export * from "./alioss-access.js";
export * from "./aliesa-access.js";

View File

@@ -7,3 +7,4 @@ export * from "./qiniu/index.js";
export * from "./ctyun/index.js"; export * from "./ctyun/index.js";
export * from "./oss/index.js"; export * from "./oss/index.js";
export * from "./s3/index.js"; export * from "./s3/index.js";
export * from "./lib/index.js";

View File

@@ -0,0 +1 @@
export * from "./ocr-api.js";

View File

@@ -0,0 +1,3 @@
export interface IOcrService {
doOcrFromImage(opts: { image: string }): Promise<{ texts: string[] }>;
}

View File

@@ -10,6 +10,7 @@ RUN cp /workspace/certd-client/dist/* /workspace/certd-server/public/ -rf
RUN cd /workspace/certd-server && pnpm install && npm run build-on-docker RUN cd /workspace/certd-server && pnpm install && npm run build-on-docker
FROM node:22-alpine FROM node:22-alpine
EXPOSE 7001 EXPOSE 7001
EXPOSE 7002 EXPOSE 7002

View File

@@ -3,6 +3,65 @@
All notable changes to this project will be documented in this file. All notable changes to this project will be documented in this file.
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines. See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
## [1.37.1](https://github.com/certd/certd/compare/v1.37.0...v1.37.1) (2025-09-29)
### Bug Fixes
* 修复版本比较bug ([109696e](https://github.com/certd/certd/commit/109696e965d68c50c8627ffd40203edd1d2daea5))
# [1.37.0](https://github.com/certd/certd/compare/v1.36.25...v1.37.0) (2025-09-28)
**Note:** Version bump only for package @certd/ui-client
## [1.36.25](https://github.com/certd/certd/compare/v1.36.24...v1.36.25) (2025-09-27)
**Note:** Version bump only for package @certd/ui-client
## [1.36.24](https://github.com/certd/certd/compare/v1.36.23...v1.36.24) (2025-09-27)
**Note:** Version bump only for package @certd/ui-client
## [1.36.23](https://github.com/certd/certd/compare/v1.36.22...v1.36.23) (2025-09-26)
### Bug Fixes
* 授权页面id列位置不在第一列的bug ([3f1722d](https://github.com/certd/certd/commit/3f1722d54debcb4849dc14521a2da0d9b304b69f))
### Performance Improvements
* 动态加载验证码script ([dcc396a](https://github.com/certd/certd/commit/dcc396afb7a23aeb8af57c01014b09af5f033e61))
* 验证码支持测试,登录验证码需要测试通过后才能开启 ([83e6476](https://github.com/certd/certd/commit/83e6476408090b741fabb1b542fb458d9a8b4134))
## [1.36.22](https://github.com/certd/certd/compare/v1.36.21...v1.36.22) (2025-09-23)
### Bug Fixes
* 选择授权对话框编辑时名称字段排在最后的bug ([31cfb09](https://github.com/certd/certd/commit/31cfb09468bda3272f5f63af65ff3e9272220b39))
### Performance Improvements
* 登录失败时清除验证码状态 ([1c15bea](https://github.com/certd/certd/commit/1c15beadc7fe8a7c6ec1903b7e722ca2f52e05b3))
## [1.36.21](https://github.com/certd/certd/compare/v1.36.20...v1.36.21) (2025-09-15)
### Bug Fixes
* 修复导入插件对话框无法打开的bug修复插件编辑页面打开多个代码编辑器消失的bug ([e5a080a](https://github.com/certd/certd/commit/e5a080aebe0d2f3e3c0f86bf863f75069c1bf7ab))
## [1.36.20](https://github.com/certd/certd/compare/v1.36.19...v1.36.20) (2025-09-13)
### Bug Fixes
* 修复商业版退出登录后丢失站点个性化设置的bug ([d75dd05](https://github.com/certd/certd/commit/d75dd058d65c85f80c49e1fa7a910e6c6f08e824))
* 修复授权类型和名称字段排到最后的bug ([43b7977](https://github.com/certd/certd/commit/43b79778ea9034065f6a15af3296274315597c6b))
### Performance Improvements
* 登录支持极验验证码 ([370db62](https://github.com/certd/certd/commit/370db62bf0aece241859244927beabba32d6a257))
* 登录注册、找回密码都支持极验验证码和图片验证码 ([7bdde68](https://github.com/certd/certd/commit/7bdde68ecea29fe2c570fd3cb082139db6c93d93))
* 优化加量包展示效果 ([3c65f37](https://github.com/certd/certd/commit/3c65f37d84177ba107d4a6462648af12d2fc4b7a))
* 证书到期剩余天数进度条根据实际证书有效期计算 ([#528](https://github.com/certd/certd/issues/528)) nicheng-he ([2d4586b](https://github.com/certd/certd/commit/2d4586b1c42c39f97d2a95b9453cca4bc8bfbe61))
## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05) ## [1.36.19](https://github.com/certd/certd/compare/v1.36.18...v1.36.19) (2025-09-05)
### Bug Fixes ### Bug Fixes

View File

@@ -23,6 +23,7 @@
</div> </div>
<script type="module" src="/src/main.ts"></script> <script type="module" src="/src/main.ts"></script>
<script src="https://static.geetest.com/v4/gt4.js"></script> <!--<script src="https://static.geetest.com/v4/gt4.js"></script>-->
<!--<script src="https://turing.captcha.qcloud.com/TJCaptcha.js"></script>-->
</body> </body>
</html> </html>

View File

@@ -1,6 +1,6 @@
{ {
"name": "@certd/ui-client", "name": "@certd/ui-client",
"version": "1.36.19", "version": "1.37.1",
"private": true, "private": true,
"scripts": { "scripts": {
"dev": "vite --open", "dev": "vite --open",
@@ -32,10 +32,11 @@
"@aws-sdk/s3-request-presigner": "^3.535.0", "@aws-sdk/s3-request-presigner": "^3.535.0",
"@certd/vue-js-cron-light": "^4.0.14", "@certd/vue-js-cron-light": "^4.0.14",
"@ctrl/tinycolor": "^4.1.0", "@ctrl/tinycolor": "^4.1.0",
"@fast-crud/fast-crud": "^1.25.13", "@fast-crud/editor-code": "^1.26.6",
"@fast-crud/fast-extends": "^1.25.13", "@fast-crud/fast-crud": "^1.26.6",
"@fast-crud/ui-antdv4": "^1.25.13", "@fast-crud/fast-extends": "^1.26.6",
"@fast-crud/ui-interface": "^1.25.13", "@fast-crud/ui-antdv4": "^1.26.6",
"@fast-crud/ui-interface": "^1.26.6",
"@iconify/tailwind": "^1.2.0", "@iconify/tailwind": "^1.2.0",
"@iconify/vue": "^4.1.1", "@iconify/vue": "^4.1.1",
"@manypkg/get-packages": "^2.2.2", "@manypkg/get-packages": "^2.2.2",
@@ -96,6 +97,7 @@
"vue-cropperjs": "^5.0.0", "vue-cropperjs": "^5.0.0",
"vue-echarts": "^7.0.3", "vue-echarts": "^7.0.3",
"vue-i18n": "^9.10.2", "vue-i18n": "^9.10.2",
"vue-plugin-load-script": "2.1.1",
"vue-router": "^4.3.0", "vue-router": "^4.3.0",
"vuedraggable": "^4.1.0", "vuedraggable": "^4.1.0",
"watermark-js-plus": "^1.5.8", "watermark-js-plus": "^1.5.8",
@@ -103,8 +105,8 @@
"zod-defaults": "^0.1.3" "zod-defaults": "^0.1.3"
}, },
"devDependencies": { "devDependencies": {
"@certd/lib-iframe": "^1.36.19", "@certd/lib-iframe": "^1.37.1",
"@certd/pipeline": "^1.36.19", "@certd/pipeline": "^1.37.1",
"@rollup/plugin-commonjs": "^25.0.7", "@rollup/plugin-commonjs": "^25.0.7",
"@rollup/plugin-node-resolve": "^15.2.3", "@rollup/plugin-node-resolve": "^15.2.3",
"@types/chai": "^4.3.12", "@types/chai": "^4.3.12",
@@ -138,7 +140,7 @@
"prettier": "3.3.3", "prettier": "3.3.3",
"pretty-quick": "^4.0.0", "pretty-quick": "^4.0.0",
"rimraf": "^5.0.5", "rimraf": "^5.0.5",
"rollup": "^4.13.0", "rollup": "4.50.0",
"rollup-plugin-visualizer": "^5.12.0", "rollup-plugin-visualizer": "^5.12.0",
"stylelint": "^15.11.0", "stylelint": "^15.11.0",
"stylelint-order": "^6.0.4", "stylelint-order": "^6.0.4",
@@ -148,7 +150,7 @@
"tslint": "^6.1.3", "tslint": "^6.1.3",
"typescript": "^5.4.2", "typescript": "^5.4.2",
"unplugin-vue-define-options": "^1.4.2", "unplugin-vue-define-options": "^1.4.2",
"vite": "^5.3.1", "vite": "5.4.19",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-html": "^3.2.2", "vite-plugin-html": "^3.2.2",
"vite-plugin-theme": "^0.8.6", "vite-plugin-theme": "^0.8.6",

View File

@@ -1,5 +1,5 @@
<template> <template>
<component :is="captchaComponent" v-if="settingStore.inited" ref="captchaRef" class="captcha_input" :captcha-get="getCaptcha" @change="onChange" /> <component :is="captchaComponent" v-if="settingStore.inited" ref="captchaRef" :model-value="modelValue" class="captcha_input" :captcha-get="getCaptcha" @change="onChange" />
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, computed, defineAsyncComponent } from "vue"; import { ref, computed, defineAsyncComponent } from "vue";
@@ -7,6 +7,20 @@ import { useSettingStore } from "/@/store/settings";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
import { request } from "/@/api/service"; import { request } from "/@/api/service";
const props = defineProps({
modelValue: {
type: Object,
default: () => ({}),
},
type: {
type: String,
default: "image",
},
addonId: {
type: Number,
default: 0,
},
});
const captchaRef = ref(null); const captchaRef = ref(null);
const settingStore = useSettingStore(); const settingStore = useSettingStore();
@@ -17,7 +31,7 @@ const captchaAddonId = computed(() => {
return settingStore.sysPublic.captchaAddonId ?? 0; return settingStore.sysPublic.captchaAddonId ?? 0;
}); });
const captchaComponent = computed(() => { const captchaComponent = computed(() => {
let type = "image"; let type: any = props.type ?? "image";
if (settingStore.sysPublic.captchaAddonId && settingStore.sysPublic.captchaType) { if (settingStore.sysPublic.captchaAddonId && settingStore.sysPublic.captchaType) {
type = settingStore.sysPublic.captchaType; type = settingStore.sysPublic.captchaType;
} }
@@ -36,7 +50,7 @@ async function getCaptcha(): Promise<any> {
}); });
} }
function onChange(data) { function onChange(data: any) {
emits("update:modelValue", data); emits("update:modelValue", data);
emits("change", data); emits("change", data);
} }
@@ -44,7 +58,11 @@ function onChange(data) {
async function getCaptchaForm() { async function getCaptchaForm() {
return await captchaRef.value.getCaptchaForm(); return await captchaRef.value.getCaptchaForm();
} }
async function reset() {
await captchaRef.value.reset();
}
defineExpose({ defineExpose({
getCaptchaForm, getCaptchaForm,
reset,
}); });
</script> </script>

View File

@@ -2,28 +2,32 @@
<div ref="captchaRef" class="geetest_captcha_wrapper"></div> <div ref="captchaRef" class="geetest_captcha_wrapper"></div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { onMounted, defineProps, defineEmits, ref, onUnmounted } from "vue"; import { onMounted, defineProps, defineEmits, ref, onUnmounted, Ref, watch } from "vue";
import { useSettingStore } from "/@/store/settings"; import { useSettingStore } from "/@/store/settings";
import { request } from "/src/api/service"; import { request } from "/src/api/service";
import { notification } from "ant-design-vue"; import { notification } from "ant-design-vue";
import { loadScript } from "vue-plugin-load-script";
const loaded = ref(false);
async function loadCaptchaScript() {
// 加载验证码js
await loadScript("https://static.geetest.com/v4/gt4.js");
loaded.value = true;
}
defineOptions({ defineOptions({
name: "GeetestCaptcha", name: "GeetestCaptcha",
}); });
const emit = defineEmits(["update:modelValue", "change"]); const emit = defineEmits(["update:modelValue", "change"]);
const props = defineProps<{ const props = defineProps<{
modelValue: any;
captchaGet: () => Promise<any>; captchaGet: () => Promise<any>;
}>(); }>();
const captchaRef = ref(null); const captchaRef = ref(null);
// const addonApi = createAddonApi();
const settingStore = useSettingStore();
const captchaInstanceRef = ref({}); const captchaInstanceRef: Ref = ref({});
async function init() { async function init() {
// if (!initGeetest4) { await loadCaptchaScript();
// await import("https://static.geetest.com/v4/gt4.js");
// }
const { captchaId } = await props.captchaGet(); const { captchaId } = await props.captchaGet();
// @ts-ignore // @ts-ignore
initGeetest4( initGeetest4(
@@ -35,6 +39,13 @@ async function init() {
captcha.appendTo(captchaRef.value); // 调用appendTo将验证码插入到页的某一个元素中这个元素用户可以自定义 captcha.appendTo(captchaRef.value); // 调用appendTo将验证码插入到页的某一个元素中这个元素用户可以自定义
captchaInstanceRef.value.instance = captcha; captchaInstanceRef.value.instance = captcha;
captchaInstanceRef.value.captchaId = captchaId; captchaInstanceRef.value.captchaId = captchaId;
captcha.onSuccess(function () {
const form = getCaptchaForm();
if (form) {
emitChange(form);
}
});
} }
); );
} }
@@ -58,29 +69,51 @@ function getCaptchaForm() {
return result; return result;
} }
const valueRef = ref(null); // const valueRef = ref(null);
const timeoutId = setInterval(() => { // const timeoutId = setInterval(() => {
const form = getCaptchaForm(); // const form = getCaptchaForm();
if (form && valueRef.value != form) { // if (form && valueRef.value != form) {
console.log("form", form); // console.log("form", form);
valueRef.value = form; // valueRef.value = form;
emitChange(form); // emitChange(form);
} // }
}, 1000); // }, 1000);
onUnmounted(() => { // onUnmounted(() => {
clearTimeout(timeoutId); // clearTimeout(timeoutId);
}); // });
function emitChange(value: string) { function emitChange(value: string) {
emit("update:modelValue", value); emit("update:modelValue", value);
emit("change", value); emit("change", value);
} }
function reset() {
captchaInstanceRef.value.instance.reset();
}
watch(
() => {
return props.modelValue;
},
value => {
if (value == null) {
reset();
}
}
);
defineExpose({ defineExpose({
getCaptchaForm, getCaptchaForm,
reset,
}); });
watch(
() => [props.captchaGet],
async () => {
await init();
}
);
onMounted(async () => { onMounted(async () => {
await init(); await init();
}); });

View File

@@ -11,10 +11,11 @@
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { defineEmits, defineExpose, defineProps, ref } from "vue"; import { defineEmits, defineExpose, defineProps, ref, watch } from "vue";
import { nanoid } from "nanoid"; import { nanoid } from "nanoid";
const props = defineProps<{ const props = defineProps<{
modelValue: any;
captchaGet?: () => Promise<any>; captchaGet?: () => Promise<any>;
}>(); }>();
defineOptions({ defineOptions({
@@ -42,6 +43,7 @@ function getCaptchaForm() {
defineExpose({ defineExpose({
resetImageCode, resetImageCode,
getCaptchaForm, getCaptchaForm,
reset: resetImageCode,
}); });
resetImageCode(); resetImageCode();
@@ -52,7 +54,18 @@ function onChange(value: string) {
emitChange(form); emitChange(form);
} }
function emitChange(value) { watch(
() => {
return props.modelValue;
},
value => {
if (value == null) {
resetImageCode();
}
}
);
function emitChange(value: any) {
emit("update:modelValue", value); emit("update:modelValue", value);
emit("change", value); emit("change", value);
} }

View File

@@ -0,0 +1,226 @@
<template>
<div ref="captchaRef" class="tencent_captcha_wrapper" :class="{ tencent_captcha_ok: modelValue }" @click="triggerCaptcha">
<div class="validation-box" :class="{ validated: modelValue != null }">
<div class="sweep-animation"></div>
<div class="box-content">
<div class="box-icon"></div>
<span v-if="modelValue == null" class="status-text">点击进行验证</span>
<span v-else class="status-text">验证成功</span>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { onMounted, defineProps, defineEmits, ref, onUnmounted, Ref, watch } from "vue";
import { notification } from "ant-design-vue";
import { loadScript } from "vue-plugin-load-script";
const loaded = ref(false);
async function loadCaptchaScript() {
// 加载验证码js
// var appid = "您的CaptchaAppId";
// loadScript("https://turing.captcha.qq.com/TJCaptcha.js?appid=" + appid);
await loadScript("https://turing.captcha.qcloud.com/TJCaptcha.js");
loaded.value = true;
}
loadCaptchaScript();
defineOptions({
name: "TencentCaptcha",
});
const emit = defineEmits(["update:modelValue", "change"]);
const props = defineProps<{
modelValue: any;
captchaGet: () => Promise<any>;
}>();
const captchaRef = ref(null);
const captchaInstanceRef: Ref = ref({});
// 定义回调函数
function callback(res: { ret: number; ticket: string; randstr: string; errorCode?: number; errorMessage?: string }) {
// 第一个参数传入回调结果,结果如下:
// ret Int 验证结果0验证成功。2用户主动关闭验证码。
// ticket String 验证成功的票据,当且仅当 ret = 0 时 ticket 有值。
// CaptchaAppId String 验证码应用ID。
// bizState Any 自定义透传参数。
// randstr String 本次验证的随机串,后续票据校验时需传递该参数。
// verifyDuration Int 验证码校验接口耗时ms
// actionDuration Int 操作校验成功耗时(用户动作+校验完成)(ms)。
// sid String 链路sid。
console.log("callback:", res);
// res用户主动关闭验证码= {ret: 2, ticket: null}
// res验证成功 = {ret: 0, ticket: "String", randstr: "String"}
// res请求验证码发生错误验证码自动返回trerror_前缀的容灾票据 = {ret: 0, ticket: "String", randstr: "String", errorCode: Number, errorMessage: "String"}
// 此处代码仅为验证结果的展示示例真实业务接入建议基于ticket和errorCode情况做不同的业务处理
if (res.ret === 0) {
emitChange({
ticket: res.ticket,
randstr: res.randstr,
});
} else if (res.ret === 2) {
console.log("用户主动关闭验证码");
}
}
// 定义验证码js加载错误处理函数
function loadErrorCallback(error: any) {
// var appid = "您的CaptchaAppId";
// // 生成容灾票据或自行做其它处理
// var ticket = "trerror_1001_" + appid + "_" + Math.floor(new Date().getTime() / 1000);
// callback({
// ret: 0,
// randstr: "@" + Math.random().toString(36).substr(2),
// ticket: ticket,
// errorCode: 1001,
// errorMessage: "jsload_error",
// });
notification.error({
message: `验证码加载失败:${error?.message || error}`,
});
}
async function triggerCaptcha() {
if (!loaded.value) {
notification.error({
message: "验证码还未加载完成,请稍后再试",
});
return;
}
const { captchaAppId } = await props.captchaGet();
try {
// 生成一个验证码对象
// CaptchaAppId登录验证码控制台从【验证管理】页面进行查看。如果未创建过验证请先新建验证。注意不可使用客户端类型为小程序的CaptchaAppId会导致数据统计错误。
//callback定义的回调函数
// @ts-ignore
var captcha = new TencentCaptcha(captchaAppId + "", callback, {
userLanguage: "zh-cn",
// showFn: (ret: any) => {
// const {
// duration, // 验证码渲染完成的耗时(ms)
// sid, // 链路sid
// } = ret;
// },
});
// 调用方法,显示验证码
captcha.show();
} catch (error) {
// 加载异常调用验证码js加载错误处理函数
loadErrorCallback(error);
}
}
function emitChange(value: any) {
emit("update:modelValue", value);
emit("change", value);
}
function reset() {
captchaInstanceRef.value.instance.reset();
}
watch(
() => {
return props.modelValue;
},
value => {
if (value == null) {
reset();
}
}
);
defineExpose({
reset,
});
</script>
<style lang="less">
.tencent_captcha_wrapper {
.validation-box {
width: 100%;
height: 40px;
margin: 0 auto 30px;
border: 1px solid #ddd;
border-radius: 8px;
display: flex;
justify-content: center;
align-items: center;
cursor: pointer;
transition: all 0.3s ease;
position: relative;
overflow: hidden;
background-color: #f9f9f9;
}
.validation-box:hover {
border-color: #aaa;
background-color: #f0f0f0;
}
.validation-box.validated {
border-color: #4caf50;
background-color: #f1f8e9;
}
.box-content {
display: flex;
align-items: center;
justify-content: center;
z-index: 2;
position: relative;
}
.box-icon {
font-size: 18px;
color: #bbb;
margin-right: 15px;
transition: all 0.3s ease;
}
.validation-box.validated .box-icon {
color: #4caf50;
}
.status-text {
font-size: 14px;
font-weight: 500;
color: #888;
transition: all 0.3s ease;
}
.validation-box.validated .status-text {
color: #4caf50;
font-weight: 600;
}
/* 划过动画效果 */
.sweep-animation {
position: absolute;
top: 0;
left: -100%;
width: 100%;
height: 100%;
background: linear-gradient(90deg, transparent, rgba(76, 175, 80, 0.2), transparent);
z-index: 1;
opacity: 0;
transition: opacity 0.3s;
}
.validation-box.validated .sweep-animation {
animation: sweep 0.8s ease forwards;
opacity: 1;
}
@keyframes sweep {
0% {
left: -100%;
}
50% {
left: 0;
}
100% {
left: 100%;
}
}
}
</style>

View File

@@ -45,6 +45,16 @@ export async function DoVerify(id: number) {
}); });
} }
export async function ResetStatus(id: number) {
return await request({
url: apiPrefix + "/resetStatus",
method: "post",
data: {
id,
},
});
}
export async function ParseDomain(fullDomain: string) { export async function ParseDomain(fullDomain: string) {
return await request({ return await request({
url: subDomainApiPrefix + "/parseDomain", url: subDomainApiPrefix + "/parseDomain",

View File

@@ -16,6 +16,9 @@
<a-tooltip v-if="cnameRecord.error" :title="cnameRecord.error"> <a-tooltip v-if="cnameRecord.error" :title="cnameRecord.error">
<fs-icon class="ml-5 color-red" icon="ion:warning-outline"></fs-icon> <fs-icon class="ml-5 color-red" icon="ion:warning-outline"></fs-icon>
</a-tooltip> </a-tooltip>
<a-tooltip v-if="cnameRecord.status === 'valid'" title="重置校验状态,重新校验">
<fs-icon class="ml-2 color-yellow text-md pointer" icon="solar:undo-left-square-bold" @click="resetStatus"></fs-icon>
</a-tooltip>
</td> </td>
<td class="center"> <td class="center">
<template v-if="cnameRecord.status !== 'valid'"> <template v-if="cnameRecord.status !== 'valid'">
@@ -35,6 +38,7 @@ import { ref, watch } from "vue";
import { dict } from "@fast-crud/fast-crud"; import { dict } from "@fast-crud/fast-crud";
import * as api from "./api.js"; import * as api from "./api.js";
import CnameTip from "./cname-tip.vue"; import CnameTip from "./cname-tip.vue";
import { Modal } from "ant-design-vue";
const statusDict = dict({ const statusDict = dict({
data: [ data: [
{ label: "待设置CNAME", value: "cname", color: "warning" }, { label: "待设置CNAME", value: "cname", color: "warning" },
@@ -71,12 +75,15 @@ function onRecordChange() {
}); });
} }
async function loadRecord() {
cnameRecord.value = await GetByDomain(props.domain);
}
let refreshIntervalId: any = null; let refreshIntervalId: any = null;
async function doRefresh() { async function doRefresh() {
if (!props.domain) { if (!props.domain) {
return; return;
} }
cnameRecord.value = await GetByDomain(props.domain); await loadRecord();
onRecordChange(); onRecordChange();
if (cnameRecord.value.status === "validating") { if (cnameRecord.value.status === "validating") {
@@ -114,6 +121,17 @@ async function doVerify() {
} }
await doRefresh(); await doRefresh();
} }
async function resetStatus() {
Modal.confirm({
title: "重置状态",
content: "确定要重置校验状态吗?",
onOk: async () => {
await api.ResetStatus(cnameRecord.value.id);
await loadRecord();
},
});
}
</script> </script>
<style lang="less"> <style lang="less">

View File

@@ -119,6 +119,7 @@ export default {
scheduledTaskCount: "Scheduled Task Count", scheduledTaskCount: "Scheduled Task Count",
deployTaskCount: "Deployment Task Count", deployTaskCount: "Deployment Task Count",
remainingValidity: "Remaining Validity", remainingValidity: "Remaining Validity",
effectiveTime: "Effective time",
expiryTime: "Expiry Time", expiryTime: "Expiry Time",
status: "Status", status: "Status",
lastRun: "Last Run", lastRun: "Last Run",
@@ -250,7 +251,9 @@ export default {
ok: "Valid", ok: "Valid",
expired: "Expired", expired: "Expired",
}, },
certEffectiveTime: "Certificate Effective",
certExpiresTime: "Certificate Expiration", certExpiresTime: "Certificate Expiration",
remainingValidity: "Remaining Validity",
expired: "expired", expired: "expired",
days: "days", days: "days",
lastCheckTime: "Last Check Time", lastCheckTime: "Last Check Time",
@@ -444,6 +447,7 @@ export default {
description: "Description", description: "Description",
createTime: "Creation Time", createTime: "Creation Time",
updateTime: "Update Time", updateTime: "Update Time",
mainDomain: "Main Domain",
edit: "Edit", edit: "Edit",
groupName: "Group Name", groupName: "Group Name",
enterGroupName: "Please enter group name", enterGroupName: "Please enter group name",
@@ -465,6 +469,7 @@ export default {
validDays: "Valid Days", validDays: "Valid Days",
expires: " expires", expires: " expires",
days: " days", days: " days",
effectiveTime: "Effective Time",
expireTime: "Expiration Time", expireTime: "Expiration Time",
certIssuer: "Certificate Issuer", certIssuer: "Certificate Issuer",
applyTime: "Application Time", applyTime: "Application Time",
@@ -707,6 +712,10 @@ export default {
pipeline: "Pipeline", pipeline: "Pipeline",
}, },
addonType: "Type",
addonName: "Name",
addonNameHelper: "Fill freely, helps to distinguish when multiple same type exist",
addonTypeSelect: "Select type",
sys: { sys: {
setting: { setting: {
showRunStrategy: "Show RunStrategy", showRunStrategy: "Show RunStrategy",
@@ -714,7 +723,15 @@ export default {
captchaEnabled: "Enable Login Captcha", captchaEnabled: "Enable Login Captcha",
captchaHelper: "Whether to enable captcha verification for login", captchaHelper: "Whether to enable captcha verification for login",
captchaType: "Captcha Type", captchaType: "Captcha Setting",
captchaTest: "Captcha Test",
// 保存后再点击测试,请务必测试通过了,再开启登录验证码
captchaTestHelper: "Save and click test, please make sure the test is passed before enabling login captcha",
baseSetting: "Base Settings",
registerSetting: "Register Settings",
safeSetting: "Safe Settings",
paymentSetting: "Payment Settings",
captchaSetting: "Captcha Setting",
}, },
}, },
modal: { modal: {

View File

@@ -125,6 +125,7 @@ export default {
scheduledTaskCount: "定时任务数", scheduledTaskCount: "定时任务数",
deployTaskCount: "部署任务数", deployTaskCount: "部署任务数",
remainingValidity: "到期剩余", remainingValidity: "到期剩余",
effectiveTime: "生效时间",
expiryTime: "过期时间", expiryTime: "过期时间",
status: "状态", status: "状态",
lastRun: "最后运行", lastRun: "最后运行",
@@ -255,7 +256,9 @@ export default {
ok: "正常", ok: "正常",
expired: "过期", expired: "过期",
}, },
certEffectiveTime: "证书生效时间",
certExpiresTime: "证书到期时间", certExpiresTime: "证书到期时间",
remainingValidity: "到期剩余",
expired: "过期", expired: "过期",
days: "天", days: "天",
lastCheckTime: "上次检查时间", lastCheckTime: "上次检查时间",
@@ -450,6 +453,7 @@ export default {
description: "说明", description: "说明",
createTime: "创建时间", createTime: "创建时间",
updateTime: "更新时间", updateTime: "更新时间",
mainDomain: "主域名",
edit: "编辑", edit: "编辑",
groupName: "分组名称", groupName: "分组名称",
enterGroupName: "请输入分组名称", enterGroupName: "请输入分组名称",
@@ -458,7 +462,7 @@ export default {
batchDeleteConfirm: "确定要批量删除这{count}条记录吗", batchDeleteConfirm: "确定要批量删除这{count}条记录吗",
selectRecordFirst: "请先勾选记录", selectRecordFirst: "请先勾选记录",
subdomainHosted: "托管的子域名", subdomainHosted: "托管的子域名",
subdomainHelpText: "如果您不理解什么是子域托管,请不要随意设置可能导致证书无法申请,可以参考文档", subdomainHelpText: "如果您不理解什么是子域托管,请不要随意设置可能导致证书无法申请,以前设置过的cname记录也需要重新配置可以参考文档",
subdomainManagement: "子域管理", subdomainManagement: "子域管理",
isDisabled: "是否禁用", isDisabled: "是否禁用",
enabled: "启用", enabled: "启用",
@@ -471,6 +475,7 @@ export default {
validDays: "有效天数", validDays: "有效天数",
expires: "过期", expires: "过期",
days: "天", days: "天",
effectiveTime: "生效时间",
expireTime: "过期时间", expireTime: "过期时间",
certIssuer: "证书颁发机构", certIssuer: "证书颁发机构",
applyTime: "申请时间", applyTime: "申请时间",
@@ -695,7 +700,10 @@ export default {
setAsDefault: "设为默认", setAsDefault: "设为默认",
disabledLabel: "禁用", disabledLabel: "禁用",
confirmToggleStatus: "确定要{action}吗?", confirmToggleStatus: "确定要{action}吗?",
addonType: "类型",
addonName: "名称",
addonNameHelper: "随意填写,相同类型助于区分即可",
addonTypeSelect: "请选择",
template: { template: {
title: "流水线模版", title: "流水线模版",
edit: "流水线模版编辑", edit: "流水线模版编辑",
@@ -717,7 +725,14 @@ export default {
captchaEnabled: "启用登录验证码", captchaEnabled: "启用登录验证码",
captchaHelper: "登录时是否启用验证码", captchaHelper: "登录时是否启用验证码",
captchaType: "验证码类型", captchaType: "验证码配置",
captchaTest: "测试验证码",
captchaTestHelper: "保存后再点击测试,请务必测试通过了,再开启登录验证码",
baseSetting: "基本设置",
registerSetting: "注册设置",
safeSetting: "安全设置",
paymentSetting: "支付设置",
captchaSetting: "验证码设置",
}, },
}, },
modal: { modal: {

View File

@@ -2,7 +2,7 @@ import { request } from "/src/api/service";
// import "/src/mock"; // import "/src/mock";
import { ColumnCompositionProps, CrudOptions, FastCrud, PageQuery, PageRes, setLogger, TransformResProps, useColumns, UseCrudProps, UserPageQuery, useTypes, utils } from "@fast-crud/fast-crud"; import { ColumnCompositionProps, CrudOptions, FastCrud, PageQuery, PageRes, setLogger, TransformResProps, useColumns, UseCrudProps, UserPageQuery, useTypes, utils } from "@fast-crud/fast-crud";
import "@fast-crud/fast-crud/dist/style.css"; import "@fast-crud/fast-crud/dist/style.css";
import { FsExtendsCopyable, FsExtendsEditor, FsExtendsJson, FsExtendsTime, FsExtendsUploader, FsExtendsInput, FsUploaderS3SignedUrlType, FsUploaderGetAuthContext, FsUploaderAliossSTS } from "@fast-crud/fast-extends"; import { FsExtendsCopyable, FsExtendsEditor, FsExtendsJson, FsExtendsTime, FsExtendsUploader, FsExtendsInput } from "@fast-crud/fast-extends";
import "@fast-crud/fast-extends/dist/style.css"; import "@fast-crud/fast-extends/dist/style.css";
import UiAntdv from "@fast-crud/ui-antdv4"; import UiAntdv from "@fast-crud/ui-antdv4";
import "@fast-crud/ui-antdv4/dist/style.css"; import "@fast-crud/ui-antdv4/dist/style.css";
@@ -13,6 +13,9 @@ import { notification } from "ant-design-vue";
import { usePreferences } from "/@/vben/preferences"; import { usePreferences } from "/@/vben/preferences";
import { LocalStorage } from "/@/utils/util.storage"; import { LocalStorage } from "/@/utils/util.storage";
import { FsEditorCode } from "@fast-crud/editor-code";
import "@fast-crud/editor-code/dist/style.css"
class ColumnSizeSaver { class ColumnSizeSaver {
save: (key: string, size: number) => void; save: (key: string, size: number) => void;
constructor() { constructor() {
@@ -272,6 +275,7 @@ function install(app: App, options: any = {}) {
app.use(FsExtendsTime); app.use(FsExtendsTime);
app.use(FsExtendsCopyable); app.use(FsExtendsCopyable);
app.use(FsExtendsInput); app.use(FsExtendsInput);
app.use(FsEditorCode);
const { addTypes, getType } = useTypes(); const { addTypes, getType } = useTypes();
//此处演示修改官方字段类型 //此处演示修改官方字段类型

View File

@@ -5,11 +5,12 @@ import permission from "./permission";
import { App } from "vue"; import { App } from "vue";
import "./validator/index.js"; import "./validator/index.js";
import directives from "./directive/index"; import directives from "./directive/index";
import { setupMonaco } from "./monaco";
function install(app: App, options: any = {}) { function install(app: App, options: any = {}) {
app.use(FastCrud, options); app.use(FastCrud, options);
app.use(permission); app.use(permission);
app.use(directives); app.use(directives);
setupMonaco();
} }
export default { export default {

View File

@@ -0,0 +1,15 @@
import editorWorker from "monaco-editor/esm/vs/editor/editor.worker?worker";
import jsonWorker from "monaco-editor/esm/vs/language/json/json.worker?worker";
import cssWorker from "monaco-editor/esm/vs/language/css/css.worker?worker";
import htmlWorker from "monaco-editor/esm/vs/language/html/html.worker?worker";
import yamlWorker from "./yaml.worker?worker";
import tsWorker from "monaco-editor/esm/vs/language/typescript/ts.worker?worker";
import { registerWorker } from "@fast-crud/editor-code";
export function setupMonaco() {
registerWorker("json", jsonWorker);
registerWorker(["css", "less", "scss"], cssWorker);
registerWorker(["html", "handlebars", "razor"], htmlWorker);
registerWorker(["yaml", "yml"], yamlWorker);
registerWorker(["typescript", "javascript"], tsWorker);
registerWorker("*", editorWorker);
}

View File

@@ -0,0 +1 @@
export * from "monaco-yaml/yaml.worker.js";

View File

@@ -91,6 +91,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
type: "number", type: "number",
column: { column: {
width: 50, width: 50,
order: -999,
}, },
form: { form: {
show: false, show: false,
@@ -105,9 +106,11 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
form: { form: {
rules: [{ required: true, message: t("certd.pleaseEnterName") }], rules: [{ required: true, message: t("certd.pleaseEnterName") }],
helper: t("certd.nameHelper"), helper: t("certd.nameHelper"),
order: -2,
}, },
column: { column: {
width: 200, width: 200,
order: -2,
}, },
}, },
from: { from: {

View File

@@ -87,6 +87,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
order: -1, order: -1,
}, },
form: { form: {
order: -1,
component: { component: {
disabled: false, disabled: false,
showSearch: true, showSearch: true,

View File

@@ -66,6 +66,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
type: "number", type: "number",
column: { column: {
width: 100, width: 100,
order: -999,
}, },
form: { form: {
show: false, show: false,
@@ -79,6 +80,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}, },
form: { form: {
rules: [{ required: true, message: "必填项" }], rules: [{ required: true, message: "必填项" }],
order: -11,
}, },
column: { column: {
width: 300, width: 300,

View File

@@ -127,3 +127,16 @@ export function createAddonApi(opts: { from: any; addonType: string }) {
}, },
}; };
} }
export const AddonTypeDefines = {
captcha: {
name: "captcha",
title: "验证码",
showDefault: false,
showTest: false,
},
};
export function getAddonTypeDefine(addonType: string) {
return AddonTypeDefines[addonType];
}

View File

@@ -6,6 +6,7 @@ import { Modal } from "ant-design-vue";
import { mitter } from "/@/utils/util.mitt"; import { mitter } from "/@/utils/util.mitt";
import { useI18n } from "/src/locales"; import { useI18n } from "/src/locales";
import * as pipelineApi from "/@/views/certd/pipeline/api"; import * as pipelineApi from "/@/views/certd/pipeline/api";
import { getAddonTypeDefine } from "/@/views/certd/addon/api";
export function addonProvide(api: any) { export function addonProvide(api: any) {
provide("addonApi", api); provide("addonApi", api);
@@ -74,6 +75,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
type: "number", type: "number",
column: { column: {
width: 100, width: 100,
order: -999,
}, },
form: { form: {
show: false, show: false,
@@ -104,7 +106,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
// }, // },
// }, // },
type: { type: {
title: t("certd.notificationType"), title: t("certd.addonType"),
type: "dict-select", type: "dict-select",
dict: addonTypeDictRef, dict: addonTypeDictRef,
search: { search: {
@@ -122,6 +124,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
}, },
}, },
form: { form: {
order: -22,
component: { component: {
disabled: false, disabled: false,
showSearch: true, showSearch: true,
@@ -138,7 +141,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
); );
}, },
}, },
rules: [{ required: true, message: t("certd.selectNotificationType") }], rules: [{ required: true, message: t("certd.addonTypeSelect") }],
valueChange: { valueChange: {
immediate: true, immediate: true,
async handle({ value, mode, form, immediate }) { async handle({ value, mode, form, immediate }) {
@@ -173,14 +176,15 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
}, },
} as ColumnCompositionProps, } as ColumnCompositionProps,
name: { name: {
title: t("certd.notificationName"), title: t("certd.addonName"),
search: { search: {
show: true, show: true,
}, },
type: ["text"], type: ["text"],
form: { form: {
order: -2,
rules: [{ required: true, message: t("certd.enterName") }], rules: [{ required: true, message: t("certd.enterName") }],
helper: t("certd.helperNotificationName"), helper: t("certd.addonNameHelper"),
}, },
column: { column: {
width: 200, width: 200,
@@ -196,6 +200,9 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
], ],
}), }),
form: { form: {
show: computed(() => {
return getAddonTypeDefine(addonType)?.showDefault ?? false;
}),
value: false, value: false,
rules: [{ required: true, message: t("certd.selectIsDefault") }], rules: [{ required: true, message: t("certd.selectIsDefault") }],
order: 999, order: 999,
@@ -203,6 +210,9 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
column: { column: {
align: "center", align: "center",
width: 100, width: 100,
show: computed(() => {
return getAddonTypeDefine(addonType)?.showDefault ?? false;
}),
component: { component: {
name: "a-switch", name: "a-switch",
vModel: "checked", vModel: "checked",
@@ -210,6 +220,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
return value === true; return value === true;
}), }),
on: { on: {
// @ts-ignore
change({ row }) { change({ row }) {
Modal.confirm({ Modal.confirm({
title: t("certd.prompt"), title: t("certd.prompt"),
@@ -226,12 +237,12 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any, a
}, },
}, },
}, },
} as ColumnCompositionProps, },
test: { test: {
title: t("certd.test"), title: t("certd.test"),
form: { form: {
show: compute(({ form }) => { show: compute(({ form }) => {
return !!form.type; return !!form.type && currentDefine.value?.showTest === true;
}), }),
component: { component: {
name: "api-test", name: "api-test",

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