import os from "node:os"; import { inject, injectAll, injectable } from "tsyringe"; import { OnLoad } from "@spt-aki/di/OnLoad"; import { OnUpdate } from "@spt-aki/di/OnUpdate"; import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes"; import { ICoreConfig } from "@spt-aki/models/spt/config/ICoreConfig"; import { ILogger } from "@spt-aki/models/spt/utils/ILogger"; import { ConfigServer } from "@spt-aki/servers/ConfigServer"; import { LocalisationService } from "@spt-aki/services/LocalisationService"; import { EncodingUtil } from "@spt-aki/utils/EncodingUtil"; import { TimeUtil } from "@spt-aki/utils/TimeUtil"; @injectable() export class App { protected onUpdateLastRun = {}; protected coreConfig: ICoreConfig; constructor( @inject("WinstonLogger") protected logger: ILogger, @inject("TimeUtil") protected timeUtil: TimeUtil, @inject("LocalisationService") protected localisationService: LocalisationService, @inject("ConfigServer") protected configServer: ConfigServer, @inject("EncodingUtil") protected encodingUtil: EncodingUtil, @injectAll("OnLoad") protected onLoadComponents: OnLoad[], @injectAll("OnUpdate") protected onUpdateComponents: OnUpdate[], ) { this.coreConfig = this.configServer.getConfig(ConfigTypes.CORE); } public async load(): Promise { // execute onLoad callbacks this.logger.info(this.localisationService.getText("executing_startup_callbacks")); this.logger.debug(`OS: ${os.arch()} | ${os.version()} | ${process.platform}`); this.logger.debug(`CPU: ${os.cpus()[0]?.model} cores: ${os.cpus().length}`); this.logger.debug(`RAM: ${(os.totalmem() / 1024 / 1024 / 1024).toFixed(2)}GB`); this.logger.debug(`PATH: ${this.encodingUtil.toBase64(process.argv[0])}`); this.logger.debug(`PATH: ${this.encodingUtil.toBase64(process.execPath)}`); this.logger.debug(`Server: ${this.coreConfig.akiVersion}`); if (this.coreConfig.buildTime) { this.logger.debug(`Date: ${this.coreConfig.buildTime}`); } if (this.coreConfig.commit) { this.logger.debug(`Commit: ${this.coreConfig.commit}`); } for (const onLoad of this.onLoadComponents) { await onLoad.onLoad(); } setInterval(() => { this.update(this.onUpdateComponents); }, 5000); } protected async update(onUpdateComponents: OnUpdate[]): Promise { for (const updateable of onUpdateComponents) { let success = false; const lastRunTimeTimestamp = this.onUpdateLastRun[updateable.getRoute()] || 0; // 0 on first load so all update() calls occur on first load const secondsSinceLastRun = this.timeUtil.getTimestamp() - lastRunTimeTimestamp; try { success = await updateable.onUpdate(secondsSinceLastRun); } catch (err) { this.logUpdateException(err, updateable); } if (success) { this.onUpdateLastRun[updateable.getRoute()] = this.timeUtil.getTimestamp(); } else { /* temporary for debug */ const warnTime = 20 * 60; if (success === void 0 && !(secondsSinceLastRun % warnTime)) { this.logger.debug( this.localisationService.getText("route_onupdate_no_response", updateable.getRoute()), ); } } } } protected logUpdateException(err: any, updateable: OnUpdate): void { this.logger.error(this.localisationService.getText("scheduled_event_failed_to_run", updateable.getRoute())); if (err.message) { this.logger.error(err.message); } if (err.stack) { this.logger.error(err.stack); } } }