2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 17:21:17 +00:00
|
|
|
import { OnLoad } from "@spt-aki/di/OnLoad";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { IHttpConfig } from "@spt-aki/models/spt/config/IHttpConfig";
|
|
|
|
import { IDatabaseTables } from "@spt-aki/models/spt/server/IDatabaseTables";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ImageRouter } from "@spt-aki/routers/ImageRouter";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { EncodingUtil } from "@spt-aki/utils/EncodingUtil";
|
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
import { ImporterUtil } from "@spt-aki/utils/ImporterUtil";
|
|
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
|
|
|
import { VFS } from "@spt-aki/utils/VFS";
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class DatabaseImporter implements OnLoad
|
|
|
|
{
|
|
|
|
private hashedFile: any;
|
|
|
|
private valid = VaildationResult.UNDEFINED;
|
2023-04-15 14:07:56 +01:00
|
|
|
private filepath: string;
|
|
|
|
protected httpConfig: IHttpConfig;
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("VFS") protected vfs: VFS,
|
|
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("ImageRouter") protected imageRouter: ImageRouter,
|
|
|
|
@inject("EncodingUtil") protected encodingUtil: EncodingUtil,
|
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
2023-04-15 14:07:56 +01:00
|
|
|
@inject("ImporterUtil") protected importerUtil: ImporterUtil,
|
2023-11-13 11:14:58 -05:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 15:23:46 +00:00
|
|
|
)
|
|
|
|
{
|
2023-04-15 14:07:56 +01:00
|
|
|
this.httpConfig = this.configServer.getConfig(ConfigTypes.HTTP);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
2023-07-12 19:50:33 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get path to aki data
|
|
|
|
* @returns path to data
|
|
|
|
*/
|
|
|
|
public getSptDataPath(): string
|
|
|
|
{
|
2023-11-13 11:14:58 -05:00
|
|
|
return (globalThis.G_RELEASE_CONFIGURATION) ?
|
|
|
|
"Aki_Data/Server/" :
|
|
|
|
"./assets/";
|
2023-07-12 19:50:33 +01:00
|
|
|
}
|
2023-11-13 11:14:58 -05:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
public async onLoad(): Promise<void>
|
|
|
|
{
|
2023-07-12 19:50:33 +01:00
|
|
|
this.filepath = this.getSptDataPath();
|
2023-11-13 11:14:58 -05:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
if (globalThis.G_RELEASE_CONFIGURATION)
|
|
|
|
{
|
2023-11-13 11:14:58 -05:00
|
|
|
try
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-15 14:07:56 +01:00
|
|
|
// Reading the dynamic SHA1 file
|
2023-03-03 15:23:46 +00:00
|
|
|
const file = "checks.dat";
|
|
|
|
const fileWithPath = `${this.filepath}${file}`;
|
|
|
|
if (this.vfs.exists(fileWithPath))
|
2023-04-15 14:07:56 +01:00
|
|
|
{
|
2023-11-13 11:14:58 -05:00
|
|
|
this.hashedFile = this.jsonUtil.deserialize(
|
|
|
|
this.encodingUtil.fromBase64(this.vfs.readFile(fileWithPath)),
|
|
|
|
file,
|
|
|
|
);
|
2023-04-15 14:07:56 +01:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
this.valid = VaildationResult.NOT_FOUND;
|
|
|
|
this.logger.debug(this.localisationService.getText("validation_not_found"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (e)
|
|
|
|
{
|
|
|
|
this.valid = VaildationResult.FAILED;
|
|
|
|
this.logger.warning(this.localisationService.getText("validation_error_decode"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
await this.hydrateDatabase(this.filepath);
|
|
|
|
|
2023-07-12 19:50:33 +01:00
|
|
|
const imageFilePath = `${this.filepath}images/`;
|
|
|
|
const directories = this.vfs.getDirs(imageFilePath);
|
|
|
|
this.loadImages(imageFilePath, directories, [
|
2023-04-15 14:07:56 +01:00
|
|
|
"/files/CONTENT/banners/",
|
|
|
|
"/files/handbook/",
|
|
|
|
"/files/Hideout/",
|
|
|
|
"/files/launcher/",
|
|
|
|
"/files/quest/icon/",
|
2023-11-13 11:14:58 -05:00
|
|
|
"/files/trader/avatar/",
|
2023-04-15 14:07:56 +01:00
|
|
|
]);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Read all json files in database folder and map into a json object
|
|
|
|
* @param filepath path to database folder
|
|
|
|
*/
|
|
|
|
protected async hydrateDatabase(filepath: string): Promise<void>
|
|
|
|
{
|
|
|
|
this.logger.info(this.localisationService.getText("importing_database"));
|
2023-11-13 11:14:58 -05:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
const dataToImport = await this.importerUtil.loadAsync<IDatabaseTables>(
|
|
|
|
`${filepath}database/`,
|
|
|
|
this.filepath,
|
2023-11-13 11:14:58 -05:00
|
|
|
(fileWithPath: string, data: string) => this.onReadValidate(fileWithPath, data),
|
2023-03-03 15:23:46 +00:00
|
|
|
);
|
|
|
|
|
2023-11-13 11:14:58 -05:00
|
|
|
const validation = (this.valid === VaildationResult.FAILED || this.valid === VaildationResult.NOT_FOUND) ?
|
|
|
|
"." :
|
|
|
|
"";
|
2023-03-03 15:23:46 +00:00
|
|
|
this.logger.info(`${this.localisationService.getText("importing_database_finish")}${validation}`);
|
|
|
|
this.databaseServer.setTables(dataToImport);
|
|
|
|
}
|
2023-11-13 11:14:58 -05:00
|
|
|
|
2023-07-15 10:45:33 +01:00
|
|
|
protected onReadValidate(fileWithPath: string, data: string): void
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
// Validate files
|
|
|
|
if (globalThis.G_RELEASE_CONFIGURATION && this.hashedFile && !this.validateFile(fileWithPath, data))
|
2023-11-13 11:14:58 -05:00
|
|
|
{
|
2023-03-03 15:23:46 +00:00
|
|
|
this.valid = VaildationResult.FAILED;
|
2023-11-13 11:14:58 -05:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public getRoute(): string
|
|
|
|
{
|
|
|
|
return "aki-database";
|
|
|
|
}
|
2023-11-13 11:14:58 -05:00
|
|
|
|
2023-07-15 10:45:33 +01:00
|
|
|
protected validateFile(filePathAndName: string, fileData: any): boolean
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-11-13 11:14:58 -05:00
|
|
|
try
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
const finalPath = filePathAndName.replace(this.filepath, "").replace(".json", "");
|
|
|
|
let tempObject;
|
|
|
|
for (const prop of finalPath.split("/"))
|
|
|
|
{
|
|
|
|
if (!tempObject)
|
2023-11-13 11:14:58 -05:00
|
|
|
{
|
2023-03-03 15:23:46 +00:00
|
|
|
tempObject = this.hashedFile[prop];
|
2023-11-13 11:14:58 -05:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
else
|
2023-11-13 11:14:58 -05:00
|
|
|
{
|
2023-03-03 15:23:46 +00:00
|
|
|
tempObject = tempObject[prop];
|
2023-11-13 11:14:58 -05:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tempObject !== this.hashUtil.generateSha1ForData(fileData))
|
|
|
|
{
|
|
|
|
this.logger.debug(this.localisationService.getText("validation_error_file", filePathAndName));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (e)
|
|
|
|
{
|
|
|
|
this.logger.warning(this.localisationService.getText("validation_error_exception", filePathAndName));
|
|
|
|
this.logger.warning(e);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-04-15 14:07:56 +01:00
|
|
|
/**
|
|
|
|
* Find and map files with image router inside a designated path
|
|
|
|
* @param filepath Path to find files in
|
|
|
|
*/
|
2023-07-12 19:50:33 +01:00
|
|
|
public loadImages(filepath: string, directories: string[], routes: string[]): void
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-15 14:07:56 +01:00
|
|
|
for (const directoryIndex in directories)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-15 14:07:56 +01:00
|
|
|
// Get all files in directory
|
|
|
|
const filesInDirectory = this.vfs.getFiles(`${filepath}${directories[directoryIndex]}`);
|
|
|
|
for (const file of filesInDirectory)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-15 14:07:56 +01:00
|
|
|
// Register each file in image router
|
2023-03-03 15:23:46 +00:00
|
|
|
const filename = this.vfs.stripExtension(file);
|
2023-04-15 14:07:56 +01:00
|
|
|
const routeKey = `${routes[directoryIndex]}${filename}`;
|
|
|
|
let imagePath = `${filepath}${directories[directoryIndex]}/${file}`;
|
|
|
|
|
|
|
|
const pathOverride = this.getImagePathOverride(imagePath);
|
|
|
|
if (pathOverride)
|
|
|
|
{
|
2023-07-12 19:50:33 +01:00
|
|
|
this.logger.debug(`overrode route: ${routeKey} endpoint: ${imagePath} with ${pathOverride}`);
|
2023-04-15 14:07:56 +01:00
|
|
|
imagePath = pathOverride;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.imageRouter.addRoute(routeKey, imagePath);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
}
|
2023-04-15 14:07:56 +01:00
|
|
|
|
|
|
|
// Map icon file separately
|
2023-03-03 15:23:46 +00:00
|
|
|
this.imageRouter.addRoute("/favicon.ico", `${filepath}icon.ico`);
|
|
|
|
}
|
2023-04-15 14:07:56 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check for a path override in the http json config file
|
|
|
|
* @param imagePath Key
|
|
|
|
* @returns override for key
|
|
|
|
*/
|
|
|
|
protected getImagePathOverride(imagePath: string): string
|
|
|
|
{
|
|
|
|
return this.httpConfig.serverImagePathOverride[imagePath];
|
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
enum VaildationResult
|
2023-11-13 11:14:58 -05:00
|
|
|
{
|
2023-10-31 16:23:00 +00:00
|
|
|
SUCCESS = 0,
|
|
|
|
FAILED = 1,
|
|
|
|
NOT_FOUND = 2,
|
2023-11-13 11:14:58 -05:00
|
|
|
UNDEFINED = 3,
|
2023-10-31 16:23:00 +00:00
|
|
|
}
|