mirror of https://github.com/certd/certd
				
				
				
			perf: 支持新网代理方式
							parent
							
								
									f415190483
								
							
						
					
					
						commit
						f612509cac
					
				| 
						 | 
				
			
			@ -117,11 +117,11 @@ export class CertApplyPlugin extends CertApplyBasePlugin {
 | 
			
		|||
      ],
 | 
			
		||||
    },
 | 
			
		||||
    required: true,
 | 
			
		||||
    helper: `1. <b>DNS直接验证</b>:域名dns解析是在阿里云/腾讯云/华为云/CF/NameSilo/西数/火山/dns.la/京东云/51dns的,选它
 | 
			
		||||
2.  <b>CNAME代理验证</b>:支持任何注册商的域名,第一次需要手动添加[CNAME记录](#/certd/cname/record)(建议将DNS服务器修改为阿里云/腾讯云的,然后使用DNS直接验证)
 | 
			
		||||
    helper: `1. <b>DNS直接验证</b>:当域名dns解析已被本系统支持时(即下方DNS解析服务商选项中可选),推荐选择此方式
 | 
			
		||||
2.  <b>CNAME代理验证</b>:支持任何注册商的域名,第一次需要手动添加[CNAME记录](#/certd/cname/record)(如果经常申请失败,建议将DNS服务器修改为阿里云/腾讯云的,然后使用DNS直接验证)
 | 
			
		||||
3.  <b>HTTP文件验证</b>:不支持泛域名,需要配置网站文件上传
 | 
			
		||||
4.  <b>多DNS提供商</b>:每个域名可以选择独立的DNS提供商
 | 
			
		||||
5.  <b>自动匹配</b>:需要在[域名管理](#/certd/cert/domain)中事先配置好校验方式
 | 
			
		||||
5.  <b>自动匹配</b>:此处无需选择校验方式,需要在[域名管理](#/certd/cert/domain)中提前配置好校验方式
 | 
			
		||||
`,
 | 
			
		||||
  })
 | 
			
		||||
  challengeType!: string;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,79 +1,157 @@
 | 
			
		|||
// import { IsAccess, AccessInput, BaseAccess } from "@certd/pipeline";
 | 
			
		||||
// import { XinnetClient } from "@certd/plugin-plus";
 | 
			
		||||
import { IsAccess, AccessInput, BaseAccess, Pager, PageSearch } from "@certd/pipeline";
 | 
			
		||||
import crypto from "crypto";
 | 
			
		||||
/**
 | 
			
		||||
 * 这个注解将注册一个授权配置
 | 
			
		||||
 * 在certd的后台管理系统中,用户可以选择添加此类型的授权
 | 
			
		||||
 */
 | 
			
		||||
@IsAccess({
 | 
			
		||||
  name: "xinnetagent",
 | 
			
		||||
  title: "新网授权(代理方式)",
 | 
			
		||||
  icon: "lsicon:badge-new-filled",
 | 
			
		||||
  desc: ""
 | 
			
		||||
})
 | 
			
		||||
export class XinnetAgentAccess extends BaseAccess {
 | 
			
		||||
 | 
			
		||||
// /**
 | 
			
		||||
//  * 这个注解将注册一个授权配置
 | 
			
		||||
//  * 在certd的后台管理系统中,用户可以选择添加此类型的授权
 | 
			
		||||
//  */
 | 
			
		||||
// @IsAccess({
 | 
			
		||||
//   name: "xinnetagent",
 | 
			
		||||
//   title: "新网授权(代理方式)",
 | 
			
		||||
//   icon: "lsicon:badge-new-filled",
 | 
			
		||||
//   desc: ""
 | 
			
		||||
// })
 | 
			
		||||
// export class XinnetAccess extends BaseAccess {
 | 
			
		||||
  /**
 | 
			
		||||
   * 授权属性配置
 | 
			
		||||
   */
 | 
			
		||||
  @AccessInput({
 | 
			
		||||
    title: "代理账号",
 | 
			
		||||
    component: {
 | 
			
		||||
      placeholder: "代理账号,如:agent0001"
 | 
			
		||||
    },
 | 
			
		||||
    required: true,
 | 
			
		||||
    encrypt: false
 | 
			
		||||
  })
 | 
			
		||||
  agentCode = "";
 | 
			
		||||
 | 
			
		||||
//   /**
 | 
			
		||||
//    * 授权属性配置
 | 
			
		||||
//    */
 | 
			
		||||
//   @AccessInput({
 | 
			
		||||
//     title: "代理账号",
 | 
			
		||||
//     component: {
 | 
			
		||||
//       placeholder: "代理账号,如:agent0001"
 | 
			
		||||
//     },
 | 
			
		||||
//     required: true,
 | 
			
		||||
//     encrypt: false
 | 
			
		||||
//   })
 | 
			
		||||
//   username = "";
 | 
			
		||||
  @AccessInput({
 | 
			
		||||
    title: "API密钥",
 | 
			
		||||
    component: {
 | 
			
		||||
      name: "a-input-password",
 | 
			
		||||
      vModel: "value",
 | 
			
		||||
      placeholder: "API密钥"
 | 
			
		||||
    },
 | 
			
		||||
    required: true,
 | 
			
		||||
    encrypt: true
 | 
			
		||||
  })
 | 
			
		||||
  appSecret = "";
 | 
			
		||||
 | 
			
		||||
//   @AccessInput({
 | 
			
		||||
//     title: "API密钥",
 | 
			
		||||
//     component: {
 | 
			
		||||
//       name: "a-input-password",
 | 
			
		||||
//       vModel: "value",
 | 
			
		||||
//       placeholder: "API密钥"
 | 
			
		||||
//     },
 | 
			
		||||
//     required: true,
 | 
			
		||||
//     encrypt: true
 | 
			
		||||
//   })
 | 
			
		||||
//   apikey = "";
 | 
			
		||||
  @AccessInput({
 | 
			
		||||
    title: "测试",
 | 
			
		||||
    component: {
 | 
			
		||||
      name: "api-test",
 | 
			
		||||
      action: "TestRequest"
 | 
			
		||||
    },
 | 
			
		||||
    helper: "点击测试接口是否正常"
 | 
			
		||||
  })
 | 
			
		||||
  testRequest = true;
 | 
			
		||||
 | 
			
		||||
//   @AccessInput({
 | 
			
		||||
//     title: "测试",
 | 
			
		||||
//     component: {
 | 
			
		||||
//       name: "api-test",
 | 
			
		||||
//       action: "TestRequest"
 | 
			
		||||
//     },
 | 
			
		||||
//     helper: "点击测试接口是否正常"
 | 
			
		||||
//   })
 | 
			
		||||
//   testRequest = true;
 | 
			
		||||
  async onTestRequest() {
 | 
			
		||||
 | 
			
		||||
//   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({
 | 
			
		||||
//     //   access: this,
 | 
			
		||||
//     //   logger: this.ctx.logger,
 | 
			
		||||
//     //   http: this.ctx.http
 | 
			
		||||
//     // });
 | 
			
		||||
 | 
			
		||||
//     await client.getDomainList({ pageNo: 1, pageSize: 1 });
 | 
			
		||||
 | 
			
		||||
//     return "ok";
 | 
			
		||||
//   }
 | 
			
		||||
    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 './access.js';
 | 
			
		||||
 | 
			
		||||
export * from './access-agent.js';
 | 
			
		||||
export * from './dns-provider-agent.js';
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue