2023-03-03 16:23:46 +01:00
|
|
|
import { inject, injectable } from "tsyringe";
|
|
|
|
|
2023-10-19 19:21:17 +02:00
|
|
|
import { HideoutHelper } from "@spt-aki/helpers/HideoutHelper";
|
|
|
|
import { InventoryHelper } from "@spt-aki/helpers/InventoryHelper";
|
|
|
|
import { ItemHelper } from "@spt-aki/helpers/ItemHelper";
|
|
|
|
import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper";
|
|
|
|
import { TraderHelper } from "@spt-aki/helpers/TraderHelper";
|
|
|
|
import { IPmcData } from "@spt-aki/models/eft/common/IPmcData";
|
|
|
|
import { Bonus, HideoutSlot, IQuestStatus } from "@spt-aki/models/eft/common/tables/IBotBase";
|
|
|
|
import { IPmcDataRepeatableQuest, IRepeatableQuest } from "@spt-aki/models/eft/common/tables/IRepeatableQuests";
|
|
|
|
import { StageBonus } from "@spt-aki/models/eft/hideout/IHideoutArea";
|
|
|
|
import { IAkiProfile } from "@spt-aki/models/eft/profile/IAkiProfile";
|
|
|
|
import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes";
|
|
|
|
import { HideoutAreas } from "@spt-aki/models/enums/HideoutAreas";
|
|
|
|
import { QuestStatus } from "@spt-aki/models/enums/QuestStatus";
|
|
|
|
import { Traders } from "@spt-aki/models/enums/Traders";
|
|
|
|
import { ICoreConfig } from "@spt-aki/models/spt/config/ICoreConfig";
|
|
|
|
import { IRagfairConfig } from "@spt-aki/models/spt/config/IRagfairConfig";
|
|
|
|
import { ILogger } from "@spt-aki/models/spt/utils/ILogger";
|
|
|
|
import { ConfigServer } from "@spt-aki/servers/ConfigServer";
|
|
|
|
import { DatabaseServer } from "@spt-aki/servers/DatabaseServer";
|
|
|
|
import { LocalisationService } from "@spt-aki/services/LocalisationService";
|
|
|
|
import { HashUtil } from "@spt-aki/utils/HashUtil";
|
|
|
|
import { JsonUtil } from "@spt-aki/utils/JsonUtil";
|
|
|
|
import { TimeUtil } from "@spt-aki/utils/TimeUtil";
|
|
|
|
import { Watermark } from "@spt-aki/utils/Watermark";
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
@injectable()
|
|
|
|
export class ProfileFixerService
|
|
|
|
{
|
2023-07-15 11:53:08 +02:00
|
|
|
protected coreConfig: ICoreConfig;
|
2023-07-30 16:22:19 +02:00
|
|
|
protected ragfairConfig: IRagfairConfig;
|
2023-07-15 11:53:08 +02:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
constructor(
|
|
|
|
@inject("WinstonLogger") protected logger: ILogger,
|
|
|
|
@inject("Watermark") protected watermark: Watermark,
|
|
|
|
@inject("HideoutHelper") protected hideoutHelper: HideoutHelper,
|
2023-07-15 11:53:08 +02:00
|
|
|
@inject("InventoryHelper") protected inventoryHelper: InventoryHelper,
|
2023-08-04 12:19:27 +02:00
|
|
|
@inject("TraderHelper") protected traderHelper: TraderHelper,
|
2023-10-10 13:03:20 +02:00
|
|
|
@inject("ProfileHelper") protected profileHelper: ProfileHelper,
|
2023-07-30 16:22:19 +02:00
|
|
|
@inject("ItemHelper") protected itemHelper: ItemHelper,
|
2023-03-03 16:23:46 +01:00
|
|
|
@inject("LocalisationService") protected localisationService: LocalisationService,
|
|
|
|
@inject("TimeUtil") protected timeUtil: TimeUtil,
|
2023-10-10 13:03:20 +02:00
|
|
|
@inject("JsonUtil") protected jsonUtil: JsonUtil,
|
|
|
|
@inject("HashUtil") protected hashUtil: HashUtil,
|
2023-07-15 11:53:08 +02:00
|
|
|
@inject("DatabaseServer") protected databaseServer: DatabaseServer,
|
2023-11-13 17:13:25 +01:00
|
|
|
@inject("ConfigServer") protected configServer: ConfigServer,
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
2023-07-15 11:53:08 +02:00
|
|
|
{
|
|
|
|
this.coreConfig = this.configServer.getConfig(ConfigTypes.CORE);
|
2023-07-30 16:22:19 +02:00
|
|
|
this.ragfairConfig = this.configServer.getConfig(ConfigTypes.RAGFAIR);
|
2023-07-15 11:53:08 +02:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Find issues in the pmc profile data that may cause issues and fix them
|
|
|
|
* @param pmcProfile profile to check and fix
|
|
|
|
*/
|
|
|
|
public checkForAndFixPmcProfileIssues(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
this.removeDanglingConditionCounters(pmcProfile);
|
|
|
|
this.removeDanglingBackendCounters(pmcProfile);
|
|
|
|
this.addMissingRepeatableQuestsProperty(pmcProfile);
|
|
|
|
this.addLighthouseKeeperIfMissing(pmcProfile);
|
|
|
|
this.addUnlockedInfoObjectIfMissing(pmcProfile);
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
if (pmcProfile.Inventory)
|
|
|
|
{
|
|
|
|
this.addHideoutAreaStashes(pmcProfile);
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
if (pmcProfile.Hideout)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
this.migrateImprovements(pmcProfile);
|
2023-03-03 16:23:46 +01:00
|
|
|
this.addMissingBonusesProperty(pmcProfile);
|
|
|
|
this.addMissingWallImprovements(pmcProfile);
|
2023-10-10 13:03:20 +02:00
|
|
|
this.addMissingHideoutWallAreas(pmcProfile);
|
|
|
|
this.addMissingGunStandContainerImprovements(pmcProfile);
|
|
|
|
this.ensureGunStandLevelsMatch(pmcProfile);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
this.removeResourcesFromSlotsInHideoutWithoutLocationIndexValue(pmcProfile);
|
|
|
|
|
|
|
|
this.reorderHideoutAreasWithResouceInputs(pmcProfile);
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
if (
|
|
|
|
pmcProfile.Hideout.Areas[HideoutAreas.GENERATOR].slots.length <
|
|
|
|
(6 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.Generator.Slots)
|
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
this.logger.debug("Updating generator area slots to a size of 6 + hideout management skill");
|
2023-11-13 17:13:25 +01:00
|
|
|
this.addEmptyObjectsToHideoutAreaSlots(
|
|
|
|
HideoutAreas.GENERATOR,
|
|
|
|
6 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.Generator
|
|
|
|
.Slots,
|
|
|
|
pmcProfile,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
if (
|
|
|
|
pmcProfile.Hideout.Areas[HideoutAreas.WATER_COLLECTOR].slots.length <
|
|
|
|
(1 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.WaterCollector.Slots)
|
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
this.logger.debug("Updating water collector area slots to a size of 1 + hideout management skill");
|
2023-11-13 17:13:25 +01:00
|
|
|
this.addEmptyObjectsToHideoutAreaSlots(
|
|
|
|
HideoutAreas.WATER_COLLECTOR,
|
|
|
|
1 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.WaterCollector.Slots,
|
|
|
|
pmcProfile,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
if (
|
|
|
|
pmcProfile.Hideout.Areas[HideoutAreas.AIR_FILTERING].slots.length <
|
|
|
|
(3 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.AirFilteringUnit.Slots)
|
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
this.logger.debug("Updating air filter area slots to a size of 3 + hideout management skill");
|
2023-11-13 17:13:25 +01:00
|
|
|
this.addEmptyObjectsToHideoutAreaSlots(
|
|
|
|
HideoutAreas.AIR_FILTERING,
|
|
|
|
3 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.AirFilteringUnit.Slots,
|
|
|
|
pmcProfile,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// BTC Farm doesnt have extra slots for hideout management, but we still check for modded stuff!!
|
2023-11-13 17:13:25 +01:00
|
|
|
if (
|
|
|
|
pmcProfile.Hideout.Areas[HideoutAreas.BITCOIN_FARM].slots.length <
|
|
|
|
(50 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.BitcoinFarm.Slots)
|
|
|
|
)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
this.logger.debug("Updating bitcoin farm area slots to a size of 50 + hideout management skill");
|
2023-11-13 17:13:25 +01:00
|
|
|
this.addEmptyObjectsToHideoutAreaSlots(
|
|
|
|
HideoutAreas.BITCOIN_FARM,
|
|
|
|
50 +
|
|
|
|
this.databaseServer.getTables().globals.config.SkillsSettings.HideoutManagement.EliteSlots
|
|
|
|
.BitcoinFarm.Slots,
|
|
|
|
pmcProfile,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.fixNullTraderSalesSums(pmcProfile);
|
|
|
|
this.updateProfilePocketsToNewId(pmcProfile);
|
|
|
|
this.updateProfileQuestDataValues(pmcProfile);
|
2023-07-30 16:22:19 +02:00
|
|
|
|
2023-08-06 13:36:07 +02:00
|
|
|
if (Object.keys(this.ragfairConfig.dynamic.unreasonableModPrices).length > 0)
|
2023-07-30 16:22:19 +02:00
|
|
|
{
|
|
|
|
this.adjustUnreasonableModFleaPrices();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
protected addMissingGunStandContainerImprovements(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
const weaponStandArea = pmcProfile.Hideout.Areas.find((x) => x.type === HideoutAreas.WEAPON_STAND);
|
2023-10-10 13:03:20 +02:00
|
|
|
if (!weaponStandArea || weaponStandArea.level === 0)
|
|
|
|
{
|
|
|
|
// No stand in profile or its level 0, skip
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const db = this.databaseServer.getTables();
|
2023-11-13 17:13:25 +01:00
|
|
|
const hideoutStandAreaDb = db.hideout.areas.find((x) => x.type === HideoutAreas.WEAPON_STAND);
|
|
|
|
const hideoutStandSecondaryAreaDb = db.hideout.areas.find((x) => x.parentArea === hideoutStandAreaDb._id);
|
2023-10-10 13:03:20 +02:00
|
|
|
const stageCurrentAt = hideoutStandAreaDb.stages[weaponStandArea.level];
|
|
|
|
const hideoutStandStashId = pmcProfile.Inventory.hideoutAreaStashes[HideoutAreas.WEAPON_STAND];
|
|
|
|
const hideoutSecondaryStashId = pmcProfile.Inventory.hideoutAreaStashes[HideoutAreas.WEAPON_STAND_SECONDARY];
|
|
|
|
|
|
|
|
// `hideoutAreaStashes` empty but profile has built gun stand
|
|
|
|
if (!hideoutStandStashId && stageCurrentAt)
|
|
|
|
{
|
|
|
|
// Value is missing, add it
|
|
|
|
pmcProfile.Inventory.hideoutAreaStashes[HideoutAreas.WEAPON_STAND] = hideoutStandAreaDb._id;
|
2023-11-13 17:13:25 +01:00
|
|
|
pmcProfile.Inventory.hideoutAreaStashes[HideoutAreas.WEAPON_STAND_SECONDARY] =
|
|
|
|
hideoutStandSecondaryAreaDb._id;
|
2023-10-10 13:03:20 +02:00
|
|
|
|
|
|
|
// Add stash item to profile
|
2023-11-13 17:13:25 +01:00
|
|
|
const gunStandStashItem = pmcProfile.Inventory.items.find((x) => x._id === hideoutStandAreaDb._id);
|
2023-10-10 13:03:20 +02:00
|
|
|
if (gunStandStashItem)
|
|
|
|
{
|
|
|
|
gunStandStashItem._tpl = stageCurrentAt.container;
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`Updated existing gun stand inventory stash: ${gunStandStashItem._id} tpl to ${stageCurrentAt.container}`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
pmcProfile.Inventory.items.push({_id: hideoutStandAreaDb._id, _tpl: stageCurrentAt.container});
|
|
|
|
this.logger.debug(
|
|
|
|
`Added missing gun stand inventory stash: ${hideoutStandAreaDb._id} tpl to ${stageCurrentAt.container}`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add secondary stash item to profile
|
2023-11-13 17:13:25 +01:00
|
|
|
const gunStandStashSecondaryItem = pmcProfile.Inventory.items.find((x) =>
|
|
|
|
x._id === hideoutStandSecondaryAreaDb._id
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
if (gunStandStashItem)
|
|
|
|
{
|
|
|
|
gunStandStashSecondaryItem._tpl = stageCurrentAt.container;
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`Updated gun stand existing inventory secondary stash: ${gunStandStashSecondaryItem._id} tpl to ${stageCurrentAt.container}`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
pmcProfile.Inventory.items.push({_id: hideoutStandSecondaryAreaDb._id, _tpl: stageCurrentAt.container});
|
|
|
|
this.logger.debug(
|
|
|
|
`Added missing gun stand inventory secondary stash: ${hideoutStandSecondaryAreaDb._id} tpl to ${stageCurrentAt.container}`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
const stashItem = pmcProfile.Inventory.items?.find((x) => x._id === hideoutStandAreaDb._id);
|
2023-10-10 13:03:20 +02:00
|
|
|
// `hideoutAreaStashes` has value related stash inventory items tpl doesnt match what's expected
|
|
|
|
if (hideoutStandStashId && stashItem?._tpl !== stageCurrentAt.container)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`primary Stash tpl was: ${stashItem._tpl}, but should be ${stageCurrentAt.container}, updating`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
// The id inside the profile does not match what the hideout db value is, out of sync, adjust
|
|
|
|
stashItem._tpl = stageCurrentAt.container;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
const stashSecondaryItem = pmcProfile.Inventory.items?.find((x) => x._id === hideoutStandSecondaryAreaDb._id);
|
2023-10-10 13:03:20 +02:00
|
|
|
// `hideoutAreaStashes` has value related stash inventory items tpl doesnt match what's expected
|
|
|
|
if (hideoutSecondaryStashId && stashSecondaryItem?._tpl !== stageCurrentAt.container)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`Secondary stash tpl was: ${stashSecondaryItem._tpl}, but should be ${stageCurrentAt.container}, updating`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
// The id inside the profile does not match what the hideout db value is, out of sync, adjust
|
|
|
|
stashSecondaryItem._tpl = stageCurrentAt.container;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected ensureGunStandLevelsMatch(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
// only proceed if stand is level 1 or above
|
2023-11-13 17:13:25 +01:00
|
|
|
const gunStandParent = pmcProfile.Hideout.Areas.find((x) => x.type === HideoutAreas.WEAPON_STAND);
|
2023-10-10 13:03:20 +02:00
|
|
|
if (gunStandParent && gunStandParent.level > 0)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
const gunStandChild = pmcProfile.Hideout.Areas.find((x) => x.type === HideoutAreas.WEAPON_STAND_SECONDARY);
|
2023-10-10 13:03:20 +02:00
|
|
|
if (gunStandChild && gunStandParent.level !== gunStandChild.level)
|
|
|
|
{
|
|
|
|
this.logger.success("Upgraded gun stand levels to match");
|
|
|
|
gunStandChild.level = gunStandParent.level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-11-13 17:13:25 +01:00
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
protected addHideoutAreaStashes(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (!pmcProfile?.Inventory?.hideoutAreaStashes)
|
|
|
|
{
|
|
|
|
this.logger.debug("Added missing hideoutAreaStashes to inventory");
|
|
|
|
pmcProfile.Inventory.hideoutAreaStashes = {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected addMissingHideoutWallAreas(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
if (!pmcProfile.Hideout.Areas.find((x) => x.type === HideoutAreas.WEAPON_STAND))
|
2023-10-10 13:03:20 +02:00
|
|
|
{
|
|
|
|
pmcProfile.Hideout.Areas.push(
|
|
|
|
{
|
|
|
|
type: 24,
|
|
|
|
level: 0,
|
|
|
|
active: true,
|
|
|
|
passiveBonusesEnabled: true,
|
|
|
|
completeTime: 0,
|
|
|
|
constructing: false,
|
|
|
|
slots: [],
|
2023-11-13 17:13:25 +01:00
|
|
|
lastRecipe: "",
|
|
|
|
},
|
2023-10-10 13:03:20 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
if (!pmcProfile.Hideout.Areas.find((x) => x.type === HideoutAreas.WEAPON_STAND_SECONDARY))
|
2023-10-10 13:03:20 +02:00
|
|
|
{
|
|
|
|
pmcProfile.Hideout.Areas.push(
|
|
|
|
{
|
|
|
|
type: 25,
|
|
|
|
level: 0,
|
|
|
|
active: true,
|
|
|
|
passiveBonusesEnabled: true,
|
|
|
|
completeTime: 0,
|
|
|
|
constructing: false,
|
|
|
|
slots: [],
|
2023-11-13 17:13:25 +01:00
|
|
|
lastRecipe: "",
|
|
|
|
},
|
2023-10-10 13:03:20 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-30 16:22:19 +02:00
|
|
|
protected adjustUnreasonableModFleaPrices(): void
|
|
|
|
{
|
|
|
|
const db = this.databaseServer.getTables();
|
|
|
|
const fleaPrices = db.templates.prices;
|
|
|
|
const handbookPrices = db.templates.handbook.Items;
|
2023-08-06 13:36:07 +02:00
|
|
|
|
|
|
|
for (const itemTypeKey in this.ragfairConfig.dynamic.unreasonableModPrices)
|
2023-07-30 16:22:19 +02:00
|
|
|
{
|
2023-08-06 13:36:07 +02:00
|
|
|
const details = this.ragfairConfig.dynamic.unreasonableModPrices[itemTypeKey];
|
|
|
|
if (!details?.enabled)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const itemTpl in fleaPrices)
|
2023-07-30 16:22:19 +02:00
|
|
|
{
|
2023-08-06 13:36:07 +02:00
|
|
|
if (!this.itemHelper.isOfBaseclass(itemTpl, itemTypeKey))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
const itemHandbookPrice = handbookPrices.find((x) => x.Id === itemTpl);
|
2023-07-30 16:22:19 +02:00
|
|
|
if (!itemHandbookPrice)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-08-06 13:36:07 +02:00
|
|
|
if (fleaPrices[itemTpl] > (itemHandbookPrice.Price * details.handbookPriceOverMultiplier))
|
2023-07-30 16:22:19 +02:00
|
|
|
{
|
|
|
|
if (fleaPrices[itemTpl] <= 1)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
continue;
|
2023-07-30 16:22:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Price is over limit, adjust
|
2023-08-06 13:36:07 +02:00
|
|
|
fleaPrices[itemTpl] = itemHandbookPrice.Price * details.newPriceHandbookMultiplier;
|
2023-07-30 16:22:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add tag to profile to indicate when it was made
|
2023-11-13 17:13:25 +01:00
|
|
|
* @param fullProfile
|
2023-03-03 16:23:46 +01:00
|
|
|
*/
|
|
|
|
public addMissingAkiVersionTagToProfile(fullProfile: IAkiProfile): void
|
|
|
|
{
|
|
|
|
if (!fullProfile.aki)
|
|
|
|
{
|
|
|
|
this.logger.debug("Adding aki object to profile");
|
|
|
|
fullProfile.aki = {
|
2023-07-21 19:08:32 +02:00
|
|
|
version: this.watermark.getVersionTag(),
|
2023-11-13 17:13:25 +01:00
|
|
|
receivedGifts: [],
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* TODO - make this non-public - currently used by RepeatableQuestController
|
|
|
|
* Remove unused condition counters
|
|
|
|
* @param pmcProfile profile to remove old counters from
|
|
|
|
*/
|
|
|
|
public removeDanglingConditionCounters(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (pmcProfile.ConditionCounters)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
pmcProfile.ConditionCounters.Counters = pmcProfile.ConditionCounters.Counters.filter((c) => c.qid !== null);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public addLighthouseKeeperIfMissing(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (!pmcProfile.TradersInfo)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
// only add if other traders exist, means this is pre-patch 13 profile
|
2023-03-03 16:23:46 +01:00
|
|
|
if (!pmcProfile.TradersInfo[Traders.LIGHTHOUSEKEEPER] && Object.keys(pmcProfile.TradersInfo).length > 0)
|
|
|
|
{
|
|
|
|
this.logger.warning("Added missing Lighthouse keeper trader to pmc profile");
|
|
|
|
pmcProfile.TradersInfo[Traders.LIGHTHOUSEKEEPER] = {
|
|
|
|
unlocked: false,
|
|
|
|
disabled: false,
|
|
|
|
salesSum: 0,
|
|
|
|
standing: 0.2,
|
|
|
|
loyaltyLevel: 1,
|
2023-11-13 17:13:25 +01:00
|
|
|
nextResupply: this.timeUtil.getTimestamp() + 3600, // now + 1 hour
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected addUnlockedInfoObjectIfMissing(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (!pmcProfile.UnlockedInfo)
|
|
|
|
{
|
|
|
|
this.logger.debug("Adding UnlockedInfo object to profile");
|
|
|
|
pmcProfile.UnlockedInfo = {
|
2023-11-13 17:13:25 +01:00
|
|
|
unlockedProductionRecipe: [],
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected removeDanglingBackendCounters(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (pmcProfile.BackendCounters)
|
|
|
|
{
|
|
|
|
const counterKeysToRemove: string[] = [];
|
|
|
|
const activeQuests = this.getActiveRepeatableQuests(pmcProfile.RepeatableQuests);
|
|
|
|
for (const [key, backendCounter] of Object.entries(pmcProfile.BackendCounters))
|
|
|
|
{
|
|
|
|
if (pmcProfile.RepeatableQuests && activeQuests.length > 0)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
const existsInActiveRepeatableQuests = activeQuests.some((x) => x._id === backendCounter.qid);
|
|
|
|
const existsInQuests = pmcProfile.Quests.some((q) => q.qid === backendCounter.qid);
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
// if BackendCounter's quest is neither in activeQuests nor Quests it's stale
|
2023-10-10 13:03:20 +02:00
|
|
|
if (!existsInActiveRepeatableQuests && !existsInQuests)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
counterKeysToRemove.push(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const counterKeyToRemove of counterKeysToRemove)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
this.logger.debug(`Removed ${counterKeyToRemove} backend count object`);
|
2023-11-13 17:13:25 +01:00
|
|
|
delete pmcProfile.BackendCounters[counterKeyToRemove];
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected getActiveRepeatableQuests(repeatableQuests: IPmcDataRepeatableQuest[]): IRepeatableQuest[]
|
|
|
|
{
|
|
|
|
let activeQuests = [];
|
2023-10-31 18:46:14 +01:00
|
|
|
for (const repeatableQuest of repeatableQuests)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-10-31 18:46:14 +01:00
|
|
|
if (repeatableQuest.activeQuests.length > 0)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
// daily/weekly collection has active quests in them, add to array and return
|
2023-10-31 18:46:14 +01:00
|
|
|
activeQuests = activeQuests.concat(repeatableQuest.activeQuests);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
2023-10-31 18:46:14 +01:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
return activeQuests;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected fixNullTraderSalesSums(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
for (const traderId in pmcProfile.TradersInfo)
|
|
|
|
{
|
|
|
|
const trader = pmcProfile.TradersInfo[traderId];
|
2023-11-13 17:13:25 +01:00
|
|
|
if (trader && trader.salesSum === null)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-07-19 14:16:45 +02:00
|
|
|
this.logger.warning(`trader ${traderId} has a null salesSum value, resetting to 0`);
|
2023-03-03 16:23:46 +01:00
|
|
|
trader.salesSum = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected addMissingBonusesProperty(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-11-13 17:51:02 +01:00
|
|
|
if (typeof pmcProfile.Bonuses === "undefined")
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-11-13 17:51:02 +01:00
|
|
|
pmcProfile.Bonuses = [];
|
2023-03-03 16:23:46 +01:00
|
|
|
this.logger.debug("Missing Bonuses property added to profile");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adjust profile quest status and statusTimers object values
|
|
|
|
* quest.status is numeric e.g. 2
|
|
|
|
* quest.statusTimers keys are numeric as strings e.g. "2"
|
|
|
|
* @param pmcProfile profile to update
|
|
|
|
*/
|
|
|
|
protected updateProfileQuestDataValues(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (!pmcProfile.Quests)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2023-10-10 13:03:20 +02:00
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
const fixes = new Map<any, number>();
|
2023-10-10 13:03:20 +02:00
|
|
|
const questsToDelete: IQuestStatus[] = [];
|
|
|
|
const fullProfile = this.profileHelper.getFullProfile(pmcProfile.sessionId);
|
|
|
|
const isDevProfile = fullProfile?.info.edition.toLowerCase() === "spt developer";
|
2023-03-03 16:23:46 +01:00
|
|
|
for (const quest of pmcProfile.Quests)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
// Old profiles had quests with a bad status of 0 (invalid) added to profile, remove them
|
|
|
|
// E.g. compensation for damage showing before standing check was added to getClientQuests()
|
|
|
|
if (quest.status === 0 && !isDevProfile)
|
|
|
|
{
|
|
|
|
questsToDelete.push(quest);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
if (quest.status && !Number(quest.status))
|
|
|
|
{
|
|
|
|
if (fixes.has(quest.status))
|
2023-11-13 17:13:25 +01:00
|
|
|
{
|
2023-03-03 16:23:46 +01:00
|
|
|
fixes.set(quest.status, fixes.get(quest.status) + 1);
|
2023-11-13 17:13:25 +01:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
else
|
2023-11-13 17:13:25 +01:00
|
|
|
{
|
2023-03-03 16:23:46 +01:00
|
|
|
fixes.set(quest.status, 1);
|
2023-11-13 17:13:25 +01:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
const newQuestStatus = QuestStatus[quest.status];
|
|
|
|
quest.status = <QuestStatus><unknown>newQuestStatus;
|
|
|
|
|
|
|
|
for (const statusTimer in quest.statusTimers)
|
|
|
|
{
|
|
|
|
if (!Number(statusTimer))
|
|
|
|
{
|
|
|
|
const newKey = QuestStatus[statusTimer];
|
|
|
|
quest.statusTimers[newKey] = quest.statusTimers[statusTimer];
|
2023-11-13 17:13:25 +01:00
|
|
|
delete quest.statusTimers[statusTimer];
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
for (const questToDelete of questsToDelete)
|
|
|
|
{
|
|
|
|
pmcProfile.Quests.splice(pmcProfile.Quests.indexOf(questToDelete), 1);
|
|
|
|
}
|
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
if (fixes.size > 0)
|
2023-11-13 17:13:25 +01:00
|
|
|
{
|
|
|
|
this.logger.debug(
|
|
|
|
`Updated quests values: ${
|
|
|
|
Array.from(fixes.entries()).map(([k, v]) => `(${k}: ${v} times)`).join(", ")
|
|
|
|
}`,
|
|
|
|
);
|
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected addMissingRepeatableQuestsProperty(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
if (pmcProfile.RepeatableQuests)
|
|
|
|
{
|
|
|
|
let repeatablesCompatible = true;
|
|
|
|
for (const currentRepeatable of pmcProfile.RepeatableQuests)
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
!(currentRepeatable.changeRequirement &&
|
2023-11-13 17:13:25 +01:00
|
|
|
currentRepeatable.activeQuests.every(
|
|
|
|
(x) => (typeof x.changeCost !== "undefined" && typeof x.changeStandingCost !== "undefined"),
|
|
|
|
))
|
2023-03-03 16:23:46 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
repeatablesCompatible = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!repeatablesCompatible)
|
|
|
|
{
|
|
|
|
pmcProfile.RepeatableQuests = [];
|
|
|
|
this.logger.debug("Missing RepeatableQuests property added to profile");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Some profiles have hideout maxed and therefore no improvements
|
|
|
|
* @param pmcProfile Profile to add improvement data to
|
|
|
|
*/
|
|
|
|
protected addMissingWallImprovements(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
const profileWallArea = pmcProfile.Hideout.Areas[HideoutAreas.EMERGENCY_WALL];
|
2023-11-13 17:13:25 +01:00
|
|
|
const wallDb = this.databaseServer.getTables().hideout.areas.find((x) =>
|
|
|
|
x.type === HideoutAreas.EMERGENCY_WALL
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
if (profileWallArea.level > 0)
|
|
|
|
{
|
|
|
|
for (let i = 0; i < profileWallArea.level; i++)
|
|
|
|
{
|
|
|
|
// Get wall stage from db
|
|
|
|
const wallStageDb = wallDb.stages[i];
|
|
|
|
if (wallStageDb.improvements.length === 0)
|
|
|
|
{
|
|
|
|
// No improvements, skip
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const improvement of wallStageDb.improvements)
|
|
|
|
{
|
|
|
|
// Don't overwrite existing improvement
|
2023-10-10 13:03:20 +02:00
|
|
|
if (pmcProfile.Hideout.Improvement[improvement.id])
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
pmcProfile.Hideout.Improvement[improvement.id] = {
|
2023-03-03 16:23:46 +01:00
|
|
|
completed: true,
|
2023-11-13 17:13:25 +01:00
|
|
|
improveCompleteTimestamp: this.timeUtil.getTimestamp() + i, // add some variability
|
2023-03-03 16:23:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
this.logger.debug(`Added wall improvement ${improvement.id} to profile`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A new property was added to slot items "locationIndex", if this is missing, the hideout slot item must be removed
|
|
|
|
* @param pmcProfile Profile to find and remove slots from
|
|
|
|
*/
|
|
|
|
protected removeResourcesFromSlotsInHideoutWithoutLocationIndexValue(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
for (const area of pmcProfile.Hideout.Areas)
|
|
|
|
{
|
|
|
|
// Skip areas with no resource slots
|
|
|
|
if (area.slots.length === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only slots with location index
|
2023-11-13 17:13:25 +01:00
|
|
|
area.slots = area.slots.filter((x) => "locationIndex" in x);
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Only slots that:
|
|
|
|
// Have an item property and it has at least one item in it
|
|
|
|
// Or
|
|
|
|
// Have no item property
|
2023-11-13 17:13:25 +01:00
|
|
|
area.slots = area.slots.filter((x) => "item" in x && x.item?.length > 0 || !("item" in x));
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hideout slots need to be in a specific order, locationIndex in ascending order
|
|
|
|
* @param pmcProfile profile to edit
|
|
|
|
*/
|
|
|
|
protected reorderHideoutAreasWithResouceInputs(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
const areasToCheck = [
|
|
|
|
HideoutAreas.AIR_FILTERING,
|
|
|
|
HideoutAreas.GENERATOR,
|
|
|
|
HideoutAreas.BITCOIN_FARM,
|
|
|
|
HideoutAreas.WATER_COLLECTOR,
|
|
|
|
];
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
for (const areaId of areasToCheck)
|
|
|
|
{
|
|
|
|
const area = pmcProfile.Hideout.Areas[areaId];
|
|
|
|
|
|
|
|
if (!area)
|
|
|
|
{
|
|
|
|
this.logger.debug(`unable to sort ${areaId} slots, no area found`);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
if (!area.slots || area.slots.length === 0)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
this.logger.debug(`unable to sort ${areaId} slots, no slots found`);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
area.slots = area.slots.sort((a, b) =>
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return a.locationIndex > b.locationIndex ? 1 : -1;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* add in objects equal to the number of slots
|
|
|
|
* @param areaType area to check
|
|
|
|
* @param pmcProfile profile to update
|
|
|
|
*/
|
2023-11-13 17:13:25 +01:00
|
|
|
protected addEmptyObjectsToHideoutAreaSlots(
|
|
|
|
areaType: HideoutAreas,
|
|
|
|
emptyItemCount: number,
|
|
|
|
pmcProfile: IPmcData,
|
|
|
|
): void
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
const area = pmcProfile.Hideout.Areas.find((x) => x.type === areaType);
|
2023-03-03 16:23:46 +01:00
|
|
|
area.slots = this.addObjectsToArray(emptyItemCount, area.slots);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected addObjectsToArray(count: number, slots: HideoutSlot[]): HideoutSlot[]
|
|
|
|
{
|
|
|
|
for (let i = 0; i < count; i++)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
if (!slots.find((x) => x.locationIndex === i))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
slots.push({locationIndex: i});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return slots;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* In 18876 bsg changed the pockets tplid to be one that has 3 additional special slots
|
2023-11-13 17:13:25 +01:00
|
|
|
* @param pmcProfile
|
2023-03-03 16:23:46 +01:00
|
|
|
*/
|
|
|
|
protected updateProfilePocketsToNewId(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
const pocketItem = pmcProfile.Inventory?.items?.find((x) => x.slotId === "Pockets");
|
2023-03-03 16:23:46 +01:00
|
|
|
if (pocketItem)
|
|
|
|
{
|
|
|
|
if (pocketItem._tpl === "557ffd194bdc2d28148b457f")
|
|
|
|
{
|
|
|
|
this.logger.success(this.localisationService.getText("fixer-updated_pockets"));
|
|
|
|
pocketItem._tpl = "627a4e6b255f7527fb05a0f6";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over players hideout areas and find what's build, look for missing bonuses those areas give and add them if missing
|
|
|
|
* @param pmcProfile Profile to update
|
|
|
|
*/
|
|
|
|
public addMissingHideoutBonusesToProfile(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
const profileHideoutAreas = pmcProfile.Hideout.Areas;
|
|
|
|
const profileBonuses = pmcProfile.Bonuses;
|
|
|
|
const dbHideoutAreas = this.databaseServer.getTables().hideout.areas;
|
|
|
|
|
|
|
|
for (const area of profileHideoutAreas)
|
|
|
|
{
|
|
|
|
const areaType = area.type;
|
|
|
|
const level = area.level;
|
|
|
|
|
|
|
|
if (level === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get array of hideout area upgrade levels to check for bonuses
|
|
|
|
// Zero indexed
|
|
|
|
const areaLevelsToCheck: number[] = [];
|
|
|
|
for (let index = 0; index < level + 1; index++)
|
|
|
|
{
|
|
|
|
areaLevelsToCheck.push(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate over area levels, check for bonuses, add if needed
|
2023-11-13 17:13:25 +01:00
|
|
|
const dbArea = dbHideoutAreas.find((x) => x.type === areaType);
|
2023-03-03 16:23:46 +01:00
|
|
|
if (!dbArea)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const level of areaLevelsToCheck)
|
|
|
|
{
|
|
|
|
// Get areas level bonuses from db
|
|
|
|
const levelBonuses = dbArea.stages[level]?.bonuses;
|
|
|
|
if (!levelBonuses || levelBonuses.length === 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate over each bonus for the areas level
|
|
|
|
for (const bonus of levelBonuses)
|
|
|
|
{
|
|
|
|
// Check if profile has bonus
|
|
|
|
const profileBonus = this.getBonusFromProfile(profileBonuses, bonus);
|
|
|
|
if (!profileBonus)
|
|
|
|
{
|
|
|
|
// no bonus, add to profile
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`Profile has level ${level} area ${
|
|
|
|
HideoutAreas[area.type]
|
|
|
|
} but no bonus found, adding ${bonus.type}`,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
this.hideoutHelper.applyPlayerUpgradesBonuses(pmcProfile, bonus);
|
|
|
|
}
|
2023-11-13 17:13:25 +01:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param profileBonuses bonuses from profile
|
|
|
|
* @param bonus bonus to find
|
|
|
|
* @returns matching bonus
|
|
|
|
*/
|
|
|
|
protected getBonusFromProfile(profileBonuses: Bonus[], bonus: StageBonus): Bonus
|
|
|
|
{
|
|
|
|
// match by id first, used by "TextBonus" bonuses
|
|
|
|
if (bonus.id)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
return profileBonuses.find((x) => x.id === bonus.id);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bonus.type.toLowerCase() === "stashsize")
|
|
|
|
{
|
|
|
|
return profileBonuses.find(
|
2023-11-13 17:13:25 +01:00
|
|
|
(x) =>
|
|
|
|
x.type === bonus.type &&
|
|
|
|
x.templateId === bonus.templateId,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bonus.type.toLowerCase() === "additionalslots")
|
|
|
|
{
|
|
|
|
return profileBonuses.find(
|
2023-11-13 17:13:25 +01:00
|
|
|
(x) =>
|
|
|
|
x.type === bonus.type &&
|
|
|
|
x.value === bonus.value &&
|
|
|
|
x.visible === bonus.visible,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return profileBonuses.find(
|
2023-11-13 17:13:25 +01:00
|
|
|
(x) =>
|
|
|
|
x.type === bonus.type &&
|
|
|
|
x.value === bonus.value,
|
|
|
|
);
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks profile inventiory for items that do not exist inside the items db
|
2023-07-15 11:53:08 +02:00
|
|
|
* @param sessionId Session id
|
2023-03-03 16:23:46 +01:00
|
|
|
* @param pmcProfile Profile to check inventory of
|
|
|
|
*/
|
2023-08-03 15:36:53 +02:00
|
|
|
public checkForOrphanedModdedItems(sessionId: string, fullProfile: IAkiProfile): void
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
const itemsDb = this.databaseServer.getTables().templates.items;
|
2023-08-03 15:36:53 +02:00
|
|
|
const pmcProfile = fullProfile.characters.pmc;
|
2023-03-03 16:23:46 +01:00
|
|
|
|
|
|
|
// Get items placed in root of stash
|
|
|
|
// TODO: extend to other areas / sub items
|
2023-11-13 17:13:25 +01:00
|
|
|
const inventoryItemsToCheck = pmcProfile.Inventory.items.filter((x) => ["hideout", "main"].includes(x.slotId));
|
2023-04-12 16:41:48 +02:00
|
|
|
if (!inventoryItemsToCheck)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:53:08 +02:00
|
|
|
// Check each item in inventory to ensure item exists in itemdb
|
2023-04-12 16:41:48 +02:00
|
|
|
for (const item of inventoryItemsToCheck)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
if (!itemsDb[item._tpl])
|
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("fixer-mod_item_found", item._tpl));
|
2023-04-12 16:41:48 +02:00
|
|
|
|
2023-07-15 11:53:08 +02:00
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.success(
|
|
|
|
`Deleting item from inventory and insurance with id: ${item._id} tpl: ${item._tpl}`,
|
|
|
|
);
|
2023-07-15 11:53:08 +02:00
|
|
|
|
|
|
|
// Also deletes from insured array
|
|
|
|
this.inventoryHelper.removeItem(pmcProfile, item._id, sessionId);
|
2023-08-03 15:36:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate over player-made weapon builds, look for missing items and remove weapon preset if found
|
2023-10-10 13:03:20 +02:00
|
|
|
for (const buildId in fullProfile.userbuilds?.weaponBuilds)
|
2023-08-03 15:36:53 +02:00
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
for (const item of fullProfile.userbuilds.weaponBuilds[buildId].items)
|
2023-08-03 15:36:53 +02:00
|
|
|
{
|
|
|
|
// Check item exists in itemsDb
|
|
|
|
if (!itemsDb[item._tpl])
|
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("fixer-mod_item_found", item._tpl));
|
|
|
|
|
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
delete fullProfile.userbuilds.weaponBuilds[buildId];
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.warning(
|
|
|
|
`Item: ${item._tpl} has resulted in the deletion of weapon build: ${buildId}`,
|
|
|
|
);
|
2023-08-03 15:36:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-07-15 11:53:08 +02:00
|
|
|
|
2023-08-03 15:36:53 +02:00
|
|
|
// Iterate over dialogs, looking for messages with items not found in item db, remove message if item found
|
|
|
|
for (const dialogId in fullProfile.dialogues)
|
|
|
|
{
|
|
|
|
const dialog = fullProfile.dialogues[dialogId];
|
|
|
|
if (!dialog?.messages)
|
|
|
|
{
|
|
|
|
continue; // Skip dialog with no messages
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate over all messages in dialog
|
|
|
|
for (const message of dialog.messages)
|
|
|
|
{
|
|
|
|
if (!message.items?.data)
|
|
|
|
{
|
|
|
|
continue; // Skip message with no items
|
|
|
|
}
|
|
|
|
|
2023-10-10 13:03:20 +02:00
|
|
|
// Fix message with no items but have the flags to indicate items to collect
|
|
|
|
if (message.items.data.length === 0 && message.hasRewards)
|
|
|
|
{
|
|
|
|
message.hasRewards = false;
|
|
|
|
message.rewardCollected = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-08-03 15:36:53 +02:00
|
|
|
// Iterate over all items in message
|
|
|
|
for (const item of message.items.data)
|
|
|
|
{
|
|
|
|
// Check item exists in itemsDb
|
|
|
|
if (!itemsDb[item._tpl])
|
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("fixer-mod_item_found", item._tpl));
|
|
|
|
|
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
dialog.messages.splice(dialog.messages.findIndex((x) => x._id === message._id), 1);
|
|
|
|
this.logger.warning(
|
|
|
|
`Item: ${item._tpl} has resulted in the deletion of message: ${message._id} from dialog ${dialogId}`,
|
|
|
|
);
|
2023-08-03 15:36:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2023-07-15 11:53:08 +02:00
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
2023-08-04 11:25:21 +02:00
|
|
|
|
|
|
|
const clothing = this.databaseServer.getTables().templates.customization;
|
|
|
|
for (const suitId of fullProfile.suits)
|
|
|
|
{
|
|
|
|
if (!clothing[suitId])
|
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("fixer-mod_item_found", suitId));
|
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
|
|
|
fullProfile.suits.splice(fullProfile.suits.indexOf(suitId), 1);
|
|
|
|
this.logger.warning(`Non-default suit purchase: ${suitId} removed from profile`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const repeatable of fullProfile.characters.pmc.RepeatableQuests)
|
|
|
|
{
|
|
|
|
for (const activeQuest of repeatable.activeQuests ?? [])
|
|
|
|
{
|
2023-08-04 12:19:27 +02:00
|
|
|
if (!this.traderHelper.traderEnumHasValue(activeQuest.traderId))
|
2023-08-04 11:25:21 +02:00
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("fixer-mod_item_found", activeQuest.traderId));
|
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.warning(
|
|
|
|
`Non-default quest: ${activeQuest._id} from trader: ${activeQuest.traderId} removed from RepeatableQuests list in profile`,
|
|
|
|
);
|
|
|
|
repeatable.activeQuests.splice(
|
|
|
|
repeatable.activeQuests.findIndex((x) => x._id === activeQuest._id),
|
|
|
|
1,
|
|
|
|
);
|
2023-08-04 11:25:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const successReward of activeQuest.rewards.Success)
|
|
|
|
{
|
|
|
|
if (successReward.type === "Item")
|
|
|
|
{
|
|
|
|
for (const rewardItem of successReward.items)
|
|
|
|
{
|
|
|
|
if (!itemsDb[rewardItem._tpl])
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.error(
|
|
|
|
this.localisationService.getText("fixer-mod_item_found", rewardItem._tpl),
|
|
|
|
);
|
2023-08-04 11:25:21 +02:00
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.warning(
|
|
|
|
`Non-default quest: ${activeQuest._id} from trader: ${activeQuest.traderId} removed from RepeatableQuests list in profile`,
|
|
|
|
);
|
|
|
|
repeatable.activeQuests.splice(
|
|
|
|
repeatable.activeQuests.findIndex((x) => x._id === activeQuest._id),
|
|
|
|
1,
|
|
|
|
);
|
2023-08-04 11:25:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const traderId in fullProfile.traderPurchases)
|
|
|
|
{
|
2023-08-04 12:19:27 +02:00
|
|
|
if (!this.traderHelper.traderEnumHasValue(traderId))
|
2023-08-04 11:25:21 +02:00
|
|
|
{
|
|
|
|
this.logger.error(this.localisationService.getText("fixer-mod_item_found", traderId));
|
|
|
|
if (this.coreConfig.fixes.removeModItemsFromProfile)
|
|
|
|
{
|
|
|
|
this.logger.warning(`Non-default trader: ${traderId} removed from traderPurchases list in profile`);
|
|
|
|
delete fullProfile.traderPurchases[traderId];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
2023-11-13 17:13:25 +01:00
|
|
|
|
2023-03-03 16:23:46 +01:00
|
|
|
/**
|
|
|
|
* Add `Improvements` object to hideout if missing - added in eft 13.0.21469
|
|
|
|
* @param pmcProfile profile to update
|
|
|
|
*/
|
|
|
|
public addMissingUpgradesPropertyToHideout(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
if (!pmcProfile.Hideout.Improvement)
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
pmcProfile.Hideout.Improvement = {};
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over associated profile template and check all hideout areas exist, add if not
|
|
|
|
* @param fullProfile Profile to update
|
|
|
|
*/
|
|
|
|
public addMissingHideoutAreasToProfile(fullProfile: IAkiProfile): void
|
|
|
|
{
|
2023-11-13 17:51:02 +01:00
|
|
|
const pmcProfile = fullProfile.characters.pmc;
|
2023-03-03 16:23:46 +01:00
|
|
|
// No profile, probably new account being created
|
|
|
|
if (!pmcProfile?.Hideout)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const profileTemplates = this.databaseServer.getTables().templates.profiles[fullProfile.info.edition];
|
|
|
|
if (!profileTemplates)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const profileTemplate = profileTemplates[pmcProfile.Info.Side.toLowerCase()];
|
|
|
|
if (!profileTemplate)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get all areas from templates/profiles.json
|
|
|
|
for (const area of profileTemplate.character.Hideout.Areas)
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
if (!pmcProfile.Hideout.Areas.find((x) => x.type === area.type))
|
2023-03-03 16:23:46 +01:00
|
|
|
{
|
|
|
|
pmcProfile.Hideout.Areas.push(area);
|
|
|
|
this.logger.debug(`Added missing hideout area ${area.type} to profile`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* These used to be used for storing scav case rewards, rewards are now generated on pickup
|
|
|
|
* @param pmcProfile Profile to update
|
|
|
|
*/
|
|
|
|
public removeLegacyScavCaseProductionCrafts(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
for (const prodKey in pmcProfile.Hideout?.Production)
|
|
|
|
{
|
|
|
|
if (prodKey.startsWith("ScavCase"))
|
|
|
|
{
|
2023-11-13 17:13:25 +01:00
|
|
|
delete pmcProfile.Hideout.Production[prodKey];
|
2023-03-03 16:23:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-10 13:03:20 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 3.7.0 moved AIDs to be numeric, old profiles need to be migrated
|
|
|
|
* We store the old AID value in new field `sessionId`
|
|
|
|
* @param fullProfile Profile to update
|
|
|
|
*/
|
|
|
|
public fixIncorrectAidValue(fullProfile: IAkiProfile): void
|
|
|
|
{
|
|
|
|
// Not a number, regenerate
|
2023-10-31 23:54:59 +01:00
|
|
|
if (Number.isNaN(fullProfile.characters.pmc.aid))
|
2023-10-10 13:03:20 +02:00
|
|
|
{
|
|
|
|
fullProfile.characters.pmc.sessionId = <string><unknown>fullProfile.characters.pmc.aid;
|
|
|
|
fullProfile.characters.pmc.aid = this.hashUtil.generateAccountId();
|
|
|
|
|
|
|
|
fullProfile.characters.scav.sessionId = <string><unknown>fullProfile.characters.pmc.sessionId;
|
|
|
|
fullProfile.characters.scav.aid = fullProfile.characters.pmc.aid;
|
|
|
|
|
|
|
|
fullProfile.info.aid = fullProfile.characters.pmc.aid;
|
|
|
|
|
2023-11-13 17:13:25 +01:00
|
|
|
this.logger.debug(
|
|
|
|
`Migrated AccountId from: ${fullProfile.characters.pmc.sessionId} to: ${fullProfile.characters.pmc.aid}`,
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bsg nested `stats` into a sub object called 'eft'
|
|
|
|
* @param fullProfile Profile to check for and migrate stats data
|
|
|
|
*/
|
|
|
|
public migrateStatsToNewStructure(fullProfile: IAkiProfile): void
|
|
|
|
{
|
|
|
|
// Data is in old structure, migrate
|
|
|
|
if ("OverallCounters" in fullProfile.characters.pmc.Stats)
|
|
|
|
{
|
|
|
|
this.logger.debug("Migrating stats object into new structure");
|
|
|
|
const statsCopy = this.jsonUtil.clone(fullProfile.characters.pmc.Stats);
|
|
|
|
|
|
|
|
// Clear stats object
|
|
|
|
fullProfile.characters.pmc.Stats = {Eft: null};
|
|
|
|
|
|
|
|
fullProfile.characters.pmc.Stats.Eft = <any><unknown>statsCopy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 26126 (7th August) requires bonuses to have an ID, these were not included in the default profile presets
|
|
|
|
* @param pmcProfile Profile to add missing IDs to
|
|
|
|
*/
|
|
|
|
public addMissingIdsToBonuses(pmcProfile: IPmcData): void
|
|
|
|
{
|
|
|
|
let foundBonus = false;
|
|
|
|
for (const bonus of pmcProfile.Bonuses)
|
|
|
|
{
|
|
|
|
if (bonus.id)
|
|
|
|
{
|
|
|
|
// Exists already, skip
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bonus lacks id, find matching hideout area / stage / bonus
|
|
|
|
for (const area of this.databaseServer.getTables().hideout.areas)
|
2023-11-13 17:13:25 +01:00
|
|
|
{
|
2023-10-10 13:03:20 +02:00
|
|
|
// TODO: skip if no stages
|
|
|
|
for (const stageIndex in area.stages)
|
|
|
|
{
|
|
|
|
const stageInfo = area.stages[stageIndex];
|
2023-11-13 17:13:25 +01:00
|
|
|
const matchingBonus = stageInfo.bonuses.find((x) =>
|
|
|
|
x.templateId === bonus.templateId && x.type === bonus.type
|
|
|
|
);
|
2023-10-10 13:03:20 +02:00
|
|
|
if (matchingBonus)
|
|
|
|
{
|
|
|
|
// Add id to bonus, flag bonus as found and exit stage loop
|
|
|
|
bonus.id = matchingBonus.id;
|
|
|
|
this.logger.debug(`Added missing Id: ${bonus.id} to bonus: ${bonus.type}`);
|
|
|
|
foundBonus = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We've found the bonus we're after, break out of area loop
|
|
|
|
if (foundBonus)
|
|
|
|
{
|
|
|
|
foundBonus = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* At some point the property name was changed,migrate data across to new name
|
|
|
|
* @param pmcProfile Profile to migrate improvements in
|
|
|
|
*/
|
|
|
|
protected migrateImprovements(pmcProfile: IPmcData): void
|
|
|
|
{
|
2023-11-13 17:51:02 +01:00
|
|
|
if (pmcProfile.Hideout.Improvements)
|
2023-10-10 13:03:20 +02:00
|
|
|
{
|
|
|
|
// Correct name is `Improvement`
|
2023-11-13 17:51:02 +01:00
|
|
|
pmcProfile.Hideout.Improvement = this.jsonUtil.clone(pmcProfile.Hideout.Improvements);
|
|
|
|
delete pmcProfile.Hideout.Improvements;
|
2023-10-10 13:03:20 +02:00
|
|
|
this.logger.success("Successfully migrated hideout Improvements data to new location, deleted old data");
|
|
|
|
}
|
|
|
|
}
|
2023-11-13 17:13:25 +01:00
|
|
|
}
|