2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 17:21:17 +00:00
|
|
|
import { SellResult } from "@spt-aki/models/eft/ragfair/IRagfairOffer";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { IRagfairConfig } from "@spt-aki/models/spt/config/IRagfairConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
2023-12-17 10:36:54 +00:00
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
2023-10-19 17:21:17 +00:00
|
|
|
import { RandomUtil } from "@spt-aki/utils/RandomUtil";
|
|
|
|
import { TimeUtil } from "@spt-aki/utils/TimeUtil";
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class RagfairSellHelper
|
|
|
|
{
|
|
|
|
protected ragfairConfig: IRagfairConfig;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
2023-12-17 10:36:54 +00:00
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
2023-11-16 21:42:06 +00:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 15:23:46 +00:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.ragfairConfig = this.configServer.getConfig(ConfigTypes.RAGFAIR);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the percent chance to sell an item based on its average listed price vs player chosen listing price
|
|
|
|
* @param averageOfferPriceRub Price of average offer in roubles
|
|
|
|
* @param playerListedPriceRub Price player listed item for in roubles
|
2023-10-10 11:03:20 +00:00
|
|
|
* @param qualityMultiplier Quality multipler of item being sold
|
2023-03-03 15:23:46 +00:00
|
|
|
* @returns percent value
|
|
|
|
*/
|
2023-11-16 21:42:06 +00:00
|
|
|
public calculateSellChance(
|
|
|
|
averageOfferPriceRub: number,
|
|
|
|
playerListedPriceRub: number,
|
|
|
|
qualityMultiplier: number,
|
|
|
|
): number
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-12-17 10:36:54 +00:00
|
|
|
const sellConfig = this.ragfairConfig.sell.chance;
|
|
|
|
|
|
|
|
// Base sell chance modified by items quality
|
|
|
|
const baseSellChancePercent = sellConfig.base * qualityMultiplier;
|
2024-02-02 13:54:07 -05:00
|
|
|
|
2023-12-17 10:36:54 +00:00
|
|
|
// Modfier gets applied twice to either penalize or incentivize over/under pricing (Probably a cleaner way to do this)
|
|
|
|
const sellModifier = (averageOfferPriceRub / playerListedPriceRub) * sellConfig.sellMultiplier;
|
|
|
|
let sellChance = Math.round((baseSellChancePercent * sellModifier) * sellModifier);
|
2024-02-02 13:54:07 -05:00
|
|
|
|
2023-12-17 10:36:54 +00:00
|
|
|
// Adjust sell chance if below config value
|
|
|
|
if (sellChance < sellConfig.minSellChancePercent)
|
|
|
|
{
|
|
|
|
sellChance = sellConfig.minSellChancePercent;
|
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-12-17 10:36:54 +00:00
|
|
|
// Adjust sell chance if above config value
|
|
|
|
if (sellChance > sellConfig.maxSellChancePercent)
|
|
|
|
{
|
|
|
|
sellChance = sellConfig.maxSellChancePercent;
|
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-12-17 10:36:54 +00:00
|
|
|
return sellChance;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-06-04 16:30:54 +01:00
|
|
|
* Get array of item count and sell time (empty array = no sell)
|
2023-03-03 15:23:46 +00:00
|
|
|
* @param sellChancePercent chance item will sell
|
|
|
|
* @param itemSellCount count of items to sell
|
|
|
|
* @returns Array of purchases of item(s) listed
|
|
|
|
*/
|
|
|
|
public rollForSale(sellChancePercent: number, itemSellCount: number): SellResult[]
|
|
|
|
{
|
|
|
|
const startTime = this.timeUtil.getTimestamp();
|
|
|
|
|
|
|
|
// Get a time in future to stop simulating sell chances at
|
2024-02-02 13:54:07 -05:00
|
|
|
const endTime = startTime
|
|
|
|
+ this.timeUtil.getHoursAsSeconds(
|
|
|
|
this.databaseServer.getTables().globals.config.RagFair.offerDurationTimeInHour,
|
|
|
|
);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
let sellTime = startTime;
|
|
|
|
let remainingCount = itemSellCount;
|
|
|
|
const result: SellResult[] = [];
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
// Value can sometimes be NaN for whatever reason, default to base chance if that happens
|
2023-11-16 21:42:06 +00:00
|
|
|
if (Number.isNaN(sellChancePercent))
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-11-16 21:42:06 +00:00
|
|
|
this.logger.warning(
|
|
|
|
`Sell chance was not a number: ${sellChancePercent}, defaulting to ${this.ragfairConfig.sell.chance.base} %`,
|
|
|
|
);
|
2023-03-03 15:23:46 +00:00
|
|
|
sellChancePercent = this.ragfairConfig.sell.chance.base;
|
|
|
|
}
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-06-13 19:33:42 +01:00
|
|
|
this.logger.debug(`Rolling to sell: ${itemSellCount} items (chance: ${sellChancePercent}%)`);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// No point rolling for a sale on a 0% chance item, exit early
|
|
|
|
if (sellChancePercent === 0)
|
|
|
|
{
|
|
|
|
return result;
|
|
|
|
}
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
while (remainingCount > 0 && sellTime < endTime)
|
|
|
|
{
|
2023-06-04 16:30:54 +01:00
|
|
|
const boughtAmount = this.randomUtil.getInt(1, remainingCount);
|
2023-03-03 15:23:46 +00:00
|
|
|
if (this.randomUtil.getChance100(sellChancePercent))
|
|
|
|
{
|
2023-06-04 16:30:54 +01:00
|
|
|
// Passed roll check, item will be sold
|
2023-12-17 10:36:54 +00:00
|
|
|
// Weight time to sell towards selling faster based on how cheap the item sold
|
|
|
|
const weighting = (100 - sellChancePercent) / 100;
|
|
|
|
let maximumTime = weighting * (this.ragfairConfig.sell.time.max * 60);
|
|
|
|
const minimumTime = this.ragfairConfig.sell.time.min * 60;
|
2024-02-02 13:54:07 -05:00
|
|
|
if (maximumTime < minimumTime)
|
|
|
|
{
|
|
|
|
maximumTime = minimumTime + 5;
|
|
|
|
}
|
2023-12-17 10:36:54 +00:00
|
|
|
// Sell time will be random between min/max
|
|
|
|
sellTime += Math.floor(Math.random() * (maximumTime - minimumTime) + minimumTime);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-11-16 21:42:06 +00:00
|
|
|
result.push({ sellTime: sellTime, amount: boughtAmount });
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-06-04 16:30:54 +01:00
|
|
|
this.logger.debug(`Offer will sell at: ${new Date(sellTime * 1000).toLocaleTimeString("en-US")}`);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
2023-06-04 16:30:54 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
this.logger.debug("Offer will not sell");
|
|
|
|
}
|
|
|
|
|
|
|
|
remainingCount -= boughtAmount;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2023-11-16 21:42:06 +00:00
|
|
|
}
|