Refactor of InsuranceController & New ItemHelper Methods (!151)

This commit is my second go-around at refactoring the `InsuranceController`, attempting to improving the code's modularity, maintainability, and efficiency while squashing a few bugs along the way.

1. **InsuranceController.ts**
    - Removed `ITemplateItem` import, as it is no longer used.
    - Introduced the `adoptOrphanedItems` method to manage orphaned items in the insurance list.
        - Since "normal" items are individually rolled for deletion, and can be nested within one another, there are situations where a parent item is deleted, leaving its children orphaned. This method moves those orphaned children from their missing parent into the root of the insurance container.
    - Overhauled `findItemsToDelete` method to improve its efficiency and readability:
        - Divided the original monolithic method into smaller, specialized methods like `populateItemsMap`, `populateParentAttachmentsMap`, `processRegularItems`, and `processAttachments`.
        - Changed the return type to `Set<string>` for better performance.
        - Introduced `EnrichedItem` interface (a simple extension of the `Item` interface) to add additional item data, like `name` and `maxPrice` to `Item` objects as they're processed throughout the class. This is done in place of repeatedly querying for this data, or complicating return types.
        - Enhanced logging capabilities to debug the item deletion process. Due to the *current* lack of testing available I've stepped up the amount of debug logging that is done. This will hopefully help us find issues in the future.
    - Modified the `rollForItemDelete` method, now renamed to `rollForDelete`, to include more detailed logging, return a boolean directly, and changed the `insuredItem` parameter to be optional.
    - Added new methods for dealing with some of the particulars that arise from item adoption and creating item maps.
    - Improved inline comments and documentation for better code maintainability.

2. **ItemHelper.ts**
    - Added the `isRaidModdable` method to check if an item is *actually* modifiable in-raid, which takes into account not just the item, but the item that it's attached to.
    - Added the `getAttachmentMainParent` method to fetch the main parent item of a given attachment, useful for item hierarchy traversal. For example, if you pass it an item ID of a suppressor, it will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately attached to, even if that gun is located within other multiple containers.
    - Added the `isAttachmentAttached` method to check if an item is an attachment that is currently attached to its parent.

**Fixes:**
 - Resolved an issue that caused item attachments from being property grouped together for deletion rolls. This issue prevented valuable attachments from being taken first.
 - Resolved an issue that caused child items being orphaned when their parent was removed due to an insurance roll. Probable cause of the bug that made the client spaz out and send repeated insurance packages to the profile---Though I'm still unable to reproduce.
 - Probably more...

Co-authored-by: Refringe <brownelltyler@gmail.com>
Reviewed-on: https://dev.sp-tarkov.com/SPT-AKI/Server/pulls/151
Co-authored-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
Co-committed-by: Refringe <refringe@noreply.dev.sp-tarkov.com>
This commit is contained in:
Refringe 2023-10-14 09:05:49 +00:00 committed by chomp
parent 723f8db572
commit 7948e3473c
2 changed files with 375 additions and 165 deletions

View File

