using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Collections.Immutable; using System.Linq; using API.Data; using API.DTOs.Metadata; using API.Entities; using API.Extensions; using API.Helpers.Builders; using API.Services.Tasks.Scanner.Parser; namespace API.Helpers; #nullable enable public static class TagHelper { public static void UpdateTag(Dictionary allTags, IEnumerable names, Action action) { foreach (var name in names) { if (string.IsNullOrEmpty(name.Trim())) continue; var normalizedName = name.ToNormalized(); allTags.TryGetValue(normalizedName, out var tag); var added = tag == null; if (tag == null) { tag = new TagBuilder(name).Build(); allTags.Add(normalizedName, tag); } action(tag, added); } } public static void KeepOnlySameTagBetweenLists(ICollection existingTags, ICollection removeAllExcept, Action? action = null) { var existing = existingTags.ToList(); foreach (var genre in existing) { var existingPerson = removeAllExcept.FirstOrDefault(g => genre.NormalizedTitle.Equals(g.NormalizedTitle)); if (existingPerson != null) continue; existingTags.Remove(genre); action?.Invoke(genre); } } /// /// Adds the tag to the list if it's not already in there. This will ignore the ExternalTag. /// /// /// public static void AddTagIfNotExists(ICollection metadataTags, Tag tag) { var existingGenre = metadataTags.FirstOrDefault(p => p.NormalizedTitle == tag.Title.ToNormalized()); if (existingGenre == null) { metadataTags.Add(tag); } } public static void AddTagIfNotExists(BlockingCollection metadataTags, Tag tag) { var existingGenre = metadataTags.FirstOrDefault(p => p.NormalizedTitle == tag.Title.ToNormalized()); if (existingGenre == null) { metadataTags.Add(tag); } } public static IList GetTagValues(string comicInfoTagSeparatedByComma) { // TODO: Unit tests needed if (string.IsNullOrEmpty(comicInfoTagSeparatedByComma)) { return ImmutableList.Empty; } return comicInfoTagSeparatedByComma.Split(",") .Select(s => s.Trim()) .DistinctBy(Parser.Normalize) .ToList(); } /// /// Remove tags on a list /// /// Used to remove before we update/add new tags /// Existing tags on Entity /// Tags from metadata /// Callback which will be executed for each tag removed public static void RemoveTags(ICollection existingTags, IEnumerable tags, Action? action = null) { var normalizedTags = tags.Select(Services.Tasks.Scanner.Parser.Parser.Normalize).ToList(); foreach (var person in normalizedTags) { var existingTag = existingTags.FirstOrDefault(p => person.Equals(p.NormalizedTitle)); if (existingTag == null) continue; existingTags.Remove(existingTag); action?.Invoke(existingTag); } } public static void UpdateTagList(ICollection? tags, Series series, IReadOnlyCollection allTags, Action handleAdd, Action onModified) { if (tags == null) return; var isModified = false; // I want a union of these 2 lists. Return only elements that are in both lists, but the list types are different var existingTags = series.Metadata.Tags.ToList(); foreach (var existing in existingTags.Where(existing => tags.SingleOrDefault(t => t.Id == existing.Id) == null)) { // Remove tag series.Metadata.Tags.Remove(existing); isModified = true; } // At this point, all tags that aren't in dto have been removed. foreach (var tagTitle in tags.Select(t => t.Title)) { var normalizedTitle = tagTitle.ToNormalized(); var existingTag = allTags.SingleOrDefault(t => t.NormalizedTitle.Equals(normalizedTitle)); if (existingTag != null) { if (series.Metadata.Tags.All(t => t.NormalizedTitle != normalizedTitle)) { handleAdd(existingTag); isModified = true; } } else { // Add new tag handleAdd(new TagBuilder(tagTitle).Build()); isModified = true; } } if (isModified) { onModified(); } } } #nullable disable