2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 17:21:17 +00:00
|
|
|
import { ApplicationContext } from "@spt-aki/context/ApplicationContext";
|
|
|
|
import { ContextVariableType } from "@spt-aki/context/ContextVariableType";
|
|
|
|
import { LootGenerator } from "@spt-aki/generators/LootGenerator";
|
|
|
|
import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper";
|
|
|
|
import { TraderHelper } from "@spt-aki/helpers/TraderHelper";
|
|
|
|
import { IPmcData } from "@spt-aki/models/eft/common/IPmcData";
|
|
|
|
import { Item } from "@spt-aki/models/eft/common/tables/IItem";
|
|
|
|
import { ICreateGroupRequestData } from "@spt-aki/models/eft/match/ICreateGroupRequestData";
|
|
|
|
import { IEndOfflineRaidRequestData } from "@spt-aki/models/eft/match/IEndOfflineRaidRequestData";
|
|
|
|
import { IGetGroupStatusRequestData } from "@spt-aki/models/eft/match/IGetGroupStatusRequestData";
|
|
|
|
import { IGetProfileRequestData } from "@spt-aki/models/eft/match/IGetProfileRequestData";
|
|
|
|
import { IGetRaidConfigurationRequestData } from "@spt-aki/models/eft/match/IGetRaidConfigurationRequestData";
|
|
|
|
import { IJoinMatchRequestData } from "@spt-aki/models/eft/match/IJoinMatchRequestData";
|
|
|
|
import { IJoinMatchResult } from "@spt-aki/models/eft/match/IJoinMatchResult";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { MessageType } from "@spt-aki/models/enums/MessageType";
|
|
|
|
import { Traders } from "@spt-aki/models/enums/Traders";
|
|
|
|
import { IInRaidConfig } from "@spt-aki/models/spt/config/IInRaidConfig";
|
|
|
|
import { IMatchConfig } from "@spt-aki/models/spt/config/IMatchConfig";
|
|
|
|
import { IPmcConfig } from "@spt-aki/models/spt/config/IPmcConfig";
|
|
|
|
import { ITraderConfig } from "@spt-aki/models/spt/config/ITraderConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { SaveServer } from "@spt-aki/servers/SaveServer";
|
|
|
|
import { BotGenerationCacheService } from "@spt-aki/services/BotGenerationCacheService";
|
|
|
|
import { BotLootCacheService } from "@spt-aki/services/BotLootCacheService";
|
|
|
|
import { MailSendService } from "@spt-aki/services/MailSendService";
|
|
|
|
import { MatchLocationService } from "@spt-aki/services/MatchLocationService";
|
|
|
|
import { ProfileSnapshotService } from "@spt-aki/services/ProfileSnapshotService";
|
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
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 MatchController
|
|
|
|
{
|
|
|
|
protected matchConfig: IMatchConfig;
|
|
|
|
protected inraidConfig: IInRaidConfig;
|
2023-10-12 20:04:50 +01:00
|
|
|
protected traderConfig: ITraderConfig;
|
2023-10-10 11:03:20 +00:00
|
|
|
protected pmcConfig: IPmcConfig;
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("SaveServer") protected saveServer: SaveServer,
|
2023-10-12 20:04:50 +01:00
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
|
|
|
@inject("RandomUtil") protected randomUtil: RandomUtil,
|
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
2023-03-03 15:23:46 +00:00
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
|
|
|
@inject("MatchLocationService") protected matchLocationService: MatchLocationService,
|
|
|
|
@inject("TraderHelper") protected traderHelper: TraderHelper,
|
|
|
|
@inject("BotLootCacheService") protected botLootCacheService: BotLootCacheService,
|
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
|
|
|
@inject("ProfileSnapshotService") protected profileSnapshotService: ProfileSnapshotService,
|
|
|
|
@inject("BotGenerationCacheService") protected botGenerationCacheService: BotGenerationCacheService,
|
2023-10-12 20:04:50 +01:00
|
|
|
@inject("MailSendService") protected mailSendService: MailSendService,
|
|
|
|
@inject("LootGenerator") protected lootGenerator: LootGenerator,
|
2023-03-03 15:23:46 +00:00
|
|
|
@inject("ApplicationContext") protected applicationContext: ApplicationContext
|
|
|
|
)
|
|
|
|
{
|
|
|
|
this.matchConfig = this.configServer.getConfig(ConfigTypes.MATCH);
|
|
|
|
this.inraidConfig = this.configServer.getConfig(ConfigTypes.IN_RAID);
|
2023-10-12 20:04:50 +01:00
|
|
|
this.traderConfig = this.configServer.getConfig(ConfigTypes.TRADER);
|
2023-10-10 11:03:20 +00:00
|
|
|
this.pmcConfig = this.configServer.getConfig(ConfigTypes.PMC);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public getEnabled(): boolean
|
|
|
|
{
|
|
|
|
return this.matchConfig.enabled;
|
|
|
|
}
|
|
|
|
|
2023-07-15 14:49:25 +01:00
|
|
|
/** Handle raid/profile/list */
|
2023-03-03 15:23:46 +00:00
|
|
|
public getProfile(info: IGetProfileRequestData): IPmcData[]
|
|
|
|
{
|
|
|
|
if (info.profileId.includes("pmcAID"))
|
|
|
|
{
|
|
|
|
return this.profileHelper.getCompleteProfile(info.profileId.replace("pmcAID", "AID"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info.profileId.includes("scavAID"))
|
|
|
|
{
|
|
|
|
return this.profileHelper.getCompleteProfile(info.profileId.replace("scavAID", "AID"));
|
|
|
|
}
|
|
|
|
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2023-07-15 14:49:25 +01:00
|
|
|
/** Handle client/match/group/create */
|
2023-03-03 15:23:46 +00:00
|
|
|
public createGroup(sessionID: string, info: ICreateGroupRequestData): any
|
|
|
|
{
|
|
|
|
return this.matchLocationService.createGroup(sessionID, info);
|
|
|
|
}
|
|
|
|
|
2023-07-15 14:49:25 +01:00
|
|
|
/** Handle client/match/group/delete */
|
2023-03-03 15:23:46 +00:00
|
|
|
public deleteGroup(info: any): void
|
|
|
|
{
|
|
|
|
this.matchLocationService.deleteGroup(info);
|
|
|
|
}
|
|
|
|
|
2023-07-15 14:49:25 +01:00
|
|
|
/** Handle match/group/start_game */
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
2023-05-20 18:37:39 +01:00
|
|
|
public joinMatch(info: IJoinMatchRequestData, sessionId: string): IJoinMatchResult
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-05-20 18:37:39 +01:00
|
|
|
const output: IJoinMatchResult = {
|
|
|
|
maxPveCountExceeded: false,
|
|
|
|
profiles: []
|
|
|
|
};
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// get list of players joining into the match
|
2023-05-20 18:37:39 +01:00
|
|
|
output.profiles.push({
|
2023-10-31 22:52:09 +00:00
|
|
|
profileid: "TODO",
|
2023-05-20 18:37:39 +01:00
|
|
|
profileToken: "TODO",
|
2023-10-31 22:52:09 +00:00
|
|
|
status: "MatchWait",
|
|
|
|
sid: "",
|
|
|
|
ip: "",
|
|
|
|
port: 0,
|
|
|
|
version: "live",
|
|
|
|
location: "TODO get location",
|
2023-03-03 15:23:46 +00:00
|
|
|
raidMode: "Online",
|
2023-10-31 22:52:09 +00:00
|
|
|
mode: "deathmatch",
|
|
|
|
shortid: null,
|
2023-03-03 15:23:46 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
2023-05-20 18:37:39 +01:00
|
|
|
additional_info: null
|
2023-03-03 15:23:46 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2023-07-15 14:49:25 +01:00
|
|
|
/** Handle client/match/group/status */
|
2023-03-03 15:23:46 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
|
|
public getGroupStatus(info: IGetGroupStatusRequestData): any
|
|
|
|
{
|
|
|
|
return {
|
2023-07-15 14:49:25 +01:00
|
|
|
players: [],
|
2023-05-20 18:37:39 +01:00
|
|
|
maxPveCountExceeded: false
|
2023-03-03 15:23:46 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle /client/raid/configuration
|
2023-07-15 14:49:25 +01:00
|
|
|
* @param request Raid config request
|
|
|
|
* @param sessionID Session id
|
2023-03-03 15:23:46 +00:00
|
|
|
*/
|
|
|
|
public startOfflineRaid(request: IGetRaidConfigurationRequestData, sessionID: string): void
|
|
|
|
{
|
2023-07-15 14:49:25 +01:00
|
|
|
// Store request data for access during bot generation
|
2023-03-03 15:23:46 +00:00
|
|
|
this.applicationContext.addValue(ContextVariableType.RAID_CONFIGURATION, request);
|
|
|
|
|
|
|
|
//TODO: add code to strip PMC of equipment now they've started the raid
|
|
|
|
|
|
|
|
// Set pmcs to difficulty set in pre-raid screen if override in bot config isnt enabled
|
2023-10-10 11:03:20 +00:00
|
|
|
if (!this.pmcConfig.useDifficultyOverride)
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-10-10 11:03:20 +00:00
|
|
|
this.pmcConfig.difficulty = this.convertDifficultyDropdownIntoBotDifficulty(request.wavesSettings.botDifficulty);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Store the profile as-is for later use on the post-raid exp screen
|
|
|
|
const currentProfile = this.saveServer.getProfile(sessionID);
|
|
|
|
this.profileSnapshotService.storeProfileSnapshot(sessionID, currentProfile);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert a difficulty value from pre-raid screen to a bot difficulty
|
|
|
|
* @param botDifficulty dropdown difficulty value
|
|
|
|
* @returns bot difficulty
|
|
|
|
*/
|
|
|
|
protected convertDifficultyDropdownIntoBotDifficulty(botDifficulty: string): string
|
|
|
|
{
|
|
|
|
// Edge case medium - must be altered
|
|
|
|
if (botDifficulty.toLowerCase() === "medium")
|
|
|
|
{
|
|
|
|
return "normal";
|
|
|
|
}
|
|
|
|
|
|
|
|
return botDifficulty;
|
|
|
|
}
|
|
|
|
|
2023-07-15 14:49:25 +01:00
|
|
|
/** Handle client/match/offline/end */
|
2023-04-24 11:57:19 +01:00
|
|
|
public endOfflineRaid(info: IEndOfflineRaidRequestData, sessionId: string): void
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-24 11:57:19 +01:00
|
|
|
const pmcData: IPmcData = this.profileHelper.getPmcProfile(sessionId);
|
|
|
|
const extractName = info.exitName;
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-07-17 09:36:10 +01:00
|
|
|
// Save time spent in raid
|
2023-10-10 11:03:20 +00:00
|
|
|
pmcData.Stats.Eft.TotalInGameTime += info.raidSeconds;
|
2023-07-17 09:36:10 +01:00
|
|
|
|
2023-10-11 10:47:54 +01:00
|
|
|
// Clean up cached bots now raid is over
|
2023-03-03 15:23:46 +00:00
|
|
|
this.botGenerationCacheService.clearStoredBots();
|
|
|
|
|
2023-10-11 10:47:54 +01:00
|
|
|
// Clear bot loot cache
|
2023-04-24 11:57:19 +01:00
|
|
|
this.botLootCacheService.clearCache();
|
|
|
|
|
|
|
|
if (this.extractWasViaCar(extractName))
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-24 11:57:19 +01:00
|
|
|
this.handleCarExtract(extractName, pmcData, sessionId);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
2023-10-12 20:04:50 +01:00
|
|
|
|
2023-10-14 10:31:01 +01:00
|
|
|
if (extractName && this.extractWasViaCoop(extractName) && this.traderConfig.fence.coopExtractGift.sendGift)
|
2023-10-12 20:04:50 +01:00
|
|
|
{
|
|
|
|
this.sendCoopTakenFenceMessage(sessionId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Did player take a COOP extract
|
|
|
|
* @param extractName Name of extract player took
|
|
|
|
* @returns True if coop extract
|
|
|
|
*/
|
|
|
|
protected extractWasViaCoop(extractName: string): boolean
|
|
|
|
{
|
2023-10-14 10:31:01 +01:00
|
|
|
// No extract name, not a coop extract
|
|
|
|
if (!extractName)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-10-12 20:04:50 +01:00
|
|
|
return (this.inraidConfig.coopExtracts.includes(extractName.trim()));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected sendCoopTakenFenceMessage(sessionId: string): void
|
|
|
|
{
|
|
|
|
// Generate reward for taking coop extract
|
|
|
|
const loot = this.lootGenerator.createRandomLoot(this.traderConfig.fence.coopExtractGift);
|
|
|
|
const mailableLoot: Item[] = [];
|
|
|
|
|
|
|
|
const parentId = this.hashUtil.generate();
|
|
|
|
for (const item of loot)
|
|
|
|
{
|
|
|
|
mailableLoot.push(
|
|
|
|
{
|
|
|
|
_id: item.id,
|
|
|
|
_tpl: item.tpl,
|
|
|
|
slotId: "main",
|
|
|
|
parentId: parentId,
|
|
|
|
upd: {
|
2023-10-25 10:14:19 +01:00
|
|
|
StackObjectsCount: item.stackCount,
|
|
|
|
SpawnedInSession: true
|
2023-10-12 20:04:50 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send message from fence giving player reward generated above
|
|
|
|
this.mailSendService.sendLocalisedNpcMessageToPlayer(
|
|
|
|
sessionId,
|
|
|
|
this.traderHelper.getTraderById(Traders.FENCE),
|
|
|
|
MessageType.MESSAGE_WITH_ITEMS,
|
|
|
|
this.randomUtil.getArrayValue(this.traderConfig.fence.coopExtractGift.messageLocaleIds),
|
|
|
|
mailableLoot,
|
|
|
|
this.timeUtil.getHoursAsSeconds(this.traderConfig.fence.coopExtractGift.giftExpiryHours)
|
|
|
|
);
|
2023-04-24 11:57:19 +01:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-04-24 11:57:19 +01:00
|
|
|
/**
|
2023-07-17 09:36:10 +01:00
|
|
|
* Was extract by car
|
2023-04-24 11:57:19 +01:00
|
|
|
* @param extractName name of extract
|
|
|
|
* @returns true if car extract
|
|
|
|
*/
|
|
|
|
protected extractWasViaCar(extractName: string): boolean
|
|
|
|
{
|
2023-10-10 11:03:20 +00:00
|
|
|
// exit name is null on death
|
|
|
|
if (!extractName)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (extractName.toLowerCase().includes("v-ex"))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.inraidConfig.carExtracts.includes(extractName.trim());
|
2023-04-24 11:57:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle when a player extracts using a car - Add rep to fence
|
|
|
|
* @param extractName name of the extract used
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @param sessionId Session id
|
|
|
|
*/
|
|
|
|
protected handleCarExtract(extractName: string, pmcData: IPmcData, sessionId: string): void
|
|
|
|
{
|
|
|
|
// Ensure key exists for extract
|
|
|
|
if (!(extractName in pmcData.CarExtractCounts))
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
2023-04-24 11:57:19 +01:00
|
|
|
pmcData.CarExtractCounts[extractName] = 0;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
2023-04-24 11:57:19 +01:00
|
|
|
// Increment extract count value
|
|
|
|
pmcData.CarExtractCounts[extractName] += 1;
|
|
|
|
|
|
|
|
const fenceId: string = Traders.FENCE;
|
|
|
|
this.updateFenceStandingInProfile(pmcData, fenceId, extractName);
|
|
|
|
|
2023-10-10 11:03:20 +00:00
|
|
|
this.traderHelper.lvlUp(fenceId, pmcData);
|
2023-04-24 11:57:19 +01:00
|
|
|
pmcData.TradersInfo[fenceId].loyaltyLevel = Math.max(pmcData.TradersInfo[fenceId].loyaltyLevel, 1);
|
2023-10-10 11:03:20 +00:00
|
|
|
|
|
|
|
this.logger.debug(`Car extract: ${extractName} used, total times taken: ${pmcData.CarExtractCounts[extractName]}`);
|
2023-04-24 11:57:19 +01:00
|
|
|
}
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-04-24 11:57:19 +01:00
|
|
|
/**
|
|
|
|
* Update players fence trader standing value in profile
|
|
|
|
* @param pmcData Player profile
|
|
|
|
* @param fenceId Id of fence trader
|
|
|
|
* @param extractName Name of extract used
|
|
|
|
*/
|
|
|
|
protected updateFenceStandingInProfile(pmcData: IPmcData, fenceId: string, extractName: string): void
|
|
|
|
{
|
|
|
|
let fenceStanding = Number(pmcData.TradersInfo[fenceId].standing);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// Not exact replica of Live behaviour
|
|
|
|
// Simplified for now, no real reason to do the whole (unconfirmed) extra 0.01 standing per day regeneration mechanic
|
|
|
|
const baseGain: number = this.inraidConfig.carExtractBaseStandingGain;
|
2023-04-24 11:57:19 +01:00
|
|
|
const extractCount: number = pmcData.CarExtractCounts[extractName];
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-04-24 11:57:19 +01:00
|
|
|
fenceStanding += Math.max(baseGain / extractCount, 0.01);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-04-24 11:57:19 +01:00
|
|
|
// Ensure fence loyalty level is not above/below the range -7 - 15
|
2023-10-10 11:03:20 +00:00
|
|
|
const newFenceStanding = Math.min(Math.max(fenceStanding, -7), 15);
|
|
|
|
this.logger.debug(`Old vs new fence standing: ${pmcData.TradersInfo[fenceId].standing}, ${newFenceStanding}`);
|
|
|
|
pmcData.TradersInfo[fenceId].standing = newFenceStanding;
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
}
|