You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Hydro/packages/hydrojudge/src/daemon.ts

108 lines
3.3 KiB
TypeScript

/* eslint-disable no-constant-condition */
/* eslint-disable no-await-in-loop */
/* ..
.' @`._
~ ...._.' ,__.-;
_..- - - /` .-' ~
: __./' , .'-'- .._
~ `- -(.-'''- -. \`._ `. ~
_.- '( .______.'.-' `-.` `.
: `-..____`-. ;
`. ```` 稻花香里说丰年, ; ~
`-.__ 听取人生经验。 __.-'
````- - -.......- - -''' ~
~ */
import fs from 'fs-extra';
import yaml from 'js-yaml';
import * as Session from './hosts/index';
import { sleep, Queue } from './utils';
import log from './log';
declare global {
namespace NodeJS {
interface Global {
onDestory: Function[]
hosts: any
}
}
}
if (!global.onDestory) global.onDestory = [];
if (!global.hosts) global.hosts = [];
let SI = false;
const { RETRY_DELAY_SEC, CONFIG_FILE, CONFIG } = require('./config');
const terminate = async () => {
log.info('正在保存数据');
try {
await Promise.all(global.onDestory.map((f) => f()));
process.exit(1);
} catch (e) {
if (SI) process.exit(1);
log.error(e.stack);
log.error('发生了错误。');
log.error('再次按下 Ctrl-C 可强制退出。');
SI = true;
}
};
process.on('SIGINT', terminate);
process.on('SIGTERM', terminate);
process.on('unhandledRejection', (reason, p) => {
console.log('Unhandled Rejection at: Promise ', p);
});
async function daemon() {
let config;
try {
if (CONFIG) config = CONFIG;
else {
config = (await fs.readFile(CONFIG_FILE)).toString();
config = yaml.load(config);
}
} catch (e) {
log.error('配置文件无效或未找到。');
process.exit(1);
}
const hosts = {};
const queue = new Queue<any>();
for (const i in config.hosts) {
config.hosts[i].host = i;
hosts[i] = new Session[config.hosts[i].type || 'vj4'](config.hosts[i]);
await hosts[i].init();
}
global.hosts = hosts;
if (!CONFIG) {
global.onDestory.push(async () => {
const cfg = { hosts: {} };
for (const i in hosts) {
cfg.hosts[i] = {
host: i,
type: hosts[i].config.type || 'default',
uname: hosts[i].config.uname,
password: hosts[i].config.password,
server_url: hosts[i].config.server_url,
};
if (hosts[i].config.cookie) cfg.hosts[i].cookie = hosts[i].config.cookie;
if (hosts[i].config.detail) cfg.hosts[i].detail = hosts[i].config.detail;
}
await fs.writeFile(CONFIG_FILE, yaml.dump(cfg));
});
}
while ('Orz twd2') {
try {
for (const i in hosts) await hosts[i].consume(queue);
while ('Orz iceb0y') {
const [task] = await queue.get();
task.handle();
}
} catch (e) {
log.error(e, e.stack);
log.info(`${RETRY_DELAY_SEC} 秒后重试`);
await sleep(RETRY_DELAY_SEC * 1000);
}
}
}
if (!module.parent) daemon();
export = daemon;