2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
2023-10-19 17:21:17 +00:00
|
|
|
import { BotGenerator } from "@spt-aki/generators/BotGenerator";
|
|
|
|
import { BotGeneratorHelper } from "@spt-aki/helpers/BotGeneratorHelper";
|
|
|
|
import { BotHelper } from "@spt-aki/helpers/BotHelper";
|
|
|
|
import { ItemHelper } from "@spt-aki/helpers/ItemHelper";
|
|
|
|
import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper";
|
|
|
|
import { IPmcData } from "@spt-aki/models/eft/common/IPmcData";
|
2024-02-16 09:59:43 +00:00
|
|
|
import { IBotBase, Settings, Skills, Stats } from "@spt-aki/models/eft/common/tables/IBotBase";
|
2023-10-19 17:21:17 +00:00
|
|
|
import { IBotType } from "@spt-aki/models/eft/common/tables/IBotType";
|
|
|
|
import { Item } from "@spt-aki/models/eft/common/tables/IItem";
|
|
|
|
import { AccountTypes } from "@spt-aki/models/enums/AccountTypes";
|
2024-01-29 10:42:02 +00:00
|
|
|
import { BonusType } from "@spt-aki/models/enums/BonusType";
|
2023-10-19 17:21:17 +00:00
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
2023-11-24 16:05:58 +00:00
|
|
|
import { ItemAddedResult } from "@spt-aki/models/enums/ItemAddedResult";
|
2023-10-19 17:21:17 +00:00
|
|
|
import { MemberCategory } from "@spt-aki/models/enums/MemberCategory";
|
|
|
|
import { Traders } from "@spt-aki/models/enums/Traders";
|
|
|
|
import { IPlayerScavConfig, KarmaLevel } from "@spt-aki/models/spt/config/IPlayerScavConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { SaveServer } from "@spt-aki/servers/SaveServer";
|
|
|
|
import { BotLootCacheService } from "@spt-aki/services/BotLootCacheService";
|
|
|
|
import { FenceService } from "@spt-aki/services/FenceService";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
2024-05-13 17:58:17 +00:00
|
|
|
import { ICloner } from "@spt-aki/utils/cloners/ICloner";
|
2023-10-19 17:21:17 +00:00
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
import { RandomUtil } from "@spt-aki/utils/RandomUtil";
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class PlayerScavGenerator
|
|
|
|
{
|
|
|
|
protected playerScavConfig: IPlayerScavConfig;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
|
|
|
@inject("BotGeneratorHelper") protected botGeneratorHelper: BotGeneratorHelper,
|
|
|
|
@inject("SaveServer") protected saveServer: SaveServer,
|
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
|
|
|
@inject("BotHelper") protected botHelper: BotHelper,
|
|
|
|
@inject("FenceService") protected fenceService: FenceService,
|
|
|
|
@inject("BotLootCacheService") protected botLootCacheService: BotLootCacheService,
|
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
@inject("BotGenerator") protected botGenerator: BotGenerator,
|
2023-11-16 21:42:06 +00:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2024-05-13 17:58:17 +00:00
|
|
|
@inject("RecursiveCloner") protected cloner: ICloner,
|
2023-03-03 15:23:46 +00:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.playerScavConfig = this.configServer.getConfig(ConfigTypes.PLAYERSCAV);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update a player profile to include a new player scav profile
|
|
|
|
* @param sessionID session id to specify what profile is updated
|
|
|
|
* @returns profile object
|
|
|
|
*/
|
|
|
|
public generate(sessionID: string): IPmcData
|
|
|
|
{
|
|
|
|
// get karma level from profile
|
|
|
|
const profile = this.saveServer.getProfile(sessionID);
|
2024-05-13 17:58:17 +00:00
|
|
|
const pmcDataClone = this.cloner.clone(profile.characters.pmc);
|
|
|
|
const existingScavDataClone = this.cloner.clone(profile.characters.scav);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2024-02-16 09:59:43 +00:00
|
|
|
const scavKarmaLevel = this.getScavKarmaLevel(pmcDataClone);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// use karma level to get correct karmaSettings
|
|
|
|
const playerScavKarmaSettings = this.playerScavConfig.karmaLevel[scavKarmaLevel];
|
|
|
|
if (!playerScavKarmaSettings)
|
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("scav-missing_karma_settings", scavKarmaLevel));
|
|
|
|
}
|
|
|
|
|
|
|
|
this.logger.debug(`generated player scav loadout with karma level ${scavKarmaLevel}`);
|
|
|
|
|
2023-10-21 18:23:58 +01:00
|
|
|
// Edit baseBotNode values
|
2023-03-03 15:23:46 +00:00
|
|
|
const baseBotNode: IBotType = this.constructBotBaseTemplate(playerScavKarmaSettings.botTypeForLoot);
|
|
|
|
this.adjustBotTemplateWithKarmaSpecificSettings(playerScavKarmaSettings, baseBotNode);
|
|
|
|
|
2023-11-16 21:42:06 +00:00
|
|
|
let scavData = this.botGenerator.generatePlayerScav(
|
|
|
|
sessionID,
|
|
|
|
playerScavKarmaSettings.botTypeForLoot.toLowerCase(),
|
|
|
|
"easy",
|
|
|
|
baseBotNode,
|
|
|
|
);
|
2023-10-21 18:23:58 +01:00
|
|
|
|
|
|
|
// Remove cached bot data after scav was generated
|
2023-03-03 15:23:46 +00:00
|
|
|
this.botLootCacheService.clearCache();
|
|
|
|
|
2023-10-21 18:23:58 +01:00
|
|
|
// Add scav metadata
|
2023-10-10 11:03:20 +00:00
|
|
|
scavData.savage = null;
|
2024-02-05 14:43:46 +00:00
|
|
|
scavData.aid = pmcDataClone.aid;
|
|
|
|
scavData.TradersInfo = pmcDataClone.TradersInfo;
|
2023-10-14 13:18:07 +01:00
|
|
|
scavData.Info.Settings = {} as Settings;
|
|
|
|
scavData.Info.Bans = [];
|
2024-02-05 14:43:46 +00:00
|
|
|
scavData.Info.RegistrationDate = pmcDataClone.Info.RegistrationDate;
|
|
|
|
scavData.Info.GameVersion = pmcDataClone.Info.GameVersion;
|
2023-10-14 13:29:27 +01:00
|
|
|
scavData.Info.MemberCategory = MemberCategory.UNIQUE_ID;
|
2023-10-21 18:23:58 +01:00
|
|
|
scavData.Info.lockedMoveCommands = true;
|
2024-02-05 14:43:46 +00:00
|
|
|
scavData.RagfairInfo = pmcDataClone.RagfairInfo;
|
|
|
|
scavData.UnlockedInfo = pmcDataClone.UnlockedInfo;
|
2023-10-21 18:23:58 +01:00
|
|
|
|
|
|
|
// Persist previous scav data into new scav
|
2024-02-05 14:43:46 +00:00
|
|
|
scavData._id = existingScavDataClone._id ?? pmcDataClone.savage;
|
|
|
|
scavData.sessionId = existingScavDataClone.sessionId ?? pmcDataClone.sessionId;
|
|
|
|
scavData.Skills = this.getScavSkills(existingScavDataClone);
|
|
|
|
scavData.Stats = this.getScavStats(existingScavDataClone);
|
|
|
|
scavData.Info.Level = this.getScavLevel(existingScavDataClone);
|
|
|
|
scavData.Info.Experience = this.getScavExperience(existingScavDataClone);
|
|
|
|
scavData.Quests = existingScavDataClone.Quests ?? [];
|
|
|
|
scavData.TaskConditionCounters = existingScavDataClone.TaskConditionCounters ?? {};
|
|
|
|
scavData.Notes = existingScavDataClone.Notes ?? { Notes: [] };
|
|
|
|
scavData.WishList = existingScavDataClone.WishList ?? [];
|
2024-03-20 17:58:32 +00:00
|
|
|
scavData.Encyclopedia = pmcDataClone.Encyclopedia ?? {};
|
2023-10-21 18:23:58 +01:00
|
|
|
|
2024-02-16 09:59:43 +00:00
|
|
|
// Add additional items to player scav as loot
|
|
|
|
this.addAdditionalLootToPlayerScavContainers(playerScavKarmaSettings.lootItemsToAddChancePercent, scavData, [
|
|
|
|
"TacticalVest",
|
|
|
|
"Pockets",
|
|
|
|
"Backpack",
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Remove secure container
|
|
|
|
scavData = this.profileHelper.removeSecureContainer(scavData);
|
|
|
|
|
|
|
|
// Set cooldown timer
|
|
|
|
scavData = this.setScavCooldownTimer(scavData, pmcDataClone);
|
|
|
|
|
|
|
|
// Add scav to the profile
|
|
|
|
this.saveServer.getProfile(sessionID).characters.scav = scavData;
|
|
|
|
|
|
|
|
return scavData;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add items picked from `playerscav.lootItemsToAddChancePercent`
|
|
|
|
* @param possibleItemsToAdd dict of tpl + % chance to be added
|
|
|
|
* @param scavData
|
|
|
|
* @param containersToAddTo Possible slotIds to add loot to
|
|
|
|
*/
|
|
|
|
protected addAdditionalLootToPlayerScavContainers(
|
|
|
|
possibleItemsToAdd: Record<string, number>,
|
|
|
|
scavData: IBotBase,
|
|
|
|
containersToAddTo: string[],
|
|
|
|
): void
|
|
|
|
{
|
|
|
|
for (const tpl in possibleItemsToAdd)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2024-02-16 09:59:43 +00:00
|
|
|
const shouldAdd = this.randomUtil.getChance100(possibleItemsToAdd[tpl]);
|
|
|
|
if (!shouldAdd)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const itemResult = this.itemHelper.getItem(tpl);
|
|
|
|
if (!itemResult[0])
|
|
|
|
{
|
|
|
|
this.logger.warning(`Unable to add ${tpl} to player scav, not an item`);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const itemTemplate = itemResult[1];
|
2024-05-17 15:32:41 -04:00
|
|
|
const itemsToAdd: Item[] = [
|
|
|
|
{
|
|
|
|
_id: this.hashUtil.generate(),
|
|
|
|
_tpl: itemTemplate._id,
|
|
|
|
...this.botGeneratorHelper.generateExtraPropertiesForItem(itemTemplate),
|
|
|
|
},
|
|
|
|
];
|
2024-02-16 09:59:43 +00:00
|
|
|
|
2024-02-25 11:45:34 +00:00
|
|
|
const result = this.botGeneratorHelper.addItemWithChildrenToEquipmentSlot(
|
2024-02-16 09:59:43 +00:00
|
|
|
containersToAddTo,
|
2023-11-16 21:42:06 +00:00
|
|
|
itemsToAdd[0]._id,
|
2024-02-16 09:59:43 +00:00
|
|
|
itemTemplate._id,
|
2023-11-16 21:42:06 +00:00
|
|
|
itemsToAdd,
|
|
|
|
scavData.Inventory,
|
|
|
|
);
|
2023-11-24 16:05:58 +00:00
|
|
|
|
|
|
|
if (result !== ItemAddedResult.SUCCESS)
|
|
|
|
{
|
|
|
|
this.logger.debug(`Unable to add keycard to bot. Reason: ${ItemAddedResult[result]}`);
|
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the scav karama level for a profile
|
|
|
|
* Is also the fence trader rep level
|
|
|
|
* @param pmcData pmc profile
|
|
|
|
* @returns karma level
|
|
|
|
*/
|
|
|
|
protected getScavKarmaLevel(pmcData: IPmcData): number
|
|
|
|
{
|
|
|
|
const fenceInfo = pmcData.TradersInfo[Traders.FENCE];
|
|
|
|
|
|
|
|
// Can be empty during profile creation
|
|
|
|
if (!fenceInfo)
|
|
|
|
{
|
|
|
|
this.logger.warning(this.localisationService.getText("scav-missing_karma_level_getting_default"));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fenceInfo.standing > 6)
|
|
|
|
{
|
|
|
|
return 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
// e.g. 2.09 becomes 2
|
|
|
|
return Math.floor(fenceInfo.standing);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a baseBot template
|
|
|
|
* If the parameter doesnt match "assault", take parts from the loot type and apply to the return bot template
|
|
|
|
* @param botTypeForLoot bot type to use for inventory/chances
|
|
|
|
* @returns IBotType object
|
|
|
|
*/
|
|
|
|
protected constructBotBaseTemplate(botTypeForLoot: string): IBotType
|
|
|
|
{
|
|
|
|
const baseScavType = "assault";
|
2024-05-13 17:58:17 +00:00
|
|
|
const assaultBase = this.cloner.clone(this.botHelper.getBotTemplate(baseScavType));
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// Loot bot is same as base bot, return base with no modification
|
|
|
|
if (botTypeForLoot === baseScavType)
|
|
|
|
{
|
|
|
|
return assaultBase;
|
|
|
|
}
|
|
|
|
|
2024-05-13 17:58:17 +00:00
|
|
|
const lootBase = this.cloner.clone(this.botHelper.getBotTemplate(botTypeForLoot));
|
2023-03-03 15:23:46 +00:00
|
|
|
assaultBase.inventory = lootBase.inventory;
|
|
|
|
assaultBase.chances = lootBase.chances;
|
|
|
|
assaultBase.generation = lootBase.generation;
|
|
|
|
|
|
|
|
return assaultBase;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adjust equipment/mod/item generation values based on scav karma levels
|
|
|
|
* @param karmaSettings Values to modify the bot template with
|
|
|
|
* @param baseBotNode bot template to modify according to karama level settings
|
|
|
|
*/
|
|
|
|
protected adjustBotTemplateWithKarmaSpecificSettings(karmaSettings: KarmaLevel, baseBotNode: IBotType): void
|
|
|
|
{
|
|
|
|
// Adjust equipment chance values
|
|
|
|
for (const equipmentKey in karmaSettings.modifiers.equipment)
|
|
|
|
{
|
|
|
|
if (karmaSettings.modifiers.equipment[equipmentKey] === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
baseBotNode.chances.equipment[equipmentKey] += karmaSettings.modifiers.equipment[equipmentKey];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Adjust mod chance values
|
|
|
|
for (const modKey in karmaSettings.modifiers.mod)
|
|
|
|
{
|
|
|
|
if (karmaSettings.modifiers.mod[modKey] === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-09 15:31:56 +00:00
|
|
|
baseBotNode.chances.weaponMods[modKey] += karmaSettings.modifiers.mod[modKey];
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Adjust item spawn quantity values
|
|
|
|
for (const itemLimitkey in karmaSettings.itemLimits)
|
|
|
|
{
|
2023-10-10 11:03:20 +00:00
|
|
|
baseBotNode.generation.items[itemLimitkey] = karmaSettings.itemLimits[itemLimitkey];
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Blacklist equipment
|
|
|
|
for (const equipmentKey in karmaSettings.equipmentBlacklist)
|
|
|
|
{
|
|
|
|
const blacklistedItemTpls = karmaSettings.equipmentBlacklist[equipmentKey];
|
|
|
|
for (const itemToRemove of blacklistedItemTpls)
|
|
|
|
{
|
|
|
|
delete baseBotNode.inventory.equipment[equipmentKey][itemToRemove];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected getScavSkills(scavProfile: IPmcData): Skills
|
|
|
|
{
|
|
|
|
if (scavProfile.Skills)
|
|
|
|
{
|
|
|
|
return scavProfile.Skills;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.getDefaultScavSkills();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected getDefaultScavSkills(): Skills
|
|
|
|
{
|
2023-11-16 21:42:06 +00:00
|
|
|
return { Common: [], Mastering: [], Points: 0 };
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
protected getScavStats(scavProfile: IPmcData): Stats
|
|
|
|
{
|
|
|
|
if (scavProfile.Stats)
|
|
|
|
{
|
|
|
|
return scavProfile.Stats;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.profileHelper.getDefaultCounters();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected getScavLevel(scavProfile: IPmcData): number
|
|
|
|
{
|
|
|
|
// Info can be null on initial account creation
|
2024-05-07 23:57:08 -04:00
|
|
|
if (!scavProfile.Info?.Level)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return scavProfile.Info.Level;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected getScavExperience(scavProfile: IPmcData): number
|
|
|
|
{
|
|
|
|
// Info can be null on initial account creation
|
2024-05-07 23:57:08 -04:00
|
|
|
if (!scavProfile.Info?.Experience)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return scavProfile.Info.Experience;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set cooldown till pscav is playable
|
|
|
|
* take into account scav cooldown bonus
|
|
|
|
* @param scavData scav profile
|
|
|
|
* @param pmcData pmc profile
|
2023-11-16 21:42:06 +00:00
|
|
|
* @returns
|
2023-03-03 15:23:46 +00:00
|
|
|
*/
|
|
|
|
protected setScavCooldownTimer(scavData: IPmcData, pmcData: IPmcData): IPmcData
|
|
|
|
{
|
|
|
|
// Set cooldown time.
|
|
|
|
// Make sure to apply ScavCooldownTimer bonus from Hideout if the player has it.
|
|
|
|
let scavLockDuration = this.databaseServer.getTables().globals.config.SavagePlayCooldown;
|
|
|
|
let modifier = 1;
|
|
|
|
|
|
|
|
for (const bonus of pmcData.Bonuses)
|
|
|
|
{
|
2024-01-29 10:42:02 +00:00
|
|
|
if (bonus.type === BonusType.SCAV_COOLDOWN_TIMER)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
// Value is negative, so add.
|
|
|
|
// Also note that for scav cooldown, multiple bonuses stack additively.
|
|
|
|
modifier += bonus.value / 100;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const fenceInfo = this.fenceService.getFenceInfo(pmcData);
|
|
|
|
modifier *= fenceInfo.SavageCooldownModifier;
|
|
|
|
scavLockDuration *= modifier;
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-10-10 11:03:20 +00:00
|
|
|
const fullProfile = this.profileHelper.getFullProfile(pmcData?.sessionId);
|
|
|
|
if (fullProfile?.info?.edition?.toLowerCase?.().startsWith?.(AccountTypes.SPT_DEVELOPER))
|
|
|
|
{
|
|
|
|
// Set scav cooldown timer to 10 seconds for spt developer account
|
|
|
|
scavLockDuration = 10;
|
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2024-05-07 23:57:08 -04:00
|
|
|
scavData.Info.SavageLockTime = Date.now() / 1000 + scavLockDuration;
|
2023-11-16 21:42:06 +00:00
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
return scavData;
|
|
|
|
}
|
2023-11-16 21:42:06 +00:00
|
|
|
}
|