mirror of
https://github.com/certd/certd.git
synced 2026-05-14 12:07:32 +08:00
272 lines
7.7 KiB
TypeScript
272 lines
7.7 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 { getAllSslProviderDomains, setSslProviderReverseProxies } from '@certd/acme-client';
|
|
import { cache, logger, mergeUtils, setGlobalProxy } from '@certd/basic';
|
|
import * as dns from 'node:dns';
|
|
import { BaseService, setAdminMode } from '../../../basic/index.js';
|
|
import { executorQueue } from '../../basic/service/executor-queue.js';
|
|
import { isComm, isPlus } from '@certd/plus-core';
|
|
const { merge } = mergeUtils;
|
|
|
|
let lastSaveEnvVars = {};
|
|
/**
|
|
* 设置
|
|
*/
|
|
@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) {
|
|
if (isComm()) {
|
|
if (bean.adminMode === 'enterprise') {
|
|
throw new Error("商业版不支持使用企业管理模式")
|
|
}
|
|
}
|
|
|
|
await this.saveSetting(bean);
|
|
//让设置生效
|
|
await this.reloadPublicSettings();
|
|
}
|
|
|
|
async getPrivateSettings(): Promise<SysPrivateSettings> {
|
|
const res = await this.getSetting<SysPrivateSettings>(SysPrivateSettings);
|
|
const sslProviderDomains = getAllSslProviderDomains();
|
|
for (const domain of sslProviderDomains) {
|
|
if (!res.reverseProxies[domain]) {
|
|
res.reverseProxies[domain] = "";
|
|
}
|
|
}
|
|
return res
|
|
}
|
|
|
|
async savePrivateSettings(bean: SysPrivateSettings) {
|
|
await this.saveSetting(bean);
|
|
|
|
//让设置生效
|
|
await this.reloadPrivateSettings();
|
|
}
|
|
|
|
async reloadSettings() {
|
|
await this.reloadPrivateSettings()
|
|
await this.reloadPublicSettings()
|
|
}
|
|
|
|
async reloadPublicSettings() {
|
|
const publicSetting = await this.getPublicSettings()
|
|
if (isPlus()){
|
|
setAdminMode(publicSetting.adminMode )
|
|
}
|
|
}
|
|
|
|
async reloadPrivateSettings() {
|
|
const privateSetting = await this.getPrivateSettings();
|
|
const opts = {
|
|
httpProxy: privateSetting.httpProxy,
|
|
httpsProxy: privateSetting.httpsProxy,
|
|
};
|
|
setGlobalProxy(opts);
|
|
|
|
if (privateSetting.dnsResultOrder) {
|
|
dns.setDefaultResultOrder(privateSetting.dnsResultOrder as any);
|
|
}
|
|
|
|
if (privateSetting.pipelineMaxRunningCount) {
|
|
executorQueue.setMaxRunningCount(privateSetting.pipelineMaxRunningCount);
|
|
}
|
|
|
|
setSslProviderReverseProxies(privateSetting.reverseProxies);
|
|
|
|
//加载环境变量
|
|
this.setEnvironmentVars(privateSetting.environmentVars);
|
|
}
|
|
|
|
setEnvironmentVars(vars: string) {
|
|
const envVars = {}
|
|
if (typeof vars !== 'string') {
|
|
vars = ""
|
|
}
|
|
vars.split('\n').forEach(line => {
|
|
line = line.trim();
|
|
if (!line || line.startsWith('#')) {
|
|
return
|
|
}
|
|
|
|
const arr = line.split("#")
|
|
if (arr.length > 0) {
|
|
line = arr[0].trim();
|
|
}
|
|
if (!line.includes("=")) {
|
|
return
|
|
}
|
|
|
|
const [key, value] = line.split('=');
|
|
if (key && value) {
|
|
envVars[key.trim()] = value.trim();
|
|
}
|
|
});
|
|
//先删除旧环境变量
|
|
if (lastSaveEnvVars) {
|
|
for (const key in lastSaveEnvVars) {
|
|
delete process.env[key];
|
|
}
|
|
}
|
|
|
|
merge(process.env, envVars);
|
|
lastSaveEnvVars = envVars;
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|