mirror of
https://github.com/certd/certd.git
synced 2026-05-13 19:47:55 +08:00
202 lines
5.9 KiB
TypeScript
202 lines
5.9 KiB
TypeScript
import { Provide, Scope, ScopeEnum } from '@midwayjs/core';
|
|
import { InjectEntityModel } from '@midwayjs/typeorm';
|
|
import { Repository } from 'typeorm';
|
|
import { SysSettingsEntity } from '../entity/sys-settings.js';
|
|
import { BaseSettings, SysInstallInfo, SysPrivateSettings, SysPublicSettings, SysSecret, SysSecretBackup } from './models.js';
|
|
|
|
import { BaseService } from '../../../basic/index.js';
|
|
import { cache, logger, setGlobalProxy } from '@certd/basic';
|
|
import * as dns from 'node:dns';
|
|
import {mergeUtils} from "@certd/basic";
|
|
import { executorQueue } from '../../basic/service/executor-queue.js';
|
|
const {merge} = mergeUtils;
|
|
/**
|
|
* 设置
|
|
*/
|
|
@Provide()
|
|
@Scope(ScopeEnum.Singleton)
|
|
export class SysSettingsService extends BaseService<SysSettingsEntity> {
|
|
@InjectEntityModel(SysSettingsEntity)
|
|
repository: Repository<SysSettingsEntity>;
|
|
|
|
getRepository() {
|
|
return this.repository;
|
|
}
|
|
async getById(id: any): Promise<SysSettingsEntity | null> {
|
|
const entity = await this.info(id);
|
|
if (!entity) {
|
|
return null;
|
|
}
|
|
const setting = JSON.parse(entity.setting);
|
|
return {
|
|
id: entity.id,
|
|
...setting,
|
|
};
|
|
}
|
|
|
|
async getByKey(key: string): Promise<SysSettingsEntity | null> {
|
|
if (!key) {
|
|
return null;
|
|
}
|
|
return await this.repository.findOne({
|
|
where: {
|
|
key,
|
|
},
|
|
});
|
|
}
|
|
|
|
async getSettingByKey(key: string): Promise<any | null> {
|
|
const entity = await this.getByKey(key);
|
|
if (!entity) {
|
|
return null;
|
|
}
|
|
return JSON.parse(entity.setting);
|
|
}
|
|
|
|
async save(bean: SysSettingsEntity) {
|
|
const entity = await this.repository.findOne({
|
|
where: {
|
|
key: bean.key,
|
|
},
|
|
});
|
|
if (entity) {
|
|
entity.setting = bean.setting;
|
|
await this.repository.save(entity);
|
|
} else {
|
|
bean.title = bean.key;
|
|
await this.repository.save(bean);
|
|
}
|
|
}
|
|
|
|
async getSetting<T>(type: any): Promise<T> {
|
|
const key = type.__key__;
|
|
const cacheKey = type.getCacheKey();
|
|
const settings: T = cache.get(cacheKey);
|
|
if (settings) {
|
|
return settings;
|
|
}
|
|
let newSetting: T = new type();
|
|
const savedSettings = await this.getSettingByKey(key);
|
|
newSetting = merge(newSetting, savedSettings);
|
|
await this.saveSetting(newSetting);
|
|
cache.set(cacheKey, newSetting);
|
|
return newSetting;
|
|
}
|
|
|
|
async saveSetting<T extends BaseSettings>(bean: T) {
|
|
const type: any = bean.constructor;
|
|
const key = type.__key__;
|
|
const cacheKey = type.getCacheKey();
|
|
|
|
const entity = await this.getByKey(key);
|
|
if (entity) {
|
|
entity.setting = JSON.stringify(bean);
|
|
entity.access = type.__access__;
|
|
|
|
if (key === SysSecretBackup.__key__) {
|
|
//备份密钥不允许更新
|
|
return;
|
|
}
|
|
|
|
await this.repository.save(entity);
|
|
} else {
|
|
const newEntity = new SysSettingsEntity();
|
|
newEntity.key = key;
|
|
newEntity.title = type.__title__;
|
|
newEntity.setting = JSON.stringify(bean);
|
|
newEntity.access = type.__access__;
|
|
await this.repository.save(newEntity);
|
|
}
|
|
|
|
cache.set(cacheKey, bean);
|
|
}
|
|
|
|
async getPublicSettings(): Promise<SysPublicSettings> {
|
|
return await this.getSetting(SysPublicSettings);
|
|
}
|
|
|
|
async savePublicSettings(bean: SysPublicSettings) {
|
|
await this.saveSetting(bean);
|
|
}
|
|
|
|
async getPrivateSettings(): Promise<SysPrivateSettings> {
|
|
return await this.getSetting(SysPrivateSettings);
|
|
}
|
|
|
|
async savePrivateSettings(bean: SysPrivateSettings) {
|
|
await this.saveSetting(bean);
|
|
|
|
//让设置生效
|
|
await this.reloadPrivateSettings();
|
|
}
|
|
|
|
async reloadPrivateSettings() {
|
|
const bean = await this.getPrivateSettings();
|
|
const opts = {
|
|
httpProxy: bean.httpProxy,
|
|
httpsProxy: bean.httpsProxy,
|
|
};
|
|
setGlobalProxy(opts);
|
|
|
|
if (bean.dnsResultOrder) {
|
|
dns.setDefaultResultOrder(bean.dnsResultOrder as any);
|
|
}
|
|
|
|
if (bean.pipelineMaxRunningCount){
|
|
executorQueue.setMaxRunningCount(bean.pipelineMaxRunningCount);
|
|
}
|
|
}
|
|
|
|
async updateByKey(key: string, setting: any) {
|
|
const entity = await this.getByKey(key);
|
|
if (entity) {
|
|
entity.setting = JSON.stringify(setting);
|
|
await this.repository.save(entity);
|
|
} else {
|
|
throw new Error('该设置不存在');
|
|
}
|
|
cache.delete(`settings.${key}`);
|
|
}
|
|
|
|
async backupSecret() {
|
|
const settings = await this.getSettingByKey(SysSecretBackup.__key__);
|
|
const privateSettings = await this.getPrivateSettings();
|
|
const installInfo = await this.getSetting<SysInstallInfo>(SysInstallInfo);
|
|
if (settings == null) {
|
|
const backup = new SysSecretBackup();
|
|
if (installInfo.siteId == null || privateSettings.encryptSecret == null) {
|
|
logger.error('备份密钥失败,siteId或encryptSecret为空');
|
|
return;
|
|
}
|
|
backup.siteId = installInfo.siteId;
|
|
backup.encryptSecret = privateSettings.encryptSecret;
|
|
await this.saveSetting(backup);
|
|
logger.info('备份密钥成功');
|
|
} else {
|
|
//校验是否有变化
|
|
if (settings.siteId !== installInfo.siteId) {
|
|
throw new Error(`siteId与备份不一致,可能是数据异常,请检查:backup=${settings.siteId}, current=${installInfo.siteId}`);
|
|
}
|
|
if (settings.encryptSecret !== privateSettings.encryptSecret) {
|
|
throw new Error('encryptSecret与备份不一致,可能是数据异常,请检查');
|
|
}
|
|
}
|
|
}
|
|
async getSecret() {
|
|
const sysSecret = await this.getSetting<SysSecret>(SysSecret);
|
|
if (sysSecret.encryptSecret) {
|
|
return sysSecret;
|
|
}
|
|
//从备份中读取
|
|
const settings = await this.getSettingByKey(SysSecretBackup.__key__);
|
|
if (settings == null || !settings.encryptSecret) {
|
|
throw new Error('密钥备份不存在');
|
|
}
|
|
sysSecret.siteId = settings.siteId;
|
|
sysSecret.encryptSecret = settings.encryptSecret;
|
|
await this.saveSetting(sysSecret);
|
|
logger.info('密钥恢复成功');
|
|
return sysSecret;
|
|
}
|
|
}
|