using System; using System.Linq; using System.Reflection; using System.Xml.Serialization; using SabreTools.Models; using SabreTools.Models.Internal; namespace SabreTools.Filter { public static class FilterParser { /// /// Parse a filter ID string into the item name and field name, if possible /// public static (string?, string?) ParseFilterId(string filterId) { // If we don't have a filter ID, we can't do anything if (string.IsNullOrWhiteSpace(filterId)) return (null, null); // If we only have one part, we can't do anything string[] splitFilter = filterId.Split('.'); if (splitFilter.Length != 2) return (null, null); // Return santized values based on the split ID return splitFilter[0].ToLowerInvariant() switch { // Header "header" => ParseHeaderFilterId(splitFilter), // Machine "game" => ParseMachineFilterId(splitFilter), "machine" => ParseMachineFilterId(splitFilter), "resource" => ParseMachineFilterId(splitFilter), "set" => ParseMachineFilterId(splitFilter), // DatItem _ => ParseDatItemFilterId(splitFilter), }; } /// /// Parse and validate header fields /// private static (string?, string?) ParseHeaderFilterId(string[] filterId) { // Get the set of constants var constants = GetConstants(typeof(Header)); if (constants == null) return (null, null); // Get if there's a match to the constant string? constantMatch = constants.FirstOrDefault(c => string.Equals(c, filterId[1], StringComparison.OrdinalIgnoreCase)); if (constantMatch == null) return (null, null); // Return the sanitized ID return (MetadataFile.HeaderKey, constantMatch); } /// /// Parse and validate machine/game fields /// private static (string?, string?) ParseMachineFilterId(string[] filterId) { // Get the set of constants var constants = GetConstants(typeof(Machine)); if (constants == null) return (null, null); // Get if there's a match to the constant string? constantMatch = constants.FirstOrDefault(c => string.Equals(c, filterId[1], StringComparison.OrdinalIgnoreCase)); if (constantMatch == null) return (null, null); // Return the sanitized ID return (MetadataFile.MachineKey, constantMatch); } /// /// Parse and validate item fields /// private static (string?, string?) ParseDatItemFilterId(string[] filterId) { // Get the correct item type var itemType = GetDatItemType(filterId[0].ToLowerInvariant()); if (itemType == null) return (null, null); // Get the set of constants var constants = GetConstants(itemType); if (constants == null) return (null, null); // Get if there's a match to the constant string? constantMatch = constants.FirstOrDefault(c => string.Equals(c, filterId[1], StringComparison.OrdinalIgnoreCase)); if (constantMatch == null) return (null, null); // Return the sanitized ID return (filterId[0].ToLowerInvariant(), constantMatch); } /// /// Get constant values for the given type, if possible /// private static string[]? GetConstants(Type? type) { if (type == null) return null; var fields = type.GetFields(BindingFlags.Static | BindingFlags.Public | BindingFlags.FlattenHierarchy); if (fields == null) return null; return fields .Where(f => f.IsLiteral && !f.IsInitOnly) .Where(f => f.CustomAttributes.Any(a => a.AttributeType == typeof(NoFilterAttribute))) .Select(f => f.GetRawConstantValue() as string) .Where(v => v != null) .ToArray()!; } /// /// Attempt to get the DatItem type from the name /// private static Type? GetDatItemType(string? itemType) { if (string.IsNullOrWhiteSpace(itemType)) return null; return AppDomain.CurrentDomain.GetAssemblies() .SelectMany(a => a.GetTypes()) .Where(t => t.IsAssignableFrom(typeof(DatItem)) && t.IsClass) .FirstOrDefault(t => t.GetCustomAttribute()?.ElementName == itemType); } } }