mirror of https://github.com/certd/certd
perf: 新增部署到百度云CDN插件
parent
5b148b7ed9
commit
f126f9f932
|
@ -36,6 +36,7 @@ _acme-challenge.cert.com ---> xxxxx.cname.proxy.com ----> txt-record-abcdefg
|
|||

|
||||

|
||||
4. 申请过程中,Certd会在`xxxxxx.cname.proxy.com`下自动添加TXT记录。
|
||||
5. 到此即可自动化申请证书了
|
||||
|
||||
|
||||
|
||||
|
|
|
@ -182,7 +182,7 @@ export function createAxiosService({ logger }: { logger: Logger }) {
|
|||
|
||||
export const http = createAxiosService({ logger }) as HttpClient;
|
||||
export type HttpClientResponse<R> = any;
|
||||
export type HttpRequestConfig<D> = {
|
||||
export type HttpRequestConfig<D=any> = {
|
||||
skipSslVerify?: boolean;
|
||||
skipCheckRes?: boolean;
|
||||
logParams?: boolean;
|
||||
|
|
|
@ -42,6 +42,7 @@
|
|||
"china-division": "^2.7.0",
|
||||
"core-js": "^3.36.0",
|
||||
"cos-js-sdk-v5": "^1.7.0",
|
||||
"cron-parser": "^4.9.0",
|
||||
"cropperjs": "^1.6.1",
|
||||
"dayjs": "^1.11.10",
|
||||
"highlight.js": "^11.9.0",
|
||||
|
|
|
@ -199,7 +199,7 @@ const steps = ref<Step[]>([
|
|||
{
|
||||
image: "/static/doc/images/15-1-email.png",
|
||||
title: "设置邮件通知",
|
||||
descriptions: ["建议选择监听'错误时'和'错误转成功'两种即可,在意外失败时可以尽快去排查问题,(免费版需要配置邮件服务器)"]
|
||||
descriptions: ["建议选择监听'错误时'和'错误转成功'两种即可,在意外失败时可以尽快去排查问题,(基础版需要配置邮件服务器)"]
|
||||
},
|
||||
{
|
||||
title: "教程结束",
|
||||
|
|
|
@ -59,7 +59,7 @@ const text = computed<Text>(() => {
|
|||
title: "此为专业版功能"
|
||||
},
|
||||
nav: {
|
||||
name: "免费版",
|
||||
name: "基础版",
|
||||
title: "升级专业版,享受更多VIP特权"
|
||||
}
|
||||
}
|
||||
|
@ -93,7 +93,7 @@ const formState = reactive({
|
|||
|
||||
const vipTypeDefine = {
|
||||
free: {
|
||||
title: "免费版",
|
||||
title: "基础版",
|
||||
type: "free",
|
||||
privilege: ["证书申请功能无限制", "证书流水线数量10条", "常用的主机、cdn等部署插件"]
|
||||
},
|
||||
|
|
|
@ -111,7 +111,7 @@ export const useSettingStore = defineStore({
|
|||
},
|
||||
vipLabel(): string {
|
||||
const vipLabelMap: any = {
|
||||
free: "免费版",
|
||||
free: "基础版",
|
||||
plus: "专业版",
|
||||
comm: "商业版"
|
||||
};
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
CNAME服务配置
|
||||
<span class="sub">
|
||||
此处配置的域名作为其他域名校验的代理,当别的域名需要申请证书时,通过CNAME映射到此域名上来验证所有权。好处是任何域名都可以通过此方式申请证书,也无需填写AccessSecret。
|
||||
<a href="https://certd.docmirror.cn/guide/feature/cname/" taget="_blank">CNAME功能原理及使用说明</a>
|
||||
</span>
|
||||
</div>
|
||||
</template>
|
||||
|
|
|
@ -128,7 +128,7 @@ export class PipelineService extends BaseService<PipelineEntity> {
|
|||
count += 1;
|
||||
}
|
||||
if (count > freeCount) {
|
||||
throw new NeedVIPException('免费版最多只能创建10个pipeline');
|
||||
throw new NeedVIPException('基础版最多只能创建10个pipeline');
|
||||
}
|
||||
}
|
||||
if (!isUpdate) {
|
||||
|
|
|
@ -0,0 +1,59 @@
|
|||
import { TencentAccess } from '@certd/plugin-plus';
|
||||
import { CertInfo } from '@certd/plugin-cert';
|
||||
import { ILogger } from '@certd/pipeline';
|
||||
|
||||
export class TencentSslClient {
|
||||
access: TencentAccess;
|
||||
logger: ILogger;
|
||||
region?: string;
|
||||
constructor(opts: { access: TencentAccess; logger: ILogger; region?: string }) {
|
||||
this.access = opts.access;
|
||||
this.logger = opts.logger;
|
||||
this.region = opts.region;
|
||||
}
|
||||
async getSslClient(): Promise<any> {
|
||||
const sdk = await import('tencentcloud-sdk-nodejs/tencentcloud/services/ssl/v20191205/index.js');
|
||||
const SslClient = sdk.v20191205.Client;
|
||||
|
||||
const clientConfig = {
|
||||
credential: {
|
||||
secretId: this.access.secretId,
|
||||
secretKey: this.access.secretKey,
|
||||
},
|
||||
region: this.region,
|
||||
profile: {
|
||||
httpProfile: {
|
||||
endpoint: 'ssl.tencentcloudapi.com',
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
return new SslClient(clientConfig);
|
||||
}
|
||||
|
||||
checkRet(ret: any) {
|
||||
if (!ret || ret.Error) {
|
||||
throw new Error('请求失败:' + ret.Error.Code + ',' + ret.Error.Message);
|
||||
}
|
||||
}
|
||||
|
||||
async uploadToTencent(opts: { certName: string; cert: CertInfo }): Promise<string> {
|
||||
const client = await this.getSslClient();
|
||||
const params = {
|
||||
CertificatePublicKey: opts.cert.crt,
|
||||
CertificatePrivateKey: opts.cert.key,
|
||||
Alias: opts.certName,
|
||||
};
|
||||
const ret = await client.UploadCertificate(params);
|
||||
this.checkRet(ret);
|
||||
this.logger.info('证书上传成功:tencentCertId=', ret.CertificateId);
|
||||
return ret.CertificateId;
|
||||
}
|
||||
|
||||
async deployCertificateInstance(params: any) {
|
||||
const client = await this.getSslClient();
|
||||
const res = await client.DeployCertificateInstance(params);
|
||||
this.checkRet(res);
|
||||
return res;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,163 @@
|
|||
import { IsTaskPlugin, pluginGroups, RunStrategy, TaskInput } from '@certd/pipeline';
|
||||
import { CertInfo } from '@certd/plugin-cert';
|
||||
import { AbstractPlusTaskPlugin, createRemoteSelectInputDefine } from '@certd/plugin-plus';
|
||||
import { TencentSslClient } from '../../lib/index.js';
|
||||
|
||||
@IsTaskPlugin({
|
||||
name: 'DeployCertToTencentCosPlugin',
|
||||
title: '部署证书到腾讯云COS',
|
||||
needPlus: true,
|
||||
icon: 'svg:icon-tencentcloud',
|
||||
group: pluginGroups.tencent.key,
|
||||
desc: '部署到腾讯云COS源站域名证书',
|
||||
deprecated: '暂不可用',
|
||||
default: {
|
||||
strategy: {
|
||||
runStrategy: RunStrategy.SkipWhenSucceed,
|
||||
},
|
||||
},
|
||||
})
|
||||
export class DeployCertToTencentCosPlugin extends AbstractPlusTaskPlugin {
|
||||
/**
|
||||
* AccessProvider的id
|
||||
*/
|
||||
@TaskInput({
|
||||
title: 'Access授权',
|
||||
helper: 'access授权',
|
||||
component: {
|
||||
name: 'access-selector',
|
||||
type: 'tencent',
|
||||
},
|
||||
required: true,
|
||||
})
|
||||
accessId!: string;
|
||||
|
||||
@TaskInput({
|
||||
title: '存储桶名称',
|
||||
helper: '请输入存储桶名称',
|
||||
})
|
||||
bucket!: string;
|
||||
|
||||
@TaskInput({
|
||||
title: '所在地域',
|
||||
helper: '存储桶所在地域',
|
||||
component: {
|
||||
name: 'a-auto-complete',
|
||||
vModel: 'value',
|
||||
options: [
|
||||
{ value: '', label: '--------中国大陆地区-------', disabled: true },
|
||||
{ value: 'ap-beijing-1', label: '北京1区' },
|
||||
{ value: 'ap-beijing', label: '北京' },
|
||||
{ value: 'ap-nanjing', label: '南京' },
|
||||
{ value: 'ap-shanghai', label: '上海' },
|
||||
{ value: 'ap-guangzhou', label: '广州' },
|
||||
{ value: 'ap-chengdu', label: '成都' },
|
||||
{ value: 'ap-chongqing', label: '重庆' },
|
||||
{ value: 'ap-shenzhen-fsi', label: '深圳金融' },
|
||||
{ value: 'ap-shanghai-fsi', label: '上海金融' },
|
||||
{ value: 'ap-beijing-fsi', label: '北京金融' },
|
||||
{ value: '', label: '--------中国香港及境外-------', disabled: true },
|
||||
{ value: 'ap-hongkong', label: '中国香港' },
|
||||
{ value: 'ap-singapore', label: '新加坡' },
|
||||
{ value: 'ap-mumbai', label: '孟买' },
|
||||
{ value: 'ap-jakarta', label: '雅加达' },
|
||||
{ value: 'ap-seoul', label: '首尔' },
|
||||
{ value: 'ap-bangkok', label: '曼谷' },
|
||||
{ value: 'ap-tokyo', label: '东京' },
|
||||
{ value: 'na-siliconvalley', label: '硅谷' },
|
||||
{ value: 'na-ashburn', label: '弗吉尼亚' },
|
||||
{ value: 'sa-saopaulo', label: '圣保罗' },
|
||||
{ value: 'eu-frankfurt', label: '法兰克福' },
|
||||
],
|
||||
},
|
||||
})
|
||||
region!: string;
|
||||
|
||||
// @TaskInput(createCertDomainGetterInputDefine())
|
||||
// certDomains!: string[];
|
||||
|
||||
@TaskInput(
|
||||
createRemoteSelectInputDefine({
|
||||
title: 'COS域名',
|
||||
helper: '请选择域名',
|
||||
typeName: DeployCertToTencentCosPlugin.name,
|
||||
action: DeployCertToTencentCosPlugin.prototype.onGetDomainList.name,
|
||||
watches: ['bucket', 'region'],
|
||||
})
|
||||
)
|
||||
domains!: string | string[];
|
||||
|
||||
@TaskInput({
|
||||
title: '域名证书',
|
||||
helper: '请选择前置任务输出的域名证书,或者选择前置任务“上传证书到腾讯云”任务的证书ID',
|
||||
component: {
|
||||
name: 'output-selector',
|
||||
from: ['CertApply', 'CertApplyLego', 'UploadCertToTencent'],
|
||||
},
|
||||
required: true,
|
||||
})
|
||||
cert!: CertInfo | string;
|
||||
|
||||
async onInstance() {}
|
||||
|
||||
async execute(): Promise<void> {
|
||||
const access = await this.accessService.getById(this.accessId);
|
||||
|
||||
const client = new TencentSslClient({
|
||||
access,
|
||||
logger: this.logger,
|
||||
region: this.region,
|
||||
});
|
||||
|
||||
let tencentCertId: string = this.cert as string;
|
||||
if (typeof this.cert !== 'string') {
|
||||
tencentCertId = await client.uploadToTencent({
|
||||
certName: this.appendTimeSuffix('certd'),
|
||||
cert: this.cert,
|
||||
});
|
||||
}
|
||||
|
||||
const params = {
|
||||
CertificateId: tencentCertId,
|
||||
ResourceType: 'cos',
|
||||
Status: 1,
|
||||
InstanceIdList: [`${this.bucket}#${this.region}#${this.domains}`],
|
||||
};
|
||||
|
||||
const res = await client.deployCertificateInstance(params);
|
||||
this.logger.info('部署成功', res);
|
||||
}
|
||||
|
||||
async onGetDomainList(data: any) {
|
||||
const access = await this.accessService.getById(this.accessId);
|
||||
|
||||
const cosv5 = await import('cos-nodejs-sdk-v5');
|
||||
const cos = new cosv5.default({
|
||||
SecretId: access.secretId,
|
||||
SecretKey: access.secretKey,
|
||||
});
|
||||
if (!this.bucket) {
|
||||
throw new Error('存储桶名称不能为空');
|
||||
}
|
||||
if (!this.region) {
|
||||
throw new Error('所在地域不能为空');
|
||||
}
|
||||
|
||||
const res = await cos.getBucketDomain({
|
||||
Bucket: this.bucket,
|
||||
/** 存储桶所在地域 @see https://cloud.tencent.com/document/product/436/6224 */
|
||||
Region: this.region,
|
||||
});
|
||||
|
||||
this.ctx.logger.info('获取域名列表:', res);
|
||||
|
||||
return res.DomainRule.map((item: any) => {
|
||||
return {
|
||||
label: item.Name,
|
||||
value: item.Name,
|
||||
};
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// new DeployCertToTencentCosPlugin()
|
Loading…
Reference in New Issue