import { inject, injectable } from "tsyringe"; import { FenceBaseAssortGenerator } from "../generators/FenceBaseAssortGenerator"; import { ProfileHelper } from "../helpers/ProfileHelper"; import { TraderAssortHelper } from "../helpers/TraderAssortHelper"; import { TraderHelper } from "../helpers/TraderHelper"; import { ITraderAssort, ITraderBase } from "../models/eft/common/tables/ITrader"; import { Traders } from "../models/enums/Traders"; import { ILogger } from "../models/spt/utils/ILogger"; import { DatabaseServer } from "../servers/DatabaseServer"; import { FenceService } from "../services/FenceService"; import { TraderAssortService } from "../services/TraderAssortService"; import { TraderPurchasePersisterService } from "../services/TraderPurchasePersisterService"; import { JsonUtil } from "../utils/JsonUtil"; @injectable() export class TraderController { constructor( @inject("WinstonLogger") protected logger: ILogger, @inject("DatabaseServer") protected databaseServer: DatabaseServer, @inject("TraderAssortHelper") protected traderAssortHelper: TraderAssortHelper, @inject("ProfileHelper") protected profileHelper: ProfileHelper, @inject("TraderHelper") protected traderHelper: TraderHelper, @inject("TraderAssortService") protected traderAssortService: TraderAssortService, @inject("TraderPurchasePersisterService") protected traderPurchasePersisterService: TraderPurchasePersisterService, @inject("FenceService") protected fenceService: FenceService, @inject("FenceBaseAssortGenerator") protected fenceBaseAssortGenerator: FenceBaseAssortGenerator, @inject("JsonUtil") protected jsonUtil: JsonUtil ) { } /** * Runs when onLoad event is fired * Iterate over traders, ensure an unmolested copy of their assorts is stored in traderAssortService * Store timestamp of next assort refresh in nextResupply property of traders .base object */ public load(): void { for (const traderId in this.databaseServer.getTables().traders) { if (traderId === "ragfair" || traderId === Traders.LIGHTHOUSEKEEPER) { continue; } if (traderId === Traders.FENCE) { this.fenceBaseAssortGenerator.generateFenceBaseAssorts(); this.fenceService.generateFenceAssorts(); continue; } const trader = this.databaseServer.getTables().traders[traderId]; // Create dict of trader assorts on server start if (!this.traderAssortService.getPristineTraderAssort(traderId)) { const assorts = this.jsonUtil.clone(trader.assort); this.traderAssortService.setPristineTraderAssort(traderId, assorts); } this.traderPurchasePersisterService.removeStalePurchasesFromProfiles(traderId); trader.base.nextResupply = this.traderHelper.getNextUpdateTimestamp(trader.base._id); this.databaseServer.getTables().traders[trader.base._id].base = trader.base; } } /** * Runs when onUpdate is fired * If current time is > nextResupply(expire) time of trader, refresh traders assorts and * Fence is handled slightly differently * @returns has run */ public update(): boolean { for (const traderId in this.databaseServer.getTables().traders) { if (traderId === "ragfair" || traderId === Traders.LIGHTHOUSEKEEPER) { continue; } if (traderId === Traders.FENCE) { if (this.fenceService.needsPartialRefresh()) { this.fenceService.performPartialRefresh(); } continue; } const trader = this.databaseServer.getTables().traders[traderId]; // trader needs to be refreshed if (this.traderAssortHelper.traderAssortsHaveExpired(traderId)) { this.traderAssortHelper.resetExpiredTrader(trader); // Reset purchase data per trader as they have independent reset times this.traderPurchasePersisterService.resetTraderPurchasesStoredInProfile(trader.base._id); } } return true; } /** * Handle client/trading/api/traderSettings * Return an array of all traders * @param sessionID Session id * @returns array if ITraderBase objects */ public getAllTraders(sessionID: string): ITraderBase[] { const traders: ITraderBase[] = []; const pmcData = this.profileHelper.getPmcProfile(sessionID); for (const traderID in this.databaseServer.getTables().traders) { if (this.databaseServer.getTables().traders[traderID].base._id === "ragfair") { continue; } traders.push(this.traderHelper.getTrader(traderID, sessionID)); if (pmcData.Info) { this.traderHelper.lvlUp(traderID, pmcData); } } return traders.sort((a, b) => this.sortByTraderId(a,b)); } /** * Order traders by their traderId (Ttid) * @param traderA First trader to compare * @param traderB Second trader to compare * @returns 1,-1 or 0 */ protected sortByTraderId(traderA: ITraderBase, traderB: ITraderBase): number { if (traderA._id > traderB._id) { return 1; } if (traderA._id < traderB._id) { return -1; } return 0; } /** Handle client/trading/api/getTrader */ public getTrader(sessionID: string, traderID: string): ITraderBase { return this.traderHelper.getTrader(sessionID, traderID); } /** Handle client/trading/api/getTraderAssort */ public getAssort(sessionId: string, traderId: string): ITraderAssort { return this.traderAssortHelper.getAssort(sessionId, traderId); } }