mirror of
https://github.com/certd/certd.git
synced 2026-04-03 14:10:54 +08:00
perf: dcdn自动匹配部署,支持新增域名感知
This commit is contained in:
@@ -334,7 +334,7 @@ export class Executor {
|
|||||||
//参数没有变化
|
//参数没有变化
|
||||||
inputChanged = false;
|
inputChanged = false;
|
||||||
}
|
}
|
||||||
if (step.strategy?.runStrategy === RunStrategy.SkipWhenSucceed) {
|
if (step.strategy?.runStrategy === RunStrategy.SkipWhenSucceed && define.runStrategy !== RunStrategy.AlwaysRun) {
|
||||||
if (lastResult != null && lastResult === ResultType.success && !inputChanged) {
|
if (lastResult != null && lastResult === ResultType.success && !inputChanged) {
|
||||||
step.status!.output = lastNode?.status?.output;
|
step.status!.output = lastNode?.status?.output;
|
||||||
step.status!.files = lastNode?.status?.files;
|
step.status!.files = lastNode?.status?.files;
|
||||||
|
|||||||
@@ -4,13 +4,14 @@ import { FileStore } from "../core/file-store.js";
|
|||||||
import { accessRegistry, IAccessService } from "../access/index.js";
|
import { accessRegistry, IAccessService } from "../access/index.js";
|
||||||
import { ICnameProxyService, IEmailService, IServiceGetter, IUrlService } from "../service/index.js";
|
import { ICnameProxyService, IEmailService, IServiceGetter, IUrlService } from "../service/index.js";
|
||||||
import { CancelError, IContext, RunHistory, RunnableCollection } from "../core/index.js";
|
import { CancelError, IContext, RunHistory, RunnableCollection } from "../core/index.js";
|
||||||
import { HttpRequestConfig, ILogger, logger, optionsUtils, utils } from "@certd/basic";
|
import { domainUtils, HttpRequestConfig, ILogger, logger, optionsUtils, utils } from "@certd/basic";
|
||||||
import { HttpClient } from "@certd/basic";
|
import { HttpClient } from "@certd/basic";
|
||||||
import dayjs from "dayjs";
|
import dayjs from "dayjs";
|
||||||
import { IPluginConfigService } from "../service/config.js";
|
import { IPluginConfigService } from "../service/config.js";
|
||||||
import { upperFirst } from "lodash-es";
|
import { cloneDeep, upperFirst } from "lodash-es";
|
||||||
import { INotificationService } from "../notification/index.js";
|
import { INotificationService } from "../notification/index.js";
|
||||||
import { TaskEmitter } from "../service/emit.js";
|
import { TaskEmitter } from "../service/emit.js";
|
||||||
|
import { PageSearch } from "../context/index.js";
|
||||||
|
|
||||||
export type PluginRequestHandleReq<T = any> = {
|
export type PluginRequestHandleReq<T = any> = {
|
||||||
typeName: string;
|
typeName: string;
|
||||||
@@ -64,6 +65,7 @@ export type PluginDefine = Registrable & {
|
|||||||
onlyAdmin?: boolean;
|
onlyAdmin?: boolean;
|
||||||
needPlus?: boolean;
|
needPlus?: boolean;
|
||||||
showRunStrategy?: boolean;
|
showRunStrategy?: boolean;
|
||||||
|
runStrategy?: any;
|
||||||
pluginType?: string; //类型
|
pluginType?: string; //类型
|
||||||
type?: string; //来源
|
type?: string; //来源
|
||||||
};
|
};
|
||||||
@@ -81,6 +83,12 @@ export type TaskResult = {
|
|||||||
pipelineVars: Record<string, any>;
|
pipelineVars: Record<string, any>;
|
||||||
pipelinePrivateVars?: Record<string, any>;
|
pipelinePrivateVars?: Record<string, any>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type CertTargetItem = {
|
||||||
|
value: string;
|
||||||
|
label: string;
|
||||||
|
domain: string | string[];
|
||||||
|
};
|
||||||
export type TaskInstanceContext = {
|
export type TaskInstanceContext = {
|
||||||
//流水线定义
|
//流水线定义
|
||||||
pipeline: Pipeline;
|
pipeline: Pipeline;
|
||||||
@@ -316,10 +324,102 @@ export abstract class AbstractTaskPlugin implements ITaskPlugin {
|
|||||||
return this.getLastStatus().status?.output?.[key];
|
return this.getLastStatus().status?.output?.[key];
|
||||||
}
|
}
|
||||||
|
|
||||||
getMatchedDomains(domainList: string[], certDomains: string[]): string[] {
|
isDomainMatched(domainList: string | string[], certDomains: string[]): boolean {
|
||||||
const { matched } = optionsUtils.groupByDomain(domainList, certDomains);
|
const matched = domainUtils.match(domainList, certDomains);
|
||||||
return matched;
|
return matched;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isNotChanged() {
|
||||||
|
const lastResult = this.ctx?.lastStatus?.status?.status;
|
||||||
|
return !this.ctx.inputChanged && lastResult === "success";
|
||||||
|
}
|
||||||
|
|
||||||
|
async getAutoMatchedTargets(req: {
|
||||||
|
targetName: string;
|
||||||
|
certDomains: string[];
|
||||||
|
pageSize: number;
|
||||||
|
getDeployTargetList: (req: PageSearch) => Promise<{ list: CertTargetItem[]; total: number }>;
|
||||||
|
}): Promise<CertTargetItem[]> {
|
||||||
|
const matchedDomains: CertTargetItem[] = [];
|
||||||
|
let pageNo = 1;
|
||||||
|
const { certDomains } = req;
|
||||||
|
|
||||||
|
const pageSize = req.pageSize || 100;
|
||||||
|
while (true) {
|
||||||
|
const result = await req.getDeployTargetList({
|
||||||
|
pageNo,
|
||||||
|
pageSize,
|
||||||
|
});
|
||||||
|
const pageData = result.list;
|
||||||
|
this.logger.info(`获取到 ${pageData.length} 个 ${req.targetName}`);
|
||||||
|
|
||||||
|
if (!pageData || pageData.length === 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const item of pageData) {
|
||||||
|
const domainName = item.domain;
|
||||||
|
if (this.isDomainMatched(domainName, certDomains)) {
|
||||||
|
matchedDomains.push(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const totalCount = result.total || 0;
|
||||||
|
if (pageNo * pageSize >= totalCount || matchedDomains.length == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
pageNo++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return matchedDomains;
|
||||||
|
}
|
||||||
|
|
||||||
|
async autoMatchedDeploy(req: {
|
||||||
|
targetName: string;
|
||||||
|
getCertDomains: () => string[];
|
||||||
|
uploadCert: () => Promise<any>;
|
||||||
|
deployOne: (req: { target: any; cert: any }) => Promise<void>;
|
||||||
|
getDeployTargetList: (req: PageSearch) => Promise<{ list: CertTargetItem[]; total: number }>;
|
||||||
|
}): Promise<{ result: any; deployedList: any[] }> {
|
||||||
|
this.logger.info("证书匹配模式部署");
|
||||||
|
const certDomains = req.getCertDomains();
|
||||||
|
const certTargetList = await this.getAutoMatchedTargets({
|
||||||
|
targetName: req.targetName,
|
||||||
|
pageSize: 200,
|
||||||
|
certDomains,
|
||||||
|
getDeployTargetList: req.getDeployTargetList,
|
||||||
|
});
|
||||||
|
if (certTargetList.length === 0) {
|
||||||
|
this.logger.warn(`未找到匹配的${req.targetName}`);
|
||||||
|
return { result: "skip", deployedList: [] };
|
||||||
|
}
|
||||||
|
this.logger.info(`找到 ${certTargetList.length} 个匹配的${req.targetName}`);
|
||||||
|
|
||||||
|
//开始部署,检查是否已经部署过
|
||||||
|
const deployedList = cloneDeep(this.getLastStatus()?.status?.output?.deployedList || []);
|
||||||
|
const unDeployedTargets = certTargetList.filter(item => !deployedList.includes(item.value));
|
||||||
|
const count = unDeployedTargets.length;
|
||||||
|
const deployedCount = certTargetList.length - count;
|
||||||
|
if (deployedCount > 0) {
|
||||||
|
this.logger.info(`跳过 ${deployedCount} 个已部署过的${req.targetName}`);
|
||||||
|
}
|
||||||
|
this.logger.info(`需要部署 ${count} 个${req.targetName}`);
|
||||||
|
if (count === 0) {
|
||||||
|
return { result: "skip", deployedList };
|
||||||
|
}
|
||||||
|
this.logger.info(`开始部署`);
|
||||||
|
const aliCrtId = await req.uploadCert();
|
||||||
|
for (const target of unDeployedTargets) {
|
||||||
|
await req.deployOne({
|
||||||
|
cert: aliCrtId,
|
||||||
|
target,
|
||||||
|
});
|
||||||
|
deployedList.push(target.value);
|
||||||
|
}
|
||||||
|
this.logger.info(`本次成功部署 ${count} 个${req.targetName}`);
|
||||||
|
return { result: "success", deployedList };
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export type OutputVO = {
|
export type OutputVO = {
|
||||||
|
|||||||
@@ -43,6 +43,12 @@ const formats = {
|
|||||||
jks: ["jks"],
|
jks: ["jks"],
|
||||||
p7b: ["p7b", "key"],
|
p7b: ["p7b", "key"],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type SimpleCertDetail = {
|
||||||
|
notBefore: Date;
|
||||||
|
notAfter: Date;
|
||||||
|
domains: string[];
|
||||||
|
};
|
||||||
export class CertReader {
|
export class CertReader {
|
||||||
cert: CertInfo;
|
cert: CertInfo;
|
||||||
|
|
||||||
@@ -116,6 +122,15 @@ export class CertReader {
|
|||||||
return CertReader.readCertDetail(crt);
|
return CertReader.readCertDetail(crt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getSimpleDetail() {
|
||||||
|
const { detail } = this.getCrtDetail();
|
||||||
|
return {
|
||||||
|
notBefore: detail.notBefore,
|
||||||
|
notAfter: detail.notAfter,
|
||||||
|
domains: this.getAllDomains(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
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 effective = detail.notBefore;
|
||||||
|
|||||||
@@ -235,7 +235,6 @@ watch(
|
|||||||
const { form } = value;
|
const { form } = value;
|
||||||
const oldForm: any = oldValue?.form;
|
const oldForm: any = oldValue?.form;
|
||||||
let changed = oldForm == null || optionsRef.value.length == 0;
|
let changed = oldForm == null || optionsRef.value.length == 0;
|
||||||
debugger;
|
|
||||||
if (props.watches && props.watches.length > 0) {
|
if (props.watches && props.watches.length > 0) {
|
||||||
for (const key of props.watches) {
|
for (const key of props.watches) {
|
||||||
if (oldForm && JSON.stringify(form[key]) != JSON.stringify(oldForm[key])) {
|
if (oldForm && JSON.stringify(form[key]) != JSON.stringify(oldForm[key])) {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { AbstractTaskPlugin, IsTaskPlugin, PageSearch, pluginGroups, RunStrategy, TaskInput } from '@certd/pipeline';
|
import { AbstractTaskPlugin, CertTargetItem, IsTaskPlugin, PageSearch, pluginGroups, RunStrategy, TaskInput, TaskOutput } from '@certd/pipeline';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import {
|
import {
|
||||||
|
CertReader,
|
||||||
createCertDomainGetterInputDefine,
|
createCertDomainGetterInputDefine,
|
||||||
createRemoteSelectInputDefine
|
createRemoteSelectInputDefine
|
||||||
} from "@certd/plugin-lib";
|
} from "@certd/plugin-lib";
|
||||||
@@ -9,18 +10,19 @@ import { AliyunAccess } from "../../../plugin-lib/aliyun/access/index.js";
|
|||||||
import { CertInfo } from '@certd/plugin-cert';
|
import { CertInfo } from '@certd/plugin-cert';
|
||||||
import { CertApplyPluginNames } from '@certd/plugin-cert';
|
import { CertApplyPluginNames } from '@certd/plugin-cert';
|
||||||
import { optionsUtils } from "@certd/basic";
|
import { optionsUtils } from "@certd/basic";
|
||||||
import { AliyunClient, CasCertId } from "../../../plugin-lib/aliyun/lib/index.js";
|
import { AliyunClient, AliyunSslClient, CasCertId } from "../../../plugin-lib/aliyun/lib/index.js";
|
||||||
@IsTaskPlugin({
|
@IsTaskPlugin({
|
||||||
name: 'DeployCertToAliyunDCDN',
|
name: 'DeployCertToAliyunDCDN',
|
||||||
title: '阿里云-部署证书至DCDN',
|
title: '阿里云-部署证书至DCDN',
|
||||||
icon: 'svg:icon-aliyun',
|
icon: 'svg:icon-aliyun',
|
||||||
group: pluginGroups.aliyun.key,
|
group: pluginGroups.aliyun.key,
|
||||||
desc: '依赖证书申请前置任务,自动部署域名证书至阿里云DCDN',
|
desc: '依赖证书申请前置任务,自动部署域名证书至阿里云DCDN',
|
||||||
default: {
|
runStrategy: RunStrategy.AlwaysRun,
|
||||||
strategy: {
|
// default: {
|
||||||
runStrategy: RunStrategy.SkipWhenSucceed,
|
// strategy: {
|
||||||
},
|
// runStrategy: RunStrategy.SkipWhenSucceed,
|
||||||
},
|
// },
|
||||||
|
// },
|
||||||
})
|
})
|
||||||
export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
||||||
@TaskInput({
|
@TaskInput({
|
||||||
@@ -57,15 +59,15 @@ export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
|||||||
|
|
||||||
@TaskInput({
|
@TaskInput({
|
||||||
title: '域名匹配模式',
|
title: '域名匹配模式',
|
||||||
helper: '选择域名匹配方式',
|
helper: '根据证书匹配:根据证书域名自动匹配DCDN加速域名自动部署,新增加速域名自动感知,自动新增部署',
|
||||||
component: {
|
component: {
|
||||||
name: 'select',
|
name: 'a-select',
|
||||||
options: [
|
options: [
|
||||||
{ label: '手动选择', value: 'manual' },
|
{ label: '手动选择', value: 'manual' },
|
||||||
{ label: '根据证书匹配', value: 'auto' },
|
{ label: '根据证书匹配', value: 'auto' },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
default: 'manual',
|
value: 'manual',
|
||||||
})
|
})
|
||||||
domainMatchMode!: 'manual' | 'auto';
|
domainMatchMode!: 'manual' | 'auto';
|
||||||
|
|
||||||
@@ -79,7 +81,7 @@ export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
|||||||
mergeScript: `
|
mergeScript: `
|
||||||
return {
|
return {
|
||||||
show: ctx.compute(({form})=>{
|
show: ctx.compute(({form})=>{
|
||||||
return domainMatchMode === "manual"
|
return form.domainMatchMode === "manual"
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
@@ -87,44 +89,82 @@ export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
|||||||
)
|
)
|
||||||
domainName!: string | string[];
|
domainName!: string | string[];
|
||||||
|
|
||||||
|
@TaskOutput({
|
||||||
|
title: '已部署过的DCDN加速域名',
|
||||||
|
})
|
||||||
|
deployedList!: string[];
|
||||||
|
|
||||||
|
|
||||||
async onInstance() { }
|
async onInstance() { }
|
||||||
async execute(): Promise<void> {
|
async execute(): Promise<any> {
|
||||||
this.logger.info('开始部署证书到阿里云DCDN');
|
this.logger.info('开始部署证书到阿里云DCDN');
|
||||||
const access = (await this.getAccess(this.accessId)) as AliyunAccess;
|
const access = (await this.getAccess(this.accessId)) as AliyunAccess;
|
||||||
const client = await this.getClient(access);
|
const client = await this.getClient(access);
|
||||||
|
const sslClient = new AliyunSslClient({ access, logger: this.logger });
|
||||||
let domains: string[] = [];
|
|
||||||
|
|
||||||
if (this.domainMatchMode === 'auto') {
|
if (this.domainMatchMode === 'auto') {
|
||||||
this.logger.info('使用根据证书匹配模式');
|
const { result, deployedList } = await this.autoMatchedDeploy({
|
||||||
if (!this.certDomains || this.certDomains.length === 0) {
|
targetName: 'DCDN加速域名',
|
||||||
throw new Error('未获取到证书域名信息');
|
uploadCert: async () => {
|
||||||
}
|
return await sslClient.uploadCertOrGet(this.cert);
|
||||||
domains = await this.getAutoMatchedDomains(this.certDomains);
|
},
|
||||||
if (domains.length === 0) {
|
deployOne: async (req:{target:any,cert:any})=>{
|
||||||
this.logger.warn('未找到匹配的DCDN域名');
|
return await this.deployOne(client, req.target.value, req.cert);
|
||||||
return;
|
},
|
||||||
}
|
getCertDomains: ()=>{
|
||||||
this.logger.info(`找到 ${domains.length} 个匹配的DCDN域名`);
|
return this.getCertDomains();
|
||||||
|
},
|
||||||
|
getDeployTargetList: async (req: PageSearch)=>{
|
||||||
|
return await this.onGetDomainList(req);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
this.deployedList = deployedList;
|
||||||
|
return result;
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
if (this.isNotChanged()) {
|
||||||
|
this.logger.info('输入参数未变更,跳过');
|
||||||
|
return "skip";
|
||||||
|
}
|
||||||
if (!this.domainName) {
|
if (!this.domainName) {
|
||||||
throw new Error('您还未选择DCDN域名');
|
throw new Error('您还未选择DCDN域名');
|
||||||
}
|
}
|
||||||
|
let domains: string[] = [];
|
||||||
domains = typeof this.domainName === 'string' ? [this.domainName] : this.domainName;
|
domains = typeof this.domainName === 'string' ? [this.domainName] : this.domainName;
|
||||||
|
const aliCrtId = await sslClient.uploadCertOrGet(this.cert);
|
||||||
|
for (const domainName of domains) {
|
||||||
|
await this.deployOne(client, domainName, aliCrtId);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const domainName of domains) {
|
|
||||||
this.logger.info(`[${domainName}]开始部署`)
|
|
||||||
const params = await this.buildParams(domainName);
|
|
||||||
await this.doRequest(client, params);
|
|
||||||
await this.ctx.utils.sleep(1000);
|
|
||||||
this.logger.info(`[${domainName}]部署成功`)
|
|
||||||
}
|
|
||||||
|
|
||||||
this.logger.info('部署完成');
|
this.logger.info('部署完成');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getCertDomains(): string[]{
|
||||||
|
const casCert = this.cert as CasCertId;
|
||||||
|
const certInfo = this.cert as CertInfo;
|
||||||
|
if (casCert.certId) {
|
||||||
|
if (!casCert.detail){
|
||||||
|
throw new Error('未获取到证书域名列表,请尝试强制重新运行一下流水线');
|
||||||
|
}
|
||||||
|
return casCert.detail?.domains || [];
|
||||||
|
}else if (certInfo.crt){
|
||||||
|
return new CertReader(certInfo).getSimpleDetail().domains || [];
|
||||||
|
}else{
|
||||||
|
throw new Error('未获取到证书域名列表,请尝试强制重新运行一下流水线');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async deployOne(client: any, domainName: string, aliCrtId: CasCertId){
|
||||||
|
this.logger.info(`[${domainName}]开始部署`)
|
||||||
|
const params = await this.buildParams(domainName, aliCrtId);
|
||||||
|
await this.doRequest(client, params);
|
||||||
|
await this.ctx.utils.sleep(1000);
|
||||||
|
this.logger.info(`[${domainName}]部署成功`)
|
||||||
|
}
|
||||||
|
|
||||||
async getClient(access: AliyunAccess) {
|
async getClient(access: AliyunAccess) {
|
||||||
const client = new AliyunClient({ logger: this.logger });
|
const client = new AliyunClient({ logger: this.logger });
|
||||||
await client.init({
|
await client.init({
|
||||||
@@ -136,30 +176,9 @@ export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
|||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
async buildParams(domainName: string) {
|
async buildParams(domainName: string, aliCrtId: CasCertId) {
|
||||||
const CertName = (this.certName ?? 'certd') + '-' + dayjs().format('YYYYMMDDHHmmss');
|
const CertName = (this.certName ?? 'certd') + '-' + dayjs().format('YYYYMMDDHHmmss');
|
||||||
|
const certId = aliCrtId.certId;
|
||||||
let certId: any = this.cert
|
|
||||||
if (typeof this.cert === 'object') {
|
|
||||||
const certInfo = this.cert as CertInfo;
|
|
||||||
const casCertId = this.cert as CasCertId;
|
|
||||||
if (certInfo.crt) {
|
|
||||||
this.logger.info('上传证书:', CertName);
|
|
||||||
const cert: any = this.cert;
|
|
||||||
return {
|
|
||||||
DomainName: domainName,
|
|
||||||
SSLProtocol: 'on',
|
|
||||||
CertName: CertName,
|
|
||||||
CertType: 'upload',
|
|
||||||
SSLPub: cert.crt,
|
|
||||||
SSLPri: cert.key,
|
|
||||||
};
|
|
||||||
}else if (casCertId.certId){
|
|
||||||
certId = casCertId.certId;
|
|
||||||
}else{
|
|
||||||
throw new Error('证书格式错误'+JSON.stringify(this.cert));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.logger.info('使用已上传的证书:', certId);
|
this.logger.info('使用已上传的证书:', certId);
|
||||||
return {
|
return {
|
||||||
DomainName: domainName,
|
DomainName: domainName,
|
||||||
@@ -187,36 +206,7 @@ export class DeployCertToAliyunDCDN extends AbstractTaskPlugin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async getAutoMatchedDomains(certDomains: string[]): Promise<string[]> {
|
async onGetDomainList(data: PageSearch): Promise<{list: CertTargetItem[], total: number}> {
|
||||||
const matchedDomains: string[] = [];
|
|
||||||
let pageNumber = 1;
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
const result = await this.onGetDomainList({ pageNo: pageNumber });
|
|
||||||
const pageData = result.list;
|
|
||||||
this.logger.info(`获取到 ${pageData.length} 个DCDN域名`);
|
|
||||||
|
|
||||||
if (!pageData || pageData.length === 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
const matched = this.getMatchedDomains(pageData, certDomains);
|
|
||||||
matchedDomains.push(...matched);
|
|
||||||
|
|
||||||
const totalCount = result.total || 0;
|
|
||||||
if (pageNumber * 500 >= totalCount) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
pageNumber++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return matchedDomains;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
async onGetDomainList(data: PageSearch) {
|
|
||||||
if (!this.accessId) {
|
if (!this.accessId) {
|
||||||
throw new Error('请选择Access授权');
|
throw new Error('请选择Access授权');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -99,11 +99,13 @@ export class UploadCertToAliyun extends AbstractTaskPlugin {
|
|||||||
endpoint,
|
endpoint,
|
||||||
});
|
});
|
||||||
let certName = ""
|
let certName = ""
|
||||||
|
const certReader = new CertReader(this.cert);
|
||||||
if (this.name){
|
if (this.name){
|
||||||
certName = this.appendTimeSuffix(this.name)
|
certName = this.appendTimeSuffix(this.name)
|
||||||
}else {
|
}else {
|
||||||
certName = this.buildCertName(CertReader.getMainDomain(this.cert.crt))
|
certName = this.buildCertName(certReader.getMainDomain())
|
||||||
}
|
}
|
||||||
|
|
||||||
const certIdRes = await client.uploadCertificate({
|
const certIdRes = await client.uploadCertificate({
|
||||||
name: certName,
|
name: certName,
|
||||||
cert: this.cert,
|
cert: this.cert,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { ILogger, utils } from "@certd/basic";
|
import { ILogger, utils } from "@certd/basic";
|
||||||
import { AliyunAccess } from "../access/index.js";
|
import { AliyunAccess } from "../access/index.js";
|
||||||
import { AliyunClient } from "./index.js";
|
import { AliyunClient } from "./index.js";
|
||||||
import { CertInfo, CertReader } from "@certd/plugin-lib";
|
import { CertInfo, CertReader, SimpleCertDetail } from "@certd/plugin-lib";
|
||||||
|
|
||||||
export type AliyunCertInfo = {
|
export type AliyunCertInfo = {
|
||||||
crt: string; //fullchain证书
|
crt: string; //fullchain证书
|
||||||
@@ -37,6 +37,7 @@ export type CasCertId = {
|
|||||||
certId: number;
|
certId: number;
|
||||||
certIdentifier: string;
|
certIdentifier: string;
|
||||||
certName: string;
|
certName: string;
|
||||||
|
detail?: SimpleCertDetail;
|
||||||
}
|
}
|
||||||
export class AliyunSslClient {
|
export class AliyunSslClient {
|
||||||
opts: AliyunSslClientOpts;
|
opts: AliyunSslClientOpts;
|
||||||
@@ -119,10 +120,12 @@ export class AliyunSslClient {
|
|||||||
this.checkRet(ret);
|
this.checkRet(ret);
|
||||||
this.opts.logger.info("证书上传成功:aliyunCertId=", ret.CertId);
|
this.opts.logger.info("证书上传成功:aliyunCertId=", ret.CertId);
|
||||||
//output
|
//output
|
||||||
|
const certReader = new CertReader(req.cert as any);
|
||||||
return {
|
return {
|
||||||
certId: ret.CertId,
|
certId: ret.CertId,
|
||||||
certName: req.name,
|
certName: req.name,
|
||||||
certIdentifier: this.getCertIdentifier(ret.CertId),
|
certIdentifier: this.getCertIdentifier(ret.CertId),
|
||||||
|
detail:certReader.getSimpleDetail(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,7 +139,8 @@ export class AliyunSslClient {
|
|||||||
const certInfo = cert as CertInfo;
|
const certInfo = cert as CertInfo;
|
||||||
// 上传证书到阿里云
|
// 上传证书到阿里云
|
||||||
this.logger.info(`开始上传证书`);
|
this.logger.info(`开始上传证书`);
|
||||||
const certName = CertReader.buildCertName(certInfo);
|
const certReader = new CertReader(certInfo);
|
||||||
|
const certName = certReader.buildCertName();
|
||||||
const res = await this.uploadCertificate({
|
const res = await this.uploadCertificate({
|
||||||
name: certName,
|
name: certName,
|
||||||
cert: certInfo
|
cert: certInfo
|
||||||
@@ -151,7 +155,7 @@ export class AliyunSslClient {
|
|||||||
return {
|
return {
|
||||||
certId,
|
certId,
|
||||||
certIdentifier,
|
certIdentifier,
|
||||||
certName
|
certName,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user