Files
SPT-Server-Build/Libraries/SPTarkov.Server.Core/Services/ItemBaseClassService.cs
T
2025-12-19 13:36:06 +01:00

179 lines
5.8 KiB
C#

using SPTarkov.DI.Annotations;
using SPTarkov.Server.Core.Models.Common;
using SPTarkov.Server.Core.Models.Eft.Common.Tables;
using SPTarkov.Server.Core.Models.Utils;
using LogLevel = SPTarkov.Server.Core.Models.Spt.Logging.LogLevel;
namespace SPTarkov.Server.Core.Services;
/// <summary>
/// Cache the baseids for each item in the items db inside a dictionary
/// </summary>
[Injectable(InjectionType.Singleton)]
public class ItemBaseClassService(
ISptLogger<ItemBaseClassService> logger,
DatabaseService databaseService,
ServerLocalisationService serverLocalisationService
)
{
/// <summary>
/// Key = Item tpl, values = Ids of its parents
/// </summary>
private Dictionary<MongoId, HashSet<MongoId>> _itemBaseClassesCache = [];
private readonly Lock _itemBaseClassesLock = new();
private readonly HashSet<MongoId> _rootNodeIds = [];
/// <summary>
/// Create cache and store inside ItemBaseClassService <br />
/// Store a dict of an items tpl to the base classes it and its parents have
/// </summary>
public void HydrateItemBaseClassCache()
{
// Clear existing cache
_itemBaseClassesCache = [];
var items = databaseService.GetItems();
foreach (var item in items)
{
AddItemToCache(item.Key);
}
}
public void AddItemToCache(MongoId itemTpl)
{
var itemDb = databaseService.GetItems();
if (!itemDb.TryGetValue(itemTpl, out var item))
{
logger.Error($"Could not add {itemTpl} to cache, it does not exist in the item database!");
return;
}
lock (_itemBaseClassesLock)
{
if (string.Equals(item.Type, "Item", StringComparison.OrdinalIgnoreCase))
{
_itemBaseClassesCache.TryAdd(item.Id, []);
AddBaseItems(item.Id, item);
}
else
{
_rootNodeIds.Add(item.Id);
}
}
}
/// <summary>
/// Helper method, recursively iterate through items parent items, finding and adding ids to dictionary
/// </summary>
/// <param name="itemIdToUpdate"> Item tpl to store base ids against in dictionary </param>
/// <param name="item"> Item being checked </param>
protected void AddBaseItems(MongoId itemIdToUpdate, TemplateItem item)
{
_itemBaseClassesCache[itemIdToUpdate].Add(item.Parent);
databaseService.GetItems().TryGetValue(item.Parent, out var parent);
if (parent is not null && !parent.Parent.IsEmpty)
{
AddBaseItems(itemIdToUpdate, parent);
}
}
/// <summary>
/// Does item tpl inherit from the requested base class
/// </summary>
/// <param name="itemTpl"> ItemTpl item to check base classes of </param>
/// <param name="baseClasses"> BaseClass base class to check for </param>
/// <returns> true if item inherits from base class passed in </returns>
public bool ItemHasBaseClass(MongoId itemTpl, IEnumerable<MongoId> baseClasses)
{
if (itemTpl.IsEmpty)
{
logger.Warning("Unable to check itemTpl base class as value passed is null");
return false;
}
// The cache is only generated for item templates with `_type == "Item"`, so return false for any other type,
// including item templates that simply don't exist.
if (_rootNodeIds.Contains(itemTpl))
{
return false;
}
var existsInCache = _itemBaseClassesCache.TryGetValue(itemTpl, out var baseClassList);
if (!existsInCache)
{
// Not found in cache, attempt to add first
AddItemToCache(itemTpl);
existsInCache = _itemBaseClassesCache.TryGetValue(itemTpl, out baseClassList);
}
if (existsInCache)
{
return baseClassList.Overlaps(baseClasses);
}
logger.Warning(serverLocalisationService.GetText("baseclass-item_not_found_failed", itemTpl.ToString()));
return false;
}
/// <summary>
/// Does item tpl inherit from the requested base class
/// </summary>
/// <param name="itemTpl"> ItemTpl item to check base classes of </param>
/// <param name="baseClasses"> BaseClass base class to check for </param>
/// <returns> true if item inherits from base class passed in </returns>
public bool ItemHasBaseClass(MongoId itemTpl, MongoId baseClasses)
{
if (itemTpl.IsEmpty)
{
logger.Warning("Unable to check itemTpl base class as value passed is null");
return false;
}
// The cache is only generated for item templates with `_type == "Item"`, so return false for any other type,
// including item templates that simply don't exist.
if (_rootNodeIds.Contains(itemTpl))
{
return false;
}
var existsInCache = _itemBaseClassesCache.TryGetValue(itemTpl, out var baseClassList);
if (!existsInCache)
{
// Not found in cache, attempt to add first
AddItemToCache(itemTpl);
existsInCache = _itemBaseClassesCache.TryGetValue(itemTpl, out baseClassList);
}
if (existsInCache)
{
return baseClassList.Contains(baseClasses);
}
logger.Warning(serverLocalisationService.GetText("baseclass-item_not_found_failed", itemTpl.ToString()));
return false;
}
/// <summary>
/// Get base classes item inherits from
/// </summary>
/// <param name="itemTpl"> ItemTpl item to get base classes for </param>
/// <returns> array of base classes </returns>
public HashSet<MongoId> GetItemBaseClasses(MongoId itemTpl)
{
if (!_itemBaseClassesCache.TryGetValue(itemTpl, out var value))
{
return [];
}
return value;
}
}