mirror of https://github.com/certd/certd
perf: 支持新网代理方式
parent
f415190483
commit
f612509cac
|
|
@ -117,11 +117,11 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
required: true,
|
required: true,
|
||||||
helper: `1. <b>DNS直接验证</b>:域名dns解析是在阿里云/腾讯云/华为云/CF/NameSilo/西数/火山/dns.la/京东云/51dns的,选它
|
helper: `1. <b>DNS直接验证</b>:当域名dns解析已被本系统支持时(即下方DNS解析服务商选项中可选),推荐选择此方式
|
||||||
2. <b>CNAME代理验证</b>:支持任何注册商的域名,第一次需要手动添加[CNAME记录](#/certd/cname/record)(建议将DNS服务器修改为阿里云/腾讯云的,然后使用DNS直接验证)
|
2. <b>CNAME代理验证</b>:支持任何注册商的域名,第一次需要手动添加[CNAME记录](#/certd/cname/record)(如果经常申请失败,建议将DNS服务器修改为阿里云/腾讯云的,然后使用DNS直接验证)
|
||||||
3. <b>HTTP文件验证</b>:不支持泛域名,需要配置网站文件上传
|
3. <b>HTTP文件验证</b>:不支持泛域名,需要配置网站文件上传
|
||||||
4. <b>多DNS提供商</b>:每个域名可以选择独立的DNS提供商
|
4. <b>多DNS提供商</b>:每个域名可以选择独立的DNS提供商
|
||||||
5. <b>自动匹配</b>:需要在[域名管理](#/certd/cert/domain)中事先配置好校验方式
|
5. <b>自动匹配</b>:此处无需选择校验方式,需要在[域名管理](#/certd/cert/domain)中提前配置好校验方式
|
||||||
`,
|
`,
|
||||||
})
|
})
|
||||||
challengeType!: string;
|
challengeType!: string;
|
||||||
|
|
|
||||||
|
|
@ -1,79 +1,157 @@
|
||||||
// import { IsAccess, AccessInput, BaseAccess } from "@certd/pipeline";
|
import { IsAccess, AccessInput, BaseAccess, Pager, PageSearch } from "@certd/pipeline";
|
||||||
// import { XinnetClient } from "@certd/plugin-plus";
|
import crypto from "crypto";
|
||||||
|
/**
|
||||||
|
* 这个注解将注册一个授权配置
|
||||||
|
* 在certd的后台管理系统中,用户可以选择添加此类型的授权
|
||||||
|
*/
|
||||||
|
@IsAccess({
|
||||||
|
name: "xinnetagent",
|
||||||
|
title: "新网授权(代理方式)",
|
||||||
|
icon: "lsicon:badge-new-filled",
|
||||||
|
desc: ""
|
||||||
|
})
|
||||||
|
export class XinnetAgentAccess extends BaseAccess {
|
||||||
|
|
||||||
// /**
|
/**
|
||||||
// * 这个注解将注册一个授权配置
|
* 授权属性配置
|
||||||
// * 在certd的后台管理系统中,用户可以选择添加此类型的授权
|
*/
|
||||||
// */
|
@AccessInput({
|
||||||
// @IsAccess({
|
title: "代理账号",
|
||||||
// name: "xinnetagent",
|
component: {
|
||||||
// title: "新网授权(代理方式)",
|
placeholder: "代理账号,如:agent0001"
|
||||||
// icon: "lsicon:badge-new-filled",
|
},
|
||||||
// desc: ""
|
required: true,
|
||||||
// })
|
encrypt: false
|
||||||
// export class XinnetAccess extends BaseAccess {
|
})
|
||||||
|
agentCode = "";
|
||||||
|
|
||||||
// /**
|
@AccessInput({
|
||||||
// * 授权属性配置
|
title: "API密钥",
|
||||||
// */
|
component: {
|
||||||
// @AccessInput({
|
name: "a-input-password",
|
||||||
// title: "代理账号",
|
vModel: "value",
|
||||||
// component: {
|
placeholder: "API密钥"
|
||||||
// placeholder: "代理账号,如:agent0001"
|
},
|
||||||
// },
|
required: true,
|
||||||
// required: true,
|
encrypt: true
|
||||||
// encrypt: false
|
})
|
||||||
// })
|
appSecret = "";
|
||||||
// username = "";
|
|
||||||
|
|
||||||
// @AccessInput({
|
@AccessInput({
|
||||||
// title: "API密钥",
|
title: "测试",
|
||||||
// component: {
|
component: {
|
||||||
// name: "a-input-password",
|
name: "api-test",
|
||||||
// vModel: "value",
|
action: "TestRequest"
|
||||||
// placeholder: "API密钥"
|
},
|
||||||
// },
|
helper: "点击测试接口是否正常"
|
||||||
// required: true,
|
})
|
||||||
// encrypt: true
|
testRequest = true;
|
||||||
// })
|
|
||||||
// apikey = "";
|
|
||||||
|
|
||||||
// @AccessInput({
|
async onTestRequest() {
|
||||||
// title: "测试",
|
|
||||||
// component: {
|
|
||||||
// name: "api-test",
|
|
||||||
// action: "TestRequest"
|
|
||||||
// },
|
|
||||||
// helper: "点击测试接口是否正常"
|
|
||||||
// })
|
|
||||||
// testRequest = true;
|
|
||||||
|
|
||||||
// async onTestRequest() {
|
// const client = new XinnetClient({
|
||||||
|
// access: this,
|
||||||
|
// logger: this.ctx.logger,
|
||||||
|
// http: this.ctx.http
|
||||||
|
// });
|
||||||
|
await this.getDomainList({ pageNo: 1, pageSize: 1 });
|
||||||
|
|
||||||
// // const client = new XinnetClient({
|
return "ok";
|
||||||
// // access: this,
|
}
|
||||||
// // logger: this.ctx.logger,
|
|
||||||
// // http: this.ctx.http
|
|
||||||
// // });
|
|
||||||
|
|
||||||
// await client.getDomainList({ pageNo: 1, pageSize: 1 });
|
|
||||||
|
|
||||||
// return "ok";
|
|
||||||
// }
|
|
||||||
|
|
||||||
|
|
||||||
// getCacheKey () {
|
|
||||||
// let hashStr = ""
|
|
||||||
// for (const key in this) {
|
|
||||||
// if (Object.prototype.hasOwnProperty.call(this, key)) {
|
|
||||||
// const element = this[key];
|
|
||||||
// hashStr += element;
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// const hashCode = this.ctx.utils.hash.sha256(hashStr);
|
|
||||||
// return `xinnet-${hashCode}`;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// }
|
async getDomainList(req:PageSearch) {
|
||||||
|
const pager = new Pager(req);
|
||||||
|
const conf = {
|
||||||
|
url: "/api/domain/list",
|
||||||
|
data: {
|
||||||
|
pageNo: String(pager.pageNo),
|
||||||
|
pageSize: String(pager.pageSize)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return await this.doRequest(conf);
|
||||||
|
}
|
||||||
|
|
||||||
// new XinnetAccess();
|
|
||||||
|
/**
|
||||||
|
* 生成 UTC 0 时区的时间戳
|
||||||
|
*/
|
||||||
|
generateTimestamp() {
|
||||||
|
const timestamp = new Date().toISOString().replace(/\.\d{3}Z$/, "Z").replaceAll(":", "").replaceAll("-", "");
|
||||||
|
return timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 字节转16进制字符串
|
||||||
|
*/
|
||||||
|
bytesToHex(bytes:any) {
|
||||||
|
return bytes.toString('hex');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 生成签名
|
||||||
|
*/
|
||||||
|
generateSignature(timestamp, urlPath, requestBody) {
|
||||||
|
const algorithm = 'HMAC-SHA256';
|
||||||
|
const requestMethod = 'POST';
|
||||||
|
|
||||||
|
// 构建待签名字符串
|
||||||
|
const stringToSign = `${algorithm}\n${timestamp}\n${requestMethod}\n${urlPath}\n${requestBody}`;
|
||||||
|
|
||||||
|
// 使用 HMAC-SHA256 计算签名
|
||||||
|
const hmac = crypto.createHmac('sha256', this.appSecret);
|
||||||
|
hmac.update(stringToSign);
|
||||||
|
const signatureBytes = hmac.digest();
|
||||||
|
|
||||||
|
// 转换为16进制字符串
|
||||||
|
return this.bytesToHex(signatureBytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 生成 authorization header
|
||||||
|
*/
|
||||||
|
generateAuthorization(timestamp, urlPath, requestBody) {
|
||||||
|
const signature = this.generateSignature(timestamp, urlPath, requestBody);
|
||||||
|
return `HMAC-SHA256 Access=${this.agentCode}, Signature=${signature}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 查询域名分页列表
|
||||||
|
*/
|
||||||
|
async doRequest(req:any) {
|
||||||
|
|
||||||
|
const baseURL = 'https://apiv2.xinnet.com';
|
||||||
|
const urlPath = req.url;
|
||||||
|
const requestURL = baseURL + urlPath; // 实际请求URL去掉最后的斜杠
|
||||||
|
|
||||||
|
// 请求体
|
||||||
|
const requestBody = JSON.stringify(req.data);
|
||||||
|
|
||||||
|
// 生成时间戳和授权头
|
||||||
|
const timestamp = this.generateTimestamp();
|
||||||
|
const authorization = this.generateAuthorization(timestamp, urlPath+"/", requestBody);
|
||||||
|
|
||||||
|
// 请求配置
|
||||||
|
const config = {
|
||||||
|
method: 'POST',
|
||||||
|
url: requestURL,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'timestamp': timestamp,
|
||||||
|
'authorization': authorization
|
||||||
|
},
|
||||||
|
data: requestBody,
|
||||||
|
};
|
||||||
|
|
||||||
|
const res = await this.ctx.http.request(config);
|
||||||
|
|
||||||
|
if (res.code !="0"){
|
||||||
|
throw new Error(`API Error: ${res.code} ${res.requestId} - ${JSON.stringify(res.msg)}`);
|
||||||
|
}
|
||||||
|
return res.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
new XinnetAgentAccess();
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,90 @@
|
||||||
|
import { AbstractDnsProvider, CreateRecordOptions, IsDnsProvider, RemoveRecordOptions } from "@certd/plugin-cert";
|
||||||
|
import { XinnetAgentAccess } from "./access-agent.js";
|
||||||
|
|
||||||
|
export type XinnetAgentRecord = {
|
||||||
|
recordId: number;
|
||||||
|
domainName: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
// 这里通过IsDnsProvider注册一个dnsProvider
|
||||||
|
@IsDnsProvider({
|
||||||
|
name: "xinnetagent",
|
||||||
|
title: "新网(代理方式)",
|
||||||
|
desc: "新网域名解析(代理方式)",
|
||||||
|
icon: "lsicon:badge-new-filled",
|
||||||
|
// 这里是对应的 cloudflare的access类型名称
|
||||||
|
accessType: "xinnetagent",
|
||||||
|
order: 7
|
||||||
|
})
|
||||||
|
export class XinnetAgentProvider extends AbstractDnsProvider<XinnetAgentRecord> {
|
||||||
|
access!: XinnetAgentAccess;
|
||||||
|
|
||||||
|
async onInstance() {
|
||||||
|
//一些初始化的操作
|
||||||
|
// 也可以通过ctx成员变量传递context
|
||||||
|
this.access = this.ctx.access as XinnetAgentAccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 创建dns解析记录,用于验证域名所有权
|
||||||
|
*/
|
||||||
|
async createRecord(options: CreateRecordOptions): Promise<XinnetAgentRecord> {
|
||||||
|
/**
|
||||||
|
* fullRecord: '_acme-challenge.test.example.com',
|
||||||
|
* value: 一串uuid
|
||||||
|
* type: 'TXT',
|
||||||
|
* domain: 'example.com'
|
||||||
|
*/
|
||||||
|
const { fullRecord, value, type, domain } = options;
|
||||||
|
this.logger.info("添加域名解析:", fullRecord, value, type, domain);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* /api/dns/create
|
||||||
|
* domainName 是 string 域名名称 test-xinnet-0516-ceshi.cn
|
||||||
|
recordName 是 string 记录名 test1.test-xinnet-0516-ceshi.cn,如果是@和空字符只需要传域名即可
|
||||||
|
type 是 string 解析记录的类型 可选择类型如下: NS A CNAME MX TXT URL SRV AAAA A
|
||||||
|
value 是 string 解析内容 192.168.1.50
|
||||||
|
line 是 string 线路 只能传"默认"
|
||||||
|
*/
|
||||||
|
|
||||||
|
const res = await this.access.doRequest({
|
||||||
|
url:"/api/dns/create",
|
||||||
|
data:{
|
||||||
|
domainName: domain,
|
||||||
|
recordName: fullRecord,
|
||||||
|
type: type,
|
||||||
|
value: value,
|
||||||
|
line: "默认"
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
return {
|
||||||
|
recordId:res,
|
||||||
|
domainName: domain
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除dns解析记录,清理申请痕迹
|
||||||
|
* @param options
|
||||||
|
*/
|
||||||
|
async removeRecord(options: RemoveRecordOptions<XinnetAgentRecord>): Promise<void> {
|
||||||
|
|
||||||
|
const {domainName,recordId} = options.recordRes;
|
||||||
|
await this.access.doRequest({
|
||||||
|
url:"/api/dns/delete",
|
||||||
|
data:{
|
||||||
|
recordId: recordId,
|
||||||
|
domainName: domainName
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//实例化这个provider,将其自动注册到系统中
|
||||||
|
new XinnetAgentProvider();
|
||||||
|
|
@ -1,2 +1,5 @@
|
||||||
export * from './dns-provider.js';
|
export * from './dns-provider.js';
|
||||||
export * from './access.js';
|
export * from './access.js';
|
||||||
|
|
||||||
|
export * from './access-agent.js';
|
||||||
|
export * from './dns-provider-agent.js';
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue