2023-03-03 15:23:46 +00:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
|
|
|
import { PlayerScavGenerator } from "../generators/PlayerScavGenerator";
|
|
|
|
import { DialogueHelper } from "../helpers/DialogueHelper";
|
|
|
|
import { ItemHelper } from "../helpers/ItemHelper";
|
|
|
|
import { ProfileHelper } from "../helpers/ProfileHelper";
|
|
|
|
import { QuestHelper } from "../helpers/QuestHelper";
|
|
|
|
import { TraderHelper } from "../helpers/TraderHelper";
|
|
|
|
import { IPmcData } from "../models/eft/common/IPmcData";
|
|
|
|
import { TemplateSide } from "../models/eft/common/tables/IProfileTemplate";
|
2023-03-08 22:41:57 +00:00
|
|
|
import { IItemEventRouterResponse } from "../models/eft/itemEvent/IItemEventRouterResponse";
|
2023-03-03 15:23:46 +00:00
|
|
|
import { IMiniProfile } from "../models/eft/launcher/IMiniProfile";
|
|
|
|
import { IAkiProfile, Inraid, Vitality } from "../models/eft/profile/IAkiProfile";
|
|
|
|
import {
|
|
|
|
IProfileChangeNicknameRequestData
|
|
|
|
} from "../models/eft/profile/IProfileChangeNicknameRequestData";
|
|
|
|
import {
|
|
|
|
IProfileChangeVoiceRequestData
|
|
|
|
} from "../models/eft/profile/IProfileChangeVoiceRequestData";
|
|
|
|
import { IProfileCreateRequestData } from "../models/eft/profile/IProfileCreateRequestData";
|
|
|
|
import { ISearchFriendRequestData } from "../models/eft/profile/ISearchFriendRequestData";
|
|
|
|
import { ISearchFriendResponse } from "../models/eft/profile/ISearchFriendResponse";
|
|
|
|
import { IValidateNicknameRequestData } from "../models/eft/profile/IValidateNicknameRequestData";
|
|
|
|
import { MessageType } from "../models/enums/MessageType";
|
|
|
|
import { QuestStatus } from "../models/enums/QuestStatus";
|
2023-07-15 11:01:29 +01:00
|
|
|
import { ILogger } from "../models/spt/utils/ILogger";
|
2023-03-03 15:23:46 +00:00
|
|
|
import { EventOutputHolder } from "../routers/EventOutputHolder";
|
|
|
|
import { DatabaseServer } from "../servers/DatabaseServer";
|
|
|
|
import { SaveServer } from "../servers/SaveServer";
|
2023-07-19 13:16:45 +01:00
|
|
|
import { LocalisationService } from "../services/LocalisationService";
|
2023-07-22 14:02:13 +01:00
|
|
|
import { MailSendService } from "../services/MailSendService";
|
2023-03-03 15:23:46 +00:00
|
|
|
import { ProfileFixerService } from "../services/ProfileFixerService";
|
|
|
|
import { HashUtil } from "../utils/HashUtil";
|
|
|
|
import { TimeUtil } from "../utils/TimeUtil";
|
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class ProfileController
|
|
|
|
{
|
|
|
|
constructor(
|
2023-07-15 11:01:29 +01:00
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
2023-03-03 15:23:46 +00:00
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
|
|
|
@inject("SaveServer") protected saveServer: SaveServer,
|
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
|
|
|
@inject("ProfileFixerService") protected profileFixerService: ProfileFixerService,
|
2023-07-19 13:16:45 +01:00
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
2023-07-22 14:02:13 +01:00
|
|
|
@inject("MailSendService") protected mailSendService: MailSendService,
|
2023-03-03 15:23:46 +00:00
|
|
|
@inject("PlayerScavGenerator") protected playerScavGenerator: PlayerScavGenerator,
|
|
|
|
@inject("EventOutputHolder") protected eventOutputHolder: EventOutputHolder,
|
|
|
|
@inject("TraderHelper") protected traderHelper: TraderHelper,
|
|
|
|
@inject("DialogueHelper") protected dialogueHelper: DialogueHelper,
|
|
|
|
@inject("QuestHelper") protected questHelper: QuestHelper,
|
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper
|
|
|
|
)
|
|
|
|
{ }
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle /launcher/profiles
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public getMiniProfiles(): IMiniProfile[]
|
|
|
|
{
|
|
|
|
const miniProfiles: IMiniProfile[] = [];
|
|
|
|
|
|
|
|
for (const sessionIdKey in this.saveServer.getProfiles())
|
|
|
|
{
|
|
|
|
miniProfiles.push(this.getMiniProfile(sessionIdKey));
|
|
|
|
}
|
|
|
|
|
|
|
|
return miniProfiles;
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle launcher/profile/info
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public getMiniProfile(sessionID: string): any
|
|
|
|
{
|
|
|
|
const maxlvl = this.profileHelper.getMaxLevel();
|
|
|
|
const profile = this.saveServer.getProfile(sessionID);
|
|
|
|
const pmc = profile.characters.pmc;
|
|
|
|
|
|
|
|
// make sure character completed creation
|
2023-10-19 11:37:07 +01:00
|
|
|
if (!((pmc?.Info?.Level)))
|
2023-03-03 15:23:46 +00:00
|
|
|
{
|
|
|
|
return {
|
2023-10-19 11:37:07 +01:00
|
|
|
username: profile.info.username,
|
|
|
|
nickname: "unknown",
|
|
|
|
side: "unknown",
|
|
|
|
currlvl: 0,
|
|
|
|
currexp: 0,
|
|
|
|
prevexp: 0,
|
|
|
|
nextlvl: 0,
|
|
|
|
maxlvl: maxlvl,
|
|
|
|
akiData: this.profileHelper.getDefaultAkiDataObject()
|
2023-03-03 15:23:46 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const currlvl = pmc.Info.Level;
|
|
|
|
const nextlvl = this.profileHelper.getExperience(currlvl + 1);
|
|
|
|
const result = {
|
2023-10-19 11:37:07 +01:00
|
|
|
username: profile.info.username,
|
|
|
|
nickname: pmc.Info.Nickname,
|
|
|
|
side: pmc.Info.Side,
|
|
|
|
currlvl: pmc.Info.Level,
|
|
|
|
currexp: pmc.Info.Experience ?? 0,
|
|
|
|
prevexp: (currlvl === 0) ? 0 : this.profileHelper.getExperience(currlvl),
|
|
|
|
nextlvl: nextlvl,
|
|
|
|
maxlvl: maxlvl,
|
|
|
|
akiData: profile.aki
|
2023-03-03 15:23:46 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle client/game/profile/list
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public getCompleteProfile(sessionID: string): IPmcData[]
|
|
|
|
{
|
|
|
|
return this.profileHelper.getCompleteProfile(sessionID);
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle client/game/profile/create
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public createProfile(info: IProfileCreateRequestData, sessionID: string): void
|
|
|
|
{
|
|
|
|
const account = this.saveServer.getProfile(sessionID).info;
|
|
|
|
const profile: TemplateSide = this.databaseServer.getTables().templates.profiles[account.edition][info.side.toLowerCase()];
|
|
|
|
const pmcData = profile.character;
|
|
|
|
|
2023-07-15 11:01:29 +01:00
|
|
|
// Delete existing profile
|
|
|
|
this.deleteProfileBySessionId(sessionID);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
// PMC
|
2023-03-03 15:23:46 +00:00
|
|
|
pmcData._id = `pmc${sessionID}`;
|
2023-10-10 11:03:20 +00:00
|
|
|
pmcData.aid = account.aid;
|
2023-03-03 15:23:46 +00:00
|
|
|
pmcData.savage = `scav${sessionID}`;
|
2023-10-10 11:03:20 +00:00
|
|
|
pmcData.sessionId = sessionID;
|
2023-03-03 15:23:46 +00:00
|
|
|
pmcData.Info.Nickname = info.nickname;
|
|
|
|
pmcData.Info.LowerNickname = info.nickname.toLowerCase();
|
|
|
|
pmcData.Info.RegistrationDate = this.timeUtil.getTimestamp();
|
|
|
|
pmcData.Info.Voice = this.databaseServer.getTables().templates.customization[info.voiceId]._name;
|
|
|
|
pmcData.Stats = this.profileHelper.getDefaultCounters();
|
|
|
|
pmcData.Customization.Head = info.headId;
|
|
|
|
pmcData.Health.UpdateTime = this.timeUtil.getTimestamp();
|
|
|
|
pmcData.Quests = [];
|
2023-10-10 11:03:20 +00:00
|
|
|
pmcData.Hideout.Seed = this.timeUtil.getTimestamp() + (8 * 60 * 60 * 24 * 365); // 8 years in future why? who knows, we saw it in live
|
2023-03-03 15:23:46 +00:00
|
|
|
pmcData.RepeatableQuests = [];
|
|
|
|
pmcData.CarExtractCounts = {};
|
|
|
|
|
2023-10-10 11:03:20 +00:00
|
|
|
if (!pmcData.UnlockedInfo)
|
|
|
|
{
|
|
|
|
pmcData.UnlockedInfo = { unlockedProductionRecipe: [] };
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change item id's to be unique
|
2023-03-03 15:23:46 +00:00
|
|
|
pmcData.Inventory.items = this.itemHelper.replaceIDs(pmcData, pmcData.Inventory.items, null, pmcData.Inventory.fastPanel);
|
2023-10-10 11:03:20 +00:00
|
|
|
pmcData.Inventory.hideoutAreaStashes = {};
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// Create profile
|
|
|
|
const profileDetails: IAkiProfile = {
|
|
|
|
info: account,
|
|
|
|
characters: {
|
|
|
|
pmc: pmcData,
|
|
|
|
scav: {} as IPmcData
|
|
|
|
},
|
|
|
|
suits: profile.suits,
|
2023-10-10 11:03:20 +00:00
|
|
|
userbuilds: profile.userbuilds,
|
2023-03-03 15:23:46 +00:00
|
|
|
dialogues: profile.dialogues,
|
|
|
|
aki: this.profileHelper.getDefaultAkiDataObject(),
|
|
|
|
vitality: {} as Vitality,
|
|
|
|
inraid: {} as Inraid,
|
|
|
|
insurance: [],
|
|
|
|
traderPurchases: {}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.profileFixerService.checkForAndFixPmcProfileIssues(profileDetails.characters.pmc);
|
|
|
|
this.profileFixerService.addMissingHideoutBonusesToProfile(profileDetails.characters.pmc);
|
|
|
|
|
|
|
|
this.saveServer.addProfile(profileDetails);
|
|
|
|
|
|
|
|
if (profile.trader.setQuestsAvailableForStart)
|
|
|
|
{
|
|
|
|
this.questHelper.addAllQuestsToProfile(profileDetails.characters.pmc, [QuestStatus.AvailableForStart]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Profile is flagged as wanting quests set to ready to hand in and collect rewards
|
|
|
|
if (profile.trader.setQuestsAvailableForFinish)
|
|
|
|
{
|
2023-10-10 11:03:20 +00:00
|
|
|
this.questHelper.addAllQuestsToProfile(profileDetails.characters.pmc, [QuestStatus.AvailableForStart, QuestStatus.Started, QuestStatus.AvailableForFinish]);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
|
|
|
// Make unused response so applyQuestReward works
|
|
|
|
const response = this.eventOutputHolder.getOutput(sessionID);
|
|
|
|
|
|
|
|
// Add rewards for starting quests to profile
|
2023-03-08 22:41:57 +00:00
|
|
|
this.givePlayerStartingQuestRewards(profileDetails, sessionID, response);
|
2023-03-03 15:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this.saveServer.getProfile(sessionID).characters.scav = this.generatePlayerScav(sessionID);
|
|
|
|
|
2023-07-15 11:01:29 +01:00
|
|
|
this.resetAllTradersInProfile(sessionID);
|
2023-03-03 15:23:46 +00:00
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
// Store minimal profile and reload it
|
2023-03-03 15:23:46 +00:00
|
|
|
this.saveServer.saveProfile(sessionID);
|
|
|
|
this.saveServer.loadProfile(sessionID);
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
// Completed account creation
|
2023-03-03 15:23:46 +00:00
|
|
|
this.saveServer.getProfile(sessionID).info.wipe = false;
|
|
|
|
this.saveServer.saveProfile(sessionID);
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:01:29 +01:00
|
|
|
/**
|
|
|
|
* Delete a profile
|
|
|
|
* @param sessionID Id of profile to delete
|
|
|
|
*/
|
|
|
|
protected deleteProfileBySessionId(sessionID: string): void
|
|
|
|
{
|
|
|
|
if (sessionID in this.saveServer.getProfiles())
|
|
|
|
{
|
|
|
|
this.saveServer.deleteProfileById(sessionID);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-07-19 13:16:45 +01:00
|
|
|
this.logger.warning(this.localisationService.getText("profile-unable_to_find_profile_by_id_cannot_delete", sessionID));
|
2023-07-15 11:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over all quests in player profile, inspect rewards for the quests current state (accepted/completed)
|
|
|
|
* and send rewards to them in mail
|
|
|
|
* @param profileDetails Player profile
|
|
|
|
* @param sessionID Session id
|
|
|
|
* @param response Event router response
|
|
|
|
*/
|
2023-03-08 22:41:57 +00:00
|
|
|
protected givePlayerStartingQuestRewards(profileDetails: IAkiProfile, sessionID: string, response: IItemEventRouterResponse): void
|
|
|
|
{
|
|
|
|
for (const quest of profileDetails.characters.pmc.Quests)
|
|
|
|
{
|
|
|
|
const questFromDb = this.questHelper.getQuestFromDb(quest.qid, profileDetails.characters.pmc);
|
|
|
|
|
|
|
|
// Get messageId of text to send to player as text message in game
|
|
|
|
// Copy of code from QuestController.acceptQuest()
|
|
|
|
const messageId = this.questHelper.getMessageIdForQuestStart(questFromDb.startedMessageText, questFromDb.description);
|
|
|
|
const itemRewards = this.questHelper.applyQuestReward(profileDetails.characters.pmc, quest.qid, QuestStatus.Started, sessionID, response);
|
2023-07-22 14:02:13 +01:00
|
|
|
|
|
|
|
this.mailSendService.sendLocalisedNpcMessageToPlayer(
|
|
|
|
sessionID,
|
|
|
|
this.traderHelper.getTraderById(questFromDb.traderId),
|
|
|
|
MessageType.QUEST_START,
|
|
|
|
messageId,
|
|
|
|
itemRewards,
|
|
|
|
this.timeUtil.getHoursAsSeconds(100));
|
2023-03-08 22:41:57 +00:00
|
|
|
}
|
2023-07-15 11:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For each trader reset their state to what a level 1 player would see
|
|
|
|
* @param sessionID Session id of profile to reset
|
|
|
|
*/
|
|
|
|
protected resetAllTradersInProfile(sessionID: string): void
|
|
|
|
{
|
|
|
|
for (const traderID in this.databaseServer.getTables().traders)
|
|
|
|
{
|
|
|
|
this.traderHelper.resetTrader(sessionID, traderID);
|
|
|
|
}
|
2023-03-08 22:41:57 +00:00
|
|
|
}
|
|
|
|
|
2023-03-03 15:23:46 +00:00
|
|
|
/**
|
|
|
|
* Generate a player scav object
|
2023-07-15 11:00:35 +01:00
|
|
|
* PMC profile MUST exist first before pscav can be generated
|
2023-03-03 15:23:46 +00:00
|
|
|
* @param sessionID
|
|
|
|
* @returns IPmcData object
|
|
|
|
*/
|
|
|
|
public generatePlayerScav(sessionID: string): IPmcData
|
|
|
|
{
|
|
|
|
return this.playerScavGenerator.generate(sessionID);
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle client/game/profile/nickname/validate
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public validateNickname(info: IValidateNicknameRequestData, sessionID: string): string
|
|
|
|
{
|
|
|
|
if (info.nickname.length < 3)
|
|
|
|
{
|
|
|
|
return "tooshort";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.profileHelper.isNicknameTaken(info, sessionID))
|
|
|
|
{
|
|
|
|
return "taken";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "OK";
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle client/game/profile/nickname/change event
|
|
|
|
* Client allows player to adjust their profile name
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public changeNickname(info: IProfileChangeNicknameRequestData, sessionID: string): string
|
|
|
|
{
|
|
|
|
const output = this.validateNickname(info, sessionID);
|
|
|
|
|
|
|
|
if (output === "OK")
|
|
|
|
{
|
|
|
|
const pmcData = this.profileHelper.getPmcProfile(sessionID);
|
|
|
|
|
|
|
|
pmcData.Info.Nickname = info.nickname;
|
|
|
|
pmcData.Info.LowerNickname = info.nickname.toLowerCase();
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle client/game/profile/voice/change event
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
public changeVoice(info: IProfileChangeVoiceRequestData, sessionID: string): void
|
|
|
|
{
|
|
|
|
const pmcData = this.profileHelper.getPmcProfile(sessionID);
|
|
|
|
pmcData.Info.Voice = info.voice;
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:00:35 +01:00
|
|
|
/**
|
|
|
|
* Handle client/game/profile/search
|
|
|
|
*/
|
2023-03-03 15:23:46 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
|
|
public getFriends(info: ISearchFriendRequestData, sessionID: string): ISearchFriendResponse[]
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
_id: this.hashUtil.generate(),
|
|
|
|
Info: {
|
|
|
|
Level: 1,
|
|
|
|
Side: "Bear",
|
|
|
|
Nickname: info.nickname
|
|
|
|
}
|
|
|
|
}
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|