Files
certd/packages/plugins/plugin-cert/src/plugin/cert-plugin/cert-reader.ts

145 lines
4.0 KiB
TypeScript
Raw Normal View History

2024-07-15 00:30:33 +08:00
import { CertInfo } from "./acme.js";
import fs from "fs";
import os from "os";
import path from "path";
2024-08-25 11:56:15 +08:00
import { crypto } from "@certd/acme-client";
2024-11-04 15:14:56 +08:00
import { ILogger } from "@certd/basic";
2024-09-05 18:00:45 +08:00
import dayjs from "dayjs";
2024-09-05 15:36:35 +08:00
2024-09-22 23:19:10 +08:00
export type CertReaderHandleContext = {
reader: CertReader;
tmpCrtPath: string;
tmpKeyPath: string;
tmpPfxPath?: string;
tmpDerPath?: string;
tmpIcPath?: string;
2024-10-30 01:44:02 +08:00
tmpJksPath?: string;
2024-09-22 23:19:10 +08:00
};
2024-09-05 15:36:35 +08:00
export type CertReaderHandle = (ctx: CertReaderHandleContext) => Promise<void>;
export type HandleOpts = { logger: ILogger; handle: CertReaderHandle };
2024-09-06 00:13:21 +08:00
export class CertReader {
cert: CertInfo;
crt: string;
key: string;
csr: string;
2024-09-22 23:19:10 +08:00
ic: string; //中间证书
detail: any;
expires: number;
constructor(certInfo: CertInfo) {
2024-09-06 00:13:21 +08:00
this.cert = certInfo;
this.crt = certInfo.crt;
this.key = certInfo.key;
this.csr = certInfo.csr;
2024-09-22 23:19:10 +08:00
this.ic = certInfo.ic;
if (!this.ic) {
this.ic = this.getIc();
this.cert.ic = this.ic;
}
2024-09-06 00:13:21 +08:00
const { detail, expires } = this.getCrtDetail(this.cert.crt);
this.detail = detail;
this.expires = expires.getTime();
}
2024-09-22 23:19:10 +08:00
getIc() {
//中间证书ic 就是crt的第一个 -----END CERTIFICATE----- 之后的内容
const endStr = "-----END CERTIFICATE-----";
const firstBlockEndIndex = this.crt.indexOf(endStr);
const start = firstBlockEndIndex + endStr.length + 1;
if (this.crt.length <= start) {
return "";
}
const ic = this.crt.substring(start);
return ic.trim();
}
toCertInfo(): CertInfo {
2024-09-06 00:13:21 +08:00
return this.cert;
}
2024-09-06 00:13:21 +08:00
getCrtDetail(crt: string = this.cert.crt) {
2024-08-25 11:56:15 +08:00
const detail = crypto.readCertificateInfo(crt.toString());
const expires = detail.notAfter;
return { detail, expires };
}
2024-09-30 18:00:51 +08:00
getAllDomains() {
const { detail } = this.getCrtDetail();
const domains = [detail.domains.commonName];
domains.push(...detail.domains.altNames);
return domains;
}
2024-10-30 01:44:02 +08:00
saveToFile(type: "crt" | "key" | "pfx" | "der" | "ic" | "jks", filepath?: string) {
2024-09-06 00:13:21 +08:00
if (!this.cert[type]) {
return;
}
if (filepath == null) {
//写入临时目录
2024-10-30 01:44:02 +08:00
filepath = path.join(os.tmpdir(), "/certd/tmp/", Math.floor(Math.random() * 1000000) + `_cert.${type}`);
}
const dir = path.dirname(filepath);
if (!fs.existsSync(dir)) {
fs.mkdirSync(dir, { recursive: true });
}
2024-09-22 23:19:10 +08:00
if (type === "crt" || type === "key" || type === "ic") {
2024-09-06 00:13:21 +08:00
fs.writeFileSync(filepath, this.cert[type]);
} else {
fs.writeFileSync(filepath, Buffer.from(this.cert[type], "base64"));
}
return filepath;
}
2024-09-05 15:36:35 +08:00
async readCertFile(opts: HandleOpts) {
const logger = opts.logger;
logger.info("将证书写入本地缓存文件");
const tmpCrtPath = this.saveToFile("crt");
const tmpKeyPath = this.saveToFile("key");
2024-09-06 00:13:21 +08:00
const tmpPfxPath = this.saveToFile("pfx");
2024-09-22 23:19:10 +08:00
const tmpIcPath = this.saveToFile("ic");
2024-09-05 15:36:35 +08:00
logger.info("本地文件写入成功");
2024-09-22 23:19:10 +08:00
const tmpDerPath = this.saveToFile("der");
2024-10-30 01:44:02 +08:00
const tmpJksPath = this.saveToFile("jks");
2024-09-05 15:36:35 +08:00
try {
2024-09-09 16:01:42 +08:00
return await opts.handle({
2024-09-05 15:36:35 +08:00
reader: this,
tmpCrtPath: tmpCrtPath,
tmpKeyPath: tmpKeyPath,
2024-09-06 00:13:21 +08:00
tmpPfxPath: tmpPfxPath,
tmpDerPath: tmpDerPath,
2024-09-22 23:19:10 +08:00
tmpIcPath: tmpIcPath,
2024-10-30 01:44:02 +08:00
tmpJksPath: tmpJksPath,
2024-09-05 15:36:35 +08:00
});
2024-09-09 16:01:42 +08:00
} catch (err) {
throw err;
2024-09-05 15:36:35 +08:00
} finally {
//删除临时文件
logger.info("删除临时文件");
2024-09-06 10:19:03 +08:00
function removeFile(filepath?: string) {
if (filepath) {
fs.unlinkSync(filepath);
}
}
removeFile(tmpCrtPath);
removeFile(tmpKeyPath);
removeFile(tmpPfxPath);
removeFile(tmpDerPath);
2024-09-22 23:19:10 +08:00
removeFile(tmpIcPath);
2024-10-30 01:44:02 +08:00
removeFile(tmpJksPath);
2024-09-05 15:36:35 +08:00
}
}
buildCertFileName(suffix: string, applyTime: number, prefix = "cert") {
const detail = this.getCrtDetail();
let domain = detail.detail.domains.commonName;
domain = domain.replace(".", "_").replace("*", "_");
2024-09-05 18:00:45 +08:00
const timeStr = dayjs(applyTime).format("YYYYMMDDHHmmss");
return `${prefix}_${domain}_${timeStr}.${suffix}`;
2024-09-05 15:36:35 +08:00
}
}