@ -6,7 +6,6 @@ import { ProfileHelper } from "../helpers/ProfileHelper";
import { TraderHelper } from "../helpers/TraderHelper"; import { TraderHelper } from "../helpers/TraderHelper";
import { IPmcData } from "../models/eft/common/IPmcData"; import { IPmcData } from "../models/eft/common/IPmcData";
import { Item } from "../models/eft/common/tables/IItem"; import { Item } from "../models/eft/common/tables/IItem";
import { ITemplateItem } from "../models/eft/common/tables/ITemplateItem";
import { IGetInsuranceCostRequestData } from "../models/eft/insurance/IGetInsuranceCostRequestData"; import { IGetInsuranceCostRequestData } from "../models/eft/insurance/IGetInsuranceCostRequestData";
import { import {
IGetInsuranceCostResponseData IGetInsuranceCostResponseData
@ -56,7 +55,7 @@ export class InsuranceController
/** /**
* Process insurance items of all profiles prior to being given back to the player through the mail service. * Process insurance items of all profiles prior to being given back to the player through the mail service.
* *
* @returns void * @returns void
*/ */
public processReturn(): void public processReturn(): void
@ -70,7 +69,7 @@ export class InsuranceController
/** /**
* Process insurance items of a single profile prior to being given back to the player through the mail service. * Process insurance items of a single profile prior to being given back to the player through the mail service.
* *
* @returns void * @returns void
*/ */
public processReturnByProfile(sessionID: string): void public processReturnByProfile(sessionID: string): void
@ -89,7 +88,7 @@ export class InsuranceController
/** /**
* Get all insured items that are ready to be processed in a specific profile. * Get all insured items that are ready to be processed in a specific profile.
* *
* @param sessionID Session ID of the profile to check. * @param sessionID Session ID of the profile to check.
* @param time The time to check ready status against. Current time by default. * @param time The time to check ready status against. Current time by default.
* @returns All insured items that are ready to be processed. * @returns All insured items that are ready to be processed.
@ -110,7 +109,7 @@ export class InsuranceController
/** /**
* This method orchestrates the processing of insured items in a profile. * This method orchestrates the processing of insured items in a profile.
* *
* @param insuranceDetails The insured items to process. * @param insuranceDetails The insured items to process.
* @param sessionID The session ID that should receive the processed items. * @param sessionID The session ID that should receive the processed items.
* @returns void * @returns void
@ -120,14 +119,17 @@ export class InsuranceController
this.logger.debug(`Processing ${insuranceDetails.length} insurance packages, which includes a total of ${insuranceDetails.map(ins => ins.items.length).reduce((acc, len) => acc + len, 0)} items, in profile ${sessionID}`); this.logger.debug(`Processing ${insuranceDetails.length} insurance packages, which includes a total of ${insuranceDetails.map(ins => ins.items.length).reduce((acc, len) => acc + len, 0)} items, in profile ${sessionID}`);
// Iterate over each of the insurance packages. // Iterate over each of the insurance packages.
insuranceDetails.forEach(insured => insuranceDetails.forEach(insured =>
{ {
// Find items that should be deleted from the insured items. // Find items that should be deleted from the insured items.
const itemsToDelete = this.findItemsToDelete(insured); const itemsToDelete = this.findItemsToDelete(insured);
// Actually remove them. // Actually remove them.
this.removeItemsFromInsurance(insured, itemsToDelete); this.removeItemsFromInsurance(insured, itemsToDelete);
// Fix any orphaned items.
this.adoptOrphanedItems(insured);
// Send the mail to the player. // Send the mail to the player.
this.sendMail(sessionID, insured, insured.items.length === 0); this.sendMail(sessionID, insured, insured.items.length === 0);
@ -138,7 +140,7 @@ export class InsuranceController
/** /**
* Remove an insurance package from a profile using the package's system data information. * Remove an insurance package from a profile using the package's system data information.
* *
* @param sessionID The session ID of the profile to remove the package from. * @param sessionID The session ID of the profile to remove the package from.
* @param index The array index of the insurance package to remove. * @param index The array index of the insurance package to remove.
* @returns void * @returns void
@ -146,71 +148,38 @@ export class InsuranceController
protected removeInsurancePackageFromProfile(sessionID: string, packageInfo: ISystemData): void protected removeInsurancePackageFromProfile(sessionID: string, packageInfo: ISystemData): void
{ {
const profile = this.saveServer.getProfile(sessionID); const profile = this.saveServer.getProfile(sessionID);
profile.insurance = profile.insurance.filter(insurance => profile.insurance = profile.insurance.filter(insurance =>
insurance.messageContent.systemData.date !== packageInfo.date || insurance.messageContent.systemData.date !== packageInfo.date ||
insurance.messageContent.systemData.time !== packageInfo.time || insurance.messageContent.systemData.time !== packageInfo.time ||
insurance.messageContent.systemData.location !== packageInfo.location insurance.messageContent.systemData.location !== packageInfo.location
); );
this.logger.debug(`Removed insurance package with date: ${packageInfo.date}, time: ${packageInfo.time}, and location: ${packageInfo.location} from profile ${sessionID}. Remaining packages: ${profile.insurance.length}`); this.logger.debug(`Removed insurance package with date: ${packageInfo.date}, time: ${packageInfo.time}, and location: ${packageInfo.location} from profile ${sessionID}. Remaining packages: ${profile.insurance.length}`);
} }
/** /**
* Build an array of items to delete from the insured items. * Finds the items that should be deleted based on the given Insurance object.
* *
* This method orchestrates several steps: * @param insured The insurance object containing the items to evaluate for deletion.
* - Filters items based on their presence in the database and their raid moddability. * @returns A Set containing the IDs of items that should be deleted.
* - Sorts base and independent child items to consider for deletion.
* - Groups child items by their parent for later evaluation.
* - Evaluates grouped child items to decide which should be deleted, based on their value and a random roll.
*
* @param insured - The insured items to build a removal array from.
* @returns An array of IDs representing items that should be deleted.
*/ */
protected findItemsToDelete(insured: Insurance): Set<string> protected findItemsToDelete(insured: Insurance): Set<string>
{ {
const toDelete = new Set<string>(); const toDelete = new Set<string>();
const childrenGroupedByParent = new Map<string, Item[]>();
insured.items.forEach(insuredItem =>
{
const itemDbDetails = this.itemHelper.getItem(insuredItem._tpl);
// Use the _tpl property from the parent item to get the parent item details // Populate a Map object of items for quick lookup by their ID and use it to populate a Map of main-parent items
const parentItem = insured.items.find(item => item._id === insuredItem.parentId); // and each of their attachments. For example, a gun mapped to each of its attachments.
const parentItemDbDetailsArray = parentItem ? this.itemHelper.getItem(parentItem._tpl) : null; const itemsMap = this.populateItemsMap(insured);
const parentItemDbDetails = parentItemDbDetailsArray ? parentItemDbDetailsArray[1] : null; const parentAttachmentsMap = this.populateParentAttachmentsMap(insured, itemsMap);
// Filter out items not in the database or not raid moddable. // Process all items that are not attached, attachments. Those are handled separately, by value.
if (!this.filterByRaidModdability(insuredItem, parentItemDbDetails, itemDbDetails)) return; this.processRegularItems(insured, toDelete);
// Check for base or independent child items.
if (this.isBaseOrIndependentChild(insuredItem))
{
// Find child IDs if the item is a parent.
const itemWithChildren = this.itemHelper.findAndReturnChildrenByItems(insured.items, insuredItem._id);
// Make a roll to decide if this item should be deleted, and if so, add it and its children to the deletion list.
if (this.makeRollAndMarkForDeletion(insuredItem, insured.traderId, toDelete))
{
itemWithChildren.forEach(childId => toDelete.add(childId));
}
}
else if (insuredItem.parentId)
{
// This is a child item equipped to a parent... Group this child item by its parent.
this.groupChildrenByParent(insuredItem, childrenGroupedByParent);
}
});
// Iterate through each group of children and sort and filter them for deletion.
childrenGroupedByParent.forEach((children) =>
{
this.sortAndFilterChildren(children, insured.traderId, toDelete);
});
// When items are selected for deletion, log the number of items and their names. // Process attached, attachments, by value.
if (toDelete.size) this.processAttachments(parentAttachmentsMap, itemsMap, insured.traderId, toDelete);
// Log the number of items marked for deletion, if any
if (toDelete.size)
{ {
this.logger.debug(`Marked ${toDelete.size} items for deletion from insurance.`); this.logger.debug(`Marked ${toDelete.size} items for deletion from insurance.`);
} }
@ -219,112 +188,222 @@ export class InsuranceController
} }
/** /**
* Filters an item based on its existence in the database, raid moddability, and slot requirements. * Populate a Map object of items for quick lookup by their ID.
* *
* @param item The item to be filtered. * @param insured The insurance object containing the items to populate the map with.
* @param parentItemDbDetails The database details of the parent item, or null if the item has no parent. * @returns A Map where the keys are the item IDs and the values are the corresponding Item objects.
* @param itemDbDetails A tuple where the first element is a boolean indicating if the item exists in the database,
* and the second element is the item details if it does.
* @returns true if the item exists in the database and neither of the following conditions are met:
* - The item has the RaidModdable property set to false.
* - The item is attached to a required slot in its parent item.
* Otherwise, returns false.
*/ */
protected filterByRaidModdability(item: Item, parentItemDbDetails: ITemplateItem | null, itemDbDetails: [boolean, ITemplateItem]): boolean protected populateItemsMap(insured: Insurance): Map<string, Item>
{ {
// Check for RaidModdable property. const itemsMap = new Map<string, Item>();
const isNotRaidModdable = itemDbDetails[1]?._props?.RaidModdable === false; insured.items.forEach(item => itemsMap.set(item._id, item));
return itemsMap;
}
// Check for Slots in parent item details. /**
let isRequiredSlot = false; * Initialize a Map object that holds main-parents to all of their attachments. Note that "main-parent" in this
if (parentItemDbDetails?._props?.Slots) * context refers to the parent item that an attachment is attached to. For example, a suppressor attached to a gun,
* not the backpack that the gun is located in (the gun's parent).
*
* @param insured - The insurance object containing the items to evaluate.
* @param itemsMap - A Map object for quick item look-up by item ID.
* @returns A Map object containing parent item IDs to arrays of their attachment items.
*/
protected populateParentAttachmentsMap(insured: Insurance, itemsMap: Map<string, Item>): Map<string, Item[]>
{
const mainParentToAttachmentsMap = new Map<string, Item[]>();
for (const insuredItem of insured.items)
{ {
// Check if a Slot in parent details matches the slotId of the current item and is marked as required // Use the template ID from the item to get the parent item's template details.
isRequiredSlot = parentItemDbDetails._props.Slots.some(slot => slot._name === item.slotId && slot._required); const parentItem = insured.items.find(item => item._id === insuredItem.parentId);
// Check if this is an attachment currently attached to its parent.
if (this.itemHelper.isAttachmentAttached(insuredItem))
{
// Filter out items not in the database or not raid moddable.
if (!this.itemHelper.isRaidModdable(insuredItem, parentItem))
{
continue;
}
// Get the main parent of this attachment. (e.g., The gun that this suppressor is attached to.)
const mainParent = this.itemHelper.getAttachmentMainParent(insuredItem._id, itemsMap);
if (!mainParent)
{
// Odd. The parent couldn't be found. Skip this attachment and warn.
this.logger.warning(`Could not find main-parent for insured attachment: ${this.itemHelper.getItemName(insuredItem._tpl)}`);
continue;
}
// Update (or add to) the main-parent to attachments map.
if (mainParentToAttachmentsMap.has(mainParent._id))
{
mainParentToAttachmentsMap.get(mainParent._id).push(insuredItem);
}
else
{
mainParentToAttachmentsMap.set(mainParent._id, [insuredItem]);
}
}
} }
return mainParentToAttachmentsMap;
return itemDbDetails[0] && !(isNotRaidModdable || isRequiredSlot);
} }
/** /**
* Determines if an item is either a base item or a child item that is not equipped to its parent. * Process "regular" insurance items. Any insured item that is not an attached, attachment is considered a "regular"
* * item. This method iterates over them, preforming item deletion rolls to see if they should be deleted. If so,
* @param item The item to check. * they (and their attached, attachments, if any) are marked for deletion in the toDelete Set.
* @returns true if the item is a base or an independent child item, otherwise false. *
*/ * @param insured The insurance object containing the items to evaluate.
protected isBaseOrIndependentChild(item: Item): boolean * @param toDelete A Set to keep track of items marked for deletion.
{
return item.slotId === "hideout" || item.slotId === "main" || !isNaN(Number(item.slotId));
}
/**
* Makes a roll to determine if a given item should be deleted. If the roll is successful, the item's ID is added
* to the `toDelete` array.
*
* @param item The item for which the roll is made.
* @param traderId The ID of the trader to consider in the rollForItemDelete method.
* @param toDelete The array accumulating the IDs of items to be deleted.
* @returns true if the item is marked for deletion, otherwise false.
*/
protected makeRollAndMarkForDeletion(item: Item, traderId: string, toDelete: Set<string>): boolean
{
if (this.rollForItemDelete(item, traderId, toDelete))
{
toDelete.add(item._id);
return true;
}
return false;
}
/**
* Groups child items by their parent IDs in a Map data structure.
*
* @param item The child item to be grouped by its parent.
* @param childrenGroupedByParent The Map that holds arrays of children items grouped by their parent IDs.
* @returns void * @returns void
*/ */
protected groupChildrenByParent(item: Item, childrenGroupedByParent: Map<string, Item[]>): void protected processRegularItems(insured: Insurance, toDelete: Set<string>): void
{ {
if (!childrenGroupedByParent.has(item.parentId!)) for (const insuredItem of insured.items)
{ {
childrenGroupedByParent.set(item.parentId!, []); // Skip if the item is an attachment. These are handled separately.
if (this.itemHelper.isAttachmentAttached(insuredItem))
{
continue;
}
// Check if the item has any children
const itemAndChildren = this.itemHelper.findAndReturnChildrenAsItems(insured.items, insuredItem._id);
// Roll for item deletion
const itemRoll = this.rollForDelete(insured.traderId, insuredItem);
if (itemRoll)
{
// Mark the item for deletion
toDelete.add(insuredItem._id);
// Check if the item has any children and mark those for deletion as well, but only if those
// children are currently attached attachments.
const directChildren = insured.items.filter(item => item.parentId === insuredItem._id);
const allChildrenAreAttachments = directChildren.every(child => this.itemHelper.isAttachmentAttached(child));
if (allChildrenAreAttachments)
{
itemAndChildren.forEach(item => toDelete.add(item._id));
}
}
} }
childrenGroupedByParent.get(item.parentId!)?.push(item);
} }
/** /**
* Sorts the array of children items in descending order by their maximum price. For each child, a roll is made to * Process parent items and their attachments, updating the toDelete Set accordingly.
* determine if it should be deleted. The method then deletes the most valuable children based on the number of *
* successful rolls made. * This method iterates over a map of parent items to their attachments and performs evaluations on each.
* * It marks items for deletion based on certain conditions and updates the toDelete Set accordingly.
* @param children The array of children items to sort and filter. *
* @param traderId The ID of the trader to consider in the rollForItemDelete method. * @param mainParentToAttachmentsMap A Map object containing parent item IDs to arrays of their attachment items.
* @param itemsMap A Map object for quick item look-up by item ID.
* @param traderId The trader ID from the Insurance object.
* @param toDelete A Set object to keep track of items marked for deletion.
*/
protected processAttachments(mainParentToAttachmentsMap: Map<string, Item[]>, itemsMap: Map<string, Item>, traderId: string, toDelete: Set<string>): void
{
mainParentToAttachmentsMap.forEach((attachmentItems, parentId) =>
{
// Log the parent item's name.
const parentItem = itemsMap.get(parentId);
const parentName = this.itemHelper.getItemName(parentItem._tpl);
this.logger.debug(`Processing attachments for parent item: ${parentName}`);
// Your existing logic for sorting and filtering children of this parent item
this.processAttachmentByParent(attachmentItems, traderId, toDelete);
});
}
/**
* Takes an array of attachment items that belong to the same main-parent item, sorts them in descending order by
* their maximum price. For each attachment, a roll is made to determine if a deletion should be made. Once the
* number of deletions has been counted, the attachments are added to the toDelete Set, starting with the most
* valuable attachments first.
*
* @param attachments The array of attachment items to sort, filter, and roll.
* @param traderId The ID of the trader to that has ensured these items.
* @param toDelete The array that accumulates the IDs of the items to be deleted. * @param toDelete The array that accumulates the IDs of the items to be deleted.
* @returns void * @returns void
*/ */
protected sortAndFilterChildren(children: Item[], traderId: string, toDelete: Set<string>): void protected processAttachmentByParent(attachments: Item[], traderId: string, toDelete: Set<string>): void
{ {
// Sort the children by their max price in descending order. const sortedAttachments = this.sortAttachmentsByPrice(attachments);
children.sort((a, b) => this.itemHelper.getItemMaxPrice(b._tpl) - this.itemHelper.getItemMaxPrice(a._tpl)); this.logAttachmentsDetails(sortedAttachments);
// Count the number of successful rolls. const successfulRolls = this.countSuccessfulRolls(sortedAttachments, traderId);
let successfulRolls = 0; this.logger.debug(`Number of successful rolls: ${successfulRolls}`);
for (const child of children)
this.attachmentDeletionByValue(sortedAttachments, successfulRolls, toDelete);
}
/**
* Sorts the attachment items by their max price in descending order.
*
* @param attachments The array of attachments items.
* @returns An array of items enriched with their max price and common locale-name.
*/
protected sortAttachmentsByPrice(attachments: Item[]): EnrichedItem[]
{
return attachments.map(item => ({
...item,
name: this.itemHelper.getItemName(item._tpl),
maxPrice: this.itemHelper.getItemMaxPrice(item._tpl)
})).sort((a, b) => b.maxPrice - a.maxPrice);
}
/**
* Logs the details of each attachment item.
*
* @param attachments The array of attachment items.
*/
protected logAttachmentsDetails(attachments: EnrichedItem[]): void
{
attachments.forEach(({ name, maxPrice }) =>
{ {
if (this.rollForItemDelete(child, traderId, toDelete)) this.logger.debug(`Child Item - Name: ${name}, Max Price: ${maxPrice}`);
});
}
/**
* Counts the number of successful rolls for the attachment items.
*
* @param attachments The array of attachment items.
* @param traderId The ID of the trader that has insured these attachments.
* @returns The number of successful rolls.
*/
protected countSuccessfulRolls(attachments: Item[], traderId: string): number
{
const rolls = Array.from({ length: attachments.length }, () => this.rollForDelete(traderId));
return rolls.filter(Boolean).length;
}
/**
* Marks the most valuable attachments for deletion based on the number of successful rolls made.
*
* @param attachments The array of attachment items.
* @param successfulRolls The number of successful rolls.
* @param toDelete The array that accumulates the IDs of the items to be deleted.
*/
protected attachmentDeletionByValue(attachments: EnrichedItem[], successfulRolls: number, toDelete: Set<string>): void
{
const valuableToDelete = attachments.slice(0, successfulRolls).map(({ _id }) => _id);
valuableToDelete.forEach(attachmentsId =>
{
const valuableChild = attachments.find(({ _id }) => _id === attachmentsId);
if (valuableChild)
{ {
successfulRolls++; const { name, maxPrice } = valuableChild;
this.logger.debug(`Marked for removal - Child Item: ${name}, Max Price: ${maxPrice}`);
toDelete.add(attachmentsId);
} }
} });
// Delete the most valuable children based on the number of successful rolls.
const mostValuableChildrenToDelete = children.slice(0, successfulRolls).map(child => child._id);
mostValuableChildrenToDelete.forEach(valuableChild => toDelete.add(valuableChild));
} }
/** /**
* Remove items from the insured items that should not be returned to the player. * Remove items from the insured items that should not be returned to the player.
* *
* @param insured The insured items to process. * @param insured The insured items to process.
* @param toDelete The items that should be deleted. * @param toDelete The items that should be deleted.
* @returns void * @returns void
@ -334,9 +413,48 @@ export class InsuranceController
insured.items = insured.items.filter(item => !toDelete.has(item._id)); insured.items = insured.items.filter(item => !toDelete.has(item._id));
} }
/**
* Adopts orphaned items by resetting them as base-level items. Helpful in situations where a parent has been
* deleted from insurance, but any insured items within the parent should remain. This method will remove the
* reference from the children to the parent and set item properties to main-level values.
*
* @param insured Insurance object containing items.
*/
protected adoptOrphanedItems(insured: Insurance): void
{
const hideoutParentId = this.fetchHideoutItemParent(insured.items);
insured.items.forEach(item =>
{
// Check if the item's parent exists in the insured items list.
const parentExists = insured.items.some(parentItem => parentItem._id === item.parentId);
// If the parent does not exist and the item is not already a 'hideout' item, adopt the orphaned item.
if (!parentExists && item.parentId !== hideoutParentId && item.slotId !== "hideout")
{
item.parentId = hideoutParentId;
item.slotId = "hideout";
delete item.location;
}
});
}
/**
* Fetches the parentId property of an item with a slotId "hideout". Not sure if this is actually dynamic, but this
* method should be a reliable way to fetch it, if it ever does change.
*
* @param items Array of items to search through.
* @returns The parentId of an item with slotId 'hideout'. Empty string if not found.
*/
protected fetchHideoutItemParent(items: Item[]): string
{
const hideoutItem = items.find(item => item.slotId === "hideout");
return hideoutItem ? hideoutItem?.parentId : "";
}
/** /**
* Handle sending the insurance message to the user that potentially contains the valid insurance items. * Handle sending the insurance message to the user that potentially contains the valid insurance items.
* *
* @param sessionID The session ID that should receive the insurance message. * @param sessionID The session ID that should receive the insurance message.
* @param insurance The context of insurance to use. * @param insurance The context of insurance to use.
* @param noItems Whether or not there are any items to return to the player. * @param noItems Whether or not there are any items to return to the player.
@ -344,14 +462,14 @@ export class InsuranceController
*/ */
protected sendMail(sessionID: string, insurance: Insurance, noItems: boolean): void protected sendMail(sessionID: string, insurance: Insurance, noItems: boolean): void
{ {
// After all of the item filtering that we've done, if there are no items remaining, the insurance has // After all of the item filtering that we've done, if there are no items remaining, the insurance has
// successfully "failed" to return anything and an appropriate message should be sent to the player. // successfully "failed" to return anything and an appropriate message should be sent to the player.
if (noItems) if (noItems)
{ {
const insuranceFailedTemplates = this.databaseServer.getTables().traders[insurance.traderId].dialogue.insuranceFailed; const insuranceFailedTemplates = this.databaseServer.getTables().traders[insurance.traderId].dialogue.insuranceFailed;
insurance.messageContent.templateId = this.randomUtil.getArrayValue(insuranceFailedTemplates); insurance.messageContent.templateId = this.randomUtil.getArrayValue(insuranceFailedTemplates);
} }
// Send the insurance message // Send the insurance message
this.mailSendService.sendLocalisedNpcMessageToPlayer( this.mailSendService.sendLocalisedNpcMessageToPlayer(
sessionID, sessionID,
@ -365,31 +483,35 @@ export class InsuranceController
} }
/** /**
* Determines whether a valid insured item should be removed from the player's inventory based on a random roll and * Determines whether a insured item should be removed from the player's inventory based on a random roll and
* trader-specific return chance. * trader-specific return chance.
* *
* @param insuredItem The insured item being evaluated for removal.
* @param traderId The ID of the trader who insured the item. * @param traderId The ID of the trader who insured the item.
* @param itemsBeingDeleted List of items that are already slated for removal. * @param insuredItem Optional. The item to roll for. Only used for logging.
* @returns true if the insured item should be removed from inventory, false otherwise. * @returns true if the insured item should be removed from inventory, false otherwise.
*/ */
protected rollForItemDelete(insuredItem: Item, traderId: string, itemsBeingDeleted: Set<string>): boolean protected rollForDelete(traderId: string, insuredItem?: Item): boolean
{ {
const maxRoll = 9999; const maxRoll = 9999;
const conversionFactor = 100; const conversionFactor = 100;
const returnChance = this.randomUtil.getInt(0, maxRoll) / conversionFactor; const returnChance = this.randomUtil.getInt(0, maxRoll) / conversionFactor;
const traderReturnChance = this.insuranceConfig.returnChancePercent[traderId]; const traderReturnChance = this.insuranceConfig.returnChancePercent[traderId];
const exceedsTraderReturnChance = returnChance >= traderReturnChance; const roll = returnChance >= traderReturnChance;
const isItemAlreadyBeingDeleted = itemsBeingDeleted.has(insuredItem._id);
return exceedsTraderReturnChance && !isItemAlreadyBeingDeleted; // Log the roll with as much detail as possible.
const itemName = insuredItem ? ` for "${this.itemHelper.getItemName(insuredItem._tpl)}"` : "";
const trader = this.traderHelper.getTraderById(traderId);
const status = roll ? "Delete" : "Keep";
this.logger.debug(`Rolling deletion${itemName} with ${trader} - Return ${traderReturnChance}% - Roll: ${returnChance} - Status: ${status}`);
return roll;
} }
/** /**
* Handle Insure event * Handle Insure event
* Add insurance to an item * Add insurance to an item
* *
* @param pmcData Player profile * @param pmcData Player profile
* @param body Insurance request * @param body Insurance request
* @param sessionID Session id * @param sessionID Session id
@ -450,7 +572,7 @@ export class InsuranceController
/** /**
* Handle client/insurance/items/list/cost * Handle client/insurance/items/list/cost
* Calculate insurance cost * Calculate insurance cost
* *
* @param request request object * @param request request object
* @param sessionID session id * @param sessionID session id
* @returns IGetInsuranceCostResponseData object to send to client * @returns IGetInsuranceCostResponseData object to send to client
@ -488,3 +610,10 @@ export class InsuranceController
return output; return output;
} }
} }
// Represents an insurance item that has had it's common locale-name and max price added to it.
interface EnrichedItem extends Item
{
name: string;
maxPrice: number;
}

View File

@ -132,7 +132,7 @@ class ItemHelper
{ {
const staticPrice = this.getStaticItemPrice(tpl); const staticPrice = this.getStaticItemPrice(tpl);
const dynamicPrice = this.getDynamicItemPrice(tpl); const dynamicPrice = this.getDynamicItemPrice(tpl);
return Math.max(staticPrice, dynamicPrice); return Math.max(staticPrice, dynamicPrice);
} }
@ -166,7 +166,7 @@ class ItemHelper
} }
return 0; return 0;
} }
/** /**
* Update items upd.StackObjectsCount to be 1 if its upd is missing or StackObjectsCount is undefined * Update items upd.StackObjectsCount to be 1 if its upd is missing or StackObjectsCount is undefined
@ -293,7 +293,7 @@ class ItemHelper
/** /**
* get normalized value (0-1) based on item condition * get normalized value (0-1) based on item condition
* @param item * @param item
* @returns number between 0 and 1 * @returns number between 0 and 1
*/ */
public getItemQualityModifier(item: Item): number public getItemQualityModifier(item: Item): number
@ -413,9 +413,9 @@ class ItemHelper
/** /**
* A variant of findAndReturnChildren where the output is list of item objects instead of their ids. * A variant of findAndReturnChildren where the output is list of item objects instead of their ids.
* @param items * @param items
* @param baseItemId * @param baseItemId
* @returns An array of Item objects * @returns An array of Item objects
*/ */
public findAndReturnChildrenAsItems(items: Item[], baseItemId: string): Item[] public findAndReturnChildrenAsItems(items: Item[], baseItemId: string): Item[]
{ {
@ -488,7 +488,7 @@ class ItemHelper
/** /**
* Gets the identifier for a child using slotId, locationX and locationY. * Gets the identifier for a child using slotId, locationX and locationY.
* @param item * @param item
* @returns "slotId OR slotid,locationX,locationY" * @returns "slotId OR slotid,locationX,locationY"
*/ */
public getChildId(item: Item): string public getChildId(item: Item): string
@ -558,7 +558,7 @@ class ItemHelper
* @returns Array of Item objects * @returns Array of Item objects
*/ */
public findBarterItems(by: "tpl" | "id", items: Item[], barterItemId: string): Item[] public findBarterItems(by: "tpl" | "id", items: Item[], barterItemId: string): Item[]
{ {
// find required items to take after buying (handles multiple items) // find required items to take after buying (handles multiple items)
const barterIDs = typeof barterItemId === "string" const barterIDs = typeof barterItemId === "string"
? [barterItemId] ? [barterItemId]
@ -577,11 +577,11 @@ class ItemHelper
barterItems = Object.assign(barterItems, filterResult); barterItems = Object.assign(barterItems, filterResult);
} }
if (barterItems.length === 0) if (barterItems.length === 0)
{ {
this.logger.warning(`No items found for barter Id: ${barterIDs}`); this.logger.warning(`No items found for barter Id: ${barterIDs}`);
} }
return barterItems; return barterItems;
} }
@ -590,7 +590,7 @@ class ItemHelper
* @param pmcData Player profile * @param pmcData Player profile
* @param items Items to adjust ID values of * @param items Items to adjust ID values of
* @param insuredItems insured items to not replace ids for * @param insuredItems insured items to not replace ids for
* @param fastPanel * @param fastPanel
* @returns Item[] * @returns Item[]
*/ */
public replaceIDs(pmcData: IPmcData, items: Item[], insuredItems: InsuredItem[] = null, fastPanel = null): Item[] public replaceIDs(pmcData: IPmcData, items: Item[], insuredItems: InsuredItem[] = null, fastPanel = null): Item[]
@ -760,11 +760,92 @@ class ItemHelper
return false; return false;
} }
/**
* Checks to see if the item is *actually* moddable in-raid. Checks include the items existence in the database, the
* parent items existence in the database, the existence (and value) of the items RaidModdable property, and that
* the parents slot-required property exists, matches that of the item, and it's value.
*
* Note: this function does not preform any checks to see if the item and parent are *actually* related.
*
* @param item The item to be checked
* @param parent The parent of the item to be checked
* @returns True if the item is actually moddable, false if it is not, and null if the check cannot be performed.
*/
public isRaidModdable(item: Item, parent: Item): boolean | null
{
// This check requires the item to have the slotId property populated.
if (!item.slotId)
{
return null;
}
const itemTemplate = this.getItem(item._tpl);
const parentTemplate = this.getItem(parent._tpl);
// Check for RaidModdable property on the item template.
let isNotRaidModdable = false;
if (itemTemplate[0])
{
isNotRaidModdable = itemTemplate[1]?._props?.RaidModdable === false;
}
// Check to see if the slot that the item is attached to is marked as required in the parent item's template.
let isRequiredSlot = false;
if (parentTemplate[0] && parentTemplate[1]?._props?.Slots)
{
isRequiredSlot = parentTemplate[1]._props.Slots.some(slot => slot._name === item.slotId && slot._required);
}
return itemTemplate[0] && parentTemplate[0] && !(isNotRaidModdable || isRequiredSlot);
}
/**
* Retrieves the main parent item for a given attachment item.
*
* This method traverses up the hierarchy of items starting from a given `itemId`, until it finds the main parent
* item that is not an attached attachment itself. In other words, if you pass it an item id of a suppressor, it
* will traverse up the muzzle brake, barrel, upper receiver, and return the gun that the suppressor is ultimately
* attached to, even if that gun is located within multiple containers.
*
* It's important to note that traversal is expensive, so this method requires that you pass it a Map of the items
* to traverse, where the keys are the item IDs and the values are the corresponding Item objects. This alleviates
* some of the performance concerns, as it allows for quick lookups of items by ID.
*
* To generate the map:
* ```
* const itemsMap = new Map<string, Item>();
* items.forEach(item => itemsMap.set(item._id, item));
* ```
*
* @param itemId - The unique identifier of the item for which to find the main parent.
* @param itemsMap - A Map containing item IDs mapped to their corresponding Item objects for quick lookup.
* @returns The Item object representing the top-most parent of the given item, or `null` if no such parent exists.
*/
public getAttachmentMainParent(itemId: string, itemsMap: Map<string, Item>): Item | null
{
let currentItem = itemsMap.get(itemId);
while (currentItem && this.isAttachmentAttached(currentItem))
{
currentItem = itemsMap.get(currentItem.parentId);
}
return currentItem;
}
/**
* Determines if an item is an attachment that is currently attached to it's parent item.
*
* @param item The item to check.
* @returns true if the item is attached attachment, otherwise false.
*/
public isAttachmentAttached(item: Item): boolean
{
return item.slotId !== "hideout" && item.slotId !== "main" && isNaN(Number(item.slotId));
}
/** /**
* Get the inventory size of an item * Get the inventory size of an item
* @param items Item with children * @param items Item with children
* @param rootItemId * @param rootItemId
* @returns ItemSize object (width and height) * @returns ItemSize object (width and height)
*/ */
public getItemSize(items: Item[], rootItemId: string): ItemHelper.ItemSize public getItemSize(items: Item[], rootItemId: string): ItemHelper.ItemSize
@ -1010,7 +1091,7 @@ class ItemHelper
* Create a basic cartrige object * Create a basic cartrige object
* @param parentId container cartridges will be placed in * @param parentId container cartridges will be placed in
* @param ammoTpl Cartridge to insert * @param ammoTpl Cartridge to insert
* @param stackCount Count of cartridges inside parent * @param stackCount Count of cartridges inside parent
* @param location Location inside parent (e.g. 0, 1) * @param location Location inside parent (e.g. 0, 1)
* @returns Item * @returns Item
*/ */
@ -1028,7 +1109,7 @@ class ItemHelper
/** /**
* Get the size of a stack, return 1 if no stack object count property found * Get the size of a stack, return 1 if no stack object count property found
* @param item Item to get stack size of * @param item Item to get stack size of
* @returns size of stack * @returns size of stack
*/ */
public getItemStackSize(item: Item): number public getItemStackSize(item: Item): number