import { inject, injectable } from "tsyringe"; import { HandbookHelper } from "@spt/helpers/HandbookHelper"; import { ItemHelper } from "@spt/helpers/ItemHelper"; import { ProfileHelper } from "@spt/helpers/ProfileHelper"; import { IPmcData } from "@spt/models/eft/common/IPmcData"; import { Item } from "@spt/models/eft/common/tables/IItem"; import { ProfileTraderTemplate } from "@spt/models/eft/common/tables/IProfileTemplate"; import { ITraderAssort, ITraderBase, LoyaltyLevel } from "@spt/models/eft/common/tables/ITrader"; import { ConfigTypes } from "@spt/models/enums/ConfigTypes"; import { Money } from "@spt/models/enums/Money"; import { Traders } from "@spt/models/enums/Traders"; import { ITraderConfig } from "@spt/models/spt/config/ITraderConfig"; import { ILogger } from "@spt/models/spt/utils/ILogger"; import { ConfigServer } from "@spt/servers/ConfigServer"; import { DatabaseServer } from "@spt/servers/DatabaseServer"; import { SaveServer } from "@spt/servers/SaveServer"; import { FenceService } from "@spt/services/FenceService"; import { LocalisationService } from "@spt/services/LocalisationService"; import { PlayerService } from "@spt/services/PlayerService"; import { RandomUtil } from "@spt/utils/RandomUtil"; import { TimeUtil } from "@spt/utils/TimeUtil"; @injectable() export class TraderHelper { protected traderConfig: ITraderConfig; /** Dictionary of item tpl and the highest trader sell rouble price */ protected highestTraderPriceItems: Record = null; /** Dictionary of item tpl and the highest trader buy back rouble price */ protected highestTraderBuyPriceItems: Record = null; constructor( @inject("WinstonLogger") protected logger: ILogger, @inject("DatabaseServer") protected databaseServer: DatabaseServer, @inject("SaveServer") protected saveServer: SaveServer, @inject("ProfileHelper") protected profileHelper: ProfileHelper, @inject("HandbookHelper") protected handbookHelper: HandbookHelper, @inject("ItemHelper") protected itemHelper: ItemHelper, @inject("PlayerService") protected playerService: PlayerService, @inject("LocalisationService") protected localisationService: LocalisationService, @inject("FenceService") protected fenceService: FenceService, @inject("TimeUtil") protected timeUtil: TimeUtil, @inject("RandomUtil") protected randomUtil: RandomUtil, @inject("ConfigServer") protected configServer: ConfigServer, ) { this.traderConfig = this.configServer.getConfig(ConfigTypes.TRADER); } /** * Get a trader base object, update profile to reflect players current standing in profile * when trader not found in profile * @param traderID Traders Id to get * @param sessionID Players id * @returns Trader base */ public getTrader(traderID: string, sessionID: string): ITraderBase { const pmcData = this.profileHelper.getPmcProfile(sessionID); if (!pmcData) { this.logger.error(this.localisationService.getText("trader-unable_to_find_profile_with_id", sessionID)); } // Profile has traderInfo dict (profile beyond creation stage) but no requested trader in profile if (pmcData.TradersInfo && !(traderID in pmcData.TradersInfo)) { // Add trader values to profile this.resetTrader(sessionID, traderID); this.lvlUp(traderID, pmcData); } const trader = this.databaseServer.getTables().traders?.[traderID]?.base; if (!trader) { this.logger.error(this.localisationService.getText("trader-unable_to_find_trader_by_id", traderID)); } return trader; } /** * Get all assort data for a particular trader * @param traderId Trader to get assorts for * @returns ITraderAssort */ public getTraderAssortsByTraderId(traderId: string): ITraderAssort { return traderId === Traders.FENCE ? this.fenceService.getRawFenceAssorts() : this.databaseServer.getTables().traders[traderId].assort; } /** * Retrieve the Item from a traders assort data by its id * @param traderId Trader to get assorts for * @param assortId Id of assort to find * @returns Item object */ public getTraderAssortItemByAssortId(traderId: string, assortId: string): Item { const traderAssorts = this.getTraderAssortsByTraderId(traderId); if (!traderAssorts) { this.logger.debug(`No assorts on trader: ${traderId} found`); return null; } // Find specific assort in traders data const purchasedAssort = traderAssorts.items.find((item) => item._id === assortId); if (!purchasedAssort) { this.logger.debug(`No assort ${assortId} on trader: ${traderId} found`); return null; } return purchasedAssort; } /** * Reset a profiles trader data back to its initial state as seen by a level 1 player * Does NOT take into account different profile levels * @param sessionID session id of player * @param traderID trader id to reset */ public resetTrader(sessionID: string, traderID: string): void { const account = this.saveServer.getProfile(sessionID); const pmcData = this.profileHelper.getPmcProfile(sessionID); const rawProfileTemplate: ProfileTraderTemplate = this.databaseServer.getTables().templates.profiles[account.info.edition][pmcData.Info.Side.toLowerCase()] .trader; pmcData.TradersInfo[traderID] = { disabled: false, loyaltyLevel: rawProfileTemplate.initialLoyaltyLevel[traderID] ?? 1, salesSum: rawProfileTemplate.initialSalesSum, standing: this.getStartingStanding(traderID, rawProfileTemplate), nextResupply: this.databaseServer.getTables().traders[traderID].base.nextResupply, unlocked: this.databaseServer.getTables().traders[traderID].base.unlockedByDefault, }; if (traderID === Traders.JAEGER) { pmcData.TradersInfo[traderID].unlocked = rawProfileTemplate.jaegerUnlocked; } } /** * Get the starting standing of a trader based on the current profiles type (e.g. EoD, Standard etc) * @param traderId Trader id to get standing for * @param rawProfileTemplate Raw profile from profiles.json to look up standing from * @returns Standing value */ protected getStartingStanding(traderId: string, rawProfileTemplate: ProfileTraderTemplate): number { // Edge case for Lightkeeper, 0 standing means seeing `Make Amends - Buyout` quest if (traderId === Traders.LIGHTHOUSEKEEPER && rawProfileTemplate.initialStanding === 0) { return 0.01; } return rawProfileTemplate.initialStanding; } /** * Alter a traders unlocked status * @param traderId Trader to alter * @param status New status to use * @param sessionId Session id of player */ public setTraderUnlockedState(traderId: string, status: boolean, sessionId: string): void { const pmcData = this.profileHelper.getPmcProfile(sessionId); pmcData.TradersInfo[traderId].unlocked = status; } /** * Add standing to a trader and level them up if exp goes over level threshold * @param sessionId Session id of player * @param traderId Traders id to add standing to * @param standingToAdd Standing value to add to trader */ public addStandingToTrader(sessionId: string, traderId: string, standingToAdd: number): void { const fullProfile = this.profileHelper.getFullProfile(sessionId); const pmcTraderInfo = fullProfile.characters.pmc.TradersInfo[traderId]; // Add standing to trader pmcTraderInfo.standing = this.addStandingValuesTogether(pmcTraderInfo.standing, standingToAdd); if (traderId === Traders.FENCE) { // Must add rep to scav profile to ensure consistency fullProfile.characters.scav.TradersInfo[traderId].standing = pmcTraderInfo.standing; } this.lvlUp(traderId, fullProfile.characters.pmc); } /** * Add standing to current standing and clamp value if it goes too low * @param currentStanding current trader standing * @param standingToAdd stansding to add to trader standing * @returns current standing + added standing (clamped if needed) */ protected addStandingValuesTogether(currentStanding: number, standingToAdd: number): number { const newStanding = currentStanding + standingToAdd; // Never let standing fall below 0 return newStanding < 0 ? 0 : newStanding; } /** * Calculate traders level based on exp amount and increments level if over threshold * @param traderID trader to check standing of * @param pmcData profile to update trader in */ public lvlUp(traderID: string, pmcData: IPmcData): void { const loyaltyLevels = this.databaseServer.getTables().traders[traderID].base.loyaltyLevels; // Level up player pmcData.Info.Level = this.playerService.calculateLevel(pmcData); // Level up traders let targetLevel = 0; // Round standing to 2 decimal places to address floating point inaccuracies pmcData.TradersInfo[traderID].standing = Math.round(pmcData.TradersInfo[traderID].standing * 100) / 100; for (const level in loyaltyLevels) { const loyalty = loyaltyLevels[level]; if ( loyalty.minLevel <= pmcData.Info.Level && loyalty.minSalesSum <= pmcData.TradersInfo[traderID].salesSum && loyalty.minStanding <= pmcData.TradersInfo[traderID].standing && targetLevel < 4 ) { // level reached targetLevel++; } } // set level pmcData.TradersInfo[traderID].loyaltyLevel = targetLevel; } /** * Get the next update timestamp for a trader * @param traderID Trader to look up update value for * @returns future timestamp */ public getNextUpdateTimestamp(traderID: string): number { const time = this.timeUtil.getTimestamp(); const updateSeconds = this.getTraderUpdateSeconds(traderID); return time + updateSeconds; } /** * Get the reset time between trader assort refreshes in seconds * @param traderId Trader to look up * @returns Time in seconds */ public getTraderUpdateSeconds(traderId: string): number { const traderDetails = this.traderConfig.updateTime.find((x) => x.traderId === traderId); if (!traderDetails || traderDetails.seconds.min === undefined || traderDetails.seconds.max === undefined) { this.logger.warning( this.localisationService.getText("trader-missing_trader_details_using_default_refresh_time", { traderId: traderId, updateTime: this.traderConfig.updateTimeDefault, }), ); this.traderConfig.updateTime.push( // create temporary entry to prevent logger spam { traderId: traderId, seconds: { min: this.traderConfig.updateTimeDefault, max: this.traderConfig.updateTimeDefault }, }, ); } else { return this.randomUtil.getInt(traderDetails.seconds.min, traderDetails.seconds.max); } } public getLoyaltyLevel(traderID: string, pmcData: IPmcData): LoyaltyLevel { const trader = this.databaseServer.getTables().traders[traderID].base; let loyaltyLevel = pmcData.TradersInfo[traderID].loyaltyLevel; if (!loyaltyLevel || loyaltyLevel < 1) { loyaltyLevel = 1; } if (loyaltyLevel > trader.loyaltyLevels.length) { loyaltyLevel = trader.loyaltyLevels.length; } return trader.loyaltyLevels[loyaltyLevel - 1]; } /** * Store the purchase of an assort from a trader in the player profile * @param sessionID Session id * @param newPurchaseDetails New item assort id + count */ public addTraderPurchasesToPlayerProfile( sessionID: string, newPurchaseDetails: { items: { itemId: string, count: number }[], traderId: string }, itemPurchased: Item, ): void { const profile = this.profileHelper.getFullProfile(sessionID); const traderId = newPurchaseDetails.traderId; // Iterate over assorts bought and add to profile for (const purchasedItem of newPurchaseDetails.items) { if (!profile.traderPurchases) { profile.traderPurchases = {}; } if (!profile.traderPurchases[traderId]) { profile.traderPurchases[traderId] = {}; } // Null guard when dict doesnt exist const currentTime = this.timeUtil.getTimestamp(); if (!profile.traderPurchases[traderId][purchasedItem.itemId]) { profile.traderPurchases[traderId][purchasedItem.itemId] = { count: purchasedItem.count, purchaseTimestamp: currentTime, }; continue; } if ( profile.traderPurchases[traderId][purchasedItem.itemId].count + purchasedItem.count > itemPurchased.upd.BuyRestrictionMax ) { throw new Error( this.localisationService.getText("trader-unable_to_purchase_item_limit_reached", { traderId: traderId, limit: itemPurchased.upd.BuyRestrictionMax, }), ); } profile.traderPurchases[traderId][purchasedItem.itemId].count += purchasedItem.count; profile.traderPurchases[traderId][purchasedItem.itemId].purchaseTimestamp = currentTime; } } /** * Get the highest rouble price for an item from traders * UNUSED * @param tpl Item to look up highest pride for * @returns highest rouble cost for item */ public getHighestTraderPriceRouble(tpl: string): number { if (this.highestTraderPriceItems) { return this.highestTraderPriceItems[tpl]; } if (!this.highestTraderPriceItems) { this.highestTraderPriceItems = {}; } // Init dict and fill for (const traderName in Traders) { // Skip some traders if (traderName === Traders.FENCE) { continue; } // Get assorts for trader, skip trader if no assorts found const traderAssorts = this.databaseServer.getTables().traders[Traders[traderName]].assort; if (!traderAssorts) { continue; } // Get all item assorts that have parentid of hideout (base item and not a mod of other item) for (const item of traderAssorts.items.filter((x) => x.parentId === "hideout")) { // Get barter scheme (contains cost of item) const barterScheme = traderAssorts.barter_scheme[item._id][0][0]; // Convert into roubles const roubleAmount = barterScheme._tpl === Money.ROUBLES ? barterScheme.count : this.handbookHelper.inRUB(barterScheme.count, barterScheme._tpl); // Existing price smaller in dict than current iteration, overwrite if (this.highestTraderPriceItems[item._tpl] ?? 0 < roubleAmount) { this.highestTraderPriceItems[item._tpl] = roubleAmount; } } } return this.highestTraderPriceItems[tpl]; } /** * Get the highest price item can be sold to trader for (roubles) * @param tpl Item to look up best trader sell-to price * @returns Rouble price */ public getHighestSellToTraderPrice(tpl: string): number { // Init dict if doesn't exist if (!this.highestTraderBuyPriceItems) { this.highestTraderBuyPriceItems = {}; } // Return result if it exists if (this.highestTraderBuyPriceItems[tpl]) { return this.highestTraderBuyPriceItems[tpl]; } // Find highest trader price for item for (const traderName in Traders) { // Get trader and check buy category allows tpl const traderBase = this.databaseServer.getTables().traders[Traders[traderName]]?.base; if (traderBase && this.itemHelper.isOfBaseclasses(tpl, traderBase.items_buy.category)) { // Get loyalty level details player has achieved with this trader // Uses lowest loyalty level as this function is used before a player has logged into server - we have no idea what player loyalty is with traders const relevantLoyaltyData = traderBase.loyaltyLevels[0]; const traderBuyBackPricePercent = relevantLoyaltyData.buy_price_coef; const itemHandbookPrice = this.handbookHelper.getTemplatePrice(tpl); const priceTraderBuysItemAt = Math.round( this.randomUtil.getPercentOfValue(traderBuyBackPricePercent, itemHandbookPrice), ); // Set new item to 1 rouble as default if (!this.highestTraderBuyPriceItems[tpl]) { this.highestTraderBuyPriceItems[tpl] = 1; } // Existing price smaller in dict than current iteration, overwrite if (this.highestTraderBuyPriceItems[tpl] < priceTraderBuysItemAt) { this.highestTraderBuyPriceItems[tpl] = priceTraderBuysItemAt; } } } return this.highestTraderBuyPriceItems[tpl]; } /** * Get a trader enum key by its value * @param traderId Traders id * @returns Traders key */ public getTraderById(traderId: string): Traders { const keys = Object.keys(Traders).filter((x) => Traders[x] === traderId); if (keys.length === 0) { this.logger.error(this.localisationService.getText("trader-unable_to_find_trader_in_enum", traderId)); return null; } return keys[0] as Traders; } /** * Validates that the provided traderEnumValue exists in the Traders enum. If the value is valid, it returns the * same enum value, effectively serving as a trader ID; otherwise, it logs an error and returns an empty string. * This method provides a runtime check to prevent undefined behavior when using the enum as a dictionary key. * * For example, instead of this: * `const traderId = Traders[Traders.PRAPOR];` * * You can use safely use this: * `const traderId = this.traderHelper.getValidTraderIdByEnumValue(Traders.PRAPOR);` * * @param traderEnumValue The trader enum value to validate * @returns The validated trader enum value as a string, or an empty string if invalid */ public getValidTraderIdByEnumValue(traderEnumValue: Traders): string { if (!this.traderEnumHasKey(traderEnumValue)) { this.logger.error(this.localisationService.getText("trader-unable_to_find_trader_in_enum", traderEnumValue)); return ""; } return Traders[traderEnumValue]; } /** * Does the 'Traders' enum has a value that matches the passed in parameter * @param key Value to check for * @returns True, values exists in Traders enum as a value */ public traderEnumHasKey(key: string): boolean { return Object.keys(Traders).some((x) => x === key); } /** * Accepts a trader id * @param traderId Trader id * @returns Ttrue if Traders enum has the param as a value */ public traderEnumHasValue(traderId: string): boolean { return Object.values(Traders).some((x) => x === traderId); } }