2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
|
|
|
import { HandbookHelper } from "../helpers/HandbookHelper";
|
|
|
|
import { InventoryHelper } from "../helpers/InventoryHelper";
|
|
|
|
import { ItemHelper } from "../helpers/ItemHelper";
|
|
|
|
import { PaymentHelper } from "../helpers/PaymentHelper";
|
|
|
|
import { TraderHelper } from "../helpers/TraderHelper";
|
|
|
|
import { IPmcData } from "../models/eft/common/IPmcData";
|
|
|
|
import { Item } from "../models/eft/common/tables/IItem";
|
|
|
|
import { IItemEventRouterResponse } from "../models/eft/itemEvent/IItemEventRouterResponse";
|
|
|
|
import { IProcessBuyTradeRequestData } from "../models/eft/trade/IProcessBuyTradeRequestData";
|
|
|
|
import { IProcessSellTradeRequestData } from "../models/eft/trade/IProcessSellTradeRequestData";
|
|
|
|
import { BackendErrorCodes } from "../models/enums/BackendErrorCodes";
|
|
|
|
import { ILogger } from "../models/spt/utils/ILogger";
|
|
|
|
import { DatabaseServer } from "../servers/DatabaseServer";
|
|
|
|
import { HttpResponseUtil } from "../utils/HttpResponseUtil";
|
|
|
|
import { LocalisationService } from "./LocalisationService";
|
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class PaymentService
|
|
|
|
{
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("HttpResponseUtil") protected httpResponse: HttpResponseUtil,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("HandbookHelper") protected handbookHelper: HandbookHelper,
|
|
|
|
@inject("TraderHelper") protected traderHelper: TraderHelper,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
|
|
|
@inject("InventoryHelper") protected inventoryHelper: InventoryHelper,
|
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
@inject("PaymentHelper") protected paymentHelper: PaymentHelper
|
|
|
|
)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Take money and insert items into return to server request
|
|
|
|
* @param {IPmcData} pmcData Player profile
|
|
|
|
* @param {IProcessBuyTradeRequestData} request
|
|
|
|
* @param {string} sessionID
|
2023-10-10 13:03:20 +02:00
|
|
|
* @returns IItemEventRouterResponse
|
2023-03-03 16:23:46 +01:00
|
|
|
*/
|
|
|
|
public payMoney(pmcData: IPmcData, request: IProcessBuyTradeRequestData, sessionID: string, output: IItemEventRouterResponse): IItemEventRouterResponse
|
|
|
|
{
|
|
|
|
const trader = this.traderHelper.getTrader(request.tid, sessionID);
|
|
|
|
let currencyTpl = this.paymentHelper.getCurrency(trader.currency);
|
|
|
|
|
|
|
|
// Delete barter items(not money) from inventory
|
|
|
|
if (request.Action === "TradingConfirm")
|
|
|
|
{
|
|
|
|
for (const index in request.scheme_items)
|
|
|
|
{
|
|
|
|
const item = pmcData.Inventory.items.find(i => i._id === request.scheme_items[index].id);
|
|
|
|
if (item !== undefined)
|
|
|
|
{
|
|
|
|
if (!this.paymentHelper.isMoneyTpl(item._tpl))
|
|
|
|
{
|
|
|
|
output = this.inventoryHelper.removeItem(pmcData, item._id, sessionID, output);
|
|
|
|
request.scheme_items[index].count = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
currencyTpl = item._tpl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// prepare a price for barter
|
|
|
|
let barterPrice = 0;
|
|
|
|
barterPrice = request.scheme_items.reduce((accumulator, item) => accumulator + item.count, 0);
|
|
|
|
|
|
|
|
// Nothing to do here, since we dont need to pay money.
|
|
|
|
if (barterPrice === 0)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
this.logger.debug(this.localisationService.getText("payment-zero_price_no_payment"));
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
// Only perform if paying with currency (not barters)
|
|
|
|
if (barterPrice > 0)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
output = this.addPaymentToOutput(pmcData, currencyTpl, barterPrice, sessionID, output);
|
|
|
|
if (output.warnings.length > 0)
|
|
|
|
{
|
|
|
|
// Something failed
|
|
|
|
return output;
|
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// set current sale sum
|
|
|
|
// convert barterPrice itemTpl into RUB then convert RUB into trader currency
|
2023-10-10 13:03:20 +02:00
|
|
|
const costOfPurchaseInCurrency = (barterPrice === 0)
|
|
|
|
? this.handbookHelper.fromRUB(this.getTraderItemHandbookPriceRouble(request.item_id, request.tid), this.paymentHelper.getCurrency(trader.currency))
|
|
|
|
: this.handbookHelper.fromRUB(this.handbookHelper.inRUB(barterPrice, currencyTpl), this.paymentHelper.getCurrency(trader.currency));
|
2023-03-03 16:23:46 +01:00
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
pmcData.TradersInfo[request.tid].salesSum += costOfPurchaseInCurrency;
|
|
|
|
this.traderHelper.lvlUp(request.tid, pmcData);
|
|
|
|
|
|
|
|
this.logger.debug("Item(s) taken. Status OK.");
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
/**
|
|
|
|
* Get the item price of a specific traders assort
|
|
|
|
* @param traderAssortId Id of assort to look up
|
|
|
|
* @param traderId Id of trader with assort
|
|
|
|
* @returns Handbook rouble price of item
|
|
|
|
*/
|
|
|
|
protected getTraderItemHandbookPriceRouble(traderAssortId: string, traderId: string): number
|
|
|
|
{
|
|
|
|
const purchasedAssortItem = this.traderHelper.getTraderAssortItemByAssortId(traderId, traderAssortId);
|
|
|
|
if (!purchasedAssortItem)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const assortItemPriceRouble = this.handbookHelper.getTemplatePrice(purchasedAssortItem._tpl);
|
|
|
|
if (!assortItemPriceRouble)
|
|
|
|
{
|
|
|
|
this.logger.debug(`No item price found for ${purchasedAssortItem._tpl} on trader: ${traderId} in assort: ${traderAssortId}`);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return assortItemPriceRouble;
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
/**
|
|
|
|
* Receive money back after selling
|
|
|
|
* @param {IPmcData} pmcData
|
|
|
|
* @param {number} amount
|
|
|
|
* @param {IProcessSellTradeRequestData} body
|
|
|
|
* @param {IItemEventRouterResponse} output
|
|
|
|
* @param {string} sessionID
|
|
|
|
* @returns IItemEventRouterResponse
|
|
|
|
*/
|
|
|
|
public getMoney(pmcData: IPmcData, amount: number, body: IProcessSellTradeRequestData, output: IItemEventRouterResponse, sessionID: string): IItemEventRouterResponse
|
|
|
|
{
|
|
|
|
const trader = this.traderHelper.getTrader(body.tid, sessionID);
|
|
|
|
const currency = this.paymentHelper.getCurrency(trader.currency);
|
|
|
|
let calcAmount = this.handbookHelper.fromRUB(this.handbookHelper.inRUB(amount, currency), currency);
|
|
|
|
const maxStackSize = this.databaseServer.getTables().templates.items[currency]._props.StackMaxSize;
|
|
|
|
let skip = false;
|
|
|
|
|
|
|
|
for (const item of pmcData.Inventory.items)
|
|
|
|
{
|
|
|
|
// item is not currency
|
|
|
|
if (item._tpl !== currency)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// item is not in the stash
|
|
|
|
if (!this.isItemInStash(pmcData, item))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item.upd.StackObjectsCount < maxStackSize)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (item.upd.StackObjectsCount + calcAmount > maxStackSize)
|
|
|
|
{
|
|
|
|
// calculate difference
|
|
|
|
calcAmount -= maxStackSize - item.upd.StackObjectsCount;
|
|
|
|
item.upd.StackObjectsCount = maxStackSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
skip = true;
|
|
|
|
item.upd.StackObjectsCount = item.upd.StackObjectsCount + calcAmount;
|
|
|
|
}
|
|
|
|
|
|
|
|
output.profileChanges[sessionID].items.change.push(item);
|
|
|
|
|
|
|
|
if (skip)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!skip)
|
|
|
|
{
|
|
|
|
const request = {
|
|
|
|
items: [{
|
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
|
|
item_id: currency,
|
|
|
|
count: calcAmount
|
|
|
|
}],
|
|
|
|
tid: body.tid
|
|
|
|
};
|
|
|
|
|
2023-07-23 13:29:00 +02:00
|
|
|
output = this.inventoryHelper.addItem(pmcData, request, output, sessionID, null, false, null, true);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// set current sale sum
|
|
|
|
const saleSum = pmcData.TradersInfo[body.tid].salesSum + amount;
|
|
|
|
|
|
|
|
pmcData.TradersInfo[body.tid].salesSum = saleSum;
|
2023-10-10 13:03:20 +02:00
|
|
|
this.traderHelper.lvlUp(body.tid, pmcData);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively checks if the given item is
|
|
|
|
* inside the stash, that is it has the stash as
|
|
|
|
* ancestor with slotId=hideout
|
|
|
|
*/
|
|
|
|
protected isItemInStash(pmcData: IPmcData, item: Item): boolean
|
|
|
|
{
|
|
|
|
let container = item;
|
|
|
|
|
|
|
|
while ("parentId" in container)
|
|
|
|
{
|
|
|
|
if (container.parentId === pmcData.Inventory.stash && container.slotId === "hideout")
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
container = pmcData.Inventory.items.find(i => i._id === container.parentId);
|
|
|
|
if (!container)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove currency from player stash/inventory
|
|
|
|
* @param pmcData Player profile to find and remove currency from
|
|
|
|
* @param currencyTpl Type of currency to pay
|
|
|
|
* @param amountToPay money value to pay
|
|
|
|
* @param sessionID Session id
|
|
|
|
* @param output output object to send to client
|
|
|
|
* @returns IItemEventRouterResponse
|
|
|
|
*/
|
|
|
|
public addPaymentToOutput(pmcData: IPmcData, currencyTpl: string, amountToPay: number, sessionID: string, output: IItemEventRouterResponse): IItemEventRouterResponse
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
const moneyItemsInInventory = this.getSortedMoneyItemsInInventory(pmcData, currencyTpl, pmcData.Inventory.stash);
|
2023-03-03 16:23:46 +01:00
|
|
|
const amountAvailable = moneyItemsInInventory.reduce((accumulator, item) => accumulator + item.upd.StackObjectsCount, 0);
|
|
|
|
|
2023-04-24 14:23:50 +02:00
|
|
|
// If no money in inventory or amount is not enough we return false
|
2023-03-03 16:23:46 +01:00
|
|
|
if (moneyItemsInInventory.length <= 0 || amountAvailable < amountToPay)
|
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("payment-not_enough_money_to_complete_transation", {amountToPay: amountToPay, amountAvailable: amountAvailable}));
|
|
|
|
output = this.httpResponse.appendErrorToOutput(output, this.localisationService.getText("payment-not_enough_money_to_complete_transation_short"), BackendErrorCodes.UNKNOWN_TRADING_ERROR);
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
let leftToPay = amountToPay;
|
|
|
|
for (const moneyItem of moneyItemsInInventory)
|
|
|
|
{
|
|
|
|
const itemAmount = moneyItem.upd.StackObjectsCount;
|
|
|
|
if (leftToPay >= itemAmount)
|
|
|
|
{
|
|
|
|
leftToPay -= itemAmount;
|
|
|
|
output = this.inventoryHelper.removeItem(pmcData, moneyItem._id, sessionID, output);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
moneyItem.upd.StackObjectsCount -= leftToPay;
|
|
|
|
leftToPay = 0;
|
|
|
|
output.profileChanges[sessionID].items.change.push(moneyItem);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (leftToPay === 0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2023-04-24 14:23:50 +02:00
|
|
|
/**
|
|
|
|
* Get all money stacks in inventory and prioritse items in stash
|
|
|
|
* @param pmcData
|
|
|
|
* @param currencyTpl
|
2023-10-10 13:03:20 +02:00
|
|
|
* @param playerStashId Players stash id
|
2023-04-24 14:23:50 +02:00
|
|
|
* @returns Sorting money items
|
|
|
|
*/
|
2023-10-10 13:03:20 +02:00
|
|
|
protected getSortedMoneyItemsInInventory(pmcData: IPmcData, currencyTpl: string, playerStashId: string): Item[]
|
2023-04-24 14:23:50 +02:00
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
const moneyItemsInInventory = this.itemHelper.findBarterItems("tpl", pmcData.Inventory.items, currencyTpl);
|
2023-04-24 14:23:50 +02:00
|
|
|
|
|
|
|
// Prioritise items in stash to top of array
|
2023-10-10 13:03:20 +02:00
|
|
|
moneyItemsInInventory.sort((a, b) => this.prioritiseStashSort(a, b, pmcData.Inventory.items, playerStashId));
|
2023-04-24 14:23:50 +02:00
|
|
|
|
|
|
|
return moneyItemsInInventory;
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
/**
|
|
|
|
* Prioritise player stash first over player inventory
|
|
|
|
* Post-raid healing would often take money out of the players pockets/secure container
|
|
|
|
* @param a First money stack item
|
|
|
|
* @param b Second money stack item
|
|
|
|
* @param inventoryItems players inventory items
|
2023-10-10 13:03:20 +02:00
|
|
|
* @param playerStashId Players stash id
|
2023-03-03 16:23:46 +01:00
|
|
|
* @returns sort order
|
|
|
|
*/
|
2023-10-10 13:03:20 +02:00
|
|
|
protected prioritiseStashSort(a: Item, b: Item, inventoryItems: Item[], playerStashId: string): number
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
// a in stash, prioritise
|
2023-03-03 16:23:46 +01:00
|
|
|
if (a.slotId === "hideout" && b.slotId !== "hideout")
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
// b in stash, prioritise
|
2023-03-03 16:23:46 +01:00
|
|
|
if (a.slotId !== "hideout" && b.slotId === "hideout")
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// both in containers
|
|
|
|
if (a.slotId === "main" && b.slotId === "main")
|
|
|
|
{
|
|
|
|
// Item is in inventory, not stash, deprioritise
|
2023-10-10 13:03:20 +02:00
|
|
|
const aInStash = this.isInStash(a.parentId, inventoryItems, playerStashId);
|
|
|
|
const bInStash = this.isInStash(b.parentId, inventoryItems, playerStashId);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
// a in stash, prioritise
|
|
|
|
if (aInStash && !bInStash)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
// b in stash, prioritise
|
|
|
|
if (!aInStash && bInStash)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// they match
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursivly check items parents to see if it is inside the players inventory, not stash
|
|
|
|
* @param itemId item id to check
|
|
|
|
* @param inventoryItems player inventory
|
2023-10-10 13:03:20 +02:00
|
|
|
* @param playerStashId Players stash id
|
2023-03-03 16:23:46 +01:00
|
|
|
* @returns true if its in inventory
|
|
|
|
*/
|
2023-10-10 13:03:20 +02:00
|
|
|
protected isInStash(itemId: string, inventoryItems: Item[], playerStashId: string): boolean
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const itemParent = inventoryItems.find(x => x._id === itemId);
|
2023-10-10 13:03:20 +02:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
if (itemParent)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
if (itemParent.slotId === "hideout")
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (itemParent._id === playerStashId)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
return this.isInStash(itemParent.parentId, inventoryItems, playerStashId);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|