import { inject, injectable } from "tsyringe"; import { BotWeaponGenerator } from "@spt-aki/generators/BotWeaponGenerator"; import { BotGeneratorHelper } from "@spt-aki/helpers/BotGeneratorHelper"; import { BotWeaponGeneratorHelper } from "@spt-aki/helpers/BotWeaponGeneratorHelper"; import { HandbookHelper } from "@spt-aki/helpers/HandbookHelper"; import { ItemHelper } from "@spt-aki/helpers/ItemHelper"; import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper"; import { Inventory as PmcInventory } from "@spt-aki/models/eft/common/tables/IBotBase"; import { IBotType, Inventory, ModsChances } from "@spt-aki/models/eft/common/tables/IBotType"; import { Item } from "@spt-aki/models/eft/common/tables/IItem"; import { ITemplateItem } from "@spt-aki/models/eft/common/tables/ITemplateItem"; import { BaseClasses } from "@spt-aki/models/enums/BaseClasses"; import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes"; import { EquipmentSlots } from "@spt-aki/models/enums/EquipmentSlots"; import { ItemAddedResult } from "@spt-aki/models/enums/ItemAddedResult"; import { LootCacheType } from "@spt-aki/models/spt/bots/IBotLootCache"; import { IBotConfig } from "@spt-aki/models/spt/config/IBotConfig"; import { IPmcConfig } from "@spt-aki/models/spt/config/IPmcConfig"; import { ILogger } from "@spt-aki/models/spt/utils/ILogger"; import { ConfigServer } from "@spt-aki/servers/ConfigServer"; import { DatabaseServer } from "@spt-aki/servers/DatabaseServer"; import { BotLootCacheService } from "@spt-aki/services/BotLootCacheService"; import { LocalisationService } from "@spt-aki/services/LocalisationService"; import { HashUtil } from "@spt-aki/utils/HashUtil"; import { RandomUtil } from "@spt-aki/utils/RandomUtil"; @injectable() export class BotLootGenerator { protected botConfig: IBotConfig; protected pmcConfig: IPmcConfig; constructor( @inject("WinstonLogger") protected logger: ILogger, @inject("HashUtil") protected hashUtil: HashUtil, @inject("RandomUtil") protected randomUtil: RandomUtil, @inject("ItemHelper") protected itemHelper: ItemHelper, @inject("DatabaseServer") protected databaseServer: DatabaseServer, @inject("HandbookHelper") protected handbookHelper: HandbookHelper, @inject("BotGeneratorHelper") protected botGeneratorHelper: BotGeneratorHelper, @inject("BotWeaponGenerator") protected botWeaponGenerator: BotWeaponGenerator, @inject("BotWeaponGeneratorHelper") protected botWeaponGeneratorHelper: BotWeaponGeneratorHelper, @inject("WeightedRandomHelper") protected weightedRandomHelper: WeightedRandomHelper, @inject("BotLootCacheService") protected botLootCacheService: BotLootCacheService, @inject("LocalisationService") protected localisationService: LocalisationService, @inject("ConfigServer") protected configServer: ConfigServer, ) { this.botConfig = this.configServer.getConfig(ConfigTypes.BOT); this.pmcConfig = this.configServer.getConfig(ConfigTypes.PMC); } /** * Add loot to bots containers * @param sessionId Session id * @param botJsonTemplate Base json db file for the bot having its loot generated * @param isPmc Will bot be a pmc * @param botRole Role of bot, e.g. asssult * @param botInventory Inventory to add loot to * @param botLevel Level of bot */ public generateLoot( sessionId: string, botJsonTemplate: IBotType, isPmc: boolean, botRole: string, botInventory: PmcInventory, botLevel: number, ): void { // Limits on item types to be added as loot const itemCounts = botJsonTemplate.generation.items; const backpackLootCount = Number( this.weightedRandomHelper.getWeightedValue(itemCounts.backpackLoot.weights), ); const pocketLootCount = Number( this.weightedRandomHelper.getWeightedValue(itemCounts.pocketLoot.weights), ); const vestLootCount = this.weightedRandomHelper.getWeightedValue(itemCounts.vestLoot.weights); const specialLootItemCount = Number( this.weightedRandomHelper.getWeightedValue(itemCounts.specialItems.weights), ); const healingItemCount = Number(this.weightedRandomHelper.getWeightedValue(itemCounts.healing.weights)); const drugItemCount = Number(this.weightedRandomHelper.getWeightedValue(itemCounts.drugs.weights)); const stimItemCount = Number(this.weightedRandomHelper.getWeightedValue(itemCounts.stims.weights)); const grenadeCount = Number(this.weightedRandomHelper.getWeightedValue(itemCounts.grenades.weights)); // Forced pmc healing loot if (isPmc && this.pmcConfig.forceHealingItemsIntoSecure) { this.addForcedMedicalItemsToPmcSecure(botInventory, botRole); } const containersBotHasAvailable = this.getAvailableContainersBotCanStoreItemsIn(botInventory); // Special items this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.SPECIAL, botJsonTemplate), containersBotHasAvailable, specialLootItemCount, botInventory, botRole, true, ); // Healing items / Meds this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.HEALING_ITEMS, botJsonTemplate), containersBotHasAvailable, healingItemCount, botInventory, botRole, false, 0, isPmc, ); // Drugs this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.DRUG_ITEMS, botJsonTemplate), containersBotHasAvailable, drugItemCount, botInventory, botRole, false, 0, isPmc, ); // Stims this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.STIM_ITEMS, botJsonTemplate), containersBotHasAvailable, stimItemCount, botInventory, botRole, true, 0, isPmc, ); // Grenades this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.GRENADE_ITEMS, botJsonTemplate), [EquipmentSlots.POCKETS, EquipmentSlots.TACTICAL_VEST], // Can't use containersBotHasEquipped as we dont want grenades added to backpack grenadeCount, botInventory, botRole, false, 0, isPmc, ); // Backpack - generate loot if they have one if (containersBotHasAvailable.includes(EquipmentSlots.BACKPACK)) { // Add randomly generated weapon to PMC backpacks if (isPmc && this.randomUtil.getChance100(this.pmcConfig.looseWeaponInBackpackChancePercent)) { this.addLooseWeaponsToInventorySlot( sessionId, botInventory, EquipmentSlots.BACKPACK, botJsonTemplate.inventory, botJsonTemplate.chances.weaponMods, botRole, isPmc, botLevel, ); } this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.BACKPACK, botJsonTemplate), [EquipmentSlots.BACKPACK], backpackLootCount, botInventory, botRole, true, this.pmcConfig.maxBackpackLootTotalRub, isPmc, ); } // TacticalVest - generate loot if they have one if (containersBotHasAvailable.includes(EquipmentSlots.TACTICAL_VEST)) { // Vest this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.VEST, botJsonTemplate), [EquipmentSlots.TACTICAL_VEST], vestLootCount, botInventory, botRole, true, this.pmcConfig.maxVestLootTotalRub, isPmc, ); } // Pockets this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.POCKET, botJsonTemplate), [EquipmentSlots.POCKETS], pocketLootCount, botInventory, botRole, true, this.pmcConfig.maxPocketLootTotalRub, isPmc, ); // Secure this.addLootFromPool( this.botLootCacheService.getLootFromCache(botRole, isPmc, LootCacheType.SECURE, botJsonTemplate), [EquipmentSlots.SECURED_CONTAINER], 50, botInventory, botRole, false, -1, isPmc, ); } /** * Get an array of the containers a bot has on them (pockets/backpack/vest) * @param botInventory Bot to check * @returns Array of available slots */ protected getAvailableContainersBotCanStoreItemsIn(botInventory: PmcInventory): EquipmentSlots[] { const result = [EquipmentSlots.POCKETS]; if (botInventory.items.find((item) => item.slotId === EquipmentSlots.TACTICAL_VEST)) { result.push(EquipmentSlots.TACTICAL_VEST); } if (botInventory.items.find((item) => item.slotId === EquipmentSlots.BACKPACK)) { result.push(EquipmentSlots.BACKPACK); } return result; } /** * Force healing items onto bot to ensure they can heal in-raid * @param botInventory Inventory to add items to * @param botRole Role of bot (sptBear/sptUsec) */ protected addForcedMedicalItemsToPmcSecure(botInventory: PmcInventory, botRole: string): void { // Grizzly this.addLootFromPool( // eslint-disable-next-line @typescript-eslint/naming-convention { "590c657e86f77412b013051d": 1 }, [EquipmentSlots.SECURED_CONTAINER], 1, botInventory, botRole, false, 0, true, ); // surv12 this.addLootFromPool( // eslint-disable-next-line @typescript-eslint/naming-convention { "5d02797c86f774203f38e30a": 1 }, [EquipmentSlots.SECURED_CONTAINER], 1, botInventory, botRole, false, 0, true, ); // Morphine this.addLootFromPool( // eslint-disable-next-line @typescript-eslint/naming-convention { "544fb3f34bdc2d03748b456a": 1 }, [EquipmentSlots.SECURED_CONTAINER], 2, botInventory, botRole, false, 0, true, ); // AFAK this.addLootFromPool( // eslint-disable-next-line @typescript-eslint/naming-convention { "60098ad7c2240c0fe85c570a": 1 }, [EquipmentSlots.SECURED_CONTAINER], 2, botInventory, botRole, false, 0, true, ); } /** * Get a biased random number * @param min Smallest size * @param max Biggest size * @param nValue Value to bias choice * @returns Chosen number */ protected getRandomisedCount(min: number, max: number, nValue: number): number { const range = max - min; return this.randomUtil.getBiasedRandomNumber(min, max, range, nValue); } /** * Take random items from a pool and add to an inventory until totalItemCount or totalValueLimit or space limit is reached * @param pool Pool of items to pick from with weight * @param equipmentSlots What equipment slot will the loot items be added to * @param totalItemCount Max count of items to add * @param inventoryToAddItemsTo Bot inventory loot will be added to * @param botRole Role of the bot loot is being generated for (assault/pmcbot) * @param useLimits Should item limit counts be used as defined in config/bot.json * @param totalValueLimitRub Total value of loot allowed in roubles * @param isPmc Is bot being generated for a pmc */ protected addLootFromPool( pool: Record, equipmentSlots: string[], totalItemCount: number, inventoryToAddItemsTo: PmcInventory, botRole: string, useLimits = false, totalValueLimitRub = 0, isPmc = false, ): void { // Loot pool has items if (Object.keys(pool).length > 0) { let currentTotalRub = 0; const itemLimits: Record = {}; /** Prep limits for items added to the container */ const itemSpawnLimits: Record> = {}; let fitItemIntoContainerAttempts = 0; for (let i = 0; i < totalItemCount; i++) { const weightedItemTpl = this.weightedRandomHelper.getWeightedValue(pool); const itemResult = this.itemHelper.getItem(weightedItemTpl); const itemToAddTemplate = itemResult[1]; if (!itemResult[0]) { this.logger.warning( `Unable to process item tpl: ${weightedItemTpl} for slots: ${equipmentSlots} on bot: ${botRole}`, ); continue; } const newRootItemId = this.hashUtil.generate(); const itemWithChildrenToAdd: Item[] = [{ _id: newRootItemId, _tpl: itemToAddTemplate._id, ...this.botGeneratorHelper.generateExtraPropertiesForItem(itemToAddTemplate, botRole), }]; if (useLimits) { if (Object.keys(itemLimits).length === 0) { this.initItemLimitArray(isPmc, botRole, itemLimits); } if (!itemSpawnLimits[botRole]) { itemSpawnLimits[botRole] = this.getItemSpawnLimitsForBotType(isPmc, botRole); } if ( this.itemHasReachedSpawnLimit( itemToAddTemplate, botRole, isPmc, itemLimits, itemSpawnLimits[botRole], ) ) { i--; continue; } } this.addRequiredChildItemsToParent(itemToAddTemplate, itemWithChildrenToAdd, isPmc); // Attempt to add item to container(s) const itemAddedResult = this.botWeaponGeneratorHelper.addItemWithChildrenToEquipmentSlot( equipmentSlots, newRootItemId, itemToAddTemplate._id, itemWithChildrenToAdd, inventoryToAddItemsTo, ); // Handle when item cannot be added if (itemAddedResult !== ItemAddedResult.SUCCESS) { if (itemAddedResult === ItemAddedResult.NO_CONTAINERS) { // Bot has no container to put item in, exit this.logger.debug( `Unable to add: ${totalItemCount} items to bot as it lacks a container to include them`, ); break; } fitItemIntoContainerAttempts++; if (fitItemIntoContainerAttempts >= 4) { this.logger.debug( `Failed to place item ${i} of ${totalItemCount} items into ${botRole} containers: ${ equipmentSlots.join(",") }. Tried ${fitItemIntoContainerAttempts} times, reason: ${ ItemAddedResult[itemAddedResult] }, skipping`, ); break; } // Try again, failed but still under attempt limit continue; } // Item added okay, reset counter for next item fitItemIntoContainerAttempts = 0; // Stop adding items to bots pool if rolling total is over total limit if (totalValueLimitRub > 0) { currentTotalRub += this.handbookHelper.getTemplatePrice(itemToAddTemplate._id); if (currentTotalRub > totalValueLimitRub) { break; } } } } } /** * Some items need child items to function, add them to the itemToAddChildrenTo array * @param itemToAddTemplate Db template of item to check * @param itemToAddChildrenTo Item to add children to * @param isPmc Is the item being generated for a pmc (affects money/ammo stack sizes) */ protected addRequiredChildItemsToParent( itemToAddTemplate: ITemplateItem, itemToAddChildrenTo: Item[], isPmc: boolean, ): void { // Fill ammo box if (this.itemHelper.isOfBaseclass(itemToAddTemplate._id, BaseClasses.AMMO_BOX)) { this.itemHelper.addCartridgesToAmmoBox(itemToAddChildrenTo, itemToAddTemplate); } // Make money a stack else if (this.itemHelper.isOfBaseclass(itemToAddTemplate._id, BaseClasses.MONEY)) { this.randomiseMoneyStackSize(isPmc, itemToAddTemplate, itemToAddChildrenTo[0]); } // Make ammo a stack else if (this.itemHelper.isOfBaseclass(itemToAddTemplate._id, BaseClasses.AMMO)) { this.randomiseAmmoStackSize(isPmc, itemToAddTemplate, itemToAddChildrenTo[0]); } // Must add soft inserts/plates else if (this.itemHelper.itemRequiresSoftInserts(itemToAddTemplate._id)) { this.itemHelper.addChildSlotItems(itemToAddChildrenTo, itemToAddTemplate, null, false); } } /** * Add generated weapons to inventory as loot * @param botInventory inventory to add preset to * @param equipmentSlot slot to place the preset in (backpack) * @param templateInventory bots template, assault.json * @param modChances chances for mods to spawn on weapon * @param botRole bots role .e.g. pmcBot * @param isPmc are we generating for a pmc */ protected addLooseWeaponsToInventorySlot( sessionId: string, botInventory: PmcInventory, equipmentSlot: string, templateInventory: Inventory, modChances: ModsChances, botRole: string, isPmc: boolean, botLevel: number, ): void { const chosenWeaponType = this.randomUtil.getArrayValue([ EquipmentSlots.FIRST_PRIMARY_WEAPON, EquipmentSlots.FIRST_PRIMARY_WEAPON, EquipmentSlots.FIRST_PRIMARY_WEAPON, EquipmentSlots.HOLSTER, ]); const randomisedWeaponCount = this.randomUtil.getInt( this.pmcConfig.looseWeaponInBackpackLootMinMax.min, this.pmcConfig.looseWeaponInBackpackLootMinMax.max, ); if (randomisedWeaponCount > 0) { for (let i = 0; i < randomisedWeaponCount; i++) { const generatedWeapon = this.botWeaponGenerator.generateRandomWeapon( sessionId, chosenWeaponType, templateInventory, botInventory.equipment, modChances, botRole, isPmc, botLevel, ); const result = this.botWeaponGeneratorHelper.addItemWithChildrenToEquipmentSlot( [equipmentSlot], generatedWeapon.weapon[0]._id, generatedWeapon.weapon[0]._tpl, [...generatedWeapon.weapon], botInventory, ); if (result !== ItemAddedResult.SUCCESS) { this.logger.debug( `Failed to add additional weapon ${generatedWeapon.weapon[0]._id} to bot backpack, reason: ${ ItemAddedResult[result] }`, ); } } } } /** * Hydrate item limit array to contain items that have a limit for a specific bot type * All values are set to 0 * @param isPmc Is the bot a pmc * @param botRole Role the bot has * @param limitCount */ protected initItemLimitArray(isPmc: boolean, botRole: string, limitCount: Record): void { // Init current count of items we want to limit const spawnLimits = this.getItemSpawnLimitsForBotType(isPmc, botRole); for (const limit in spawnLimits) { limitCount[limit] = 0; } } /** * Check if an item has reached its bot-specific spawn limit * @param itemTemplate Item we check to see if its reached spawn limit * @param botRole Bot type * @param isPmc Is bot we're working with a pmc * @param limitCount Spawn limits for items on bot * @param itemSpawnLimits The limits this bot is allowed to have * @returns true if item has reached spawn limit */ protected itemHasReachedSpawnLimit( itemTemplate: ITemplateItem, botRole: string, isPmc: boolean, limitCount: Record, itemSpawnLimits: Record, ): boolean { // PMCs and scavs have different sections of bot config for spawn limits if (!!itemSpawnLimits && itemSpawnLimits.length === 0) { // No items found in spawn limit, drop out return false; } // No spawn limits, skipping if (!itemSpawnLimits) { return false; } const idToCheckFor = this.getMatchingIdFromSpawnLimits(itemTemplate, itemSpawnLimits); if (!idToCheckFor) { // ParentId or tplid not found in spawnLimits, not a spawn limited item, skip return false; } // Increment item count with this bot type limitCount[idToCheckFor]++; // return true, we are over limit if (limitCount[idToCheckFor] > itemSpawnLimits[idToCheckFor]) { // Prevent edge-case of small loot pools + code trying to add limited item over and over infinitely if (limitCount[idToCheckFor] > itemSpawnLimits[idToCheckFor] * 10) { this.logger.debug( this.localisationService.getText("bot-item_spawn_limit_reached_skipping_item", { botRole: botRole, itemName: itemTemplate._name, attempts: limitCount[idToCheckFor], }), ); return false; } return true; } return false; } /** * Randomise the stack size of a money object, uses different values for pmc or scavs * @param isPmc Is money on a PMC bot * @param itemTemplate item details from db * @param moneyItem Money item to randomise */ protected randomiseMoneyStackSize(isPmc: boolean, itemTemplate: ITemplateItem, moneyItem: Item): void { // PMCs have a different stack max size const minStackSize = itemTemplate._props.StackMinRandom; const maxStackSize = isPmc ? this.pmcConfig.dynamicLoot.moneyStackLimits[itemTemplate._id] : itemTemplate._props.StackMaxRandom; const randomSize = this.randomUtil.getInt(minStackSize, maxStackSize); if (!moneyItem.upd) { moneyItem.upd = {}; } moneyItem.upd.StackObjectsCount = randomSize; } /** * Randomise the size of an ammo stack * @param isPmc Is ammo on a PMC bot * @param itemTemplate item details from db * @param ammoItem Ammo item to randomise */ protected randomiseAmmoStackSize(isPmc: boolean, itemTemplate: ITemplateItem, ammoItem: Item): void { const randomSize = itemTemplate._props.StackMaxSize === 1 ? 1 : this.randomUtil.getInt( itemTemplate._props.StackMinRandom, Math.min(itemTemplate._props.StackMaxRandom, 60), ); if (!ammoItem.upd) { ammoItem.upd = {}; } ammoItem.upd.StackObjectsCount = randomSize; } /** * Get spawn limits for a specific bot type from bot.json config * If no limit found for a non pmc bot, fall back to defaults * @param isPmc is the bot we want limits for a pmc * @param botRole what role does the bot have * @returns Dictionary of tplIds and limit */ protected getItemSpawnLimitsForBotType(isPmc: boolean, botRole: string): Record { if (isPmc) { return this.botConfig.itemSpawnLimits.pmc; } if (this.botConfig.itemSpawnLimits[botRole.toLowerCase()]) { return this.botConfig.itemSpawnLimits[botRole.toLowerCase()]; } this.logger.warning( this.localisationService.getText("bot-unable_to_find_spawn_limits_fallback_to_defaults", botRole), ); return this.botConfig.itemSpawnLimits.default; } /** * Get the parentId or tplId of item inside spawnLimits object if it exists * @param itemTemplate item we want to look for in spawn limits * @param spawnLimits Limits to check for item * @returns id as string, otherwise undefined */ protected getMatchingIdFromSpawnLimits(itemTemplate: ITemplateItem, spawnLimits: Record): string { if (itemTemplate._id in spawnLimits) { return itemTemplate._id; } // tplId not found in spawnLimits, check if parentId is if (itemTemplate._parent in spawnLimits) { return itemTemplate._parent; } // parentId and tplid not found return undefined; } }