Compare commits
1 Commits
v1.36.25
...
v2-dev-ord
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
69e6f2362e |
@@ -1,11 +0,0 @@
|
|||||||
#
|
|
||||||
# http://editorconfig.org
|
|
||||||
#
|
|
||||||
|
|
||||||
root = true
|
|
||||||
|
|
||||||
[*]
|
|
||||||
indent_style = space
|
|
||||||
indent_size = 2
|
|
||||||
trim_trailing_whitespace = true
|
|
||||||
|
|
||||||
5
.github/FUNDING.yml
vendored
@@ -1,5 +0,0 @@
|
|||||||
# These are supported funding model platforms
|
|
||||||
|
|
||||||
github: greper
|
|
||||||
buy_me_a_coffee: greper
|
|
||||||
custom: ['https://afdian.com/a/greper']
|
|
||||||
@@ -1,28 +1,21 @@
|
|||||||
---
|
|
||||||
name: Bug Report
|
|
||||||
about: 错误或问题报告
|
|
||||||
title: "[BUG] "
|
|
||||||
labels: bug
|
|
||||||
---
|
|
||||||
|
|
||||||
|
|
||||||
> 感谢您支持certd,请按如下规范提交issue
|
> 感谢您支持certd,请按如下规范提交issue
|
||||||
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
||||||
|
|
||||||
# bug提交
|
|
||||||
## 1、问题描述
|
## 一、问题描述
|
||||||
`请在此处简要描述你所遇到的问题,必要时请贴出相关截图辅助理解和定位`
|
`请在此处简要描述你所遇到的问题,必要时请贴出相关截图辅助理解和定位`
|
||||||
|
|
||||||
### 2、复现步骤
|
### 复现步骤
|
||||||
`请描述复现问题的详细步骤`
|
`请描述复现问题的详细步骤`
|
||||||
`如果非示例页面的问题,最好能提供最小复现示例的代码、或者仓库链接`
|
`如果非示例页面的问题,最好能提供最小复现示例的代码、或者仓库链接`
|
||||||
|
|
||||||
|
|
||||||
### 3.报错截图
|
### 报错截图
|
||||||
`请贴出报错日志截图`
|
`请贴出报错日志截图`
|
||||||
|
|
||||||
### 4、效果截图
|
### 效果截图
|
||||||
`请贴出效果截图`
|
`请贴出效果截图`
|
||||||
#### 4.1. 期望效果
|
#### 1. 期望效果
|
||||||
|
|
||||||
|
#### 2. 实际效果
|
||||||
|
|
||||||
#### 4.2. 实际效果
|
|
||||||
36
.github/ISSUE_TEMPLATE/1plugin.md
vendored
@@ -1,36 +0,0 @@
|
|||||||
---
|
|
||||||
name: Plugin Apply
|
|
||||||
about: 部署插件申请支持
|
|
||||||
title: "[Plugin] "
|
|
||||||
labels: feature
|
|
||||||
---
|
|
||||||
|
|
||||||
> > 感谢您支持certd,请按如下规范提交issue
|
|
||||||
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
|
||||||
|
|
||||||
# 新部署插件申请支持
|
|
||||||
|
|
||||||
## 1. 需求描述
|
|
||||||
`请在此处简要描述你的需求`
|
|
||||||
|
|
||||||
|
|
||||||
## 2. 要部署证书应用的信息
|
|
||||||
|
|
||||||
1. 应用名称:
|
|
||||||
|
|
||||||
|
|
||||||
2. 应用网址/项目地址/官方网站:
|
|
||||||
|
|
||||||
|
|
||||||
3. 管理证书界面截图(或者手动部署证书方式介绍及截图):
|
|
||||||
|
|
||||||
|
|
||||||
4. 是否有API接口,接口地址:
|
|
||||||
|
|
||||||
|
|
||||||
5. 如果没有API接口,网页登录是否需要验证码:
|
|
||||||
|
|
||||||
|
|
||||||
6. 是否可以提供测试账号?(如果可以请留下联系方式或者加作者好友)
|
|
||||||
|
|
||||||
|
|
||||||
36
.github/ISSUE_TEMPLATE/2dns.md
vendored
@@ -1,36 +0,0 @@
|
|||||||
---
|
|
||||||
name: DNS Provider Apply
|
|
||||||
about: 域名提供商申请支持
|
|
||||||
title: "[DNS] "
|
|
||||||
labels: feature
|
|
||||||
---
|
|
||||||
|
|
||||||
|
|
||||||
> 感谢您支持certd,请按如下规范提交issue
|
|
||||||
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
|
||||||
|
|
||||||
# 新域名提供商支持申请
|
|
||||||
|
|
||||||
## 1. 基本信息
|
|
||||||
请填写如下内容:
|
|
||||||
|
|
||||||
1. 域名提供商名称:
|
|
||||||
|
|
||||||
|
|
||||||
2. 管理页面地址:
|
|
||||||
|
|
||||||
|
|
||||||
3. 是否有API接口,接口地址:
|
|
||||||
|
|
||||||
|
|
||||||
4. 如果没有API接口,网页登录是否有验证码:
|
|
||||||
|
|
||||||
|
|
||||||
5. 是否可以提供测试账号?(如果可以请留下联系方式或者加作者好友)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## 2. 截图
|
|
||||||
|
|
||||||
`域名管理页面截图`
|
|
||||||
|
|
||||||
24
.github/ISSUE_TEMPLATE/4feature.md
vendored
@@ -1,24 +0,0 @@
|
|||||||
---
|
|
||||||
name: Feature Request
|
|
||||||
about: 新需求、新特性申请支持
|
|
||||||
title: "[Feature] "
|
|
||||||
labels: feature
|
|
||||||
---
|
|
||||||
|
|
||||||
> > 感谢您支持certd,请按如下规范提交issue
|
|
||||||
> 如果有条件,请尽量在[github上提交](https://github.com/certd/certd/issues)
|
|
||||||
|
|
||||||
|
|
||||||
# 新特性申请
|
|
||||||
>注意:这里仅供如果是要申请新的部署插件,请提交插件申请
|
|
||||||
|
|
||||||
## 1. 需求描述,需求背景
|
|
||||||
`请在此处简要描述你所遇到的问题,必要时请贴出相关截图辅助理解`
|
|
||||||
|
|
||||||
|
|
||||||
## 2. 期望效果
|
|
||||||
`必要时可以截图描述你的期望效果`
|
|
||||||
|
|
||||||
|
|
||||||
## 3. 你的解决方案
|
|
||||||
`如果你有解决方案,请描述你的方案`
|
|
||||||
2
.gitignore
vendored
@@ -1,5 +1,6 @@
|
|||||||
./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
|
||||||
@@ -29,4 +30,5 @@ 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
@@ -1,43 +0,0 @@
|
|||||||
{
|
|
||||||
// 使用 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
@@ -1,4 +0,0 @@
|
|||||||
{
|
|
||||||
"eslint.debug": false,
|
|
||||||
"eslint.format.enable": true
|
|
||||||
}
|
|
||||||
52
.vscode/tasks.json
vendored
@@ -1,52 +0,0 @@
|
|||||||
{
|
|
||||||
"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": []
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
210
CHANGELOG.md
@@ -3,216 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 更新我爱云CDN域名地址,和部分目录结构 [@tyjsjxh](https://github.com/tyjsjxh) ([#514](https://github.com/certd/certd/issues/514)) ([78e7a81](https://github.com/certd/certd/commit/78e7a81638c2ee779f0ab6c3ba7e5c6f6e064151))
|
|
||||||
* 修复cron选择组件星期显示错误的bug ([eb75e52](https://github.com/certd/certd/commit/eb75e52278f94a72643f7317e6740fb42666c68a))
|
|
||||||
* 修复proxmox某些情况下执行卡住的bug ([ebd6917](https://github.com/certd/certd/commit/ebd6917a1d40ae4d94555c32b7e3c093d0599b94))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到k8s支持自动创建secret ([c09c962](https://github.com/certd/certd/commit/c09c962cb676ca261610aa9f3e5105c9dae43f43))
|
|
||||||
* 短信验证码支持腾讯云 ([9108459](https://github.com/certd/certd/commit/9108459ae42bcd95a59acba164a64e82e5f2cfe6))
|
|
||||||
* 商业版支持自定义插件的参数配置 ([17f23f3](https://github.com/certd/certd/commit/17f23f37516af925d5049291d67d41e4271f81f8))
|
|
||||||
* 腾讯云插件支持国际版 ([58e82d5](https://github.com/certd/certd/commit/58e82d5dbd4ebf089ef239578ef9b68454d17b30))
|
|
||||||
* 腾讯云EO插件支持自动获取zoneid和域名列表 ([70fcdc9](https://github.com/certd/certd/commit/70fcdc9ebbfb7c883c0c8a2138f61a0776a9491b))
|
|
||||||
* 支持部署到阿里云云原生API网关、AI网关 ([2ca20be](https://github.com/certd/certd/commit/2ca20be197720201fceabcce9d927f4dbc1cc872))
|
|
||||||
* 支持部署到华为云obs ([9feb9d0](https://github.com/certd/certd/commit/9feb9d04b3c56ec95c06fcf4fd071eb0e88ffc6f))
|
|
||||||
* 支持部署到dokploy ([7dbdeae](https://github.com/certd/certd/commit/7dbdeaebe0bfee7521a863fe5e6b4a712aec5876))
|
|
||||||
* 支持删除宝塔证书夹中的过期证书 ([3575113](https://github.com/certd/certd/commit/3575113655be751d19f88c64491e98a89042d6a2))
|
|
||||||
* 支持p7b证书格式 ([d9f4a57](https://github.com/certd/certd/commit/d9f4a5793d68a017a5d80ad5385cbda603c4e165))
|
|
||||||
* lecdnv2支持api token ([e448934](https://github.com/certd/certd/commit/e4489343fee7754be07bcfc3323969dc3a30e90c))
|
|
||||||
* openapi返回证书时挑选匹配范围最小的那一个;增加format参数,增加返回值p7b格式,增加detail返回 ([2085bcc](https://github.com/certd/certd/commit/2085bcceb61c3723c9bdfec4c4cc0917631ff5e5))
|
|
||||||
* ssh 配置sudo免密提示 ([e1e7011](https://github.com/certd/certd/commit/e1e7011853ad0c5bd7b09c3690861d5aa34b2db4))
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复新部署的无法保存公共eab配置的bug ([d5dee75](https://github.com/certd/certd/commit/d5dee75df3bd635a597436e448b2de1407531f3a))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 阿里云 FC3.0 不在要求证书加密方式为旧版, 修复支持的协议类型可以正常选择 ([a34db74](https://github.com/certd/certd/commit/a34db7449eff6ad1dda01de673bf85579fa3865a))
|
|
||||||
* 部署到腾讯云cdn,每个域名增加3每秒延迟 ([f7d43ad](https://github.com/certd/certd/commit/f7d43ad5af4663d4be369820a80d1fd9817ca4ab))
|
|
||||||
* 腾讯云关闭证书通知增加开关选项,在腾讯云授权里面 ([a77c777](https://github.com/certd/certd/commit/a77c777980dd38d97d983124eeed1596879bba95))
|
|
||||||
* 证书申请任务默认不发送申请成功通知 ([0283bd2](https://github.com/certd/certd/commit/0283bd2f978dbcd13d361129135e439dd9fbc180))
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复授权配置复制功能,无法复制已加密字段的问题 ([221e068](https://github.com/certd/certd/commit/221e068bac3af6cd5d1794f8cd4c2ec5c0bc3f45))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 百度云支持上传到证书托管,支持部署到负载均衡 ([798a48a](https://github.com/certd/certd/commit/798a48aa9686fd5d11cfffb6cd93eadfc40aacb3))
|
|
||||||
* 部署到百度cdn支持自动获取域名列表选择 ([4e432ed](https://github.com/certd/certd/commit/4e432ed03f4fb564e85a2f284ee26b58400b82f5))
|
|
||||||
* 验证码可重试次数设置为3次 ([1bdceee](https://github.com/certd/certd/commit/1bdceeecf4b5daecdd621a05a2596b6eb45ce8ea))
|
|
||||||
* 增加找回密码的验证码可重试次数 [@nicheng-he](https://github.com/nicheng-he) ([#496](https://github.com/certd/certd/issues/496)) ([fe03f99](https://github.com/certd/certd/commit/fe03f9942b5662fb90cad86da10782f5dc3603f5))
|
|
||||||
* 支持阿里云API网关 ([9e1e4ee](https://github.com/certd/certd/commit/9e1e4eeec2859759ca5b07834c9d24cf88a6ad33))
|
|
||||||
* 支持部署到金山云CDN ([dfa74a6](https://github.com/certd/certd/commit/dfa74a69f7cbb9009d3e20c7eecfa1b905a00cf0))
|
|
||||||
* 支持更新金山云cdn证书 ([462e22a](https://github.com/certd/certd/commit/462e22a3b0a94887462fe6aa68e4671a365e0737))
|
|
||||||
* 支持apisix证书部署 ([9b63fb4](https://github.com/certd/certd/commit/9b63fb4ee2c6b56139160c5bf63482dab0869c2b))
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复 https://cas.undefined.aliyuncs.com 的bug ([60e6aa9](https://github.com/certd/certd/commit/60e6aa9b54a761a47e39acee4a1ff947a745be27))
|
|
||||||
* 修复阿里云clb api接口没有使用region的问题 ([0770f17](https://github.com/certd/certd/commit/0770f174a14313e28d08113e69829ef6cc02d719))
|
|
||||||
* 修复站点监控使用自定义dns解析域名报错的bug ([eb8cd53](https://github.com/certd/certd/commit/eb8cd53de27991321e36dd14e5ce95f42b51351f))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到阿里云支持选择bucket和域名 ([013b9c4](https://github.com/certd/certd/commit/013b9c4c7c2adf485d086123ccea448719577fd4))
|
|
||||||
* 清理数据库备份的临时目录 ([fd95549](https://github.com/certd/certd/commit/fd95549de9a5d8cec09772ee2630bb7521e15e1f))
|
|
||||||
* 添加免费通知,OneBot V11协议通知支持 ([#491](https://github.com/certd/certd/issues/491)) [@ayakasuki](https://github.com/ayakasuki) ([be053d4](https://github.com/certd/certd/commit/be053d47e41084f817882400882b64143d036d1a))
|
|
||||||
* 支持webhook部署证书 ([cbe0b1c](https://github.com/certd/certd/commit/cbe0b1c5a6538f232e9a63f1693d20d5acf0a306))
|
|
||||||
* 注册时支持填写用户名 ([fdcfcc7](https://github.com/certd/certd/commit/fdcfcc77a0db87954e0b026635d3ccdd9bc6cee8))
|
|
||||||
* add start:server npm script for quick server launch from root directory ([#484](https://github.com/certd/certd/issues/484)) [@orzyyyy](https://github.com/orzyyyy) ([fae1981](https://github.com/certd/certd/commit/fae1981161080f698c3f1263b712306d63baae64))
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复复制流水线为空的bug ([b070773](https://github.com/certd/certd/commit/b0707739fdfbae3d78db4efd3f180db05c4e4164))
|
|
||||||
* 修复商用证书上传第二次运行无法使用pfx格式证书的bug ([251dd1f](https://github.com/certd/certd/commit/251dd1fe457a7b152f43eb6de18f7beb9f0b194e))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 1panel支持 currenNode ([acc8907](https://github.com/certd/certd/commit/acc890730f43d492c9b1bd3668814cf10efdf7b8))
|
|
||||||
* 授权管理支持模糊查询 ([866eb62](https://github.com/certd/certd/commit/866eb6241baa7b21f6eddc649966324c188236c6))
|
|
||||||
* 新增找回密码功能 [@nicheng-he](https://github.com/nicheng-he) ([81ac240](https://github.com/certd/certd/commit/81ac240ac84db0af2f56b6352e227ecb49f38377))
|
|
||||||
* 优化start脚本 ([238ad7c](https://github.com/certd/certd/commit/238ad7ce51f17e1098c624e7f61ee2d98de1e02d))
|
|
||||||
* 运行主机脚本插件支持选择运行策略 ([86b3df1](https://github.com/certd/certd/commit/86b3df194126476e1f58e0952a77e986f62eecce))
|
|
||||||
* cdnfly 支持 账号密码登陆授权 ([e87f6d5](https://github.com/certd/certd/commit/e87f6d56f524dbbb9e3243e382b348b6e49f0d2c))
|
|
||||||
* k8s ack、tke 支持重启ingress ([95715a0](https://github.com/certd/certd/commit/95715a007d931c64fa7dd953d94957398e00a443))
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复阿里云发送短信验证码失败的bug ([2e6d03f](https://github.com/certd/certd/commit/2e6d03ff001f521f57368e7a62b97ed7b122e8d0))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 阿里云部分插件优化 [@nicheng-he](https://github.com/nicheng-he) ([e3738f6](https://github.com/certd/certd/commit/e3738f6422270d75ec414c15a343248cc4cad6e1))
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 上传到阿里云cas,证书前缀无效的bug ([b382351](https://github.com/certd/certd/commit/b382351c7b91ec10e1f61d94bec5aad075207ec8))
|
|
||||||
* 修复自定义插件onlyAdmin报错的bug ([4e5e862](https://github.com/certd/certd/commit/4e5e862f5834ad180e4428959c272d444a6f78ab))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到k8s,tke,ack忽悠证书校验 ([ab84835](https://github.com/certd/certd/commit/ab848353621869464a2c9a45fdb5e28d998b8a58))
|
|
||||||
* 首页增加更新日志按钮 ([41ce848](https://github.com/certd/certd/commit/41ce8489dc2f03a705dfa3fbb357769defb56c60))
|
|
||||||
* 增加版本过低提示 ([d1ce360](https://github.com/certd/certd/commit/d1ce36038cab72b5dc1b320d0a708c261ffbdacb))
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# Certd
|
# Certd
|
||||||
|
|
||||||
中文 | [English](./README_en.md)
|
[English](./README_en.md) | [中文](./README.md)
|
||||||
|
|
||||||
Certd® 是一个免费的全自动证书管理系统,让你的网站证书永不过期。
|
Certd® 是一个免费的全自动证书管理系统,让你的网站证书永不过期。
|
||||||
后缀d取自linux守护进程的命名风格,意为证书守护进程
|
后缀d取自linux守护进程的命名风格,意为证书守护进程
|
||||||
@@ -16,6 +16,7 @@ Certd® 是一个免费的全自动证书管理系统,让你的网站证书永
|
|||||||
|
|
||||||
> 流水线数量现已调整为无限制,欢迎大家使用
|
> 流水线数量现已调整为无限制,欢迎大家使用
|
||||||
|
|
||||||
|
|
||||||
## 一、特性
|
## 一、特性
|
||||||
本项目不仅支持证书申请过程自动化,还可以自动化部署更新证书,让你的证书永不过期。
|
本项目不仅支持证书申请过程自动化,还可以自动化部署更新证书,让你的证书永不过期。
|
||||||
|
|
||||||
@@ -86,8 +87,8 @@ https://certd.handfree.work/
|
|||||||
1. 【推荐】[Docker方式部署 ](https://certd.docmirror.cn/guide/install/docker/)
|
1. 【推荐】[Docker方式部署 ](https://certd.docmirror.cn/guide/install/docker/)
|
||||||
2. 【推荐】[宝塔面板方式部署 ](https://certd.docmirror.cn/guide/install/docker/)
|
2. 【推荐】[宝塔面板方式部署 ](https://certd.docmirror.cn/guide/install/docker/)
|
||||||
3. 【推荐】[1Panel面板方式部署](https://certd.docmirror.cn/guide/install/1panel/)
|
3. 【推荐】[1Panel面板方式部署](https://certd.docmirror.cn/guide/install/1panel/)
|
||||||
4. 【推荐】[雨云一键部署](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2) : 首充翻倍,每月仅需2.2元
|
4. 【推荐】[雨云一键部署](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2_) : 首充翻倍,每月仅需2.2元
|
||||||
[<img src="https://rainyun-apps.cn-nb1.rains3.com/materials/deploy-on-rainyun-cn.svg">](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2)
|
[<img src="https://rainyun-apps.cn-nb1.rains3.com/materials/deploy-on-rainyun-cn.svg">](https://app.rainyun.com/apps/rca/store/6646/?ref=NzExMDQ2_)
|
||||||
5. 【不推荐】[源码方式部署 ](https://certd.docmirror.cn/guide/install/source/)
|
5. 【不推荐】[源码方式部署 ](https://certd.docmirror.cn/guide/install/source/)
|
||||||
|
|
||||||
#### Docker镜像说明:
|
#### Docker镜像说明:
|
||||||
@@ -171,7 +172,6 @@ https://afdian.com/a/greper
|
|||||||
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署、宝塔、1Panel等大部分插件 | 群晖 |
|
| 自动部署插件 | 阿里云CDN、腾讯云、七牛CDN、主机部署、宝塔、1Panel等大部分插件 | 群晖 |
|
||||||
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、钉钉、飞书、anpush、server酱等 |
|
| 通知 | 邮件通知、自定义webhook | 邮件免配置、企微、钉钉、飞书、anpush、server酱等 |
|
||||||
|
|
||||||
************************
|
|
||||||
|
|
||||||
************************
|
************************
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# Certd
|
# Certd
|
||||||
|
|
||||||
[中文](./README.md) | English
|
[English](./README_en.md) | [中文](./README.md)
|
||||||
|
|
||||||
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,8 +134,6 @@ 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 +1 @@
|
|||||||
5
|
12:27
|
||||||
|
|||||||
@@ -11,9 +11,6 @@ services:
|
|||||||
# ↓↓↓↓↓ -------------------------------------------------------- 数据库以及证书存储路径,默认存在宿主机的/data/certd/目录下,【您需要定时备份此目录,以保障数据容灾】
|
# ↓↓↓↓↓ -------------------------------------------------------- 数据库以及证书存储路径,默认存在宿主机的/data/certd/目录下,【您需要定时备份此目录,以保障数据容灾】
|
||||||
# 只要修改冒号前面的,冒号后面的/app/data不要动
|
# 只要修改冒号前面的,冒号后面的/app/data不要动
|
||||||
- /data/certd:/app/data
|
- /data/certd:/app/data
|
||||||
# ↓↓↓↓↓ -------------------------------------------------------- 如果走时不准,考虑挂载localtime文件
|
|
||||||
#- /etc/localtime:/etc/localtime
|
|
||||||
#- /etc/timezone:/etc/timezone
|
|
||||||
ports: # 端口映射
|
ports: # 端口映射
|
||||||
# ↓↓↓↓ ---------------------------------------------------------- 如果端口有冲突,可以修改第一个7001为其他不冲突的端口号,第二个7001不要动
|
# ↓↓↓↓ ---------------------------------------------------------- 如果端口有冲突,可以修改第一个7001为其他不冲突的端口号,第二个7001不要动
|
||||||
- "7001:7001"
|
- "7001:7001"
|
||||||
@@ -41,11 +38,11 @@ services:
|
|||||||
# - ip6net
|
# - ip6net
|
||||||
environment:
|
environment:
|
||||||
# ↓↓↓↓ ----------------------------------------------------- 使用上海东八时区
|
# ↓↓↓↓ ----------------------------------------------------- 使用上海东八时区
|
||||||
- TZ=Asia/Shanghai
|
# - TZ=Asia/Shanghai
|
||||||
# 设置环境变量即可自定义certd配置
|
# 设置环境变量即可自定义certd配置
|
||||||
# 配置项见: packages/ui/certd-server/src/config/config.default.ts
|
# 配置项见: packages/ui/certd-server/src/config/config.default.ts
|
||||||
# 配置规则: certd_ + 配置项, 点号用_代替
|
# 配置规则: certd_ + 配置项, 点号用_代替
|
||||||
# #↓↓↓↓ ----------------------------- 如果忘记管理员密码,可以设置为true,docker compose up -d 重建容器之后,管理员密码将改成123456,然后请及时修改回false
|
# #↓↓↓↓ ----------------------------- 如果忘记管理员密码,可以设置为true,重启之后,管理员密码将改成123456,然后请及时修改回false
|
||||||
- certd_system_resetAdminPasswd=false
|
- certd_system_resetAdminPasswd=false
|
||||||
|
|
||||||
# 默认使用sqlite文件数据库,如果需要使用其他数据库,请设置以下环境变量
|
# 默认使用sqlite文件数据库,如果需要使用其他数据库,请设置以下环境变量
|
||||||
|
|||||||
@@ -107,6 +107,7 @@ 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"},
|
||||||
@@ -119,8 +120,6 @@ 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"},
|
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -3,210 +3,6 @@
|
|||||||
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.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)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 更新我爱云CDN域名地址,和部分目录结构 [@tyjsjxh](https://github.com/tyjsjxh) ([#514](https://github.com/certd/certd/issues/514)) ([78e7a81](https://github.com/certd/certd/commit/78e7a81638c2ee779f0ab6c3ba7e5c6f6e064151))
|
|
||||||
* 修复cron选择组件星期显示错误的bug ([eb75e52](https://github.com/certd/certd/commit/eb75e52278f94a72643f7317e6740fb42666c68a))
|
|
||||||
* 修复proxmox某些情况下执行卡住的bug ([ebd6917](https://github.com/certd/certd/commit/ebd6917a1d40ae4d94555c32b7e3c093d0599b94))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到k8s支持自动创建secret ([c09c962](https://github.com/certd/certd/commit/c09c962cb676ca261610aa9f3e5105c9dae43f43))
|
|
||||||
* 短信验证码支持腾讯云 ([9108459](https://github.com/certd/certd/commit/9108459ae42bcd95a59acba164a64e82e5f2cfe6))
|
|
||||||
* 商业版支持自定义插件的参数配置 ([17f23f3](https://github.com/certd/certd/commit/17f23f37516af925d5049291d67d41e4271f81f8))
|
|
||||||
* 腾讯云插件支持国际版 ([58e82d5](https://github.com/certd/certd/commit/58e82d5dbd4ebf089ef239578ef9b68454d17b30))
|
|
||||||
* 腾讯云EO插件支持自动获取zoneid和域名列表 ([70fcdc9](https://github.com/certd/certd/commit/70fcdc9ebbfb7c883c0c8a2138f61a0776a9491b))
|
|
||||||
* 支持部署到阿里云云原生API网关、AI网关 ([2ca20be](https://github.com/certd/certd/commit/2ca20be197720201fceabcce9d927f4dbc1cc872))
|
|
||||||
* 支持部署到华为云obs ([9feb9d0](https://github.com/certd/certd/commit/9feb9d04b3c56ec95c06fcf4fd071eb0e88ffc6f))
|
|
||||||
* 支持部署到dokploy ([7dbdeae](https://github.com/certd/certd/commit/7dbdeaebe0bfee7521a863fe5e6b4a712aec5876))
|
|
||||||
* 支持删除宝塔证书夹中的过期证书 ([3575113](https://github.com/certd/certd/commit/3575113655be751d19f88c64491e98a89042d6a2))
|
|
||||||
* 支持p7b证书格式 ([d9f4a57](https://github.com/certd/certd/commit/d9f4a5793d68a017a5d80ad5385cbda603c4e165))
|
|
||||||
* lecdnv2支持api token ([e448934](https://github.com/certd/certd/commit/e4489343fee7754be07bcfc3323969dc3a30e90c))
|
|
||||||
* openapi返回证书时挑选匹配范围最小的那一个;增加format参数,增加返回值p7b格式,增加detail返回 ([2085bcc](https://github.com/certd/certd/commit/2085bcceb61c3723c9bdfec4c4cc0917631ff5e5))
|
|
||||||
* ssh 配置sudo免密提示 ([e1e7011](https://github.com/certd/certd/commit/e1e7011853ad0c5bd7b09c3690861d5aa34b2db4))
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复新部署的无法保存公共eab配置的bug ([d5dee75](https://github.com/certd/certd/commit/d5dee75df3bd635a597436e448b2de1407531f3a))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 阿里云 FC3.0 不在要求证书加密方式为旧版, 修复支持的协议类型可以正常选择 ([a34db74](https://github.com/certd/certd/commit/a34db7449eff6ad1dda01de673bf85579fa3865a))
|
|
||||||
* 部署到腾讯云cdn,每个域名增加3每秒延迟 ([f7d43ad](https://github.com/certd/certd/commit/f7d43ad5af4663d4be369820a80d1fd9817ca4ab))
|
|
||||||
* 腾讯云关闭证书通知增加开关选项,在腾讯云授权里面 ([a77c777](https://github.com/certd/certd/commit/a77c777980dd38d97d983124eeed1596879bba95))
|
|
||||||
* 证书申请任务默认不发送申请成功通知 ([0283bd2](https://github.com/certd/certd/commit/0283bd2f978dbcd13d361129135e439dd9fbc180))
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复授权配置复制功能,无法复制已加密字段的问题 ([221e068](https://github.com/certd/certd/commit/221e068bac3af6cd5d1794f8cd4c2ec5c0bc3f45))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 百度云支持上传到证书托管,支持部署到负载均衡 ([798a48a](https://github.com/certd/certd/commit/798a48aa9686fd5d11cfffb6cd93eadfc40aacb3))
|
|
||||||
* 部署到百度cdn支持自动获取域名列表选择 ([4e432ed](https://github.com/certd/certd/commit/4e432ed03f4fb564e85a2f284ee26b58400b82f5))
|
|
||||||
* 验证码可重试次数设置为3次 ([1bdceee](https://github.com/certd/certd/commit/1bdceeecf4b5daecdd621a05a2596b6eb45ce8ea))
|
|
||||||
* 增加找回密码的验证码可重试次数 [@nicheng-he](https://github.com/nicheng-he) ([#496](https://github.com/certd/certd/issues/496)) ([fe03f99](https://github.com/certd/certd/commit/fe03f9942b5662fb90cad86da10782f5dc3603f5))
|
|
||||||
* 支持阿里云API网关 ([9e1e4ee](https://github.com/certd/certd/commit/9e1e4eeec2859759ca5b07834c9d24cf88a6ad33))
|
|
||||||
* 支持部署到金山云CDN ([dfa74a6](https://github.com/certd/certd/commit/dfa74a69f7cbb9009d3e20c7eecfa1b905a00cf0))
|
|
||||||
* 支持更新金山云cdn证书 ([462e22a](https://github.com/certd/certd/commit/462e22a3b0a94887462fe6aa68e4671a365e0737))
|
|
||||||
* 支持apisix证书部署 ([9b63fb4](https://github.com/certd/certd/commit/9b63fb4ee2c6b56139160c5bf63482dab0869c2b))
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复 https://cas.undefined.aliyuncs.com 的bug ([60e6aa9](https://github.com/certd/certd/commit/60e6aa9b54a761a47e39acee4a1ff947a745be27))
|
|
||||||
* 修复阿里云clb api接口没有使用region的问题 ([0770f17](https://github.com/certd/certd/commit/0770f174a14313e28d08113e69829ef6cc02d719))
|
|
||||||
* 修复站点监控使用自定义dns解析域名报错的bug ([eb8cd53](https://github.com/certd/certd/commit/eb8cd53de27991321e36dd14e5ce95f42b51351f))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到阿里云支持选择bucket和域名 ([013b9c4](https://github.com/certd/certd/commit/013b9c4c7c2adf485d086123ccea448719577fd4))
|
|
||||||
* 清理数据库备份的临时目录 ([fd95549](https://github.com/certd/certd/commit/fd95549de9a5d8cec09772ee2630bb7521e15e1f))
|
|
||||||
* 添加免费通知,OneBot V11协议通知支持 ([#491](https://github.com/certd/certd/issues/491)) [@ayakasuki](https://github.com/ayakasuki) ([be053d4](https://github.com/certd/certd/commit/be053d47e41084f817882400882b64143d036d1a))
|
|
||||||
* 支持webhook部署证书 ([cbe0b1c](https://github.com/certd/certd/commit/cbe0b1c5a6538f232e9a63f1693d20d5acf0a306))
|
|
||||||
* 注册时支持填写用户名 ([fdcfcc7](https://github.com/certd/certd/commit/fdcfcc77a0db87954e0b026635d3ccdd9bc6cee8))
|
|
||||||
* add start:server npm script for quick server launch from root directory ([#484](https://github.com/certd/certd/issues/484)) [@orzyyyy](https://github.com/orzyyyy) ([fae1981](https://github.com/certd/certd/commit/fae1981161080f698c3f1263b712306d63baae64))
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复复制流水线为空的bug ([b070773](https://github.com/certd/certd/commit/b0707739fdfbae3d78db4efd3f180db05c4e4164))
|
|
||||||
* 修复商用证书上传第二次运行无法使用pfx格式证书的bug ([251dd1f](https://github.com/certd/certd/commit/251dd1fe457a7b152f43eb6de18f7beb9f0b194e))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 1panel支持 currenNode ([acc8907](https://github.com/certd/certd/commit/acc890730f43d492c9b1bd3668814cf10efdf7b8))
|
|
||||||
* 授权管理支持模糊查询 ([866eb62](https://github.com/certd/certd/commit/866eb6241baa7b21f6eddc649966324c188236c6))
|
|
||||||
* 新增找回密码功能 [@nicheng-he](https://github.com/nicheng-he) ([81ac240](https://github.com/certd/certd/commit/81ac240ac84db0af2f56b6352e227ecb49f38377))
|
|
||||||
* 优化start脚本 ([238ad7c](https://github.com/certd/certd/commit/238ad7ce51f17e1098c624e7f61ee2d98de1e02d))
|
|
||||||
* 运行主机脚本插件支持选择运行策略 ([86b3df1](https://github.com/certd/certd/commit/86b3df194126476e1f58e0952a77e986f62eecce))
|
|
||||||
* cdnfly 支持 账号密码登陆授权 ([e87f6d5](https://github.com/certd/certd/commit/e87f6d56f524dbbb9e3243e382b348b6e49f0d2c))
|
|
||||||
* k8s ack、tke 支持重启ingress ([95715a0](https://github.com/certd/certd/commit/95715a007d931c64fa7dd953d94957398e00a443))
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复阿里云发送短信验证码失败的bug ([2e6d03f](https://github.com/certd/certd/commit/2e6d03ff001f521f57368e7a62b97ed7b122e8d0))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 阿里云部分插件优化 [@nicheng-he](https://github.com/nicheng-he) ([e3738f6](https://github.com/certd/certd/commit/e3738f6422270d75ec414c15a343248cc4cad6e1))
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 上传到阿里云cas,证书前缀无效的bug ([b382351](https://github.com/certd/certd/commit/b382351c7b91ec10e1f61d94bec5aad075207ec8))
|
|
||||||
* 修复自定义插件onlyAdmin报错的bug ([4e5e862](https://github.com/certd/certd/commit/4e5e862f5834ad180e4428959c272d444a6f78ab))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到k8s,tke,ack忽悠证书校验 ([ab84835](https://github.com/certd/certd/commit/ab848353621869464a2c9a45fdb5e28d998b8a58))
|
|
||||||
* 首页增加更新日志按钮 ([41ce848](https://github.com/certd/certd/commit/41ce8489dc2f03a705dfa3fbb357769defb56c60))
|
|
||||||
* 增加版本过低提示 ([d1ce360](https://github.com/certd/certd/commit/d1ce36038cab72b5dc1b320d0a708c261ffbdacb))
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|||||||
@@ -10,8 +10,7 @@
|
|||||||
* 登录宝塔面板,在菜单栏中点击 Docker,首次进入会提示安装Docker服务,点击立即安装,按提示完成安装
|
* 登录宝塔面板,在菜单栏中点击 Docker,首次进入会提示安装Docker服务,点击立即安装,按提示完成安装
|
||||||
|
|
||||||
### 2、部署certd
|
### 2、部署certd
|
||||||
以下两种方式任选一种:
|
以下两种方式人选一种:
|
||||||
|
|
||||||
#### 2.1 应用商店方式一键部署【推荐】
|
#### 2.1 应用商店方式一键部署【推荐】
|
||||||
|
|
||||||
* 在宝塔Docker应用商店中找到`certd`(要先点右上角更新应用)
|
* 在宝塔Docker应用商店中找到`certd`(要先点右上角更新应用)
|
||||||
@@ -82,6 +81,3 @@ admin/123456
|
|||||||
2. 确认宝塔防火墙是否放开对应端口
|
2. 确认宝塔防火墙是否放开对应端口
|
||||||
3. 尝试将Certd容器加入宝塔的`bridge`网络
|
3. 尝试将Certd容器加入宝塔的`bridge`网络
|
||||||

|

|
||||||
|
|
||||||
### 2. 动态IP无法加白名单问题
|
|
||||||
[Nginx代理解决方案](../../use/baota/white_list.md)
|
|
||||||
@@ -4,7 +4,7 @@
|
|||||||
## 一、源码安装
|
## 一、源码安装
|
||||||
|
|
||||||
### 环境要求
|
### 环境要求
|
||||||
- nodejs 22 及以上
|
- nodejs 20 及以上
|
||||||
### 源码启动
|
### 源码启动
|
||||||
```shell
|
```shell
|
||||||
# 克隆代码
|
# 克隆代码
|
||||||
@@ -58,18 +58,3 @@ kill -9 $(lsof -t -i:7001)
|
|||||||
## 四、备份恢复
|
## 四、备份恢复
|
||||||
|
|
||||||
将备份的`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
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
73
docs/guide/qa/index.md
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
# 常见报错解决
|
||||||
|
|
||||||
|
## 1. getaddrinfo ENOTFOUND错误
|
||||||
|
如果出现`getaddrinfo ENOTFOUND`/`getaddrinfo EAI_AGAIN`错误,可以尝试在`docker-compose.yaml`中设置dns
|
||||||
|
```yaml
|
||||||
|
version: '3.3' # 兼容旧版docker-compose
|
||||||
|
services:
|
||||||
|
certd:
|
||||||
|
#↓↓↓↓ ------------ # 如果出现getaddrinfo ENOTFOUND 或 EAI_AGAIN错误,可以尝试设置dns
|
||||||
|
dns:
|
||||||
|
- 223.5.5.5 # 阿里云公共dns
|
||||||
|
- 223.6.6.6
|
||||||
|
# # ↓↓↓↓ ------- # 如果你服务器在腾讯云,可以用这个替换上面阿里云的公共dns
|
||||||
|
# - 119.29.29.29 # 腾讯云公共dns
|
||||||
|
# - 182.254.116.116
|
||||||
|
# # ↓↓↓↓ ------- # 如果你服务器部署在国外,可以用这个替换上面阿里云的公共dns
|
||||||
|
# - 8.8.8.8 # 谷歌公共dns
|
||||||
|
# - 8.8.4.4
|
||||||
|
```
|
||||||
|
|
||||||
|
如果仍然有问题,按如下步骤检查是否能够ping通域名
|
||||||
|
```shell
|
||||||
|
docker exec -it certd /bin/sh
|
||||||
|
ping www.baidu.com
|
||||||
|
ping gg.px.certd.handfree.work
|
||||||
|
ping app.handfree.work
|
||||||
|
```
|
||||||
|
|
||||||
|
如果您是宝塔部署的
|
||||||
|
可以试试将容器网络加入brige网络,看是否解决问题
|
||||||
|

|
||||||
|
|
||||||
|
如果还是不行,请联系我们
|
||||||
|
|
||||||
|
|
||||||
|
## 2. 连接IPv6超时
|
||||||
|
docker-compose 需要放开IPv6网络的配置
|
||||||
|
```yaml
|
||||||
|
services:
|
||||||
|
certd:
|
||||||
|
networks:
|
||||||
|
- ip6net
|
||||||
|
# ↓↓↓↓ -------------------------------------------------------------- 启用ipv6网络,还需要把上面networks的注释放开
|
||||||
|
networks:
|
||||||
|
ip6net:
|
||||||
|
enable_ipv6: true
|
||||||
|
ipam:
|
||||||
|
config:
|
||||||
|
- subnet: 2001:db8::/64
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
## 3. SSL_CERT_NOT_MATCH_DOMAIN_ERROR
|
||||||
|
部署证书任务报类似 `SSL_CERT_NOT_MATCH_DOMAIN_ERROR`错误
|
||||||
|
这是由于当前流水线的证书域名与要部署的目标站点的域名不匹配导致的,在申请证书任务中,增加目标站点域名,重新运行流水线即可
|
||||||
|
|
||||||
|
|
||||||
|
## 4. 没有服务器配置文件,请检查是否开启了外网映射!
|
||||||
|
宝塔网站证书部署报错:`Error: 没有服务器配置文件,请检查是否开启了外网映射!`
|
||||||
|
解决方案:先手动在宝塔网站中设置一次证书
|
||||||
|
|
||||||
|
|
||||||
|
## 5. 如何查看容器日志
|
||||||
|
```shell
|
||||||
|
docker logs -f --tail 200 certd
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
# 常见问题
|
# 使用问题
|
||||||
|
|
||||||
|
|
||||||
## 1. 是否支持IP证书
|
## 1. 是否支持IP证书
|
||||||
@@ -12,22 +12,3 @@
|
|||||||
当证书到期前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" |
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 22 KiB |
|
Before Width: | Height: | Size: 30 KiB |
|
Before Width: | Height: | Size: 36 KiB |
|
Before Width: | Height: | Size: 15 KiB |
|
Before Width: | Height: | Size: 7.2 KiB |
|
Before Width: | Height: | Size: 27 KiB |
|
Before Width: | Height: | Size: 58 KiB |
|
Before Width: | Height: | Size: 23 KiB |
@@ -1,98 +0,0 @@
|
|||||||
# 宝塔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访问
|
|
||||||
|
Before Width: | Height: | Size: 49 KiB |
|
Before Width: | Height: | Size: 76 KiB |
@@ -1,10 +0,0 @@
|
|||||||
# 二级子域名托管
|
|
||||||
如果你的域名是免费的二级域名(比如:sub.handsfree.work),托管在CF或者阿里云上
|
|
||||||
在使用DNS方式校验时需要设置子域名托管
|
|
||||||
|
|
||||||
[阿里云子域名托管说明](https://help.aliyun.com/zh/dns/pubz-subdomain-management)
|
|
||||||
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||

|
|
||||||
@@ -2,24 +2,19 @@
|
|||||||
|
|
||||||
## 配置步骤
|
## 配置步骤
|
||||||
|
|
||||||
|
1. 创建应用,获取APPID
|
||||||
1. 注册支付宝商家账号
|
|
||||||
* 开通电脑网站支付产品(需营业执照): https://b.alipay.com/page/product-workspace/all-product
|
|
||||||
|
|
||||||
|
|
||||||
2. 开放平台,创建应用,获取APPID
|
|
||||||
* 登录支付宝开放平台,进入开发者中心,创建网页应用,获取应用的AppId(左上角复制)
|
* 登录支付宝开放平台,进入开发者中心,创建网页应用,获取应用的AppId(左上角复制)
|
||||||
* 开发者中心:https://open.alipay.com/develop/manage
|
* 开发者中心:https://open.alipay.com/develop/manage
|
||||||
|
|
||||||
|
|
||||||
3. 进入应用详情,选择开发设置,配置接口加签方式 (选择密钥类型)
|
2. 进入应用详情,选择开发设置,配置接口加签方式 (选择密钥类型)
|
||||||
|
|
||||||
* 参考文档:https://opendocs.alipay.com/common/02kdnc?pathHash=fb0c752a
|
* 参考文档:https://opendocs.alipay.com/common/02kdnc?pathHash=fb0c752a
|
||||||
* 此步骤完成后,可以获取应用的私钥、支付宝公钥。
|
* 此步骤完成后,可以获取应用的私钥、支付宝公钥。
|
||||||
* 注意:支付宝不会保存应用的私钥,你需要自己保管好私钥。
|
* 注意:支付宝不会保存应用的私钥,你需要自己保管好私钥。
|
||||||
|
|
||||||
|
|
||||||
4. 在Certd后台配置支付宝
|
3. 在Certd后台配置支付宝
|
||||||
|
|
||||||
* 进入“系统”->"设置"->“支付设置”
|
* 进入“系统”->"设置"->“支付设置”
|
||||||
* 启用支付宝,选择“支付宝配置”,点击添加
|
* 启用支付宝,选择“支付宝配置”,点击添加
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 52 KiB |
|
Before Width: | Height: | Size: 54 KiB |
@@ -8,11 +8,8 @@
|
|||||||

|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## 如何避免收到腾讯云证书过期邮件
|
## 如何避免收到腾讯云证书过期邮件
|
||||||
|
|
||||||
> 新版本已经自动将证书设置为免提醒,certd上传的证书后续都不会再提醒了。
|
|
||||||
|
|
||||||
腾讯云在证书有效期还剩28天时会发送过期通知邮件
|
腾讯云在证书有效期还剩28天时会发送过期通知邮件
|
||||||
您可以通过配置“腾讯云过期证书删除”任务来避免收到此类邮件。
|
您可以通过配置“腾讯云过期证书删除”任务来避免收到此类邮件。
|
||||||
|
|
||||||
@@ -22,16 +19,3 @@
|
|||||||
> 1. 选择腾讯云授权,需授权`服务角色SSL_QCSLinkedRoleInReplaceLoadCertificate`权限
|
> 1. 选择腾讯云授权,需授权`服务角色SSL_QCSLinkedRoleInReplaceLoadCertificate`权限
|
||||||
> 2. `1.26.14`版本之前Certd创建的证书流水线默认是到期前20天才更新证书,需要将之前创建的证书申请任务的更新天数修改为35天,保证删除之前就已经替换掉即将过期证书
|
> 2. `1.26.14`版本之前Certd创建的证书流水线默认是到期前20天才更新证书,需要将之前创建的证书申请任务的更新天数修改为35天,保证删除之前就已经替换掉即将过期证书
|
||||||

|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## TKE service 的 TCP_SSL Opaque类型证书授权
|
|
||||||
|
|
||||||
部署证书到腾讯云TKE,如果报以下错误:
|
|
||||||
`is forbidden: User "xxxxxx-xxxxx" cannot get resource "secrets" in API group "" in the namespace "default"'`
|
|
||||||
则需要单独从授权管理侧再授权子用户的权限
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||

|
|
||||||
|
|||||||
@@ -9,5 +9,5 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"npmClient": "pnpm",
|
"npmClient": "pnpm",
|
||||||
"version": "1.36.25"
|
"version": "1.36.11"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,6 @@
|
|||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "lerna bootstrap --hoist",
|
"start": "lerna bootstrap --hoist",
|
||||||
"start:server": "cd ./packages/ui/certd-server && npm start",
|
|
||||||
"devb": "lerna run dev-build",
|
"devb": "lerna run dev-build",
|
||||||
"i-all": "lerna link && lerna exec npm install ",
|
"i-all": "lerna link && lerna exec npm install ",
|
||||||
"publish": "npm run prepublishOnly2 && lerna publish --force-publish=pro/plus-core --conventional-commits --create-release github && npm run afterpublishOnly && npm run commitAll",
|
"publish": "npm run prepublishOnly2 && lerna publish --force-publish=pro/plus-core --conventional-commits --create-release github && npm run afterpublishOnly && npm run commitAll",
|
||||||
@@ -31,7 +30,7 @@
|
|||||||
"init": "lerna run build",
|
"init": "lerna run build",
|
||||||
"init:dev": "lerna run build",
|
"init:dev": "lerna run build",
|
||||||
"docs:dev": "vitepress dev docs",
|
"docs:dev": "vitepress dev docs",
|
||||||
"docs:build": "npm run copylogs && vitepress build docs",
|
"docs:build": "vitepress build docs",
|
||||||
"docs:preview": "vitepress preview docs",
|
"docs:preview": "vitepress preview docs",
|
||||||
"pub": "echo 1"
|
"pub": "echo 1"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ root = true
|
|||||||
|
|
||||||
[*]
|
[*]
|
||||||
indent_style = space
|
indent_style = space
|
||||||
indent_size = 2
|
indent_size = 4
|
||||||
trim_trailing_whitespace = true
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
[{*.yml,*.yaml}]
|
[{*.yml,*.yaml}]
|
||||||
|
|||||||
@@ -3,66 +3,6 @@
|
|||||||
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/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)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 支持ssl.com证书颁发机构 ([27b6dfa](https://github.com/publishlab/node-acme-client/commit/27b6dfa4d2ab3bddd284c3a34511a72e1a513a4c))
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/publishlab/node-acme-client/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/publishlab/node-acme-client/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/publishlab/node-acme-client/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到百度cdn支持自动获取域名列表选择 ([4e432ed](https://github.com/publishlab/node-acme-client/commit/4e432ed03f4fb564e85a2f284ee26b58400b82f5))
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/publishlab/node-acme-client/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/publishlab/node-acme-client/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/publishlab/node-acme-client/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/publishlab/node-acme-client/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/publishlab/node-acme-client/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/publishlab/node-acme-client/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/acme-client
|
**Note:** Version bump only for package @certd/acme-client
|
||||||
|
|||||||
@@ -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.25",
|
"version": "1.36.11",
|
||||||
"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.25",
|
"@certd/basic": "^1.36.11",
|
||||||
"@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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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挑战验证)');
|
log('Waiting for ACME challenge verification(等待ACME挑战验证)', this.backoffOpts);
|
||||||
return util.retry(verifyFn, this.backoffOpts);
|
return util.retry(verifyFn, this.backoffOpts);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -25,10 +25,6 @@ export const directory = {
|
|||||||
staging: 'https://acme.zerossl.com/v2/DV90',
|
staging: 'https://acme.zerossl.com/v2/DV90',
|
||||||
production: 'https://acme.zerossl.com/v2/DV90',
|
production: 'https://acme.zerossl.com/v2/DV90',
|
||||||
},
|
},
|
||||||
sslcom:{
|
|
||||||
staging: 'https://acme.ssl.com/sslcom-dv-rsa',
|
|
||||||
production: 'https://acme.ssl.com/sslcom-dv-rsa',
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -3,72 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复批量流水线执行时日志显示错乱的问题 ([4372adc](https://github.com/certd/certd/commit/4372adc703b9a4c785664054ab2a533626d815a8))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 去掉宝塔url后面的斜杠 ([8a0c2b9](https://github.com/certd/certd/commit/8a0c2b9b13628da750c25757e0cb8ed3038775ba))
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/basic
|
**Note:** Version bump only for package @certd/basic
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
09:38
|
12:23
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/basic",
|
"name": "@certd/basic",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.25",
|
"version": "1.36.11",
|
||||||
"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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,26 +17,10 @@ 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,
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,22 @@
|
|||||||
import log4js, { CallStack, Level } from "log4js";
|
import log4js, { LoggingEvent, Logger } from "log4js";
|
||||||
|
|
||||||
|
const OutputAppender = {
|
||||||
|
configure: (config: any, layouts: any, findAppender: any, levels: any) => {
|
||||||
|
let layout = layouts.basicLayout;
|
||||||
|
if (config.layout) {
|
||||||
|
layout = layouts.layout(config.layout.type, config.layout);
|
||||||
|
}
|
||||||
|
function customAppender(layout: any, timezoneOffset: any) {
|
||||||
|
return (loggingEvent: LoggingEvent) => {
|
||||||
|
if (loggingEvent.context.outputHandler?.write) {
|
||||||
|
const text = `${layout(loggingEvent, timezoneOffset)}\n`;
|
||||||
|
loggingEvent.context.outputHandler.write(text);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return customAppender(layout, config.timezoneOffset);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
let logFilePath = "./logs/app.log";
|
let logFilePath = "./logs/app.log";
|
||||||
export function resetLogConfigure() {
|
export function resetLogConfigure() {
|
||||||
@@ -6,6 +24,7 @@ export function resetLogConfigure() {
|
|||||||
log4js.configure({
|
log4js.configure({
|
||||||
appenders: {
|
appenders: {
|
||||||
std: { type: "stdout" },
|
std: { type: "stdout" },
|
||||||
|
output: { type: OutputAppender },
|
||||||
file: {
|
file: {
|
||||||
type: "dateFile",
|
type: "dateFile",
|
||||||
filename: logFilePath,
|
filename: logFilePath,
|
||||||
@@ -14,7 +33,7 @@ export function resetLogConfigure() {
|
|||||||
numBackups: 3,
|
numBackups: 3,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
categories: { default: { appenders: ["std", "file"], level: "info" }, pipeline: { appenders: ["std", "file"], level: "info" } },
|
categories: { default: { appenders: ["std", "file"], level: "info" }, pipeline: { appenders: ["std", "file", "output"], level: "info" } },
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
resetLogConfigure();
|
resetLogConfigure();
|
||||||
@@ -25,98 +44,15 @@ export function resetLogFilePath(filePath: string) {
|
|||||||
resetLogConfigure();
|
resetLogConfigure();
|
||||||
}
|
}
|
||||||
export function buildLogger(write: (text: string) => void) {
|
export function buildLogger(write: (text: string) => void) {
|
||||||
return new PipelineLogger("pipeline", write);
|
const logger = log4js.getLogger("pipeline");
|
||||||
}
|
const _secrets: string[] = [];
|
||||||
|
//@ts-ignore
|
||||||
export type ILogger = {
|
logger.addSecret = (secret: string) => {
|
||||||
readonly category: string;
|
_secrets.push(secret);
|
||||||
level: Level | string;
|
};
|
||||||
log(level: Level | string, ...args: any[]): void;
|
logger.addContext("outputHandler", {
|
||||||
|
write: (text: string) => {
|
||||||
isLevelEnabled(level?: string): boolean;
|
for (const item of _secrets) {
|
||||||
|
|
||||||
isTraceEnabled(): boolean;
|
|
||||||
isDebugEnabled(): boolean;
|
|
||||||
isInfoEnabled(): boolean;
|
|
||||||
isWarnEnabled(): boolean;
|
|
||||||
isErrorEnabled(): boolean;
|
|
||||||
isFatalEnabled(): boolean;
|
|
||||||
|
|
||||||
_log(level: Level, data: any): void;
|
|
||||||
|
|
||||||
addContext(key: string, value: any): void;
|
|
||||||
|
|
||||||
removeContext(key: string): void;
|
|
||||||
|
|
||||||
clearContext(): void;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Replace the basic parse function with a new custom one
|
|
||||||
* - Note that linesToSkip will be based on the origin of the Error object in addition to the callStackLinesToSkip (at least 1)
|
|
||||||
* @param parseFunction the new parseFunction. Use `undefined` to reset to the base implementation
|
|
||||||
*/
|
|
||||||
setParseCallStackFunction(parseFunction: (error: Error, linesToSkip: number) => CallStack | undefined): void;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Adjust the value of linesToSkip when the parseFunction is called.
|
|
||||||
*
|
|
||||||
* Cannot be less than 0.
|
|
||||||
*/
|
|
||||||
callStackLinesToSkip: number;
|
|
||||||
|
|
||||||
trace(message: any, ...args: any[]): void;
|
|
||||||
|
|
||||||
debug(message: any, ...args: any[]): void;
|
|
||||||
|
|
||||||
info(message: any, ...args: any[]): void;
|
|
||||||
|
|
||||||
warn(message: any, ...args: any[]): void;
|
|
||||||
|
|
||||||
error(message: any, ...args: any[]): void;
|
|
||||||
|
|
||||||
fatal(message: any, ...args: any[]): void;
|
|
||||||
|
|
||||||
mark(message: any, ...args: any[]): void;
|
|
||||||
};
|
|
||||||
|
|
||||||
const locale = Intl.DateTimeFormat().resolvedOptions().locale;
|
|
||||||
const formatter = new Intl.DateTimeFormat(locale, {
|
|
||||||
year: "numeric",
|
|
||||||
month: "2-digit",
|
|
||||||
day: "2-digit",
|
|
||||||
hour: "2-digit",
|
|
||||||
minute: "2-digit",
|
|
||||||
second: "2-digit",
|
|
||||||
hour12: false,
|
|
||||||
});
|
|
||||||
function formatDateIntl(date = new Date()) {
|
|
||||||
const milliseconds = date.getMilliseconds(); // 获取毫秒
|
|
||||||
const formattedMilliseconds = milliseconds.toString().padStart(3, "0");
|
|
||||||
return formatter.format(date) + "." + formattedMilliseconds;
|
|
||||||
}
|
|
||||||
|
|
||||||
// @ts-ignore
|
|
||||||
export class PipelineLogger implements ILogger {
|
|
||||||
callStackLinesToSkip: number = 3;
|
|
||||||
readonly category: string = "pipeline";
|
|
||||||
level: Level | string = "info";
|
|
||||||
_secrets: string[] = [];
|
|
||||||
logger: ILogger;
|
|
||||||
customWriter!: (text: string) => void;
|
|
||||||
|
|
||||||
constructor(name: string, write: (text: string) => void) {
|
|
||||||
this.customWriter = write;
|
|
||||||
this.logger = log4js.getLogger(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
addSecret(secret: string) {
|
|
||||||
this._secrets.push(secret);
|
|
||||||
}
|
|
||||||
|
|
||||||
_doLog(level: string, ...args: any[]) {
|
|
||||||
let text = args.join(" ");
|
|
||||||
if (this.customWriter) {
|
|
||||||
for (const item of this._secrets) {
|
|
||||||
if (item == null) {
|
if (item == null) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -130,88 +66,10 @@ export class PipelineLogger implements ILogger {
|
|||||||
text = text.replaceAll(item, "*".repeat(item.length));
|
text = text.replaceAll(item, "*".repeat(item.length));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
text = `[${formatDateIntl()}] [${level.toUpperCase()}] - ${text} \n`;
|
write(text);
|
||||||
this.customWriter(text);
|
},
|
||||||
}
|
});
|
||||||
// @ts-ignore
|
return logger;
|
||||||
this.logger[level](...args);
|
|
||||||
}
|
|
||||||
|
|
||||||
_log(level: Level, data: any): void {}
|
|
||||||
|
|
||||||
addContext(key: string, value: any): void {}
|
|
||||||
|
|
||||||
clearContext(): void {}
|
|
||||||
|
|
||||||
debug(message: any, ...args: any[]): void {
|
|
||||||
if (this.isDebugEnabled()) {
|
|
||||||
this._doLog("debug", message, ...args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
error(message: any, ...args: any[]): void {
|
|
||||||
if (this.isErrorEnabled()) {
|
|
||||||
this._doLog("error", message, ...args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fatal(message: any, ...args: any[]): void {
|
|
||||||
if (this.isFatalEnabled()) {
|
|
||||||
this._doLog("fatal", message, ...args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
info(message: any, ...args: any[]): void {
|
|
||||||
if (this.isInfoEnabled()) {
|
|
||||||
this._doLog("info", message, ...args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
trace(message: any, ...args: any[]): void {
|
|
||||||
if (this.isTraceEnabled()) {
|
|
||||||
this._doLog("trace", message, ...args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
warn(message: any, ...args: any[]): void {
|
|
||||||
if (this.isWarnEnabled()) {
|
|
||||||
this._doLog("warn", message, ...args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
isDebugEnabled(): boolean {
|
|
||||||
return logger.isDebugEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
isErrorEnabled(): boolean {
|
|
||||||
return logger.isErrorEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
isFatalEnabled(): boolean {
|
|
||||||
return logger.isFatalEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
isInfoEnabled(): boolean {
|
|
||||||
return logger.isInfoEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
isLevelEnabled(level?: string): boolean {
|
|
||||||
return logger.isLevelEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
isTraceEnabled(): boolean {
|
|
||||||
return logger.isTraceEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
isWarnEnabled(): boolean {
|
|
||||||
return logger.isWarnEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
log(level: Level | string, ...args: any[]): void {}
|
|
||||||
|
|
||||||
mark(message: any, ...args: any[]): void {}
|
|
||||||
|
|
||||||
removeContext(key: string): void {}
|
|
||||||
|
|
||||||
setParseCallStackFunction(parseFunction: (error: Error, linesToSkip: number) => CallStack | undefined): void {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type ILogger = Logger;
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import axios, { AxiosHeaders, AxiosRequestConfig } from "axios";
|
import axios, { AxiosHeaders, AxiosRequestConfig } from "axios";
|
||||||
import { ILogger, logger } from "./util.log.js";
|
import { ILogger, logger } from "./util.log.js";
|
||||||
|
import { Logger } from "log4js";
|
||||||
import { HttpProxyAgent } from "http-proxy-agent";
|
import { HttpProxyAgent } from "http-proxy-agent";
|
||||||
import { HttpsProxyAgent } from "https-proxy-agent";
|
import { HttpsProxyAgent } from "https-proxy-agent";
|
||||||
import nodeHttp from "http";
|
import nodeHttp from "http";
|
||||||
@@ -7,13 +8,6 @@ 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;
|
||||||
@@ -28,12 +22,11 @@ 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" && message.indexOf) {
|
if (message && typeof message === "string") {
|
||||||
for (const key in errorMap) {
|
if (message.indexOf && message.indexOf("ssl3_get_record:wrong version number") >= 0) {
|
||||||
if (message.indexOf(key) > -1) {
|
this.message = `${message}(http协议错误,服务端要求http协议,请检查是否使用了https请求)`;
|
||||||
this.message = `${this.message}(${errorMap[key]})`;
|
} else if (message.indexOf("getaddrinfo EAI_AGAIN") >= 0) {
|
||||||
break;
|
this.message = `${message}(无法解析域名,请检查网络连接或dns配置,更换docker-compose.yaml中dns配置)`;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -91,7 +84,7 @@ export function getGlobalAgents() {
|
|||||||
/**
|
/**
|
||||||
* @description 创建请求实例
|
* @description 创建请求实例
|
||||||
*/
|
*/
|
||||||
export function createAxiosService({ logger }: { logger: ILogger }) {
|
export function createAxiosService({ logger }: { logger: Logger }) {
|
||||||
// 创建一个 axios 实例
|
// 创建一个 axios 实例
|
||||||
const service = axios.create();
|
const service = axios.create();
|
||||||
|
|
||||||
@@ -207,31 +200,10 @@ 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,76 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 前置任务输出不存在时输出警告提示 ([b59052c](https://github.com/certd/certd/commit/b59052cc43b7b070fabd8b8e914e4c2a5e0ad61c))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 支持godaddy ([b7980aa](https://github.com/certd/certd/commit/b7980aad5ab50f58662eaddf5d84aa82876a98eb))
|
|
||||||
* 支持ssl.com证书颁发机构 ([27b6dfa](https://github.com/certd/certd/commit/27b6dfa4d2ab3bddd284c3a34511a72e1a513a4c))
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/pipeline
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/pipeline
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 百度云支持上传到证书托管,支持部署到负载均衡 ([798a48a](https://github.com/certd/certd/commit/798a48aa9686fd5d11cfffb6cd93eadfc40aacb3))
|
|
||||||
* 支持部署到金山云CDN ([dfa74a6](https://github.com/certd/certd/commit/dfa74a69f7cbb9009d3e20c7eecfa1b905a00cf0))
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/pipeline
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/pipeline
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/pipeline
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复自定义插件onlyAdmin报错的bug ([4e5e862](https://github.com/certd/certd/commit/4e5e862f5834ad180e4428959c272d444a6f78ab))
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/pipeline",
|
"name": "@certd/pipeline",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.25",
|
"version": "1.36.11",
|
||||||
"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.25",
|
"@certd/basic": "^1.36.11",
|
||||||
"@certd/plus-core": "^1.36.25",
|
"@certd/plus-core": "^1.36.11",
|
||||||
"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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ export type PageSearch = {
|
|||||||
// sortOrder?: "asc" | "desc";
|
// sortOrder?: "asc" | "desc";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
export type PageRes = {
|
export type PageRes = {
|
||||||
pageNo?: number;
|
pageNo?: number;
|
||||||
pageSize?: number;
|
pageSize?: number;
|
||||||
@@ -21,9 +22,9 @@ export type PageRes = {
|
|||||||
export class Pager {
|
export class Pager {
|
||||||
pageNo: number;
|
pageNo: number;
|
||||||
pageSize: number;
|
pageSize: number;
|
||||||
constructor(req?: PageSearch) {
|
constructor(req: PageSearch) {
|
||||||
this.pageNo = req?.pageNo ?? 1;
|
this.pageNo = req.pageNo ?? 1;
|
||||||
this.pageSize = req?.pageSize || 50;
|
this.pageSize = req.pageSize || 50;
|
||||||
}
|
}
|
||||||
|
|
||||||
getOffset() {
|
getOffset() {
|
||||||
|
|||||||
@@ -314,7 +314,7 @@ export class Executor {
|
|||||||
const outputKey = arr[2];
|
const outputKey = arr[2];
|
||||||
input[key] = this.currentStatusMap.get(id)?.status?.output[outputKey] ?? this.lastStatusMap.get(id)?.status?.output[outputKey];
|
input[key] = this.currentStatusMap.get(id)?.status?.output[outputKey] ?? this.lastStatusMap.get(id)?.status?.output[outputKey];
|
||||||
if (input[key] == null) {
|
if (input[key] == null) {
|
||||||
currentLogger.warn(`${item.title}的配置未找到对应的输出值,请确认对应的前置任务是否存在或者是否执行正确`);
|
this.logger.warn(`${item.title}的配置未找到对应的输出值,请确认对应的前置任务是否存在或者是否执行正确`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -120,9 +120,10 @@ 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(stack, cause);
|
this._loggers[runnable.id].error(`[${runnable.runnableType}] [${runnable.title}]<id:${runnable.id}> :`, e, 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) {
|
||||||
|
|||||||
@@ -165,7 +165,7 @@ export abstract class AbstractTaskPlugin implements ITaskPlugin {
|
|||||||
this.registerSecret(cert.one);
|
this.registerSecret(cert.one);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.ctx?.define?.onlyAdmin) {
|
if (this.ctx.define.onlyAdmin) {
|
||||||
if (!this.isAdmin()) {
|
if (!this.isAdmin()) {
|
||||||
throw new Error("只有管理员才能运行此任务");
|
throw new Error("只有管理员才能运行此任务");
|
||||||
}
|
}
|
||||||
@@ -253,9 +253,9 @@ export abstract class AbstractTaskPlugin implements ITaskPlugin {
|
|||||||
return name + "_" + dayjs().format("YYYYMMDDHHmmssSSS");
|
return name + "_" + dayjs().format("YYYYMMDDHHmmssSSS");
|
||||||
}
|
}
|
||||||
|
|
||||||
buildCertName(domain: string, prefix = "") {
|
buildCertName(domain: string) {
|
||||||
domain = domain.replaceAll("*", "_").replaceAll(".", "_");
|
domain = domain.replaceAll("*", "_").replaceAll(".", "_");
|
||||||
return `${prefix}_${domain}_${dayjs().format("YYYYMMDDHHmmssSSS")}`;
|
return `${domain}_${dayjs().format("YYYYMMDDHHmmssSSS")}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
async onRequest(req: PluginRequestHandleReq<any>) {
|
async onRequest(req: PluginRequestHandleReq<any>) {
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ export const pluginGroups = {
|
|||||||
tencent: new PluginGroup("tencent", "腾讯云", 4, "svg:icon-tencentcloud"),
|
tencent: new PluginGroup("tencent", "腾讯云", 4, "svg:icon-tencentcloud"),
|
||||||
volcengine: new PluginGroup("volcengine", "火山引擎", 4, "svg:icon-volcengine"),
|
volcengine: new PluginGroup("volcengine", "火山引擎", 4, "svg:icon-volcengine"),
|
||||||
jdcloud: new PluginGroup("jdcloud", "京东云", 4, "svg:icon-jdcloud"),
|
jdcloud: new PluginGroup("jdcloud", "京东云", 4, "svg:icon-jdcloud"),
|
||||||
baidu: new PluginGroup("baidu", "百度云", 4, "ant-design:baidu-outlined"),
|
|
||||||
qiniu: new PluginGroup("qiniu", "七牛云", 5, "svg:icon-qiniuyun"),
|
qiniu: new PluginGroup("qiniu", "七牛云", 5, "svg:icon-qiniuyun"),
|
||||||
aws: new PluginGroup("aws", "亚马逊云", 6, "svg:icon-aws"),
|
aws: new PluginGroup("aws", "亚马逊云", 6, "svg:icon-aws"),
|
||||||
other: new PluginGroup("other", "其他", 10, "clarity:plugin-line"),
|
other: new PluginGroup("other", "其他", 10, "clarity:plugin-line"),
|
||||||
|
|||||||
@@ -69,15 +69,9 @@ export class Registry<T = any> {
|
|||||||
return this.storage;
|
return this.storage;
|
||||||
}
|
}
|
||||||
|
|
||||||
getDefineList(prefix?: string) {
|
getDefineList() {
|
||||||
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) {
|
||||||
@@ -96,10 +90,7 @@ export class Registry<T = any> {
|
|||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
|
||||||
getDefine(key: string, prefix?: string) {
|
getDefine(key: string) {
|
||||||
if (prefix) {
|
|
||||||
key = prefix + ":" + key;
|
|
||||||
}
|
|
||||||
const item = this.storage[key];
|
const item = this.storage[key];
|
||||||
if (!item) {
|
if (!item) {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -17,7 +17,6 @@ export type CnameRecord = {
|
|||||||
cnameProvider: CnameProvider;
|
cnameProvider: CnameProvider;
|
||||||
status: string;
|
status: string;
|
||||||
commonDnsProvider?: any;
|
commonDnsProvider?: any;
|
||||||
mainDomain?: string;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export type ICnameProxyService = {
|
export type ICnameProxyService = {
|
||||||
|
|||||||
@@ -3,62 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-huawei
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**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.25",
|
"version": "1.36.11",
|
||||||
"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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,62 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-iframe
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**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.25",
|
"version": "1.36.11",
|
||||||
"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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,62 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/jdcloud
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**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.25",
|
"version": "1.36.11",
|
||||||
"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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,76 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复远程数据选择无法过滤的bug ([6cbb073](https://github.com/certd/certd/commit/6cbb0739f8428d51b0712f718fe4d236cc087cf9))
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 创建k8s secret 时设置type为tls ([79ebabf](https://github.com/certd/certd/commit/79ebabfcfb9e5a534049c84f5f1a642b357fc856))
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到k8s支持自动创建secret ([c09c962](https://github.com/certd/certd/commit/c09c962cb676ca261610aa9f3e5105c9dae43f43))
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-k8s
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-k8s
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-k8s
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* k8s ack、tke 支持重启ingress ([95715a0](https://github.com/certd/certd/commit/95715a007d931c64fa7dd953d94957398e00a443))
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-k8s
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 部署到k8s,tke,ack忽悠证书校验 ([ab84835](https://github.com/certd/certd/commit/ab848353621869464a2c9a45fdb5e28d998b8a58))
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-k8s
|
**Note:** Version bump only for package @certd/lib-k8s
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/lib-k8s",
|
"name": "@certd/lib-k8s",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.25",
|
"version": "1.36.11",
|
||||||
"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.25",
|
"@certd/basic": "^1.36.11",
|
||||||
"@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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { CoreV1Api, KubeConfig, NetworkingV1Api, V1Ingress, V1Secret } from "@kubernetes/client-node";
|
import { CoreV1Api, KubeConfig, NetworkingV1Api, V1Ingress, V1Secret } from "@kubernetes/client-node";
|
||||||
import dns from "dns";
|
import dns from "dns";
|
||||||
import { ILogger } from "@certd/basic";
|
import { ILogger } from "@certd/basic";
|
||||||
import { merge } from "lodash-es";
|
import _ from "lodash-es";
|
||||||
|
|
||||||
export type K8sClientOpts = {
|
export type K8sClientOpts = {
|
||||||
kubeConfigStr: string;
|
kubeConfigStr: string;
|
||||||
@@ -85,11 +85,12 @@ 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";
|
||||||
const created = await this.client.createNamespacedSecret(namespace, opts.body);
|
const created = await this.client.createNamespacedSecret(namespace, opts.body);
|
||||||
this.logger.info("new secrets:", opts.body.metadata);
|
this.logger.info("new secrets:", opts.body);
|
||||||
return created.body;
|
return created.body;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -102,39 +103,17 @@ export class K8sClient {
|
|||||||
// return await this.client.replaceNamespacedSecret(secretName, namespace, opts.body);
|
// return await this.client.replaceNamespacedSecret(secretName, namespace, opts.body);
|
||||||
// }
|
// }
|
||||||
|
|
||||||
async patchSecret(opts: { namespace: string; secretName: string; body: V1Secret; createOnNotFound?: boolean }) {
|
async patchSecret(opts: { namespace: string; secretName: string; body: V1Secret }) {
|
||||||
const namespace = opts.namespace || "default";
|
const namespace = opts.namespace || "default";
|
||||||
const secretName = opts.secretName;
|
const secretName = opts.secretName;
|
||||||
if (secretName == null) {
|
if (secretName == null) {
|
||||||
throw new Error("secretName 不能为空");
|
throw new Error("secretName 不能为空");
|
||||||
}
|
}
|
||||||
this.logger.info("patch secret:", secretName, namespace);
|
this.logger.info("patch secret:", secretName, namespace);
|
||||||
let oldSecret: any = null;
|
const oldSecret = await this.client.readNamespacedSecret(secretName, namespace);
|
||||||
try {
|
const newSecret = _.merge(oldSecret.body, opts.body);
|
||||||
oldSecret = await this.client.readNamespacedSecret(secretName, namespace);
|
|
||||||
} catch (e) {
|
|
||||||
//@ts-ignore
|
|
||||||
if (e.response?.body?.code === 404) {
|
|
||||||
this.logger.warn(`secret ${secretName} 不存在`);
|
|
||||||
if (opts.createOnNotFound) {
|
|
||||||
//没有找到,则创建
|
|
||||||
const body = merge(
|
|
||||||
{
|
|
||||||
type: "kubernetes.io/tls",
|
|
||||||
},
|
|
||||||
opts.body
|
|
||||||
);
|
|
||||||
const res = await this.createSecret({ namespace, body });
|
|
||||||
this.logger.info(`secret ${secretName} 已创建`);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
|
|
||||||
const newSecret = merge(oldSecret.body, opts.body);
|
|
||||||
const res = await this.client.replaceNamespacedSecret(secretName, namespace, newSecret);
|
const res = await this.client.replaceNamespacedSecret(secretName, namespace, newSecret);
|
||||||
this.logger.info(`secret ${secretName} 已更新`);
|
this.logger.info("secret updated");
|
||||||
return res.body;
|
return res.body;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -166,23 +145,9 @@ export class K8sClient {
|
|||||||
this.logger.info("patch ingress:", ingressName, namespace);
|
this.logger.info("patch ingress:", ingressName, namespace);
|
||||||
const client = this.kubeconfig.makeApiClient(NetworkingV1Api);
|
const client = this.kubeconfig.makeApiClient(NetworkingV1Api);
|
||||||
const oldIngress = await client.readNamespacedIngress(ingressName, namespace);
|
const oldIngress = await client.readNamespacedIngress(ingressName, namespace);
|
||||||
const newIngress = merge(oldIngress.body, opts.body);
|
const newIngress = _.merge(oldIngress.body, opts.body);
|
||||||
const res = await client.replaceNamespacedIngress(ingressName, namespace, newIngress);
|
const res = await client.replaceNamespacedIngress(ingressName, namespace, newIngress);
|
||||||
this.logger.info("ingress patched", opts.body);
|
this.logger.info("ingress patched", opts.body);
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
async restartIngress(namespace: string, ingressNames: string[], labels: any) {
|
|
||||||
const body = {
|
|
||||||
metadata: {
|
|
||||||
labels: {
|
|
||||||
...labels,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
for (const ingress of ingressNames) {
|
|
||||||
await this.patchIngress({ namespace, ingressName: ingress, body });
|
|
||||||
this.logger.info(`ingress已重启:${ingress}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,75 +3,6 @@
|
|||||||
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))
|
|
||||||
|
|
||||||
## [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)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-server
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-server
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-server
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复授权配置复制功能,无法复制已加密字段的问题 ([221e068](https://github.com/certd/certd/commit/221e068bac3af6cd5d1794f8cd4c2ec5c0bc3f45))
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-server
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 新增找回密码功能 [@nicheng-he](https://github.com/nicheng-he) ([81ac240](https://github.com/certd/certd/commit/81ac240ac84db0af2f56b6352e227ecb49f38377))
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-server
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/lib-server
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**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.25",
|
"version": "1.36.11",
|
||||||
"description": "midway with flyway, sql upgrade way ",
|
"description": "midway with flyway, sql upgrade way ",
|
||||||
"private": false,
|
"private": false,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
@@ -27,20 +27,18 @@
|
|||||||
],
|
],
|
||||||
"license": "AGPL",
|
"license": "AGPL",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@certd/acme-client": "^1.36.25",
|
"@certd/acme-client": "^1.36.11",
|
||||||
"@certd/basic": "^1.36.25",
|
"@certd/basic": "^1.36.11",
|
||||||
"@certd/pipeline": "^1.36.25",
|
"@certd/pipeline": "^1.36.11",
|
||||||
"@certd/plugin-lib": "^1.36.25",
|
"@certd/plus-core": "^1.36.11",
|
||||||
"@certd/plus-core": "^1.36.25",
|
"@midwayjs/cache": "~3.14.0",
|
||||||
"@midwayjs/cache": "3.14.0",
|
"@midwayjs/core": "~3.20.3",
|
||||||
"@midwayjs/core": "3.20.11",
|
"@midwayjs/i18n": "~3.20.3",
|
||||||
"@midwayjs/i18n": "3.20.13",
|
"@midwayjs/info": "~3.20.3",
|
||||||
"@midwayjs/info": "3.20.13",
|
"@midwayjs/koa": "~3.20.3",
|
||||||
"@midwayjs/koa": "3.20.13",
|
"@midwayjs/logger": "~3.4.2",
|
||||||
"@midwayjs/logger": "3.4.2",
|
"@midwayjs/typeorm": "~3.20.3",
|
||||||
"@midwayjs/typeorm": "3.20.11",
|
"@midwayjs/upload": "^3.20.3",
|
||||||
"@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",
|
||||||
@@ -63,5 +61,5 @@
|
|||||||
"typeorm": "^0.3.11",
|
"typeorm": "^0.3.11",
|
||||||
"typescript": "^5.4.2"
|
"typescript": "^5.4.2"
|
||||||
},
|
},
|
||||||
"gitHead": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,5 +8,3 @@ 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,9 +1,8 @@
|
|||||||
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,AddonEntity];
|
export const libServerEntities = [SysSettingsEntity, AccessEntity];
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
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';
|
|
||||||
|
|||||||
@@ -1,24 +0,0 @@
|
|||||||
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("plusService")
|
@Provide()
|
||||||
@Scope(ScopeEnum.Request, { allowDowngrade: true })
|
@Scope(ScopeEnum.Request, { allowDowngrade: true })
|
||||||
export class PlusService {
|
export class PlusService {
|
||||||
@Inject()
|
@Inject()
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ export class SysPublicSettings extends BaseSettings {
|
|||||||
mobileRegisterEnabled = false;
|
mobileRegisterEnabled = false;
|
||||||
smsLoginEnabled = false;
|
smsLoginEnabled = false;
|
||||||
emailRegisterEnabled = false;
|
emailRegisterEnabled = false;
|
||||||
selfServicePasswordRetrievalEnabled = false;
|
|
||||||
|
|
||||||
limitUserPipelineCount = 0;
|
limitUserPipelineCount = 0;
|
||||||
managerOtherUserPipeline = false;
|
managerOtherUserPipeline = false;
|
||||||
@@ -30,13 +29,6 @@ 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 {
|
||||||
@@ -214,3 +206,4 @@ export class SysSafeSetting extends BaseSettings {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -34,18 +34,7 @@ export class AccessService extends BaseService<AccessEntity> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async add(param) {
|
async add(param) {
|
||||||
let oldEntity = null;
|
this.encryptSetting(param, null);
|
||||||
if (param._copyFrom){
|
|
||||||
oldEntity = await this.info(param._copyFrom);
|
|
||||||
if (oldEntity == null) {
|
|
||||||
throw new ValidateException('该授权配置不存在,请确认是否已被删除');
|
|
||||||
}
|
|
||||||
if (oldEntity.userId !== param.userId) {
|
|
||||||
throw new ValidateException('您无权查看该授权配置');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
delete param._copyFrom
|
|
||||||
this.encryptSetting(param, oldEntity);
|
|
||||||
return await super.add(param);
|
return await super.add(param);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,142 +0,0 @@
|
|||||||
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>;
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
// 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;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
export * from "./api.js";
|
|
||||||
export * from "./registry.js";
|
|
||||||
export * from "./decorator.js";
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
import { createRegistry } from "@certd/pipeline";
|
|
||||||
|
|
||||||
export const addonRegistry = createRegistry("addon");
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
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;
|
|
||||||
}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
export * from './api/index.js'
|
|
||||||
export * from './entity/addon.js'
|
|
||||||
export * from './service/addon-service.js'
|
|
||||||
export * from './service/addon-getter.js'
|
|
||||||
export * from './service/addon-sys-getter.js'
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
import { IAddonGetter } from "../api/index.js";
|
|
||||||
|
|
||||||
export class AddonGetter implements IAddonGetter {
|
|
||||||
userId: number;
|
|
||||||
getter: <T>(id: any, userId?: number) => Promise<T>;
|
|
||||||
constructor(userId: number, getter: (id: any, userId: number) => Promise<any>) {
|
|
||||||
this.userId = userId;
|
|
||||||
this.getter = getter;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getById<T = any>(id: any) {
|
|
||||||
return await this.getter<T>(id, this.userId);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getCommonById<T = any>(id: any) {
|
|
||||||
return await this.getter<T>(id, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,236 +0,0 @@
|
|||||||
import { Inject, Provide, Scope, ScopeEnum } from "@midwayjs/core";
|
|
||||||
import { InjectEntityModel } from "@midwayjs/typeorm";
|
|
||||||
import { In, Repository } from "typeorm";
|
|
||||||
import { AddonDefine, BaseService, PageReq, PermissionException, ValidateException } from "../../../index.js";
|
|
||||||
import { addonRegistry, newAddon } from "../api/index.js";
|
|
||||||
import { AddonEntity } from "../entity/addon.js";
|
|
||||||
import { http, logger, utils } from "@certd/basic";
|
|
||||||
import { TaskServiceBuilder } from "@certd/ui-server/dist/modules/pipeline/service/getter/task-service-getter.js";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Addon
|
|
||||||
*/
|
|
||||||
@Provide()
|
|
||||||
@Scope(ScopeEnum.Request, {allowDowngrade: true})
|
|
||||||
export class AddonService extends BaseService<AddonEntity> {
|
|
||||||
@InjectEntityModel(AddonEntity)
|
|
||||||
repository: Repository<AddonEntity>;
|
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private taskServiceBuilder: TaskServiceBuilder;
|
|
||||||
|
|
||||||
//@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,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
async getAddonById(id: any, checkUserId: boolean, userId?: number): Promise<any> {
|
|
||||||
const serviceGetter = this.taskServiceBuilder.create({userId:userId??0})
|
|
||||||
const ctx = {
|
|
||||||
http: http,
|
|
||||||
logger: logger,
|
|
||||||
utils: utils,
|
|
||||||
serviceGetter
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!id){
|
|
||||||
//使用图片验证码
|
|
||||||
return await newAddon("captcha", "image", {},ctx);
|
|
||||||
}
|
|
||||||
const entity = await this.info(id);
|
|
||||||
if (entity == null) {
|
|
||||||
//使用图片验证码
|
|
||||||
return await newAddon("captcha", "image", {},ctx);
|
|
||||||
}
|
|
||||||
if (checkUserId) {
|
|
||||||
if (userId == null) {
|
|
||||||
throw new ValidateException('userId不能为空');
|
|
||||||
}
|
|
||||||
if (userId !== entity.userId) {
|
|
||||||
throw new PermissionException('您对该Addon无访问权限');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const setting = JSON.parse(entity.setting ??"{}")
|
|
||||||
const input = {
|
|
||||||
id: entity.id,
|
|
||||||
...setting,
|
|
||||||
};
|
|
||||||
|
|
||||||
return await newAddon(entity.addonType, entity.type, input,ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getById(id: any, userId: number): Promise<any> {
|
|
||||||
return await this.getAddonById(id, true, userId);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
getDefineList(addonType: string) {
|
|
||||||
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,17 +0,0 @@
|
|||||||
import { IAccessService } from '@certd/pipeline';
|
|
||||||
import { AddonService } from './addon-service.js';
|
|
||||||
|
|
||||||
export class AddonSysGetter implements IAccessService {
|
|
||||||
addonService: AddonService;
|
|
||||||
constructor(addonService: AddonService) {
|
|
||||||
this.addonService = addonService;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getById<T = any>(id: any) {
|
|
||||||
return await this.addonService.getById(id, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getCommonById<T = any>(id: any) {
|
|
||||||
return await this.addonService.getById(id, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,2 +1 @@
|
|||||||
export * from './access/index.js';
|
export * from './access/index.js';
|
||||||
export * from './addon/index.js';
|
|
||||||
|
|||||||
@@ -3,62 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/midway-flyway-js
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**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.25",
|
"version": "1.36.11",
|
||||||
"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.11",
|
"@midwayjs/core": "~3.20.3",
|
||||||
"@midwayjs/logger": "3.4.2",
|
"@midwayjs/logger": "~3.4.2",
|
||||||
"@midwayjs/typeorm": "3.20.11",
|
"@midwayjs/typeorm": "~3.20.3",
|
||||||
"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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,92 +3,6 @@
|
|||||||
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)
|
|
||||||
|
|
||||||
**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)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 支持ssl.com证书颁发机构 ([27b6dfa](https://github.com/certd/certd/commit/27b6dfa4d2ab3bddd284c3a34511a72e1a513a4c))
|
|
||||||
* 子域名托管说明 ([39a0223](https://github.com/certd/certd/commit/39a02235cf4416bb5bd1acd3831241efeaa2f602))
|
|
||||||
|
|
||||||
## [1.36.18](https://github.com/certd/certd/compare/v1.36.17...v1.36.18) (2025-08-28)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 支持p7b证书格式 ([d9f4a57](https://github.com/certd/certd/commit/d9f4a5793d68a017a5d80ad5385cbda603c4e165))
|
|
||||||
* openapi返回证书时挑选匹配范围最小的那一个;增加format参数,增加返回值p7b格式,增加detail返回 ([2085bcc](https://github.com/certd/certd/commit/2085bcceb61c3723c9bdfec4c4cc0917631ff5e5))
|
|
||||||
|
|
||||||
## [1.36.17](https://github.com/certd/certd/compare/v1.36.16...v1.36.17) (2025-08-17)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 证书申请任务默认不发送申请成功通知 ([0283bd2](https://github.com/certd/certd/commit/0283bd2f978dbcd13d361129135e439dd9fbc180))
|
|
||||||
|
|
||||||
## [1.36.16](https://github.com/certd/certd/compare/v1.36.15...v1.36.16) (2025-08-16)
|
|
||||||
|
|
||||||
### Performance Improvements
|
|
||||||
|
|
||||||
* 百度云支持上传到证书托管,支持部署到负载均衡 ([798a48a](https://github.com/certd/certd/commit/798a48aa9686fd5d11cfffb6cd93eadfc40aacb3))
|
|
||||||
|
|
||||||
## [1.36.15](https://github.com/certd/certd/compare/v1.36.14...v1.36.15) (2025-08-07)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/plugin-cert
|
|
||||||
|
|
||||||
## [1.36.14](https://github.com/certd/certd/compare/v1.36.13...v1.36.14) (2025-07-28)
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* 修复商用证书上传第二次运行无法使用pfx格式证书的bug ([251dd1f](https://github.com/certd/certd/commit/251dd1fe457a7b152f43eb6de18f7beb9f0b194e))
|
|
||||||
|
|
||||||
## [1.36.13](https://github.com/certd/certd/compare/v1.36.12...v1.36.13) (2025-07-23)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/plugin-cert
|
|
||||||
|
|
||||||
## [1.36.12](https://github.com/certd/certd/compare/v1.36.11...v1.36.12) (2025-07-22)
|
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/plugin-cert
|
|
||||||
|
|
||||||
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
## [1.36.11](https://github.com/certd/certd/compare/v1.36.10...v1.36.11) (2025-07-22)
|
||||||
|
|
||||||
**Note:** Version bump only for package @certd/plugin-cert
|
**Note:** Version bump only for package @certd/plugin-cert
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@certd/plugin-cert",
|
"name": "@certd/plugin-cert",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "1.36.25",
|
"version": "1.36.11",
|
||||||
"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.25",
|
"@certd/acme-client": "^1.36.11",
|
||||||
"@certd/basic": "^1.36.25",
|
"@certd/basic": "^1.36.11",
|
||||||
"@certd/pipeline": "^1.36.25",
|
"@certd/pipeline": "^1.36.11",
|
||||||
"@certd/plugin-lib": "^1.36.25",
|
"@certd/plugin-lib": "^1.36.11",
|
||||||
"@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": "0c0c353ecc22c8729cfe55a26341f81b02de45fb"
|
"gitHead": "7f9c4e52ac5c3837b251d3b2508457ce802e11cb"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ export class EabAccess extends BaseAccess {
|
|||||||
component: {
|
component: {
|
||||||
placeholder: "kid / keyId",
|
placeholder: "kid / keyId",
|
||||||
},
|
},
|
||||||
helper: "EAB KID, google的叫 keyId,ssl.com的叫Account/ACME Key",
|
helper: "EAB KID, google的叫 keyId",
|
||||||
required: true,
|
required: true,
|
||||||
encrypt: true,
|
encrypt: true,
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { HttpClient, ILogger, utils } from "@certd/basic";
|
import { HttpClient, ILogger, utils } from "@certd/basic";
|
||||||
import { IAccess, IServiceGetter, Registrable } from "@certd/pipeline";
|
import { IAccess, Registrable } from "@certd/pipeline";
|
||||||
|
|
||||||
export type DnsProviderDefine = Registrable & {
|
export type DnsProviderDefine = Registrable & {
|
||||||
accessType: string;
|
accessType: string;
|
||||||
@@ -25,7 +25,6 @@ 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> {
|
||||||
|
|||||||
@@ -48,9 +48,8 @@ export type CertInfo = {
|
|||||||
der?: string;
|
der?: string;
|
||||||
jks?: string;
|
jks?: string;
|
||||||
one?: string;
|
one?: string;
|
||||||
p7b?: string;
|
|
||||||
};
|
};
|
||||||
export type SSLProvider = "letsencrypt" | "google" | "zerossl" | "sslcom";
|
export type SSLProvider = "letsencrypt" | "google" | "zerossl";
|
||||||
export type PrivateKeyType = "rsa_1024" | "rsa_2048" | "rsa_3072" | "rsa_4096" | "ec_256" | "ec_384" | "ec_521";
|
export type PrivateKeyType = "rsa_1024" | "rsa_2048" | "rsa_3072" | "rsa_4096" | "ec_256" | "ec_384" | "ec_521";
|
||||||
type AcmeServiceOptions = {
|
type AcmeServiceOptions = {
|
||||||
userContext: IContext;
|
userContext: IContext;
|
||||||
@@ -329,9 +328,8 @@ 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, preferredChain } = options;
|
const { email, isTest, csrInfo, dnsProvider, domainsVerifyPlan, profile } = 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;
|
||||||
@@ -374,7 +372,6 @@ export class AcmeService {
|
|||||||
commonName,
|
commonName,
|
||||||
...csrInfo,
|
...csrInfo,
|
||||||
altNames,
|
altNames,
|
||||||
// emailAddress: email,
|
|
||||||
},
|
},
|
||||||
privateKey
|
privateKey
|
||||||
);
|
);
|
||||||
@@ -405,7 +402,6 @@ export class AcmeService {
|
|||||||
},
|
},
|
||||||
signal: this.options.signal,
|
signal: this.options.signal,
|
||||||
profile,
|
profile,
|
||||||
preferredChain,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const crtString = crt.toString();
|
const crtString = crt.toString();
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ export abstract class CertApplyBaseConvertPlugin extends AbstractTaskPlugin {
|
|||||||
"2、子域名被通配符包含的不要填写,例如:www.foo.com已经被*.foo.com包含,不要填写www.foo.com\n" +
|
"2、子域名被通配符包含的不要填写,例如:www.foo.com已经被*.foo.com包含,不要填写www.foo.com\n" +
|
||||||
"3、泛域名只能通配*号那一级(*.foo.com的证书不能用于xxx.yyy.foo.com、不能用于foo.com)\n" +
|
"3、泛域名只能通配*号那一级(*.foo.com的证书不能用于xxx.yyy.foo.com、不能用于foo.com)\n" +
|
||||||
"4、输入一个,空格之后,再输入下一个 \n" +
|
"4、输入一个,空格之后,再输入下一个 \n" +
|
||||||
"5、如果设置了子域托管解析(比如免费的二级域名托管在CF或者阿里云),请先[设置托管子域名](#/certd/pipeline/subDomain)",
|
"5、如果您配置了子域托管解析,请先[设置托管子域名](#/certd/pipeline/subDomain)",
|
||||||
})
|
})
|
||||||
domains!: string[];
|
domains!: string[];
|
||||||
|
|
||||||
@@ -99,7 +99,6 @@ 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 = {};
|
||||||
@@ -126,10 +125,6 @@ export abstract class CertApplyBaseConvertPlugin extends AbstractTaskPlugin {
|
|||||||
cert.jks = res.jks;
|
cert.jks = res.jks;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cert.p7b == null && res.p7b) {
|
|
||||||
cert.p7b = res.p7b;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.logger.info("转换证书格式成功");
|
this.logger.info("转换证书格式成功");
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logger.error("转换证书格式失败", e);
|
this.logger.error("转换证书格式失败", e);
|
||||||
@@ -155,7 +150,6 @@ export abstract class CertApplyBaseConvertPlugin extends AbstractTaskPlugin {
|
|||||||
zip.file("intermediate.crt", cert.ic);
|
zip.file("intermediate.crt", cert.ic);
|
||||||
zip.file("origin.crt", cert.oc);
|
zip.file("origin.crt", cert.oc);
|
||||||
zip.file("one.pem", cert.one);
|
zip.file("one.pem", cert.one);
|
||||||
zip.file("cert.p7b", cert.p7b);
|
|
||||||
if (cert.pfx) {
|
if (cert.pfx) {
|
||||||
zip.file("cert.pfx", Buffer.from(cert.pfx, "base64"));
|
zip.file("cert.pfx", Buffer.from(cert.pfx, "base64"));
|
||||||
}
|
}
|
||||||
@@ -200,13 +194,4 @@ cert.jks:jks格式证书文件,java服务器使用
|
|||||||
};
|
};
|
||||||
return newCert;
|
return newCert;
|
||||||
}
|
}
|
||||||
|
|
||||||
async readLastCert(): Promise<CertReader | undefined> {
|
|
||||||
const cert = this.lastStatus?.status?.output?.cert;
|
|
||||||
if (cert == null) {
|
|
||||||
this.logger.info("没有找到上次的证书");
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
return new CertReader(cert);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ export abstract class CertApplyBasePlugin extends CertApplyBaseConvertPlugin {
|
|||||||
|
|
||||||
@TaskInput({
|
@TaskInput({
|
||||||
title: "证书申请成功通知",
|
title: "证书申请成功通知",
|
||||||
value: false,
|
value: true,
|
||||||
component: {
|
component: {
|
||||||
name: "a-switch",
|
name: "a-switch",
|
||||||
vModel: "checked",
|
vModel: "checked",
|
||||||
@@ -41,7 +41,7 @@ export abstract class CertApplyBasePlugin extends CertApplyBaseConvertPlugin {
|
|||||||
order: 100,
|
order: 100,
|
||||||
helper: "证书申请成功后是否发送通知,优先使用默认通知渠道",
|
helper: "证书申请成功后是否发送通知,优先使用默认通知渠道",
|
||||||
})
|
})
|
||||||
successNotify = false;
|
successNotify = true;
|
||||||
|
|
||||||
// @TaskInput({
|
// @TaskInput({
|
||||||
// title: "CsrInfo",
|
// title: "CsrInfo",
|
||||||
@@ -130,6 +130,15 @@ export abstract class CertApplyBasePlugin extends CertApplyBaseConvertPlugin {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async readLastCert(): Promise<CertReader | undefined> {
|
||||||
|
const cert = this.lastStatus?.status?.output?.cert;
|
||||||
|
if (cert == null) {
|
||||||
|
this.logger.info("没有找到上次的证书");
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
return new CertReader(cert);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 检查是否过期,默认提前35天
|
* 检查是否过期,默认提前35天
|
||||||
* @param expires
|
* @param expires
|
||||||
|
|||||||
@@ -17,25 +17,14 @@ export type CertReaderHandleContext = {
|
|||||||
tmpIcPath?: string;
|
tmpIcPath?: string;
|
||||||
tmpJksPath?: string;
|
tmpJksPath?: string;
|
||||||
tmpOnePath?: string;
|
tmpOnePath?: string;
|
||||||
tmpP7bPath?: string;
|
|
||||||
};
|
};
|
||||||
export type CertReaderHandle = (ctx: CertReaderHandleContext) => Promise<void>;
|
export type CertReaderHandle = (ctx: CertReaderHandleContext) => Promise<void>;
|
||||||
export type HandleOpts = { logger: ILogger; handle: CertReaderHandle };
|
export type HandleOpts = { logger: ILogger; handle: CertReaderHandle };
|
||||||
|
|
||||||
const formats = {
|
|
||||||
pem: ["crt", "key", "ic"],
|
|
||||||
one: ["one"],
|
|
||||||
pfx: ["pfx"],
|
|
||||||
der: ["der"],
|
|
||||||
jks: ["jks"],
|
|
||||||
p7b: ["p7b", "key"],
|
|
||||||
};
|
|
||||||
export class CertReader {
|
export class CertReader {
|
||||||
cert: CertInfo;
|
cert: CertInfo;
|
||||||
|
|
||||||
detail: CertificateInfo;
|
detail: CertificateInfo;
|
||||||
//毫秒时间戳
|
//毫秒时间戳
|
||||||
effective: number;
|
|
||||||
expires: number;
|
expires: number;
|
||||||
constructor(certInfo: CertInfo) {
|
constructor(certInfo: CertInfo) {
|
||||||
this.cert = certInfo;
|
this.cert = certInfo;
|
||||||
@@ -53,9 +42,8 @@ export class CertReader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { detail, effective, expires } = this.getCrtDetail(this.cert.crt);
|
const { detail, 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);
|
||||||
@@ -85,17 +73,8 @@ export class CertReader {
|
|||||||
return arr[0] + endStr;
|
return arr[0] + endStr;
|
||||||
}
|
}
|
||||||
|
|
||||||
toCertInfo(format?: string): CertInfo {
|
toCertInfo(): CertInfo {
|
||||||
if (!format) {
|
return this.cert;
|
||||||
return this.cert;
|
|
||||||
}
|
|
||||||
|
|
||||||
const formatArr = formats[format];
|
|
||||||
const res: any = {};
|
|
||||||
formatArr.forEach((key: string) => {
|
|
||||||
res[key] = this.cert[key];
|
|
||||||
});
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getCrtDetail(crt: string = this.cert.crt) {
|
getCrtDetail(crt: string = this.cert.crt) {
|
||||||
@@ -104,9 +83,8 @@ 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, effective, expires };
|
return { detail, expires };
|
||||||
}
|
}
|
||||||
|
|
||||||
getAllDomains() {
|
getAllDomains() {
|
||||||
@@ -146,7 +124,7 @@ export class CertReader {
|
|||||||
return domain;
|
return domain;
|
||||||
}
|
}
|
||||||
|
|
||||||
saveToFile(type: "crt" | "key" | "pfx" | "der" | "oc" | "one" | "ic" | "jks" | "p7b", filepath?: string) {
|
saveToFile(type: "crt" | "key" | "pfx" | "der" | "oc" | "one" | "ic" | "jks", filepath?: string) {
|
||||||
if (!this.cert[type]) {
|
if (!this.cert[type]) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -160,7 +138,7 @@ export class CertReader {
|
|||||||
if (!fs.existsSync(dir)) {
|
if (!fs.existsSync(dir)) {
|
||||||
fs.mkdirSync(dir, { recursive: true });
|
fs.mkdirSync(dir, { recursive: true });
|
||||||
}
|
}
|
||||||
if (type === "crt" || type === "key" || type === "ic" || type === "oc" || type === "one" || type === "p7b") {
|
if (type === "crt" || type === "key" || type === "ic" || type === "oc" || type === "one") {
|
||||||
fs.writeFileSync(filepath, this.cert[type]);
|
fs.writeFileSync(filepath, this.cert[type]);
|
||||||
} else {
|
} else {
|
||||||
fs.writeFileSync(filepath, Buffer.from(this.cert[type], "base64"));
|
fs.writeFileSync(filepath, Buffer.from(this.cert[type], "base64"));
|
||||||
@@ -179,19 +157,17 @@ export class CertReader {
|
|||||||
const tmpDerPath = this.saveToFile("der");
|
const tmpDerPath = this.saveToFile("der");
|
||||||
const tmpJksPath = this.saveToFile("jks");
|
const tmpJksPath = this.saveToFile("jks");
|
||||||
const tmpOnePath = this.saveToFile("one");
|
const tmpOnePath = this.saveToFile("one");
|
||||||
const tmpP7bPath = this.saveToFile("p7b");
|
|
||||||
logger.info("本地文件写入成功");
|
logger.info("本地文件写入成功");
|
||||||
try {
|
try {
|
||||||
return await opts.handle({
|
return await opts.handle({
|
||||||
reader: this,
|
reader: this,
|
||||||
tmpCrtPath,
|
tmpCrtPath: tmpCrtPath,
|
||||||
tmpKeyPath,
|
tmpKeyPath: tmpKeyPath,
|
||||||
tmpPfxPath,
|
tmpPfxPath: tmpPfxPath,
|
||||||
tmpDerPath,
|
tmpDerPath: tmpDerPath,
|
||||||
tmpIcPath,
|
tmpIcPath: tmpIcPath,
|
||||||
tmpJksPath,
|
tmpJksPath: tmpJksPath,
|
||||||
tmpOcPath,
|
tmpOcPath: tmpOcPath,
|
||||||
tmpP7bPath,
|
|
||||||
tmpOnePath,
|
tmpOnePath,
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -213,7 +189,6 @@ export class CertReader {
|
|||||||
removeFile(tmpIcPath);
|
removeFile(tmpIcPath);
|
||||||
removeFile(tmpJksPath);
|
removeFile(tmpJksPath);
|
||||||
removeFile(tmpOnePath);
|
removeFile(tmpOnePath);
|
||||||
removeFile(tmpP7bPath);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -224,10 +199,10 @@ export class CertReader {
|
|||||||
return `${prefix}_${domain}_${timeStr}.${suffix}`;
|
return `${prefix}_${domain}_${timeStr}.${suffix}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
buildCertName(prefix: string = "") {
|
buildCertName() {
|
||||||
let domain = this.getMainDomain();
|
let domain = this.getMainDomain();
|
||||||
domain = domain.replaceAll(".", "_").replaceAll("*", "_");
|
domain = domain.replaceAll(".", "_").replaceAll("*", "_");
|
||||||
return `${prefix}_${domain}_${dayjs().format("YYYYMMDDHHmmssSSS")}`;
|
return `${domain}_${dayjs().format("YYYYMMDDHHmmssSSS")}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static appendTimeSuffix(name?: string) {
|
static appendTimeSuffix(name?: string) {
|
||||||
@@ -236,8 +211,4 @@ export class CertReader {
|
|||||||
}
|
}
|
||||||
return name + "_" + dayjs().format("YYYYMMDDHHmmssSSS");
|
return name + "_" + dayjs().format("YYYYMMDDHHmmssSSS");
|
||||||
}
|
}
|
||||||
|
|
||||||
static buildCertName(cert: any) {
|
|
||||||
return new CertReader(cert).buildCertName();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,13 +18,11 @@ export class CertConverter {
|
|||||||
pfx: string;
|
pfx: string;
|
||||||
der: string;
|
der: string;
|
||||||
jks: string;
|
jks: string;
|
||||||
p7b: string;
|
|
||||||
}> {
|
}> {
|
||||||
const certReader = new CertReader(opts.cert);
|
const certReader = new CertReader(opts.cert);
|
||||||
let pfx: string;
|
let pfx: string;
|
||||||
let der: string;
|
let der: string;
|
||||||
let jks: string;
|
let jks: string;
|
||||||
let p7b: string;
|
|
||||||
const handle = async (ctx: CertReaderHandleContext) => {
|
const handle = async (ctx: CertReaderHandleContext) => {
|
||||||
// 调用openssl 转pfx
|
// 调用openssl 转pfx
|
||||||
pfx = await this.convertPfx(ctx, opts.pfxPassword, opts.pfxArgs);
|
pfx = await this.convertPfx(ctx, opts.pfxPassword, opts.pfxArgs);
|
||||||
@@ -33,8 +31,6 @@ export class CertConverter {
|
|||||||
der = await this.convertDer(ctx);
|
der = await this.convertDer(ctx);
|
||||||
|
|
||||||
jks = await this.convertJks(ctx, opts.pfxPassword);
|
jks = await this.convertJks(ctx, opts.pfxPassword);
|
||||||
|
|
||||||
p7b = await this.convertP7b(ctx);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
await certReader.readCertFile({ logger: this.logger, handle });
|
await certReader.readCertFile({ logger: this.logger, handle });
|
||||||
@@ -43,7 +39,6 @@ export class CertConverter {
|
|||||||
pfx,
|
pfx,
|
||||||
der,
|
der,
|
||||||
jks,
|
jks,
|
||||||
p7b,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,23 +95,6 @@ export class CertConverter {
|
|||||||
return derCert;
|
return derCert;
|
||||||
}
|
}
|
||||||
|
|
||||||
async convertP7b(opts: CertReaderHandleContext) {
|
|
||||||
const { tmpCrtPath } = opts;
|
|
||||||
const p7bPath = path.join(os.tmpdir(), "/certd/tmp/", Math.floor(Math.random() * 1000000) + `_cert.p7b`);
|
|
||||||
const dir = path.dirname(p7bPath);
|
|
||||||
if (!fs.existsSync(dir)) {
|
|
||||||
fs.mkdirSync(dir, { recursive: true });
|
|
||||||
}
|
|
||||||
//openssl crl2pkcs7 -nocrl \
|
|
||||||
// -certfile your_domain.crt \
|
|
||||||
// -certfile intermediate.crt \
|
|
||||||
// -out chain.p7b
|
|
||||||
await this.exec(`openssl crl2pkcs7 -nocrl -certfile ${tmpCrtPath} -out ${p7bPath}`);
|
|
||||||
const fileBuffer = fs.readFileSync(p7bPath);
|
|
||||||
const p7bCert = fileBuffer.toString();
|
|
||||||
fs.unlinkSync(p7bPath);
|
|
||||||
return p7bCert;
|
|
||||||
}
|
|
||||||
async convertJks(opts: CertReaderHandleContext, pfxPassword = "") {
|
async convertJks(opts: CertReaderHandleContext, pfxPassword = "") {
|
||||||
const jksPassword = pfxPassword || "123456";
|
const jksPassword = pfxPassword || "123456";
|
||||||
try {
|
try {
|
||||||
@@ -135,7 +113,9 @@ export class CertConverter {
|
|||||||
if (!fs.existsSync(dir)) {
|
if (!fs.existsSync(dir)) {
|
||||||
fs.mkdirSync(dir, { recursive: true });
|
fs.mkdirSync(dir, { recursive: true });
|
||||||
}
|
}
|
||||||
await this.exec(`keytool -importkeystore -srckeystore ${p12Path} -srcstoretype PKCS12 -srcstorepass "${jksPassword}" -destkeystore ${jksPath} -deststoretype PKCS12 -deststorepass "${jksPassword}" `);
|
await this.exec(
|
||||||
|
`keytool -importkeystore -srckeystore ${p12Path} -srcstoretype PKCS12 -srcstorepass "${jksPassword}" -destkeystore ${jksPath} -deststoretype PKCS12 -deststorepass "${jksPassword}" `
|
||||||
|
);
|
||||||
fs.unlinkSync(p12Path);
|
fs.unlinkSync(p12Path);
|
||||||
|
|
||||||
const fileBuffer = fs.readFileSync(jksPath);
|
const fileBuffer = fs.readFileSync(jksPath);
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ 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",
|
||||||
@@ -63,19 +62,6 @@ 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: {
|
||||||
@@ -111,65 +97,45 @@ 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() {
|
||||||
let certReader = null;
|
const certReader = new CertReader(this.uploadCert);
|
||||||
try {
|
if (!certReader.expires && certReader.expires < new Date().getTime()) {
|
||||||
this.logger.info("读取上次证书");
|
throw new Error("证书已过期,停止部署,请重新上传证书");
|
||||||
certReader = await this.readLastCert();
|
|
||||||
} catch (e) {
|
|
||||||
this.logger.warn("读取cert失败:", e);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return certReader;
|
return certReader;
|
||||||
}
|
}
|
||||||
|
|
||||||
private checkExpires(certReader: CertReader) {
|
|
||||||
const renewDays = (this.renewDays ?? 10) * 24 * 60 * 60 * 1000;
|
|
||||||
if (certReader.expires) {
|
|
||||||
if (certReader.expires < new Date().getTime()) {
|
|
||||||
throw new Error("证书已过期,停止部署,请尽快上传新证书");
|
|
||||||
}
|
|
||||||
if (certReader.expires < new Date().getTime() + renewDays) {
|
|
||||||
throw new Error("证书即将已过期,停止部署,请尽快上传新证书");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async execute(): Promise<string | void> {
|
async execute(): Promise<string | void> {
|
||||||
const oldCertReader = await this.getCertFromStore();
|
const certReader = await this.getCertFromStore();
|
||||||
if (oldCertReader) {
|
const crtMd5 = this.ctx.utils.hash.md5(certReader.cert.crt);
|
||||||
const leftDays = dayjs(oldCertReader.expires).diff(dayjs(), "day");
|
|
||||||
this.logger.info(`证书过期时间${dayjs(oldCertReader.expires).format("YYYY-MM-DD HH:mm:ss")},剩余${leftDays}天`);
|
const leftDays = dayjs(certReader.expires).diff(dayjs(), "day");
|
||||||
this.checkExpires(oldCertReader);
|
this.logger.info(`证书过期时间${dayjs(certReader.expires).format("YYYY-MM-DD HH:mm:ss")},剩余${leftDays}天`);
|
||||||
if (!this.ctx.inputChanged) {
|
|
||||||
this.logger.info("输入参数无变化");
|
if (!this.ctx.inputChanged) {
|
||||||
const lastCrtMd5 = this.lastStatus?.status?.output?.certMd5;
|
this.logger.info("输入参数无变化");
|
||||||
const newCrtMd5 = this.ctx.utils.hash.md5(this.uploadCert.crt);
|
const lastCrtMd5 = this.lastStatus?.status?.output?.certMd5;
|
||||||
this.logger.info("证书MD5", newCrtMd5);
|
this.logger.info("证书MD5", crtMd5);
|
||||||
this.logger.info("上次证书MD5", lastCrtMd5);
|
this.logger.info("上次证书MD5", lastCrtMd5);
|
||||||
if (lastCrtMd5 === newCrtMd5) {
|
if (lastCrtMd5 === crtMd5) {
|
||||||
this.logger.info("证书无变化,跳过");
|
this.logger.info("证书无变化,跳过");
|
||||||
//输出证书MD5
|
//输出证书MD5
|
||||||
this.certMd5 = newCrtMd5;
|
this.certMd5 = crtMd5;
|
||||||
await this.output(oldCertReader, false);
|
await this.output(certReader, false);
|
||||||
return "skip";
|
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 = this.ctx.utils.hash.md5(newCertReader.cert.crt);
|
this.certMd5 = crtMd5;
|
||||||
const newLeftDays = dayjs(newCertReader.expires).diff(dayjs(), "day");
|
await this.output(certReader, true);
|
||||||
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,53 +38,6 @@ 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版)",
|
||||||
@@ -136,10 +89,9 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
{ value: "letsencrypt", label: "Let's Encrypt", icon: "simple-icons:letsencrypt" },
|
{ value: "letsencrypt", label: "Let's Encrypt", icon: "simple-icons:letsencrypt" },
|
||||||
{ value: "google", label: "Google", icon: "flat-color-icons:google" },
|
{ value: "google", label: "Google", icon: "flat-color-icons:google" },
|
||||||
{ value: "zerossl", label: "ZeroSSL", icon: "emojione:digit-zero" },
|
{ value: "zerossl", label: "ZeroSSL", icon: "emojione:digit-zero" },
|
||||||
{ value: "sslcom", label: "SSL.com(仅主域名和www免费)", icon: "la:expeditedssl" },
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
helper: "Let's Encrypt:申请最简单\nGoogle:大厂光环,兼容性好,仅首次需要翻墙获取EAB授权\nZeroSSL:需要EAB授权,无需翻墙\nSSL.com:仅主域名和www免费,必须设置CAA记录",
|
helper: "Let's Encrypt:申请最简单\nGoogle:大厂光环,兼容性好,仅首次需要翻墙获取EAB授权\nZeroSSL:需要EAB授权,无需翻墙",
|
||||||
required: true,
|
required: true,
|
||||||
})
|
})
|
||||||
sslProvider!: SSLProvider;
|
sslProvider!: SSLProvider;
|
||||||
@@ -242,13 +194,6 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
})
|
})
|
||||||
zerosslCommonEabAccessId!: number;
|
zerosslCommonEabAccessId!: number;
|
||||||
|
|
||||||
@TaskInput({
|
|
||||||
title: "SSL.com公共EAB授权",
|
|
||||||
isSys: true,
|
|
||||||
show: false,
|
|
||||||
})
|
|
||||||
sslcomCommonEabAccessId!: number;
|
|
||||||
|
|
||||||
@TaskInput({
|
@TaskInput({
|
||||||
title: "EAB授权",
|
title: "EAB授权",
|
||||||
component: {
|
component: {
|
||||||
@@ -258,16 +203,11 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
maybeNeed: true,
|
maybeNeed: true,
|
||||||
required: false,
|
required: false,
|
||||||
helper:
|
helper:
|
||||||
"需要提供EAB授权" +
|
"需要提供EAB授权\nZeroSSL:请前往[zerossl开发者中心](https://app.zerossl.com/developer),生成 'EAB Credentials'\n Google:请查看[google获取eab帮助文档](https://certd.docmirror.cn/guide/use/google/),用过一次后会绑定邮箱,后续复用EAB要用同一个邮箱",
|
||||||
"\nZeroSSL:请前往[zerossl开发者中心](https://app.zerossl.com/developer),生成 'EAB Credentials'" +
|
|
||||||
"\nGoogle:请查看[google获取eab帮助文档](https://certd.docmirror.cn/guide/use/google/),用过一次后会绑定邮箱,后续复用EAB要用同一个邮箱" +
|
|
||||||
"\nSSL.com:[SSL.com账号页面](https://secure.ssl.com/account),然后点击api credentials链接,然后点击编辑按钮,查看Secret key和HMAC key",
|
|
||||||
mergeScript: `
|
mergeScript: `
|
||||||
return {
|
return {
|
||||||
show: ctx.compute(({form})=>{
|
show: ctx.compute(({form})=>{
|
||||||
return (form.sslProvider === 'zerossl' && !form.zerosslCommonEabAccessId)
|
return (form.sslProvider === 'zerossl' && !form.zerosslCommonEabAccessId) || (form.sslProvider === 'google' && !form.googleCommonEabAccessId)
|
||||||
|| (form.sslProvider === 'google' && !form.googleCommonEabAccessId)
|
|
||||||
|| (form.sslProvider === 'sslcom' && !form.sslcomCommonEabAccessId)
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
@@ -339,19 +279,6 @@ 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,
|
||||||
@@ -412,8 +339,8 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
async onInit() {
|
async onInit() {
|
||||||
let eab: EabAccess = null;
|
let eab: EabAccess = null;
|
||||||
|
|
||||||
if (this.sslProvider && this.sslProvider !== "letsencrypt") {
|
if (this.sslProvider === "google") {
|
||||||
if (this.sslProvider === "google" && this.googleAccessId) {
|
if (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);
|
||||||
const googleClient = new GoogleClient({
|
const googleClient = new GoogleClient({
|
||||||
@@ -421,19 +348,24 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
logger: this.logger,
|
logger: this.logger,
|
||||||
});
|
});
|
||||||
eab = await googleClient.getEab();
|
eab = await googleClient.getEab();
|
||||||
|
} else if (this.eabAccessId) {
|
||||||
|
this.logger.info("当前正在使用 google EAB授权");
|
||||||
|
eab = await this.getAccess(this.eabAccessId);
|
||||||
|
} else if (this.googleCommonEabAccessId) {
|
||||||
|
this.logger.info("当前正在使用 google 公共EAB授权");
|
||||||
|
eab = await this.getAccess(this.googleCommonEabAccessId, true);
|
||||||
} else {
|
} else {
|
||||||
const getEab = async (type: string) => {
|
throw new Error("google需要配置EAB授权或服务账号授权");
|
||||||
if (this.eabAccessId) {
|
}
|
||||||
this.logger.info(`当前正在使用 ${type} EAB授权`);
|
} else if (this.sslProvider === "zerossl") {
|
||||||
eab = await this.getAccess(this.eabAccessId);
|
if (this.eabAccessId) {
|
||||||
} else if (this[`${type}CommonEabAccessId`]) {
|
this.logger.info("当前正在使用 zerossl EAB授权");
|
||||||
this.logger.info(`当前正在使用 ${type} 公共EAB授权`);
|
eab = await this.getAccess(this.eabAccessId);
|
||||||
eab = await this.getAccess(this[`${type}CommonEabAccessId`], true);
|
} else if (this.zerosslCommonEabAccessId) {
|
||||||
} else {
|
this.logger.info("当前正在使用 zerossl 公共EAB授权");
|
||||||
throw new Error(`${type}需要配置EAB授权`);
|
eab = await this.getAccess(this.zerosslCommonEabAccessId, true);
|
||||||
}
|
} else {
|
||||||
};
|
throw new Error("zerossl需要配置EAB授权");
|
||||||
await getEab(this.sslProvider);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.eab = eab;
|
this.eab = eab;
|
||||||
@@ -465,12 +397,12 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
|||||||
|
|
||||||
const csrInfo = _.merge(
|
const csrInfo = _.merge(
|
||||||
{
|
{
|
||||||
// country: "CN",
|
country: "CN",
|
||||||
// state: "GuangDong",
|
state: "GuangDong",
|
||||||
// locality: "ShengZhen",
|
locality: "ShengZhen",
|
||||||
// organization: "CertD Org.",
|
organization: "CertD Org.",
|
||||||
// organizationUnit: "IT Department",
|
organizationUnit: "IT Department",
|
||||||
// emailAddress: email,
|
emailAddress: email,
|
||||||
},
|
},
|
||||||
this.csrInfo ? JSON.parse(this.csrInfo) : {}
|
this.csrInfo ? JSON.parse(this.csrInfo) : {}
|
||||||
);
|
);
|
||||||
@@ -498,7 +430,6 @@ 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);
|
||||||
@@ -524,7 +455,6 @@ 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,
|
||||||
@@ -541,7 +471,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})的校验计划(如果您在流水线创建之后设置了子域名托管,需要重新编辑证书申请任务和重新校验cname记录的校验状态)`);
|
throw new Error(`没有找到域名(${domain})的校验计划`);
|
||||||
}
|
}
|
||||||
if (planSetting.type === "dns") {
|
if (planSetting.type === "dns") {
|
||||||
plan[domain] = await this.createDnsDomainVerifyPlan(planSetting, domain, mainDomain);
|
plan[domain] = await this.createDnsDomainVerifyPlan(planSetting, domain, mainDomain);
|
||||||
@@ -630,20 +560,10 @@ 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,
|
||||||
|
|||||||