2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 19:21:17 +02:00
|
|
|
import { ApplicationContext } from "@spt-aki/context/ApplicationContext";
|
|
|
|
import { ContextVariableType } from "@spt-aki/context/ContextVariableType";
|
|
|
|
import { BotGenerator } from "@spt-aki/generators/BotGenerator";
|
|
|
|
import { BotDifficultyHelper } from "@spt-aki/helpers/BotDifficultyHelper";
|
|
|
|
import { BotHelper } from "@spt-aki/helpers/BotHelper";
|
|
|
|
import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper";
|
|
|
|
import { IGenerateBotsRequestData } from "@spt-aki/models/eft/bot/IGenerateBotsRequestData";
|
|
|
|
import { IBotBase } from "@spt-aki/models/eft/common/tables/IBotBase";
|
|
|
|
import { IBotCore } from "@spt-aki/models/eft/common/tables/IBotCore";
|
|
|
|
import { Difficulty } from "@spt-aki/models/eft/common/tables/IBotType";
|
|
|
|
import { IGetRaidConfigurationRequestData } from "@spt-aki/models/eft/match/IGetRaidConfigurationRequestData";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { BotGenerationDetails } from "@spt-aki/models/spt/bots/BotGenerationDetails";
|
|
|
|
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 { BotGenerationCacheService } from "@spt-aki/services/BotGenerationCacheService";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { MatchBotDetailsCacheService } from "@spt-aki/services/MatchBotDetailsCacheService";
|
|
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class BotController
|
|
|
|
{
|
|
|
|
protected botConfig: IBotConfig;
|
2023-10-10 13:03:20 +02:00
|
|
|
protected pmcConfig: IPmcConfig;
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("BotGenerator") protected botGenerator: BotGenerator,
|
|
|
|
@inject("BotHelper") protected botHelper: BotHelper,
|
|
|
|
@inject("BotDifficultyHelper") protected botDifficultyHelper: BotDifficultyHelper,
|
|
|
|
@inject("BotGenerationCacheService") protected botGenerationCacheService: BotGenerationCacheService,
|
2023-03-22 11:25:34 +01:00
|
|
|
@inject("MatchBotDetailsCacheService") protected matchBotDetailsCacheService: MatchBotDetailsCacheService,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
|
|
|
@inject("ApplicationContext") protected applicationContext: ApplicationContext,
|
2023-11-16 22:42:06 +01:00
|
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
this.botConfig = this.configServer.getConfig(ConfigTypes.BOT);
|
2023-10-10 13:03:20 +02:00
|
|
|
this.pmcConfig = this.configServer.getConfig(ConfigTypes.PMC);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the number of bot loadout varieties to be generated
|
|
|
|
* @param type bot Type we want the loadout gen count for
|
|
|
|
* @returns number of bots to generate
|
|
|
|
*/
|
|
|
|
public getBotPresetGenerationLimit(type: string): number
|
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
return this.botConfig.presetBatch[(type === "assaultGroup") ? "assault" : type];
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-07-15 15:49:25 +02:00
|
|
|
* Handle singleplayer/settings/bot/difficulty
|
2023-03-03 16:23:46 +01:00
|
|
|
* Get the core.json difficulty settings from database\bots
|
|
|
|
* @returns IBotCore
|
|
|
|
*/
|
|
|
|
public getBotCoreDifficulty(): IBotCore
|
|
|
|
{
|
|
|
|
return this.databaseServer.getTables().bots.core;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get bot difficulty settings
|
|
|
|
* adjust PMC settings to ensure they engage the correct bot types
|
|
|
|
* @param type what bot the server is requesting settings for
|
|
|
|
* @param difficulty difficulty level server requested settings for
|
|
|
|
* @returns Difficulty object
|
|
|
|
*/
|
|
|
|
public getBotDifficulty(type: string, difficulty: string): Difficulty
|
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
const raidConfig = this.applicationContext.getLatestValue(ContextVariableType.RAID_CONFIGURATION)?.getValue<
|
|
|
|
IGetRaidConfigurationRequestData
|
|
|
|
>();
|
2023-07-28 18:11:18 +02:00
|
|
|
if (!raidConfig)
|
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
this.logger.error(
|
|
|
|
this.localisationService.getText("bot-missing_application_context", "RAID_CONFIGURATION"),
|
|
|
|
);
|
2023-07-28 18:11:18 +02:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Check value chosen in pre-raid difficulty dropdown
|
|
|
|
// If value is not 'asonline', change requested difficulty to be what was chosen in dropdown
|
|
|
|
const botDifficultyDropDownValue = raidConfig.wavesSettings.botDifficulty.toLowerCase();
|
|
|
|
if (botDifficultyDropDownValue !== "asonline")
|
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
difficulty = this.botDifficultyHelper.convertBotDifficultyDropdownToBotDifficulty(
|
|
|
|
botDifficultyDropDownValue,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
let difficultySettings: Difficulty;
|
|
|
|
const lowercasedBotType = type.toLowerCase();
|
|
|
|
switch (lowercasedBotType)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
case this.pmcConfig.bearType.toLowerCase():
|
2023-11-16 22:42:06 +01:00
|
|
|
difficultySettings = this.botDifficultyHelper.getPmcDifficultySettings(
|
|
|
|
"bear",
|
|
|
|
difficulty,
|
|
|
|
this.pmcConfig.usecType,
|
|
|
|
this.pmcConfig.bearType,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
break;
|
2023-10-10 13:03:20 +02:00
|
|
|
case this.pmcConfig.usecType.toLowerCase():
|
2023-11-16 22:42:06 +01:00
|
|
|
difficultySettings = this.botDifficultyHelper.getPmcDifficultySettings(
|
|
|
|
"usec",
|
|
|
|
difficulty,
|
|
|
|
this.pmcConfig.usecType,
|
|
|
|
this.pmcConfig.bearType,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
difficultySettings = this.botDifficultyHelper.getBotDifficultySettings(type, difficulty);
|
|
|
|
// Don't add pmcs to gifter enemy list
|
|
|
|
if (type.toLowerCase() !== "gifter")
|
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
this.botHelper.addBotToEnemyList(difficultySettings, [
|
|
|
|
this.pmcConfig.bearType,
|
|
|
|
this.pmcConfig.usecType,
|
|
|
|
], lowercasedBotType);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
2023-11-16 22:42:06 +01:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return difficultySettings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generate bot profiles and store in cache
|
|
|
|
* @param sessionId Session id
|
|
|
|
* @param info bot generation request info
|
|
|
|
* @returns IBotBase array
|
|
|
|
*/
|
|
|
|
public generate(sessionId: string, info: IGenerateBotsRequestData): IBotBase[]
|
|
|
|
{
|
|
|
|
const pmcProfile = this.profileHelper.getPmcProfile(sessionId);
|
|
|
|
|
|
|
|
const botsToReturn: IBotBase[] = [];
|
|
|
|
for (const condition of info.conditions)
|
|
|
|
{
|
|
|
|
const botGenerationDetails: BotGenerationDetails = {
|
|
|
|
isPmc: false,
|
|
|
|
side: "Savage",
|
|
|
|
role: condition.Role,
|
|
|
|
playerLevel: pmcProfile.Info.Level,
|
2023-10-10 13:03:20 +02:00
|
|
|
botRelativeLevelDeltaMax: this.pmcConfig.botRelativeLevelDeltaMax,
|
|
|
|
botCountToGenerate: this.botConfig.presetBatch[condition.Role],
|
2023-03-03 16:23:46 +01:00
|
|
|
botDifficulty: condition.Difficulty,
|
2023-11-16 22:42:06 +01:00
|
|
|
isPlayerScav: false,
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Custom map waves can have spt roles in them
|
|
|
|
// Is bot type sptusec/sptbear, set is pmc true and set side
|
|
|
|
if (this.botHelper.botRoleIsPmc(condition.Role))
|
|
|
|
{
|
|
|
|
botGenerationDetails.isPmc = true;
|
|
|
|
botGenerationDetails.side = this.botHelper.getPmcSideByRole(condition.Role);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Loop over and make x bots for this condition
|
|
|
|
let cacheKey = "";
|
2023-11-16 22:42:06 +01:00
|
|
|
for (let i = 0; i < botGenerationDetails.botCountToGenerate; i++)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const details = this.jsonUtil.clone(botGenerationDetails);
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
// Roll chance to be pmc if type is allowed to be one
|
|
|
|
const botConvertRateMinMax = this.pmcConfig.convertIntoPmcChance[details.role.toLowerCase()];
|
2023-03-03 16:23:46 +01:00
|
|
|
if (botConvertRateMinMax)
|
|
|
|
{
|
|
|
|
// Should bot become PMC
|
|
|
|
const convertToPmc = this.botHelper.rollChanceToBePmc(details.role, botConvertRateMinMax);
|
|
|
|
if (convertToPmc)
|
|
|
|
{
|
|
|
|
details.isPmc = true;
|
|
|
|
details.role = this.botHelper.getRandomizedPmcRole();
|
|
|
|
details.side = this.botHelper.getPmcSideByRole(details.role);
|
|
|
|
details.botDifficulty = this.getPMCDifficulty(details.botDifficulty);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cacheKey = `${details.role}${details.botDifficulty}`;
|
|
|
|
// Check for bot in cache, add if not
|
|
|
|
if (!this.botGenerationCacheService.cacheHasBotOfRole(cacheKey))
|
|
|
|
{
|
|
|
|
// Generate and add x bots to cache
|
|
|
|
const botsToAddToCache = this.botGenerator.prepareAndGenerateBots(sessionId, details);
|
|
|
|
this.botGenerationCacheService.storeBots(cacheKey, botsToAddToCache);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Get bot from cache, add to return array
|
2023-03-22 11:25:34 +01:00
|
|
|
const botToReturn = this.botGenerationCacheService.getBot(cacheKey);
|
|
|
|
|
|
|
|
if (info.conditions.length === 1)
|
|
|
|
{
|
|
|
|
// Cache bot when we're returning 1 bot, this indicated the bot is being requested to be spawned
|
2023-10-10 13:03:20 +02:00
|
|
|
// Used by PMC response text system
|
2023-03-22 11:25:34 +01:00
|
|
|
this.matchBotDetailsCacheService.cacheBot(botToReturn);
|
|
|
|
}
|
|
|
|
|
|
|
|
botsToReturn.push(botToReturn);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return botsToReturn;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the difficulty passed in, if its not "asoline", get selected difficulty from config
|
2023-11-16 22:42:06 +01:00
|
|
|
* @param requestedDifficulty
|
|
|
|
* @returns
|
2023-03-03 16:23:46 +01:00
|
|
|
*/
|
|
|
|
public getPMCDifficulty(requestedDifficulty: string): string
|
|
|
|
{
|
|
|
|
// maybe retrun a random difficulty...
|
2023-10-10 13:03:20 +02:00
|
|
|
if (this.pmcConfig.difficulty.toLowerCase() === "asonline")
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return requestedDifficulty;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
if (this.pmcConfig.difficulty.toLowerCase() === "random")
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return this.botDifficultyHelper.chooseRandomDifficulty();
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
return this.pmcConfig.difficulty;
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the max number of bots allowed on a map
|
|
|
|
* Looks up location player is entering when getting cap value
|
|
|
|
* @returns cap number
|
|
|
|
*/
|
|
|
|
public getBotCap(): number
|
|
|
|
{
|
|
|
|
const defaultMapCapId = "default";
|
2023-11-16 22:42:06 +01:00
|
|
|
const raidConfig = this.applicationContext.getLatestValue(ContextVariableType.RAID_CONFIGURATION).getValue<
|
|
|
|
IGetRaidConfigurationRequestData
|
|
|
|
>();
|
2023-03-03 16:23:46 +01:00
|
|
|
if (!raidConfig)
|
|
|
|
{
|
|
|
|
this.logger.warning(this.localisationService.getText("bot-missing_saved_match_info"));
|
|
|
|
}
|
|
|
|
|
2023-11-16 22:42:06 +01:00
|
|
|
const mapName = raidConfig ? raidConfig.location : defaultMapCapId;
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
let botCap = this.botConfig.maxBotCap[mapName.toLowerCase()];
|
|
|
|
if (!botCap)
|
|
|
|
{
|
2023-11-16 22:42:06 +01:00
|
|
|
this.logger.warning(
|
|
|
|
this.localisationService.getText(
|
|
|
|
"bot-no_bot_cap_found_for_location",
|
|
|
|
raidConfig.location.toLowerCase(),
|
|
|
|
),
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
botCap = this.botConfig.maxBotCap[defaultMapCapId];
|
|
|
|
}
|
|
|
|
|
|
|
|
return botCap;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
public getAiBotBrainTypes(): any
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-11-26 22:33:47 +01:00
|
|
|
return {
|
|
|
|
pmc: this.pmcConfig.pmcType,
|
|
|
|
assault: this.botConfig.assaultBrainType,
|
|
|
|
playerScav: this.botConfig.playerScavBrainType};
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|