From f9cf3242c86e4d10515533944554bb6c1e48c79d Mon Sep 17 00:00:00 2001 From: Dev Date: Tue, 28 Nov 2023 11:06:08 +0000 Subject: [PATCH] Add new core config entry `fixProfileBreakingInventoryItemIssues`, defaults to off Attempts to fix common issues that happen to profile inventory items: Duplicate items with the same _id value Item Tag names with non alphanumeric characters StackObjectsCount null values --- project/assets/configs/core.json | 5 +- project/src/controllers/GameController.ts | 85 +++++++++++++++++++- project/src/models/spt/config/ICoreConfig.ts | 2 + 3 files changed, 87 insertions(+), 5 deletions(-) diff --git a/project/assets/configs/core.json b/project/assets/configs/core.json index 6560c7f1..27be4ed7 100644 --- a/project/assets/configs/core.json +++ b/project/assets/configs/core.json @@ -7,9 +7,10 @@ "sptFriendNickname": "SPT", "fixes": { "fixShotgunDispersion": true, - "removeModItemsFromProfile": false + "removeModItemsFromProfile": false, + "fixProfileBreakingInventoryItemIssues": false }, "features": { "autoInstallModDependencies": false } -} \ No newline at end of file +} diff --git a/project/src/controllers/GameController.ts b/project/src/controllers/GameController.ts index f6b2f86b..6fc82aef 100644 --- a/project/src/controllers/GameController.ts +++ b/project/src/controllers/GameController.ts @@ -8,7 +8,7 @@ import { ProfileHelper } from "@spt-aki/helpers/ProfileHelper"; import { WeightedRandomHelper } from "@spt-aki/helpers/WeightedRandomHelper"; import { PreAkiModLoader } from "@spt-aki/loaders/PreAkiModLoader"; import { IEmptyRequestData } from "@spt-aki/models/eft/common/IEmptyRequestData"; -import { Exit, ILocationBase } from "@spt-aki/models/eft/common/ILocationBase"; +import { ILocationBase } from "@spt-aki/models/eft/common/ILocationBase"; import { ILooseLoot } from "@spt-aki/models/eft/common/ILooseLoot"; import { IPmcData } from "@spt-aki/models/eft/common/IPmcData"; import { BodyPartHealth } from "@spt-aki/models/eft/common/tables/IBotBase"; @@ -19,7 +19,6 @@ import { IGameKeepAliveResponse } from "@spt-aki/models/eft/game/IGameKeepAliveR import { IGetRaidTimeRequest } from "@spt-aki/models/eft/game/IGetRaidTimeRequest"; import { ExtractChange, IGetRaidTimeResponse } from "@spt-aki/models/eft/game/IGetRaidTimeResponse"; import { IServerDetails } from "@spt-aki/models/eft/game/IServerDetails"; -import { IGetRaidConfigurationRequestData } from "@spt-aki/models/eft/match/IGetRaidConfigurationRequestData"; import { IAkiProfile } from "@spt-aki/models/eft/profile/IAkiProfile"; import { AccountTypes } from "@spt-aki/models/enums/AccountTypes"; import { ConfigTypes } from "@spt-aki/models/enums/ConfigTypes"; @@ -42,6 +41,7 @@ import { LocalisationService } from "@spt-aki/services/LocalisationService"; import { OpenZoneService } from "@spt-aki/services/OpenZoneService"; import { ProfileFixerService } from "@spt-aki/services/ProfileFixerService"; import { SeasonalEventService } from "@spt-aki/services/SeasonalEventService"; +import { HashUtil } from "@spt-aki/utils/HashUtil"; import { JsonUtil } from "@spt-aki/utils/JsonUtil"; import { RandomUtil } from "@spt-aki/utils/RandomUtil"; import { TimeUtil } from "@spt-aki/utils/TimeUtil"; @@ -61,6 +61,7 @@ export class GameController @inject("DatabaseServer") protected databaseServer: DatabaseServer, @inject("JsonUtil") protected jsonUtil: JsonUtil, @inject("TimeUtil") protected timeUtil: TimeUtil, + @inject("HashUtil") protected hashUtil: HashUtil, @inject("PreAkiModLoader") protected preAkiModLoader: PreAkiModLoader, @inject("HttpServerHelper") protected httpServerHelper: HttpServerHelper, @inject("RandomUtil") protected randomUtil: RandomUtil, @@ -143,6 +144,11 @@ export class GameController this.logger.debug(`Started game with sessionId: ${sessionID} ${pmcProfile.Info?.Nickname}`); + if (this.coreConfig.fixes.fixProfileBreakingInventoryItemIssues) + { + this.fixProfileBreakingInventoryItemIssues(pmcProfile) + } + if (pmcProfile.Health) { this.updateProfileHealthValues(pmcProfile); @@ -242,6 +248,79 @@ export class GameController } } + /** + * Attempt to fix common item issues that corrupt profiles + * @param pmcProfile Profile to check items of + */ + protected fixProfileBreakingInventoryItemIssues(pmcProfile: IPmcData): void + { + // Create a mapping of all inventory items, keyed by _id value + const itemMapping = pmcProfile.Inventory.items.reduce((acc, curr) => + { + acc[curr._id] = acc[curr._id] || []; + acc[curr._id].push(curr); + + return acc; + }, {}); + + for (const key in itemMapping) + { + // Only one item for this id, not a dupe + if (itemMapping[key].length === 1) + { + continue; + } + + this.logger.warning(`${itemMapping[key].length - 1} duplicate(s) found for item: ${key}`); + const itemAJson = this.jsonUtil.serialize(itemMapping[key][0]); + const itemBJson = this.jsonUtil.serialize(itemMapping[key][1]); + if (itemAJson === itemBJson) + { + // Both items match, we can safely delete one + const indexOfItemToRemove = pmcProfile.Inventory.items.findIndex(x => x._id === key); + pmcProfile.Inventory.items.splice(indexOfItemToRemove, 1); + this.logger.warning(`Deleted duplicate item: ${key}`); + } + else + { + // Items are different, replace ID with unique value + // Only replace ID if items have no children, we dont want orphaned children + const itemsHaveChildren = pmcProfile.Inventory.items.some(x => x.parentId === key); + if (!itemsHaveChildren) + { + const itemToAdjustId = pmcProfile.Inventory.items.find(x => x._id === key); + itemToAdjustId._id = this.hashUtil.generate(); + this.logger.warning(`Replace duplicate item Id: ${key} with ${itemToAdjustId._id}`); + } + } + } + + // Iterate over all inventory items + for (const item of pmcProfile.Inventory.items.filter(x => x.slotId)) + { + if (!item.upd) + { + // Ignore items without a upd object + continue; + } + + // Check items with a tag that contains non alphanumeric characters + const regxp = /[^a-zA-Z0-9 .]/g; + if (regxp.test(item.upd.Tag?.Name)) + { + this.logger.warning(`Fixed item: ${item._id}s Tag value, removed invalid characters`); + item.upd.Tag.Name = item.upd.Tag.Name.replace(regxp, ''); + } + + // Check items with StackObjectsCount (null) + if (item.upd.StackObjectsCount === null) + { + this.logger.warning(`Fixed item: ${item._id}s null StackObjectsCount value, now set to 1`); + item.upd.StackObjectsCount = 1; + } + } + } + /** * Out of date/incorrectly made trader mods forget this data */ @@ -482,7 +561,7 @@ export class GameController } /** - * singleplayer/settings/getRaidTime + * Handle singleplayer/settings/getRaidTime */ public getRaidTime(sessionId: string, request: IGetRaidTimeRequest): IGetRaidTimeResponse { diff --git a/project/src/models/spt/config/ICoreConfig.ts b/project/src/models/spt/config/ICoreConfig.ts index 9c20726b..3d9c6319 100644 --- a/project/src/models/spt/config/ICoreConfig.ts +++ b/project/src/models/spt/config/ICoreConfig.ts @@ -23,6 +23,8 @@ export interface IGameFixes fixShotgunDispersion: boolean; /** Remove items added by mods when the mod no longer exists - can fix dead profiles stuck at game load*/ removeModItemsFromProfile: boolean; + /** Fix issues that cause the game to not start due to inventory item issues */ + fixProfileBreakingInventoryItemIssues: boolean; } export interface IServerFeatures