Compare commits
128 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e17cd1f298 | ||
|
|
13092e9f80 | ||
|
|
8133b8b9dd | ||
|
|
e25aafac6d | ||
|
|
037c7beb1b | ||
|
|
faac4dfc30 | ||
|
|
469a088a4d | ||
|
|
9c854f727f | ||
|
|
8f6e5bd24b | ||
|
|
992f91cf4c | ||
|
|
0c61d4c978 | ||
|
|
72d32edf9a | ||
|
|
dde39def9e | ||
|
|
6b43007c44 | ||
|
|
876558cf77 | ||
|
|
b35a146edf | ||
|
|
86cf6a9908 | ||
|
|
b0f7288ac0 | ||
|
|
32fcc1a8fb | ||
|
|
eb4d125eaf | ||
|
|
87e5cced3c | ||
|
|
bcd9ee2d48 | ||
|
|
edf3d87458 | ||
|
|
0c0c353ecc | ||
|
|
aaa4c8f899 | ||
|
|
57e3565c11 | ||
|
|
fbcf72d762 | ||
|
|
ca8daa836e | ||
|
|
c2ccdbec9d | ||
|
|
69aee36e75 | ||
|
|
063f5c3b55 | ||
|
|
6d1b8ca65e | ||
|
|
03899d4d9c | ||
|
|
2b84af977d | ||
|
|
e15b180322 | ||
|
|
f070030f6b | ||
|
|
330ac66b38 | ||
|
|
12a9e650af | ||
|
|
1e5ccd811e | ||
|
|
2902ee6ad5 | ||
|
|
90ce4fec2c | ||
|
|
a7ab26d08d | ||
|
|
dcc396afb7 | ||
|
|
3f1722d54d | ||
|
|
c79658afbb | ||
|
|
6f84ebb323 | ||
|
|
54c8d62243 | ||
|
|
83e6476408 | ||
|
|
03f317ffdb | ||
|
|
3f67c7c74a | ||
|
|
b8b4660563 | ||
|
|
3d42bfd479 | ||
|
|
2ae193092d | ||
|
|
7e1d52ff00 | ||
|
|
c98f43b984 | ||
|
|
e93f128a7a | ||
|
|
71d8e7edd2 | ||
|
|
48f4298a8d | ||
|
|
1c15beadc7 | ||
|
|
2c1600ddfb | ||
|
|
298f7d9d52 | ||
|
|
105f0bfde2 | ||
|
|
cf3a78e114 | ||
|
|
9cc5f0f889 | ||
|
|
e30db9ee77 | ||
|
|
235be757f8 | ||
|
|
e31d26a887 | ||
|
|
2293ba02ea | ||
|
|
7188997dd1 | ||
|
|
31cfb09468 | ||
|
|
b76f2e2008 | ||
|
|
4b90972341 | ||
|
|
f4ff34224c | ||
|
|
877c9c4ff9 | ||
|
|
ac0b7291dd | ||
|
|
491ef6085a | ||
|
|
3cedef4974 | ||
|
|
22ab04bd2b | ||
|
|
e5a080aebe | ||
|
|
c560cc5add | ||
|
|
0d27bc323b | ||
|
|
c71d3cef18 | ||
|
|
4e2d8daa3a | ||
|
|
d0f51da0af | ||
|
|
aeb73bca27 | ||
|
|
f239b03291 | ||
|
|
297c2965f4 | ||
|
|
daddf4d98e | ||
|
|
e05f9bfebf | ||
|
|
ef46aeae6f | ||
|
|
7edb3fd856 | ||
|
|
43b79778ea | ||
|
|
37f1f53b56 | ||
|
|
67bd1cdcd9 | ||
|
|
506385e5a2 | ||
|
|
2d4586b1c4 | ||
|
|
1476b9cb9c | ||
|
|
7bdde68ece | ||
|
|
50f92f55e2 | ||
|
|
370db62bf0 | ||
|
|
65f34f1d31 | ||
|
|
00a3908abb | ||
|
|
32034d590a | ||
|
|
3635fb3910 | ||
|
|
d2ecfe5491 | ||
|
|
1f759dce5b | ||
|
|
ae41c6038b | ||
|
|
f41f7eb2ad | ||
|
|
d04f383161 | ||
|
|
cb989d7489 | ||
|
|
b5cba19d26 | ||
|
|
b7271d7a46 | ||
|
|
768bdc2cc2 | ||
|
|
a0a093e260 | ||
|
|
0b2a7fdc15 | ||
|
|
f1876e20f8 | ||
|
|
7d6a6e53f7 | ||
|
|
6b765a1f77 | ||
|
|
3b3c93dd53 | ||
|
|
521083a309 | ||
|
|
6d35325601 | ||
|
|
3c65f37d84 | ||
|
|
d75dd058d6 | ||
|
|
40475e02ec | ||
|
|
f6ea9c1300 | ||
|
|
902359f24e | ||
|
|
bb4d5f1e93 | ||
|
|
1dec3f000e |
5
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: greper
|
||||||
|
buy_me_a_coffee: greper
|
||||||
|
custom: ['https://afdian.com/a/greper']
|
||||||
34
.github/workflows/build-image.yml
vendored
@@ -3,7 +3,7 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches: ['v2-dev']
|
branches: ['v2-dev']
|
||||||
paths:
|
paths:
|
||||||
- "build.trigger"
|
- "trigger/build1.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}}
|
|
||||||
|
|||||||
9
.github/workflows/deploy-demo.yml
vendored
@@ -1,14 +1,15 @@
|
|||||||
name: deploy-demo
|
name: deploy-demo
|
||||||
on:
|
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 * * *'
|
||||||
|
|||||||
@@ -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"
|
# - "builder/release.trigger"
|
||||||
|
# workflow_run:
|
||||||
|
# workflows: [ "build-image" ]
|
||||||
|
# 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,18 @@ 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}}
|
||||||
2
.gitignore
vendored
@@ -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
@@ -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
@@ -0,0 +1,4 @@
|
|||||||
|
{
|
||||||
|
"eslint.debug": false,
|
||||||
|
"eslint.format.enable": true
|
||||||
|
}
|
||||||
52
.vscode/tasks.json
vendored
Normal 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": []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
83
CHANGELOG.md
@@ -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
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# Certd
|
# Certd
|
||||||
|
|
||||||
[English](./README_en.md) | [中文](./README.md)
|
中文 | [English](./README_en.md)
|
||||||
|
|
||||||
Certd® 是一个免费的全自动证书管理系统,让你的网站证书永不过期。
|
Certd® 是一个免费的全自动证书管理系统,让你的网站证书永不过期。
|
||||||
后缀d取自linux守护进程的命名风格,意为证书守护进程
|
后缀d取自linux守护进程的命名风格,意为证书守护进程
|
||||||
@@ -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酱等 |
|
||||||
|
|
||||||
|
************************
|
||||||
|
|
||||||
************************
|
************************
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# Certd
|
# Certd
|
||||||
|
|
||||||
[English](./README_en.md) | [中文](./README.md)
|
[中文](./README.md) | English
|
||||||
|
|
||||||
Certd® is a free, fully automated certificate management system that ensures your website certificates never expire. The suffix 'd' is inspired by the naming convention of Linux daemons, representing a certificate daemon.
|
Certd® is a free, fully automated certificate management system that ensures your website certificates never expire. The suffix 'd' is inspired by the naming convention of Linux daemons, representing a certificate daemon.
|
||||||
|
|
||||||
@@ -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
|
||||||
|
************************
|
||||||
|
[](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
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
2
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
00:43
|
|
||||||
@@ -107,7 +107,6 @@ export default defineConfig({
|
|||||||
text: "常见问题",
|
text: "常见问题",
|
||||||
items: [
|
items: [
|
||||||
{text: "QA", link: "/guide/qa/use.md"},
|
{text: "QA", link: "/guide/qa/use.md"},
|
||||||
{text: "常见报错处理", link: "/guide/qa/"},
|
|
||||||
{text: "群晖证书部署", link: "/guide/use/synology/"},
|
{text: "群晖证书部署", link: "/guide/use/synology/"},
|
||||||
{text: "腾讯云密钥获取", link: "/guide/use/tencent/"},
|
{text: "腾讯云密钥获取", link: "/guide/use/tencent/"},
|
||||||
{text: "连接windows主机", link: "/guide/use/host/windows.md"},
|
{text: "连接windows主机", link: "/guide/use/host/windows.md"},
|
||||||
@@ -120,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"},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -3,6 +3,105 @@
|
|||||||
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.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)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* 前置任务输出不存在时输出警告提示 ([b59052c](https://github.com/certd/certd/commit/b59052cc43b7b070fabd8b8e914e4c2a5e0ad61c))
|
||||||
|
* 修复批量流水线执行时日志显示错乱的问题 ([4372adc](https://github.com/certd/certd/commit/4372adc703b9a4c785664054ab2a533626d815a8))
|
||||||
|
* 修复远程数据选择无法过滤的bug ([6cbb073](https://github.com/certd/certd/commit/6cbb0739f8428d51b0712f718fe4d236cc087cf9))
|
||||||
|
* 修复mysql下购买套餐加量包无效的bug ([c26ad4c](https://github.com/certd/certd/commit/c26ad4c8075f0606d45b8da13915737968d6191a))
|
||||||
|
|
||||||
|
### Performance Improvements
|
||||||
|
|
||||||
|
* 创建证书时支持选择通知时机 ([0e96bfd](https://github.com/certd/certd/commit/0e96bfdfa377824d204e72923d1176408ae6b300))
|
||||||
|
* 创建k8s secret 时设置type为tls ([79ebabf](https://github.com/certd/certd/commit/79ebabfcfb9e5a534049c84f5f1a642b357fc856))
|
||||||
|
* 去掉宝塔url后面的斜杠 ([8a0c2b9](https://github.com/certd/certd/commit/8a0c2b9b13628da750c25757e0cb8ed3038775ba))
|
||||||
|
* 商业版隐藏文档相关链接 ([4443a1c](https://github.com/certd/certd/commit/4443a1c0308fa6b95a05efd73d15d24b65d641c9))
|
||||||
|
* 商业版隐藏文档相关链接 ([db89561](https://github.com/certd/certd/commit/db8956148083bc4f988226ccf719940d08158a27))
|
||||||
|
* 增加健康检查探针 /health/liveliness 和 /health/readiness ([44019e1](https://github.com/certd/certd/commit/44019e104289fedd32a867db00e9c6cb71b389cc))
|
||||||
|
* 支持根据id更新证书(证书Id不变接口),不过该接口为白名单功能,普通腾讯云账户无法使用 ([fe9c4f3](https://github.com/certd/certd/commit/fe9c4f3391ff07c01dd9a252225f69a129c39050))
|
||||||
|
* 支持godaddy ([b7980aa](https://github.com/certd/certd/commit/b7980aad5ab50f58662eaddf5d84aa82876a98eb))
|
||||||
|
* 支持ssl.com证书颁发机构 ([27b6dfa](https://github.com/certd/certd/commit/27b6dfa4d2ab3bddd284c3a34511a72e1a513a4c))
|
||||||
|
* 子域名托管说明 ([39a0223](https://github.com/certd/certd/commit/39a02235cf4416bb5bd1acd3831241efeaa2f602))
|
||||||
|
* ssh 增加超时断开连接,默认10分钟超时 ([c24a040](https://github.com/certd/certd/commit/c24a040c19cacafc79228d7a7649af93837d94a1))
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|||||||
@@ -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->容器编排->添加容器编排
|
||||||

|

|
||||||
点击确定,等待启动完成
|
点击确定,等待启动完成
|
||||||

|

|
||||||
|
|
||||||
> 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`->`更新镜像`
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@@ -80,5 +80,8 @@ admin/123456
|
|||||||
### 1. 无法访问Certd
|
### 1. 无法访问Certd
|
||||||
1. 确认服务器的安全规则,是否放开了对应端口
|
1. 确认服务器的安全规则,是否放开了对应端口
|
||||||
2. 确认宝塔防火墙是否放开对应端口
|
2. 确认宝塔防火墙是否放开对应端口
|
||||||
3. 尝试将Certd容器加入宝塔的`bridge`网络
|
3. 尝试将Certd容器加入宝塔的`bridge`网络
|
||||||

|

|
||||||
|
|
||||||
|
### 2. 动态IP无法加白名单问题
|
||||||
|
[Nginx代理解决方案](../../use/baota/white_list.md)
|
||||||
@@ -12,7 +12,7 @@ git clone https://github.com/certd/certd --depth=1
|
|||||||
# git checkout v1.x.x # 当v2主干分支代码无法正常启动时,可以尝试此命令,1.x.x换成最新版本号
|
# git checkout v1.x.x # 当v2主干分支代码无法正常启动时,可以尝试此命令,1.x.x换成最新版本号
|
||||||
cd certd
|
cd certd
|
||||||
# 启动服务
|
# 启动服务
|
||||||
./start.sh
|
./start.sh
|
||||||
|
|
||||||
```
|
```
|
||||||
>如果是windows,请先安装`git for windows` ,然后右键,选择`open git bash here`打开终端,再执行`./start.sh`命令
|
>如果是windows,请先安装`git for windows` ,然后右键,选择`open git bash here`打开终端,再执行`./start.sh`命令
|
||||||
@@ -21,9 +21,9 @@ cd certd
|
|||||||
|
|
||||||
### 访问测试
|
### 访问测试
|
||||||
|
|
||||||
http://your_server_ip:7001
|
http://your_server_ip:7001
|
||||||
https://your_server_ip:7002
|
https://your_server_ip:7002
|
||||||
默认账号密码:admin/123456
|
默认账号密码:admin/123456
|
||||||
记得修改密码
|
记得修改密码
|
||||||
|
|
||||||
|
|
||||||
@@ -37,7 +37,7 @@ cp -rf ./packages/ui/certd-server/data ../certd-data-backup
|
|||||||
|
|
||||||
git pull
|
git pull
|
||||||
# 如果提示pull失败,可以尝试强制更新
|
# 如果提示pull失败,可以尝试强制更新
|
||||||
# git checkout v2 -f && git pull
|
# git checkout v2 -f && git pull
|
||||||
|
|
||||||
# 先停止旧的服务,7001是certd的默认端口
|
# 先停止旧的服务,7001是certd的默认端口
|
||||||
kill -9 $(lsof -t -i:7001)
|
kill -9 $(lsof -t -i:7001)
|
||||||
@@ -45,16 +45,31 @@ kill -9 $(lsof -t -i:7001)
|
|||||||
./start.sh
|
./start.sh
|
||||||
|
|
||||||
```
|
```
|
||||||
::: warning
|
::: warning
|
||||||
升级certd版本前,切记切记先备份一下数据
|
升级certd版本前,切记切记先备份一下数据
|
||||||
:::
|
:::
|
||||||
|
|
||||||
|
|
||||||
## 三、数据备份
|
## 三、数据备份
|
||||||
> 数据默认保存在 `./packages/ui/certd-server/data` 目录下
|
> 数据默认保存在 `./packages/ui/certd-server/data` 目录下
|
||||||
> 建议配置一条[数据库备份流水线](../../use/backup/) 自动备份
|
> 建议配置一条[数据库备份流水线](../../use/backup/) 自动备份
|
||||||
|
|
||||||
|
|
||||||
## 四、备份恢复
|
## 四、备份恢复
|
||||||
|
|
||||||
将备份的`db.sqlite`及同目录下的其他文件覆盖到原来的位置,重启certd即可
|
将备份的`db.sqlite`及同目录下的其他文件覆盖到原来的位置,重启certd即可
|
||||||
|
|
||||||
|
## 六、常见问题
|
||||||
|
|
||||||
|
### 1. npm install better-sqlite3 时,提示node-gyp需要vscode环境编译
|
||||||
|
|
||||||
|
1. 首先确保node版本为22以上
|
||||||
|
2. 将下面两行加到 ~/.npmrc 里面
|
||||||
|
3. 重新install
|
||||||
|
> better_sqlite3_binary_host=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
> better_sqlite3_binary_host_mirror=https://registry.npmmirror.com/-/binary/better-sqlite3
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# 使用问题
|
# 常见问题
|
||||||
|
|
||||||
|
|
||||||
## 1. 是否支持IP证书
|
## 1. 是否支持IP证书
|
||||||
@@ -7,8 +7,27 @@
|
|||||||
|
|
||||||
|
|
||||||
## 2. 建议设置多长时间运行一次流水线
|
## 2. 建议设置多长时间运行一次流水线
|
||||||
建议每天运行一次,检查证书过期时间
|
建议每天运行一次,检查证书过期时间
|
||||||
当证书没过期时,自动跳过部署
|
当证书没过期时,自动跳过部署
|
||||||
当证书到期前35天(创建流水线时可以修改),将会自动重新申请证书,自动部署
|
当证书到期前35天(创建流水线时可以修改),将会自动重新申请证书,自动部署
|
||||||
|
|
||||||
|
|
||||||
|
## 3. too many certificates 错误
|
||||||
|
当出现如下报错时,说明相同的域名短时间内申请超过5次
|
||||||
|
解决方案:可以加多一个子域名,重新执行就可以规避次错误
|
||||||
|
```
|
||||||
|
"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" |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
BIN
docs/guide/use/baota/images/white-1.png
Normal file
|
After Width: | Height: | Size: 22 KiB |
BIN
docs/guide/use/baota/images/white-2.png
Normal file
|
After Width: | Height: | Size: 30 KiB |
BIN
docs/guide/use/baota/images/white-3.png
Normal file
|
After Width: | Height: | Size: 36 KiB |
BIN
docs/guide/use/baota/images/white-4.png
Normal file
|
After Width: | Height: | Size: 15 KiB |
BIN
docs/guide/use/baota/images/white-5.png
Normal file
|
After Width: | Height: | Size: 7.2 KiB |
BIN
docs/guide/use/baota/images/white-6.png
Normal file
|
After Width: | Height: | Size: 27 KiB |
BIN
docs/guide/use/baota/images/white-safe-1.png
Normal file
|
After Width: | Height: | Size: 58 KiB |
BIN
docs/guide/use/baota/images/white-safe-2.png
Normal file
|
After Width: | Height: | Size: 23 KiB |
98
docs/guide/use/baota/white_list.md
Normal 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反向代理
|
||||||
|

|
||||||
|
|
||||||
|
### 2. 域名和代理目标
|
||||||
|

|
||||||
|
|
||||||
|
### 3. 设置放开哪些接口
|
||||||
|

|
||||||
|

|
||||||
|
将如下脚本填入上方文本域中,保存
|
||||||
|
```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
|
||||||
|

|
||||||
|
|
||||||
|
### 5. certd中宝塔授权配置改成新的这个域名地址
|
||||||
|
|
||||||
|

|
||||||
|
点击测试检查是否ok ,到这里就可以正常部署证书了
|
||||||
|
|
||||||
|
### 6. 安全加强(将请求地址改成https)
|
||||||
|
在宝塔中配置证书部署任务,选择刚才新建的这个网站,给他部署证书
|
||||||
|
勾选强制https
|
||||||
|

|
||||||
|
更换443端口【可选】
|
||||||
|

|
||||||
|
禁止http访问
|
||||||
@@ -9,5 +9,5 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"npmClient": "pnpm",
|
"npmClient": "pnpm",
|
||||||
"version": "1.36.19"
|
"version": "1.37.0"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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",
|
||||||
|
|||||||
@@ -3,6 +3,34 @@
|
|||||||
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/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
|
||||||
|
|||||||
@@ -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.0",
|
||||||
"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.0",
|
||||||
"@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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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.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
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
00:30
|
12:14
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/basic",
|
"name": "@certd/basic",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
@@ -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,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -3,6 +3,36 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/pipeline",
|
"name": "@certd/pipeline",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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.0",
|
||||||
"@certd/plus-core": "^1.36.19",
|
"@certd/plus-core": "^1.37.0",
|
||||||
"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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -69,9 +69,15 @@ export class Registry<T = any> {
|
|||||||
return this.storage;
|
return this.storage;
|
||||||
}
|
}
|
||||||
|
|
||||||
getDefineList() {
|
getDefineList(prefix?: string) {
|
||||||
let list = [];
|
let list = [];
|
||||||
|
if (prefix) {
|
||||||
|
prefix = prefix + ":";
|
||||||
|
}
|
||||||
for (const key in this.storage) {
|
for (const key in this.storage) {
|
||||||
|
if (prefix && !key.startsWith(prefix)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
const define = this.getDefine(key);
|
const define = this.getDefine(key);
|
||||||
if (define) {
|
if (define) {
|
||||||
if (define?.deprecated) {
|
if (define?.deprecated) {
|
||||||
@@ -90,7 +96,10 @@ export class Registry<T = any> {
|
|||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
getDefine(key: string) {
|
getDefine(key: string, prefix?: string) {
|
||||||
|
if (prefix) {
|
||||||
|
key = prefix + ":" + key;
|
||||||
|
}
|
||||||
const item = this.storage[key];
|
const item = this.storage[key];
|
||||||
if (!item) {
|
if (!item) {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -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 = {
|
||||||
|
|||||||
@@ -3,6 +3,34 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/lib-huawei",
|
"name": "@certd/lib-huawei",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,34 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/lib-iframe",
|
"name": "@certd/lib-iframe",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,34 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/jdcloud",
|
"name": "@certd/jdcloud",
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,36 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/lib-k8s",
|
"name": "@certd/lib-k8s",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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.0",
|
||||||
"@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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -85,7 +85,6 @@ export class K8sClient {
|
|||||||
/**
|
/**
|
||||||
* 创建Secret
|
* 创建Secret
|
||||||
* @param opts {namespace:default, body:yamlStr}
|
* @param opts {namespace:default, body:yamlStr}
|
||||||
* @returns {Promise<*>}
|
|
||||||
*/
|
*/
|
||||||
async createSecret(opts: { namespace: string; body: V1Secret }) {
|
async createSecret(opts: { namespace: string; body: V1Secret }) {
|
||||||
const namespace = opts.namespace || "default";
|
const namespace = opts.namespace || "default";
|
||||||
@@ -121,7 +120,7 @@ export class K8sClient {
|
|||||||
//没有找到,则创建
|
//没有找到,则创建
|
||||||
const body = merge(
|
const body = merge(
|
||||||
{
|
{
|
||||||
type: "type: kubernetes.io/tls",
|
type: "kubernetes.io/tls",
|
||||||
},
|
},
|
||||||
opts.body
|
opts.body
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -3,6 +3,45 @@
|
|||||||
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
|
||||||
|
|
||||||
|
* 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
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/lib-server",
|
"name": "@certd/lib-server",
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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.0",
|
||||||
"@certd/basic": "^1.36.19",
|
"@certd/basic": "^1.37.0",
|
||||||
"@certd/pipeline": "^1.36.19",
|
"@certd/pipeline": "^1.37.0",
|
||||||
"@certd/plus-core": "^1.36.19",
|
"@certd/plugin-lib": "^1.37.0",
|
||||||
"@midwayjs/cache": "~3.14.0",
|
"@certd/plus-core": "^1.37.0",
|
||||||
"@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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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'
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { SysSettingsEntity } from './system/index.js';
|
import { SysSettingsEntity } from './system/index.js';
|
||||||
import { AccessEntity } from './user/access/entity/access.js';
|
import { AccessEntity } from './user/access/entity/access.js';
|
||||||
|
import { AddonEntity } from "./user/index.js";
|
||||||
export * from './basic/index.js';
|
export * from './basic/index.js';
|
||||||
export * from './system/index.js';
|
export * from './system/index.js';
|
||||||
export * from './user/index.js';
|
export * from './user/index.js';
|
||||||
export { LibServerConfiguration as Configuration } from './configuration.js';
|
export { LibServerConfiguration as Configuration } from './configuration.js';
|
||||||
|
|
||||||
export const libServerEntities = [SysSettingsEntity, AccessEntity];
|
export const libServerEntities = [SysSettingsEntity, AccessEntity,AddonEntity];
|
||||||
|
|||||||
@@ -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';
|
||||||
|
|||||||
@@ -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;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -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()
|
||||||
|
|||||||
@@ -30,6 +30,13 @@ export class SysPublicSettings extends BaseSettings {
|
|||||||
mpsNo?: string;
|
mpsNo?: string;
|
||||||
robots?: boolean = true;
|
robots?: boolean = true;
|
||||||
aiChatEnabled = true;
|
aiChatEnabled = true;
|
||||||
|
|
||||||
|
|
||||||
|
//验证码是否开启
|
||||||
|
captchaEnabled = false;
|
||||||
|
//验证码类型
|
||||||
|
captchaType?: string;
|
||||||
|
captchaAddonId?:number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class SysPrivateSettings extends BaseSettings {
|
export class SysPrivateSettings extends BaseSettings {
|
||||||
@@ -207,4 +214,3 @@ export class SysSafeSetting extends BaseSettings {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
142
packages/libs/lib-server/src/user/addon/api/api.ts
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
import { HttpClient, ILogger, utils } from "@certd/basic";
|
||||||
|
import {upperFirst} from "lodash-es";
|
||||||
|
import {
|
||||||
|
accessRegistry,
|
||||||
|
FormItemProps,
|
||||||
|
IAccessService,
|
||||||
|
IServiceGetter,
|
||||||
|
PluginRequestHandleReq,
|
||||||
|
Registrable
|
||||||
|
} from "@certd/pipeline";
|
||||||
|
|
||||||
|
|
||||||
|
export type AddonRequestHandleReqInput<T = any> = {
|
||||||
|
id?: number;
|
||||||
|
title?: string;
|
||||||
|
addon: T;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AddonRequestHandleReq<T = any> = {
|
||||||
|
addonType: string;
|
||||||
|
} &PluginRequestHandleReq<AddonRequestHandleReqInput<T>>;
|
||||||
|
|
||||||
|
export type AddonInputDefine = FormItemProps & {
|
||||||
|
title: string;
|
||||||
|
required?: boolean;
|
||||||
|
};
|
||||||
|
export type AddonDefine = Registrable & {
|
||||||
|
addonType: string;
|
||||||
|
needPlus?: boolean;
|
||||||
|
input?: {
|
||||||
|
[key: string]: AddonInputDefine;
|
||||||
|
};
|
||||||
|
showTest?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AddonInstanceConfig = {
|
||||||
|
id: number;
|
||||||
|
addonType: string;
|
||||||
|
type: string;
|
||||||
|
name: string;
|
||||||
|
userId: number;
|
||||||
|
setting: {
|
||||||
|
[key: string]: any;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export interface IAddon {
|
||||||
|
ctx: AddonContext;
|
||||||
|
[key: string]: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type AddonContext = {
|
||||||
|
http: HttpClient;
|
||||||
|
logger: ILogger;
|
||||||
|
utils: typeof utils;
|
||||||
|
serviceGetter: IServiceGetter;
|
||||||
|
};
|
||||||
|
|
||||||
|
export abstract class BaseAddon implements IAddon {
|
||||||
|
define!: AddonDefine;
|
||||||
|
ctx!: AddonContext;
|
||||||
|
http!: HttpClient;
|
||||||
|
logger!: ILogger;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
||||||
|
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) {
|
||||||
|
this.ctx = ctx;
|
||||||
|
this.http = ctx.http;
|
||||||
|
this.logger = ctx.logger;
|
||||||
|
}
|
||||||
|
setDefine = (define:AddonDefine) => {
|
||||||
|
this.define = define;
|
||||||
|
};
|
||||||
|
|
||||||
|
async onRequest(req:AddonRequestHandleReq) {
|
||||||
|
if (!req.action) {
|
||||||
|
throw new Error("action is required");
|
||||||
|
}
|
||||||
|
|
||||||
|
let methodName = req.action;
|
||||||
|
if (!req.action.startsWith("on")) {
|
||||||
|
methodName = `on${upperFirst(req.action)}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const method = this[methodName];
|
||||||
|
if (method) {
|
||||||
|
// @ts-ignore
|
||||||
|
return await this[methodName](req.data);
|
||||||
|
}
|
||||||
|
throw new Error(`action ${req.action} not found`);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export interface IAddonGetter {
|
||||||
|
getById<T = any>(id: any): Promise<T>;
|
||||||
|
getCommonById<T = any>(id: any): Promise<T>;
|
||||||
|
}
|
||||||
65
packages/libs/lib-server/src/user/addon/api/decorator.ts
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
// src/decorator/memoryCache.decorator.ts
|
||||||
|
import * as _ from "lodash-es";
|
||||||
|
import { merge } from "lodash-es";
|
||||||
|
import { addonRegistry } from "./registry.js";
|
||||||
|
import { AddonContext, AddonDefine, AddonInputDefine } from "./api.js";
|
||||||
|
import { Decorator } from "@certd/pipeline";
|
||||||
|
|
||||||
|
// 提供一个唯一 key
|
||||||
|
export const ADDON_CLASS_KEY = "pipeline:addon";
|
||||||
|
export const ADDON_INPUT_KEY = "pipeline:addon:input";
|
||||||
|
|
||||||
|
export function IsAddon(define: AddonDefine): ClassDecorator {
|
||||||
|
return (target: any) => {
|
||||||
|
target = Decorator.target(target);
|
||||||
|
|
||||||
|
const inputs: any = {};
|
||||||
|
const properties = Decorator.getClassProperties(target);
|
||||||
|
for (const property in properties) {
|
||||||
|
const input = Reflect.getMetadata(ADDON_INPUT_KEY, target, property);
|
||||||
|
if (input) {
|
||||||
|
inputs[property] = input;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_.merge(define, { input: inputs });
|
||||||
|
Reflect.defineMetadata(ADDON_CLASS_KEY, define, target);
|
||||||
|
target.define = define;
|
||||||
|
const key = `${define.addonType}:${define.name}`;
|
||||||
|
addonRegistry.register(key, {
|
||||||
|
define,
|
||||||
|
target: async () => {
|
||||||
|
return target;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AddonInput(input?: AddonInputDefine): PropertyDecorator {
|
||||||
|
return (target, propertyKey) => {
|
||||||
|
target = Decorator.target(target, propertyKey);
|
||||||
|
// const _type = Reflect.getMetadata("design:type", target, propertyKey);
|
||||||
|
Reflect.defineMetadata(ADDON_INPUT_KEY, input, target, propertyKey);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function newAddon(addonType:string,type: string, input: any, ctx: AddonContext) {
|
||||||
|
const key = `${addonType}:${type}`
|
||||||
|
const register = addonRegistry.get(key);
|
||||||
|
if (register == null) {
|
||||||
|
throw new Error(`${addonType} ${type} not found`);
|
||||||
|
}
|
||||||
|
// @ts-ignore
|
||||||
|
const pluginCls = await register.target();
|
||||||
|
// @ts-ignore
|
||||||
|
const plugin = new pluginCls();
|
||||||
|
merge(plugin, input);
|
||||||
|
if (!ctx) {
|
||||||
|
throw new Error("ctx is required");
|
||||||
|
}
|
||||||
|
plugin.setDefine(register.define);
|
||||||
|
plugin.setCtx(ctx);
|
||||||
|
await plugin.onInstance();
|
||||||
|
return plugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
3
packages/libs/lib-server/src/user/addon/api/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from "./api.js";
|
||||||
|
export * from "./registry.js";
|
||||||
|
export * from "./decorator.js";
|
||||||
3
packages/libs/lib-server/src/user/addon/api/registry.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
import { createRegistry } from "@certd/pipeline";
|
||||||
|
|
||||||
|
export const addonRegistry = createRegistry("addon");
|
||||||
44
packages/libs/lib-server/src/user/addon/entity/addon.ts
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import { Column, Entity, PrimaryGeneratedColumn } from 'typeorm';
|
||||||
|
|
||||||
|
/**
|
||||||
|
*/
|
||||||
|
@Entity('cd_addon')
|
||||||
|
export class AddonEntity {
|
||||||
|
@PrimaryGeneratedColumn()
|
||||||
|
id: number;
|
||||||
|
@Column({ name: 'user_id', comment: '用户id' })
|
||||||
|
userId: number;
|
||||||
|
@Column({ comment: '名称', length: 100 })
|
||||||
|
name: string;
|
||||||
|
|
||||||
|
|
||||||
|
@Column({ name: 'addon_type', comment: 'addon类型', length: 100 })
|
||||||
|
addonType: string;
|
||||||
|
|
||||||
|
|
||||||
|
@Column({ comment: '类型', length: 100 })
|
||||||
|
type: string;
|
||||||
|
|
||||||
|
@Column({ name: 'setting', comment: '设置', length: 10240, nullable: true })
|
||||||
|
setting: string;
|
||||||
|
|
||||||
|
@Column({ name: 'is_system', comment: '是否系统级别', nullable: false, default: false })
|
||||||
|
isSystem: boolean;
|
||||||
|
|
||||||
|
@Column({ name: 'is_default', comment: '是否默认', nullable: false, default: false })
|
||||||
|
isDefault: boolean;
|
||||||
|
|
||||||
|
|
||||||
|
@Column({
|
||||||
|
name: 'create_time',
|
||||||
|
comment: '创建时间',
|
||||||
|
default: () => 'CURRENT_TIMESTAMP',
|
||||||
|
})
|
||||||
|
createTime: Date;
|
||||||
|
@Column({
|
||||||
|
name: 'update_time',
|
||||||
|
comment: '修改时间',
|
||||||
|
default: () => 'CURRENT_TIMESTAMP',
|
||||||
|
})
|
||||||
|
updateTime: Date;
|
||||||
|
}
|
||||||
3
packages/libs/lib-server/src/user/addon/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from './api/index.js'
|
||||||
|
export * from './entity/addon.js'
|
||||||
|
export * from './service/addon-service.js'
|
||||||
190
packages/libs/lib-server/src/user/addon/service/addon-service.ts
Normal file
@@ -0,0 +1,190 @@
|
|||||||
|
import { Provide, Scope, ScopeEnum } from "@midwayjs/core";
|
||||||
|
import { InjectEntityModel } from "@midwayjs/typeorm";
|
||||||
|
import { In, Repository } from "typeorm";
|
||||||
|
import { AddonDefine, BaseService, PageReq, ValidateException } from "../../../index.js";
|
||||||
|
import { addonRegistry } from "../api/index.js";
|
||||||
|
import { AddonEntity } from "../entity/addon.js";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Addon
|
||||||
|
*/
|
||||||
|
@Provide()
|
||||||
|
@Scope(ScopeEnum.Request, { allowDowngrade: true })
|
||||||
|
export class AddonService extends BaseService<AddonEntity> {
|
||||||
|
@InjectEntityModel(AddonEntity)
|
||||||
|
repository: Repository<AddonEntity>;
|
||||||
|
|
||||||
|
//@ts-ignore
|
||||||
|
getRepository() {
|
||||||
|
return this.repository;
|
||||||
|
}
|
||||||
|
|
||||||
|
async page(pageReq: PageReq<AddonEntity>) {
|
||||||
|
const res = await super.page(pageReq);
|
||||||
|
res.records = res.records.map(item => {
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
async add(param) {
|
||||||
|
let oldEntity = null;
|
||||||
|
if (param._copyFrom) {
|
||||||
|
oldEntity = await this.info(param._copyFrom);
|
||||||
|
if (oldEntity == null) {
|
||||||
|
throw new ValidateException("该Addon配置不存在,请确认是否已被删除");
|
||||||
|
}
|
||||||
|
if (oldEntity.userId !== param.userId) {
|
||||||
|
throw new ValidateException("您无权查看该Addon配置");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!param.userId) {
|
||||||
|
param.isSystem = true;
|
||||||
|
} else {
|
||||||
|
param.isSystem = false;
|
||||||
|
}
|
||||||
|
delete param._copyFrom;
|
||||||
|
return await super.add(param);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改
|
||||||
|
* @param param 数据
|
||||||
|
*/
|
||||||
|
async update(param) {
|
||||||
|
const oldEntity = await this.info(param.id);
|
||||||
|
if (oldEntity == null) {
|
||||||
|
throw new ValidateException("该Addon配置不存在,请确认是否已被删除");
|
||||||
|
}
|
||||||
|
return await super.update(param);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getSimpleInfo(id: number) {
|
||||||
|
const entity = await this.info(id);
|
||||||
|
if (entity == null) {
|
||||||
|
throw new ValidateException("该Addon配置不存在,请确认是否已被删除");
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
id: entity.id,
|
||||||
|
name: entity.name,
|
||||||
|
userId: entity.userId,
|
||||||
|
addonType: entity.addonType,
|
||||||
|
type: entity.type
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
getDefineList(addonType: string) {
|
||||||
|
return addonRegistry.getDefineList();
|
||||||
|
}
|
||||||
|
|
||||||
|
getDefineByType(type: string, prefix?: string) {
|
||||||
|
return addonRegistry.getDefine(type, prefix) as AddonDefine;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async getSimpleByIds(ids: number[], userId: any) {
|
||||||
|
if (ids.length === 0) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
if (!userId) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
return await this.repository.find({
|
||||||
|
where: {
|
||||||
|
id: In(ids),
|
||||||
|
userId
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
name: true,
|
||||||
|
addonType: true,
|
||||||
|
type: true,
|
||||||
|
userId: true,
|
||||||
|
isSystem: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async getDefault(userId: number, addonType: string): Promise<any> {
|
||||||
|
const res = await this.repository.findOne({
|
||||||
|
where: {
|
||||||
|
userId,
|
||||||
|
addonType
|
||||||
|
},
|
||||||
|
order: {
|
||||||
|
isDefault: "DESC"
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (!res) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return this.buildAddonInstanceConfig(res);
|
||||||
|
}
|
||||||
|
|
||||||
|
private buildAddonInstanceConfig(res: AddonEntity) {
|
||||||
|
const setting = JSON.parse(res.setting);
|
||||||
|
return {
|
||||||
|
id: res.id,
|
||||||
|
addonType: res.addonType,
|
||||||
|
type: res.type,
|
||||||
|
name: res.name,
|
||||||
|
userId: res.userId,
|
||||||
|
setting
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async setDefault(id: number, userId: number, addonType: string) {
|
||||||
|
if (!id) {
|
||||||
|
throw new ValidateException("id不能为空");
|
||||||
|
}
|
||||||
|
if (!userId) {
|
||||||
|
throw new ValidateException("userId不能为空");
|
||||||
|
}
|
||||||
|
await this.repository.update(
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
addonType
|
||||||
|
},
|
||||||
|
{
|
||||||
|
isDefault: false
|
||||||
|
}
|
||||||
|
);
|
||||||
|
await this.repository.update(
|
||||||
|
{
|
||||||
|
id,
|
||||||
|
userId,
|
||||||
|
addonType
|
||||||
|
},
|
||||||
|
{
|
||||||
|
isDefault: true
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getOrCreateDefault(opts: { addonType: string, type: string, inputs: any, userId: any }) {
|
||||||
|
const { addonType, type, inputs, userId } = opts;
|
||||||
|
|
||||||
|
const addonDefine = this.getDefineByType(type, addonType);
|
||||||
|
|
||||||
|
const defaultConfig = await this.getDefault(userId, addonType);
|
||||||
|
if (defaultConfig) {
|
||||||
|
return defaultConfig;
|
||||||
|
}
|
||||||
|
const setting = {
|
||||||
|
...inputs
|
||||||
|
};
|
||||||
|
const res = await this.repository.save({
|
||||||
|
userId,
|
||||||
|
addonType,
|
||||||
|
type: type,
|
||||||
|
name: addonDefine.title,
|
||||||
|
setting: JSON.stringify(setting),
|
||||||
|
isDefault: true
|
||||||
|
});
|
||||||
|
return this.buildAddonInstanceConfig(res);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1 +1,2 @@
|
|||||||
export * from './access/index.js';
|
export * from './access/index.js';
|
||||||
|
export * from './addon/index.js';
|
||||||
|
|||||||
@@ -3,6 +3,34 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/midway-flyway-js",
|
"name": "@certd/midway-flyway-js",
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,52 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/plugin-cert",
|
"name": "@certd/plugin-cert",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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.0",
|
||||||
"@certd/basic": "^1.36.19",
|
"@certd/basic": "^1.37.0",
|
||||||
"@certd/pipeline": "^1.36.19",
|
"@certd/pipeline": "^1.37.0",
|
||||||
"@certd/plugin-lib": "^1.36.19",
|
"@certd/plugin-lib": "^1.37.0",
|
||||||
"@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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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> {
|
||||||
|
|||||||
@@ -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";
|
||||||
|
|||||||
@@ -329,8 +329,9 @@ export class AcmeService {
|
|||||||
isTest?: boolean;
|
isTest?: boolean;
|
||||||
privateKeyType?: string;
|
privateKeyType?: string;
|
||||||
profile?: string;
|
profile?: string;
|
||||||
|
preferredChain?: string;
|
||||||
}): Promise<CertInfo> {
|
}): Promise<CertInfo> {
|
||||||
const { email, isTest, csrInfo, dnsProvider, domainsVerifyPlan, profile } = options;
|
const { email, isTest, csrInfo, dnsProvider, domainsVerifyPlan, profile, preferredChain } = options;
|
||||||
const client: acme.Client = await this.getAcmeClient(email, isTest);
|
const client: acme.Client = await this.getAcmeClient(email, isTest);
|
||||||
|
|
||||||
let domains = options.domains;
|
let domains = options.domains;
|
||||||
@@ -373,7 +374,7 @@ export class AcmeService {
|
|||||||
commonName,
|
commonName,
|
||||||
...csrInfo,
|
...csrInfo,
|
||||||
altNames,
|
altNames,
|
||||||
emailAddress: email,
|
// emailAddress: email,
|
||||||
},
|
},
|
||||||
privateKey
|
privateKey
|
||||||
);
|
);
|
||||||
@@ -404,6 +405,7 @@ export class AcmeService {
|
|||||||
},
|
},
|
||||||
signal: this.options.signal,
|
signal: this.options.signal,
|
||||||
profile,
|
profile,
|
||||||
|
preferredChain,
|
||||||
});
|
});
|
||||||
|
|
||||||
const crtString = crt.toString();
|
const crtString = crt.toString();
|
||||||
|
|||||||
@@ -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 = {};
|
||||||
|
|||||||
@@ -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() {
|
||||||
|
|||||||
@@ -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();
|
||||||
|
|||||||
@@ -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;
|
||||||
@@ -292,6 +339,19 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
})
|
})
|
||||||
certProfile!: string;
|
certProfile!: string;
|
||||||
|
|
||||||
|
@TaskInput({
|
||||||
|
title: "首选链",
|
||||||
|
component: {
|
||||||
|
name: "a-select",
|
||||||
|
vModel: "value",
|
||||||
|
options: preferredChainConfigs.letsencrypt.options,
|
||||||
|
},
|
||||||
|
helper: preferredChainConfigs.letsencrypt.helper,
|
||||||
|
required: false,
|
||||||
|
mergeScript: preferredChainMergeScript,
|
||||||
|
})
|
||||||
|
preferredChain!: string;
|
||||||
|
|
||||||
@TaskInput({
|
@TaskInput({
|
||||||
title: "使用代理",
|
title: "使用代理",
|
||||||
value: false,
|
value: false,
|
||||||
@@ -352,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);
|
||||||
@@ -438,6 +498,7 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
isTest: false,
|
isTest: false,
|
||||||
privateKeyType: this.privateKeyType,
|
privateKeyType: this.privateKeyType,
|
||||||
profile: this.certProfile,
|
profile: this.certProfile,
|
||||||
|
preferredChain: this.preferredChain,
|
||||||
});
|
});
|
||||||
|
|
||||||
const certInfo = this.formatCerts(cert);
|
const certInfo = this.formatCerts(cert);
|
||||||
@@ -463,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,
|
||||||
@@ -479,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);
|
||||||
@@ -568,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) {
|
||||||
|
throw new Error(`CNAME记录${domain}的域名与配置的主域名不一致,请确认是否在流水线创建之后修改了子域名托管,您需要重新校验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,
|
||||||
|
|||||||
@@ -3,6 +3,36 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/plugin-lib",
|
"name": "@certd/plugin-lib",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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.0",
|
||||||
"@certd/pipeline": "^1.36.19",
|
"@certd/pipeline": "^1.37.0",
|
||||||
"@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": "ea18a5ad151b296fda54fb5bcbe64c7d80cdff2f"
|
"gitHead": "6b43007c44e1b8ccdb87081b0cf3f99fc2c09d53"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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";
|
||||||
|
|||||||
1
packages/plugins/plugin-lib/src/lib/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from "./ocr-api.js";
|
||||||
3
packages/plugins/plugin-lib/src/lib/ocr-api.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export interface IOcrService {
|
||||||
|
doOcrFromImage(opts: { image: string }): Promise<{ texts: string[] }>;
|
||||||
|
}
|
||||||
@@ -64,6 +64,22 @@ export class SshAccess extends BaseAccess {
|
|||||||
})
|
})
|
||||||
passphrase!: string;
|
passphrase!: string;
|
||||||
|
|
||||||
|
@AccessInput({
|
||||||
|
title: "脚本类型",
|
||||||
|
helper: "bash 、sh 、fish",
|
||||||
|
component: {
|
||||||
|
name: "a-select",
|
||||||
|
vModel: "value",
|
||||||
|
options: [
|
||||||
|
{ value: "default", label: "默认" },
|
||||||
|
{ value: "sh", label: "sh" },
|
||||||
|
{ value: "bash", label: "bash" },
|
||||||
|
{ value: "fish", label: "fish(不支持set -e)" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
scriptType: string;
|
||||||
|
|
||||||
@AccessInput({
|
@AccessInput({
|
||||||
title: "伪终端",
|
title: "伪终端",
|
||||||
helper: "如果登录报错:all authentication methods failed,可以尝试开启伪终端模式进行keyboard-interactive方式登录\n开启后对日志输出有一定的影响",
|
helper: "如果登录报错:all authentication methods failed,可以尝试开启伪终端模式进行keyboard-interactive方式登录\n开启后对日志输出有一定的影响",
|
||||||
|
|||||||
@@ -543,8 +543,16 @@ export class SshClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isLinux && options.stopOnError !== false) {
|
if (isLinux) {
|
||||||
script = "set -e\n" + script;
|
if (options.connectConf.scriptType == "bash") {
|
||||||
|
script = "#!/usr/bin/env bash \n" + script;
|
||||||
|
} else if (options.connectConf.scriptType == "sh") {
|
||||||
|
script = "#!/bin/sh\n" + script;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (options.connectConf.scriptType != "fish" && options.stopOnError !== false) {
|
||||||
|
script = "set -e\n" + script;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return await conn.exec(script as string, { throwOnStdErr });
|
return await conn.exec(script as string, { throwOnStdErr });
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -3,6 +3,59 @@
|
|||||||
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)
|
||||||
|
|
||||||
|
**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
|
||||||
|
|||||||
@@ -23,5 +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://turing.captcha.qcloud.com/TJCaptcha.js"></script>-->
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/ui-client",
|
"name": "@certd/ui-client",
|
||||||
"version": "1.36.19",
|
"version": "1.37.0",
|
||||||
"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.0",
|
||||||
"@certd/pipeline": "^1.36.19",
|
"@certd/pipeline": "^1.37.0",
|
||||||
"@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",
|
||||||
|
|||||||
@@ -0,0 +1,68 @@
|
|||||||
|
<template>
|
||||||
|
<component :is="captchaComponent" v-if="settingStore.inited" ref="captchaRef" :model-value="modelValue" class="captcha_input" :captcha-get="getCaptcha" @change="onChange" />
|
||||||
|
</template>
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, computed, defineAsyncComponent } from "vue";
|
||||||
|
import { useSettingStore } from "/@/store/settings";
|
||||||
|
import { nanoid } from "nanoid";
|
||||||
|
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 settingStore = useSettingStore();
|
||||||
|
|
||||||
|
const emits = defineEmits(["update:modelValue", "change"]);
|
||||||
|
const captchaImpls = import.meta.glob("./captchas/*.vue");
|
||||||
|
|
||||||
|
const captchaAddonId = computed(() => {
|
||||||
|
return settingStore.sysPublic.captchaAddonId ?? 0;
|
||||||
|
});
|
||||||
|
const captchaComponent = computed(() => {
|
||||||
|
let type: any = props.type ?? "image";
|
||||||
|
if (settingStore.sysPublic.captchaAddonId && settingStore.sysPublic.captchaType) {
|
||||||
|
type = settingStore.sysPublic.captchaType;
|
||||||
|
}
|
||||||
|
const componentName = `${type}_captcha`;
|
||||||
|
return defineAsyncComponent(captchaImpls[`./captchas/${componentName}.vue`]);
|
||||||
|
});
|
||||||
|
|
||||||
|
async function getCaptcha(): Promise<any> {
|
||||||
|
const randomStr = nanoid(10);
|
||||||
|
return await request({
|
||||||
|
url: `/basic/code/captcha/get?randomStr=${randomStr}`,
|
||||||
|
method: "post",
|
||||||
|
data: {
|
||||||
|
captchaAddonId: captchaAddonId.value,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function onChange(data: any) {
|
||||||
|
emits("update:modelValue", data);
|
||||||
|
emits("change", data);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getCaptchaForm() {
|
||||||
|
return await captchaRef.value.getCaptchaForm();
|
||||||
|
}
|
||||||
|
async function reset() {
|
||||||
|
await captchaRef.value.reset();
|
||||||
|
}
|
||||||
|
defineExpose({
|
||||||
|
getCaptchaForm,
|
||||||
|
reset,
|
||||||
|
});
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,129 @@
|
|||||||
|
<template>
|
||||||
|
<div ref="captchaRef" class="geetest_captcha_wrapper"></div>
|
||||||
|
</template>
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { onMounted, defineProps, defineEmits, ref, onUnmounted, Ref, watch } from "vue";
|
||||||
|
import { useSettingStore } from "/@/store/settings";
|
||||||
|
import { request } from "/src/api/service";
|
||||||
|
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({
|
||||||
|
name: "GeetestCaptcha",
|
||||||
|
});
|
||||||
|
const emit = defineEmits(["update:modelValue", "change"]);
|
||||||
|
const props = defineProps<{
|
||||||
|
modelValue: any;
|
||||||
|
captchaGet: () => Promise<any>;
|
||||||
|
}>();
|
||||||
|
const captchaRef = ref(null);
|
||||||
|
|
||||||
|
const captchaInstanceRef: Ref = ref({});
|
||||||
|
async function init() {
|
||||||
|
await loadCaptchaScript();
|
||||||
|
const { captchaId } = await props.captchaGet();
|
||||||
|
// @ts-ignore
|
||||||
|
initGeetest4(
|
||||||
|
{
|
||||||
|
captchaId: captchaId,
|
||||||
|
},
|
||||||
|
(captcha: any) => {
|
||||||
|
// captcha为验证码实例
|
||||||
|
captcha.appendTo(captchaRef.value); // 调用appendTo将验证码插入到页的某一个元素中,这个元素用户可以自定义
|
||||||
|
captchaInstanceRef.value.instance = captcha;
|
||||||
|
captchaInstanceRef.value.captchaId = captchaId;
|
||||||
|
|
||||||
|
captcha.onSuccess(function () {
|
||||||
|
const form = getCaptchaForm();
|
||||||
|
if (form) {
|
||||||
|
emitChange(form);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCaptchaForm() {
|
||||||
|
if (!captchaInstanceRef.value?.instance) {
|
||||||
|
// notification.error({
|
||||||
|
// message: "验证码还未初始化",
|
||||||
|
// });
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const result = captchaInstanceRef.value.instance.getValidate();
|
||||||
|
if (!result) {
|
||||||
|
// notification.error({
|
||||||
|
// message: "请先完成验证码验证",
|
||||||
|
// });
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
result.captcha_id = captchaInstanceRef.value.captchaId;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
// const valueRef = ref(null);
|
||||||
|
// const timeoutId = setInterval(() => {
|
||||||
|
// const form = getCaptchaForm();
|
||||||
|
// if (form && valueRef.value != form) {
|
||||||
|
// console.log("form", form);
|
||||||
|
// valueRef.value = form;
|
||||||
|
// emitChange(form);
|
||||||
|
// }
|
||||||
|
// }, 1000);
|
||||||
|
|
||||||
|
// onUnmounted(() => {
|
||||||
|
// clearTimeout(timeoutId);
|
||||||
|
// });
|
||||||
|
|
||||||
|
function emitChange(value: string) {
|
||||||
|
emit("update:modelValue", value);
|
||||||
|
emit("change", value);
|
||||||
|
}
|
||||||
|
function reset() {
|
||||||
|
captchaInstanceRef.value.instance.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => {
|
||||||
|
return props.modelValue;
|
||||||
|
},
|
||||||
|
value => {
|
||||||
|
if (value == null) {
|
||||||
|
reset();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
defineExpose({
|
||||||
|
getCaptchaForm,
|
||||||
|
reset,
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => [props.captchaGet],
|
||||||
|
async () => {
|
||||||
|
await init();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
await init();
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<style lang="less">
|
||||||
|
.geetest_captcha_wrapper {
|
||||||
|
.geetest_captcha {
|
||||||
|
.geetest_holder {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
<template>
|
||||||
|
<div class="flex">
|
||||||
|
<a-input :value="valueRef" placeholder="请输入图片验证码" autocomplete="off" @update:value="onChange">
|
||||||
|
<template #prefix>
|
||||||
|
<fs-icon icon="ion:image-outline"></fs-icon>
|
||||||
|
</template>
|
||||||
|
</a-input>
|
||||||
|
<div class="input-right pointer" title="点击刷新">
|
||||||
|
<img class="image-code" :src="imageCodeSrc" @click="resetImageCode" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { defineEmits, defineExpose, defineProps, ref, watch } from "vue";
|
||||||
|
import { nanoid } from "nanoid";
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
modelValue: any;
|
||||||
|
captchaGet?: () => Promise<any>;
|
||||||
|
}>();
|
||||||
|
defineOptions({
|
||||||
|
name: "ImageCaptcha",
|
||||||
|
});
|
||||||
|
const emit = defineEmits(["update:modelValue", "change"]);
|
||||||
|
|
||||||
|
const valueRef = ref("");
|
||||||
|
const randomStrRef = ref();
|
||||||
|
const imageCodeSrc = ref();
|
||||||
|
async function resetImageCode() {
|
||||||
|
const res = await props.captchaGet();
|
||||||
|
randomStrRef.value = res.randomStr;
|
||||||
|
valueRef.value = "";
|
||||||
|
emitChange(null);
|
||||||
|
imageCodeSrc.value = "data:image/svg+xml," + encodeURIComponent(res.imageData);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCaptchaForm() {
|
||||||
|
return {
|
||||||
|
imageCode: valueRef.value,
|
||||||
|
randomStr: randomStrRef.value,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
defineExpose({
|
||||||
|
resetImageCode,
|
||||||
|
getCaptchaForm,
|
||||||
|
reset: resetImageCode,
|
||||||
|
});
|
||||||
|
|
||||||
|
resetImageCode();
|
||||||
|
|
||||||
|
function onChange(value: string) {
|
||||||
|
valueRef.value = value;
|
||||||
|
const form = getCaptchaForm();
|
||||||
|
emitChange(form);
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => {
|
||||||
|
return props.modelValue;
|
||||||
|
},
|
||||||
|
value => {
|
||||||
|
if (value == null) {
|
||||||
|
resetImageCode();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
function emitChange(value: any) {
|
||||||
|
emit("update:modelValue", value);
|
||||||
|
emit("change", value);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
@@ -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>
|
||||||
@@ -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",
|
||||||
|
|||||||
@@ -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">
|
||||||
|
|||||||
@@ -53,7 +53,6 @@ const pagerRef: Ref = ref({
|
|||||||
current: 1,
|
current: 1,
|
||||||
});
|
});
|
||||||
const getOptions = async () => {
|
const getOptions = async () => {
|
||||||
debugger;
|
|
||||||
if (loading.value) {
|
if (loading.value) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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",
|
||||||
@@ -465,6 +468,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,10 +711,26 @@ 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",
|
||||||
showRunStrategyHelper: "Allow modify the run strategy of the task",
|
showRunStrategyHelper: "Allow modify the run strategy of the task",
|
||||||
|
|
||||||
|
captchaEnabled: "Enable Login Captcha",
|
||||||
|
captchaHelper: "Whether to enable captcha verification for login",
|
||||||
|
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: {
|
||||||
@@ -731,4 +751,8 @@ export default {
|
|||||||
challengeSetting: "Challenge Setting",
|
challengeSetting: "Challenge Setting",
|
||||||
gotoCnameTip: "Please go to CNAME Record Page",
|
gotoCnameTip: "Please go to CNAME Record Page",
|
||||||
},
|
},
|
||||||
|
addonSelector: {
|
||||||
|
select: "Select",
|
||||||
|
placeholder: "select please",
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -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: "上次检查时间",
|
||||||
@@ -458,7 +461,7 @@ export default {
|
|||||||
batchDeleteConfirm: "确定要批量删除这{count}条记录吗",
|
batchDeleteConfirm: "确定要批量删除这{count}条记录吗",
|
||||||
selectRecordFirst: "请先勾选记录",
|
selectRecordFirst: "请先勾选记录",
|
||||||
subdomainHosted: "托管的子域名",
|
subdomainHosted: "托管的子域名",
|
||||||
subdomainHelpText: "如果您不理解什么是子域托管,请不要随意设置,可能导致证书无法申请,可以参考文档",
|
subdomainHelpText: "如果您不理解什么是子域托管,请不要随意设置(可能导致证书无法申请,以前设置过的cname记录也需要重新配置),可以参考文档",
|
||||||
subdomainManagement: "子域管理",
|
subdomainManagement: "子域管理",
|
||||||
isDisabled: "是否禁用",
|
isDisabled: "是否禁用",
|
||||||
enabled: "启用",
|
enabled: "启用",
|
||||||
@@ -471,6 +474,7 @@ export default {
|
|||||||
validDays: "有效天数",
|
validDays: "有效天数",
|
||||||
expires: "过期",
|
expires: "过期",
|
||||||
days: "天",
|
days: "天",
|
||||||
|
effectiveTime: "生效时间",
|
||||||
expireTime: "过期时间",
|
expireTime: "过期时间",
|
||||||
certIssuer: "证书颁发机构",
|
certIssuer: "证书颁发机构",
|
||||||
applyTime: "申请时间",
|
applyTime: "申请时间",
|
||||||
@@ -695,7 +699,10 @@ export default {
|
|||||||
setAsDefault: "设为默认",
|
setAsDefault: "设为默认",
|
||||||
disabledLabel: "禁用",
|
disabledLabel: "禁用",
|
||||||
confirmToggleStatus: "确定要{action}吗?",
|
confirmToggleStatus: "确定要{action}吗?",
|
||||||
|
addonType: "类型",
|
||||||
|
addonName: "名称",
|
||||||
|
addonNameHelper: "随意填写,相同类型助于区分即可",
|
||||||
|
addonTypeSelect: "请选择",
|
||||||
template: {
|
template: {
|
||||||
title: "流水线模版",
|
title: "流水线模版",
|
||||||
edit: "流水线模版编辑",
|
edit: "流水线模版编辑",
|
||||||
@@ -714,6 +721,17 @@ export default {
|
|||||||
setting: {
|
setting: {
|
||||||
showRunStrategy: "显示运行策略选择",
|
showRunStrategy: "显示运行策略选择",
|
||||||
showRunStrategyHelper: "任务设置中是否允许选择运行策略",
|
showRunStrategyHelper: "任务设置中是否允许选择运行策略",
|
||||||
|
|
||||||
|
captchaEnabled: "启用登录验证码",
|
||||||
|
captchaHelper: "登录时是否启用验证码",
|
||||||
|
captchaType: "验证码配置",
|
||||||
|
captchaTest: "测试验证码",
|
||||||
|
captchaTestHelper: "保存后再点击测试,请务必测试通过了,再开启登录验证码",
|
||||||
|
baseSetting: "基本设置",
|
||||||
|
registerSetting: "注册设置",
|
||||||
|
safeSetting: "安全设置",
|
||||||
|
paymentSetting: "支付设置",
|
||||||
|
captchaSetting: "验证码设置",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
modal: {
|
modal: {
|
||||||
@@ -734,4 +752,8 @@ export default {
|
|||||||
challengeSetting: "校验配置",
|
challengeSetting: "校验配置",
|
||||||
gotoCnameTip: "CNAME域名配置请前往CNAME记录页面添加",
|
gotoCnameTip: "CNAME域名配置请前往CNAME记录页面添加",
|
||||||
},
|
},
|
||||||
|
addonSelector: {
|
||||||
|
select: "选择",
|
||||||
|
placeholder: "请选择",
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -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();
|
||||||
//此处演示修改官方字段类型
|
//此处演示修改官方字段类型
|
||||||
|
|||||||