| | | 1 | | #pragma warning disable RS0030 // Do not use banned APIs |
| | | 2 | | |
| | | 3 | | using System; |
| | | 4 | | using System.Collections.Generic; |
| | | 5 | | using System.Linq; |
| | | 6 | | using System.Threading; |
| | | 7 | | using System.Threading.Tasks; |
| | | 8 | | using Jellyfin.Database.Implementations; |
| | | 9 | | using Jellyfin.Database.Implementations.Entities; |
| | | 10 | | using Jellyfin.Extensions; |
| | | 11 | | using MediaBrowser.Controller; |
| | | 12 | | using MediaBrowser.Controller.Entities; |
| | | 13 | | using MediaBrowser.Controller.Entities.Audio; |
| | | 14 | | using MediaBrowser.Controller.Persistence; |
| | | 15 | | using MediaBrowser.Controller.Playlists; |
| | | 16 | | using Microsoft.EntityFrameworkCore; |
| | | 17 | | using Microsoft.Extensions.Logging; |
| | | 18 | | using BaseItemDto = MediaBrowser.Controller.Entities.BaseItem; |
| | | 19 | | using DbLinkedChildType = Jellyfin.Database.Implementations.Entities.LinkedChildType; |
| | | 20 | | using LinkedChildType = MediaBrowser.Controller.Entities.LinkedChildType; |
| | | 21 | | |
| | | 22 | | namespace Jellyfin.Server.Implementations.Item; |
| | | 23 | | |
| | | 24 | | /// <summary> |
| | | 25 | | /// Handles item persistence operations (save, delete, update). |
| | | 26 | | /// </summary> |
| | | 27 | | public class ItemPersistenceService : IItemPersistenceService |
| | | 28 | | { |
| | | 29 | | private readonly IDbContextFactory<JellyfinDbContext> _dbProvider; |
| | | 30 | | private readonly IServerApplicationHost _appHost; |
| | | 31 | | private readonly ILogger<ItemPersistenceService> _logger; |
| | | 32 | | |
| | | 33 | | /// <summary> |
| | | 34 | | /// Initializes a new instance of the <see cref="ItemPersistenceService"/> class. |
| | | 35 | | /// </summary> |
| | | 36 | | /// <param name="dbProvider">The database context factory.</param> |
| | | 37 | | /// <param name="appHost">The application host.</param> |
| | | 38 | | /// <param name="logger">The logger.</param> |
| | | 39 | | public ItemPersistenceService( |
| | | 40 | | IDbContextFactory<JellyfinDbContext> dbProvider, |
| | | 41 | | IServerApplicationHost appHost, |
| | | 42 | | ILogger<ItemPersistenceService> logger) |
| | | 43 | | { |
| | 21 | 44 | | _dbProvider = dbProvider; |
| | 21 | 45 | | _appHost = appHost; |
| | 21 | 46 | | _logger = logger; |
| | 21 | 47 | | } |
| | | 48 | | |
| | | 49 | | /// <inheritdoc /> |
| | | 50 | | public void DeleteItem(params IReadOnlyList<Guid> ids) |
| | | 51 | | { |
| | 1 | 52 | | if (ids is null || ids.Count == 0 || ids.Any(f => f.Equals(BaseItemRepository.PlaceholderId))) |
| | | 53 | | { |
| | 0 | 54 | | throw new ArgumentException("Guid can't be empty or the placeholder id.", nameof(ids)); |
| | | 55 | | } |
| | | 56 | | |
| | 1 | 57 | | using var context = _dbProvider.CreateDbContext(); |
| | 1 | 58 | | using var transaction = context.Database.BeginTransaction(); |
| | | 59 | | |
| | 1 | 60 | | var date = (DateTime?)DateTime.UtcNow; |
| | | 61 | | |
| | 1 | 62 | | var descendantIds = DescendantQueryHelper.GetOwnedDescendantIdsBatch(context, ids); |
| | 4 | 63 | | foreach (var id in ids) |
| | | 64 | | { |
| | 1 | 65 | | descendantIds.Add(id); |
| | | 66 | | } |
| | | 67 | | |
| | 1 | 68 | | var extraIds = context.BaseItems |
| | 1 | 69 | | .Where(e => e.OwnerId.HasValue && descendantIds.Contains(e.OwnerId.Value)) |
| | 1 | 70 | | .Select(e => e.Id) |
| | 1 | 71 | | .ToArray(); |
| | | 72 | | |
| | 2 | 73 | | foreach (var extraId in extraIds) |
| | | 74 | | { |
| | 0 | 75 | | descendantIds.Add(extraId); |
| | | 76 | | } |
| | | 77 | | |
| | 1 | 78 | | var relatedItems = descendantIds.ToArray(); |
| | | 79 | | |
| | | 80 | | // When batch-deleting, multiple items may have UserData for the same (UserId, CustomDataKey). |
| | | 81 | | // Moving all of them to PlaceholderId would violate the UNIQUE constraint. |
| | | 82 | | // Deduplicate by loading keys client-side, keeping the best row per group. |
| | 1 | 83 | | var batchUserData = context.UserData.WhereOneOrMany(relatedItems, e => e.ItemId); |
| | | 84 | | |
| | 1 | 85 | | var allRows = batchUserData |
| | 1 | 86 | | .Select(ud => new { ud.ItemId, ud.UserId, ud.CustomDataKey, ud.LastPlayedDate, ud.PlayCount }) |
| | 1 | 87 | | .ToList(); |
| | | 88 | | |
| | 1 | 89 | | var duplicateRows = allRows |
| | 1 | 90 | | .GroupBy(ud => new { ud.UserId, ud.CustomDataKey }) |
| | 1 | 91 | | .Where(g => g.Count() > 1) |
| | 1 | 92 | | .SelectMany(g => g |
| | 1 | 93 | | .OrderByDescending(ud => ud.LastPlayedDate) |
| | 1 | 94 | | .ThenByDescending(ud => ud.PlayCount) |
| | 1 | 95 | | .Skip(1)) |
| | 1 | 96 | | .ToList(); |
| | | 97 | | |
| | 2 | 98 | | foreach (var dup in duplicateRows) |
| | | 99 | | { |
| | 0 | 100 | | context.UserData |
| | 0 | 101 | | .Where(ud => ud.ItemId == dup.ItemId && ud.UserId == dup.UserId && ud.CustomDataKey == dup.CustomDataKey |
| | 0 | 102 | | .ExecuteDelete(); |
| | | 103 | | } |
| | | 104 | | |
| | | 105 | | // Delete existing placeholder rows that would conflict with the incoming ones |
| | 1 | 106 | | context.UserData |
| | 1 | 107 | | .Join( |
| | 1 | 108 | | batchUserData, |
| | 1 | 109 | | placeholder => new { placeholder.UserId, placeholder.CustomDataKey }, |
| | 1 | 110 | | userData => new { userData.UserId, userData.CustomDataKey }, |
| | 1 | 111 | | (placeholder, userData) => placeholder) |
| | 1 | 112 | | .Where(e => e.ItemId == BaseItemRepository.PlaceholderId) |
| | 1 | 113 | | .ExecuteDelete(); |
| | | 114 | | |
| | 1 | 115 | | batchUserData |
| | 1 | 116 | | .ExecuteUpdate(e => e |
| | 1 | 117 | | .SetProperty(f => f.RetentionDate, date) |
| | 1 | 118 | | .SetProperty(f => f.ItemId, BaseItemRepository.PlaceholderId)); |
| | | 119 | | |
| | 1 | 120 | | context.AncestorIds.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 121 | | context.AncestorIds.WhereOneOrMany(relatedItems, e => e.ParentItemId).ExecuteDelete(); |
| | 1 | 122 | | context.AttachmentStreamInfos.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 123 | | context.BaseItemImageInfos.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 124 | | context.BaseItemMetadataFields.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 125 | | context.BaseItemProviders.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 126 | | context.BaseItemTrailerTypes.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 127 | | context.Chapters.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 128 | | context.CustomItemDisplayPreferences.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 129 | | context.ItemDisplayPreferences.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 130 | | context.ItemValues.Where(e => e.BaseItemsMap!.Count == 0).ExecuteDelete(); |
| | 1 | 131 | | context.ItemValuesMap.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 132 | | context.LinkedChildren.WhereOneOrMany(relatedItems, e => e.ParentId).ExecuteDelete(); |
| | 1 | 133 | | context.LinkedChildren.WhereOneOrMany(relatedItems, e => e.ChildId).ExecuteDelete(); |
| | 1 | 134 | | context.BaseItems.WhereOneOrMany(relatedItems, e => e.Id).ExecuteDelete(); |
| | 1 | 135 | | context.KeyframeData.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 136 | | context.MediaSegments.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 137 | | context.MediaStreamInfos.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 138 | | var query = context.PeopleBaseItemMap.WhereOneOrMany(relatedItems, e => e.ItemId).Select(f => f.PeopleId).Distin |
| | 1 | 139 | | context.PeopleBaseItemMap.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 140 | | context.Peoples.WhereOneOrMany(query, e => e.Id).Where(e => e.BaseItems!.Count == 0).ExecuteDelete(); |
| | 1 | 141 | | context.TrickplayInfos.WhereOneOrMany(relatedItems, e => e.ItemId).ExecuteDelete(); |
| | 1 | 142 | | context.SaveChanges(); |
| | 1 | 143 | | transaction.Commit(); |
| | 2 | 144 | | } |
| | | 145 | | |
| | | 146 | | /// <inheritdoc /> |
| | | 147 | | public void UpdateInheritedValues() |
| | | 148 | | { |
| | 17 | 149 | | using var context = _dbProvider.CreateDbContext(); |
| | 17 | 150 | | using var transaction = context.Database.BeginTransaction(); |
| | | 151 | | |
| | 17 | 152 | | context.ItemValuesMap.Where(e => e.ItemValue.Type == ItemValueType.InheritedTags).ExecuteDelete(); |
| | 17 | 153 | | context.SaveChanges(); |
| | | 154 | | |
| | 17 | 155 | | transaction.Commit(); |
| | 34 | 156 | | } |
| | | 157 | | |
| | | 158 | | /// <inheritdoc /> |
| | | 159 | | public void SaveItems(IReadOnlyList<BaseItemDto> items, CancellationToken cancellationToken) |
| | | 160 | | { |
| | 113 | 161 | | UpdateOrInsertItems(items, cancellationToken); |
| | 113 | 162 | | } |
| | | 163 | | |
| | | 164 | | /// <inheritdoc /> |
| | | 165 | | public async Task SaveImagesAsync(BaseItem item, CancellationToken cancellationToken = default) |
| | | 166 | | { |
| | 0 | 167 | | ArgumentNullException.ThrowIfNull(item); |
| | | 168 | | |
| | 0 | 169 | | var images = item.ImageInfos.Select(e => BaseItemMapper.MapImageToEntity(item.Id, e)).ToArray(); |
| | | 170 | | |
| | 0 | 171 | | var context = await _dbProvider.CreateDbContextAsync(cancellationToken).ConfigureAwait(false); |
| | 0 | 172 | | await using (context.ConfigureAwait(false)) |
| | | 173 | | { |
| | 0 | 174 | | if (!await context.BaseItems |
| | 0 | 175 | | .AnyAsync(bi => bi.Id == item.Id, cancellationToken) |
| | 0 | 176 | | .ConfigureAwait(false)) |
| | | 177 | | { |
| | 0 | 178 | | _logger.LogWarning("Unable to save ImageInfo for non existing BaseItem"); |
| | 0 | 179 | | return; |
| | | 180 | | } |
| | | 181 | | |
| | 0 | 182 | | await context.BaseItemImageInfos |
| | 0 | 183 | | .Where(e => e.ItemId == item.Id) |
| | 0 | 184 | | .ExecuteDeleteAsync(cancellationToken) |
| | 0 | 185 | | .ConfigureAwait(false); |
| | | 186 | | |
| | 0 | 187 | | await context.BaseItemImageInfos |
| | 0 | 188 | | .AddRangeAsync(images, cancellationToken) |
| | 0 | 189 | | .ConfigureAwait(false); |
| | | 190 | | |
| | 0 | 191 | | await context.SaveChangesAsync(cancellationToken).ConfigureAwait(false); |
| | | 192 | | } |
| | 0 | 193 | | } |
| | | 194 | | |
| | | 195 | | /// <inheritdoc /> |
| | | 196 | | public async Task ReattachUserDataAsync(BaseItemDto item, CancellationToken cancellationToken) |
| | | 197 | | { |
| | 34 | 198 | | ArgumentNullException.ThrowIfNull(item); |
| | 34 | 199 | | cancellationToken.ThrowIfCancellationRequested(); |
| | | 200 | | |
| | 33 | 201 | | var dbContext = await _dbProvider.CreateDbContextAsync(cancellationToken).ConfigureAwait(false); |
| | | 202 | | |
| | 33 | 203 | | await using (dbContext.ConfigureAwait(false)) |
| | | 204 | | { |
| | 33 | 205 | | var transaction = await dbContext.Database.BeginTransactionAsync(cancellationToken).ConfigureAwait(false); |
| | 33 | 206 | | await using (transaction.ConfigureAwait(false)) |
| | | 207 | | { |
| | 33 | 208 | | var userKeys = item.GetUserDataKeys().ToArray(); |
| | 33 | 209 | | var retentionDate = (DateTime?)null; |
| | | 210 | | |
| | 33 | 211 | | await dbContext.UserData |
| | 33 | 212 | | .Where(e => e.ItemId == BaseItemRepository.PlaceholderId) |
| | 33 | 213 | | .Where(e => userKeys.Contains(e.CustomDataKey)) |
| | 33 | 214 | | .ExecuteUpdateAsync( |
| | 33 | 215 | | e => e |
| | 33 | 216 | | .SetProperty(f => f.ItemId, item.Id) |
| | 33 | 217 | | .SetProperty(f => f.RetentionDate, retentionDate), |
| | 33 | 218 | | cancellationToken).ConfigureAwait(false); |
| | | 219 | | |
| | 33 | 220 | | item.UserData = await dbContext.UserData |
| | 33 | 221 | | .AsNoTracking() |
| | 33 | 222 | | .Where(e => e.ItemId == item.Id) |
| | 33 | 223 | | .ToArrayAsync(cancellationToken) |
| | 33 | 224 | | .ConfigureAwait(false); |
| | | 225 | | |
| | 33 | 226 | | await transaction.CommitAsync(cancellationToken).ConfigureAwait(false); |
| | 33 | 227 | | } |
| | 33 | 228 | | } |
| | 33 | 229 | | } |
| | | 230 | | |
| | | 231 | | private void UpdateOrInsertItems(IReadOnlyList<BaseItemDto> items, CancellationToken cancellationToken) |
| | | 232 | | { |
| | 113 | 233 | | ArgumentNullException.ThrowIfNull(items); |
| | 113 | 234 | | cancellationToken.ThrowIfCancellationRequested(); |
| | | 235 | | |
| | 113 | 236 | | var tuples = new List<(BaseItemDto Item, List<Guid>? AncestorIds, BaseItemDto TopParent, IEnumerable<string> Use |
| | 452 | 237 | | foreach (var item in items.GroupBy(e => e.Id).Select(e => e.Last()).Where(e => e.Id != BaseItemRepository.Placeh |
| | | 238 | | { |
| | 113 | 239 | | var ancestorIds = item.SupportsAncestors ? |
| | 113 | 240 | | item.GetAncestorIds().Distinct().ToList() : |
| | 113 | 241 | | null; |
| | | 242 | | |
| | 113 | 243 | | var topParent = item.GetTopParent(); |
| | | 244 | | |
| | 113 | 245 | | var userdataKey = item.GetUserDataKeys(); |
| | 113 | 246 | | var inheritedTags = item.GetInheritedTags(); |
| | | 247 | | |
| | 113 | 248 | | tuples.Add((item, ancestorIds, topParent, userdataKey, inheritedTags)); |
| | | 249 | | } |
| | | 250 | | |
| | 113 | 251 | | using var context = _dbProvider.CreateDbContext(); |
| | 113 | 252 | | using var transaction = context.Database.BeginTransaction(); |
| | | 253 | | |
| | 113 | 254 | | var ids = tuples.Select(f => f.Item.Id).ToArray(); |
| | 113 | 255 | | var existingItems = context.BaseItems.Where(e => ids.Contains(e.Id)).Select(f => f.Id).ToArray(); |
| | | 256 | | |
| | 452 | 257 | | foreach (var item in tuples) |
| | | 258 | | { |
| | 113 | 259 | | var entity = BaseItemMapper.Map(item.Item, _appHost); |
| | 113 | 260 | | entity.TopParentId = item.TopParent?.Id; |
| | | 261 | | |
| | 113 | 262 | | if (!existingItems.Any(e => e == entity.Id)) |
| | | 263 | | { |
| | 60 | 264 | | context.BaseItems.Add(entity); |
| | | 265 | | } |
| | | 266 | | else |
| | | 267 | | { |
| | 53 | 268 | | context.BaseItemProviders.Where(e => e.ItemId == entity.Id).ExecuteDelete(); |
| | 53 | 269 | | context.BaseItemImageInfos.Where(e => e.ItemId == entity.Id).ExecuteDelete(); |
| | 53 | 270 | | context.BaseItemMetadataFields.Where(e => e.ItemId == entity.Id).ExecuteDelete(); |
| | | 271 | | |
| | 53 | 272 | | if (entity.Images is { Count: > 0 }) |
| | | 273 | | { |
| | 0 | 274 | | context.BaseItemImageInfos.AddRange(entity.Images); |
| | | 275 | | } |
| | | 276 | | |
| | 53 | 277 | | if (entity.LockedFields is { Count: > 0 }) |
| | | 278 | | { |
| | 0 | 279 | | context.BaseItemMetadataFields.AddRange(entity.LockedFields); |
| | | 280 | | } |
| | | 281 | | |
| | 53 | 282 | | context.BaseItems.Attach(entity).State = EntityState.Modified; |
| | | 283 | | } |
| | | 284 | | } |
| | | 285 | | |
| | 113 | 286 | | var itemValueMaps = tuples |
| | 113 | 287 | | .Select(e => (e.Item, Values: GetItemValuesToSave(e.Item, e.InheritedTags))) |
| | 113 | 288 | | .ToArray(); |
| | 113 | 289 | | var allListedItemValues = itemValueMaps |
| | 113 | 290 | | .SelectMany(f => f.Values) |
| | 113 | 291 | | .Distinct() |
| | 113 | 292 | | .ToArray(); |
| | | 293 | | |
| | 113 | 294 | | var types = allListedItemValues.Select(e => e.MagicNumber).Distinct().ToArray(); |
| | 113 | 295 | | var values = allListedItemValues.Select(e => e.Value).Distinct().ToArray(); |
| | 113 | 296 | | var allListedItemValuesSet = allListedItemValues.ToHashSet(); |
| | | 297 | | |
| | 113 | 298 | | var existingValues = context.ItemValues |
| | 113 | 299 | | .Where(e => types.Contains(e.Type) && values.Contains(e.Value)) |
| | 113 | 300 | | .AsEnumerable() |
| | 113 | 301 | | .Where(e => allListedItemValuesSet.Contains((e.Type, e.Value))) |
| | 113 | 302 | | .ToArray(); |
| | 113 | 303 | | var missingItemValues = allListedItemValues.Except(existingValues.Select(f => (MagicNumber: f.Type, f.Value))).S |
| | 113 | 304 | | { |
| | 113 | 305 | | CleanValue = f.Value.GetCleanValue(), |
| | 113 | 306 | | ItemValueId = Guid.NewGuid(), |
| | 113 | 307 | | Type = f.MagicNumber, |
| | 113 | 308 | | Value = f.Value |
| | 113 | 309 | | }).ToArray(); |
| | 113 | 310 | | context.ItemValues.AddRange(missingItemValues); |
| | | 311 | | |
| | 113 | 312 | | var itemValuesStore = existingValues.Concat(missingItemValues).ToArray(); |
| | 113 | 313 | | var valueMap = itemValueMaps |
| | 113 | 314 | | .Select(f => (f.Item, Values: f.Values.Select(e => itemValuesStore.First(g => g.Value == e.Value && g.Type = |
| | 113 | 315 | | .ToArray(); |
| | | 316 | | |
| | 113 | 317 | | var mappedValues = context.ItemValuesMap.Where(e => ids.Contains(e.ItemId)).ToList(); |
| | | 318 | | |
| | 452 | 319 | | foreach (var item in valueMap) |
| | | 320 | | { |
| | 113 | 321 | | var itemMappedValues = mappedValues.Where(e => e.ItemId == item.Item.Id).ToList(); |
| | 226 | 322 | | foreach (var itemValue in item.Values) |
| | | 323 | | { |
| | 0 | 324 | | var existingItem = itemMappedValues.FirstOrDefault(f => f.ItemValueId == itemValue.ItemValueId); |
| | 0 | 325 | | if (existingItem is null) |
| | | 326 | | { |
| | 0 | 327 | | context.ItemValuesMap.Add(new ItemValueMap() |
| | 0 | 328 | | { |
| | 0 | 329 | | Item = null!, |
| | 0 | 330 | | ItemId = item.Item.Id, |
| | 0 | 331 | | ItemValue = null!, |
| | 0 | 332 | | ItemValueId = itemValue.ItemValueId |
| | 0 | 333 | | }); |
| | | 334 | | } |
| | | 335 | | else |
| | | 336 | | { |
| | 0 | 337 | | itemMappedValues.Remove(existingItem); |
| | | 338 | | } |
| | | 339 | | } |
| | | 340 | | |
| | 113 | 341 | | context.ItemValuesMap.RemoveRange(itemMappedValues); |
| | | 342 | | } |
| | | 343 | | |
| | 113 | 344 | | var itemsWithAncestors = tuples |
| | 113 | 345 | | .Where(t => t.Item.SupportsAncestors && t.AncestorIds != null) |
| | 113 | 346 | | .Select(t => t.Item.Id) |
| | 113 | 347 | | .ToList(); |
| | | 348 | | |
| | 113 | 349 | | var allExistingAncestorIds = itemsWithAncestors.Count > 0 |
| | 113 | 350 | | ? context.AncestorIds |
| | 113 | 351 | | .Where(e => itemsWithAncestors.Contains(e.ItemId)) |
| | 113 | 352 | | .ToList() |
| | 113 | 353 | | .GroupBy(e => e.ItemId) |
| | 113 | 354 | | .ToDictionary(g => g.Key, g => g.ToList()) |
| | 113 | 355 | | : new Dictionary<Guid, List<AncestorId>>(); |
| | | 356 | | |
| | 113 | 357 | | var allRequestedAncestorIds = tuples |
| | 113 | 358 | | .Where(t => t.Item.SupportsAncestors && t.AncestorIds != null) |
| | 113 | 359 | | .SelectMany(t => t.AncestorIds!) |
| | 113 | 360 | | .Distinct() |
| | 113 | 361 | | .ToList(); |
| | | 362 | | |
| | 113 | 363 | | var validAncestorIdsSet = allRequestedAncestorIds.Count > 0 |
| | 113 | 364 | | ? context.BaseItems |
| | 113 | 365 | | .Where(e => allRequestedAncestorIds.Contains(e.Id)) |
| | 113 | 366 | | .Select(f => f.Id) |
| | 113 | 367 | | .ToHashSet() |
| | 113 | 368 | | : new HashSet<Guid>(); |
| | | 369 | | |
| | 452 | 370 | | foreach (var item in tuples) |
| | | 371 | | { |
| | 113 | 372 | | if (item.Item.SupportsAncestors && item.AncestorIds != null) |
| | | 373 | | { |
| | 113 | 374 | | var existingAncestorIds = allExistingAncestorIds.GetValueOrDefault(item.Item.Id) ?? new List<AncestorId> |
| | 113 | 375 | | var validAncestorIds = item.AncestorIds.Where(id => validAncestorIdsSet.Contains(id)).ToArray(); |
| | 280 | 376 | | foreach (var ancestorId in validAncestorIds) |
| | | 377 | | { |
| | 27 | 378 | | var existingAncestorId = existingAncestorIds.FirstOrDefault(e => e.ParentItemId == ancestorId); |
| | 27 | 379 | | if (existingAncestorId is null) |
| | | 380 | | { |
| | 23 | 381 | | context.AncestorIds.Add(new AncestorId() |
| | 23 | 382 | | { |
| | 23 | 383 | | ParentItemId = ancestorId, |
| | 23 | 384 | | ItemId = item.Item.Id, |
| | 23 | 385 | | Item = null!, |
| | 23 | 386 | | ParentItem = null! |
| | 23 | 387 | | }); |
| | | 388 | | } |
| | | 389 | | else |
| | | 390 | | { |
| | 4 | 391 | | existingAncestorIds.Remove(existingAncestorId); |
| | | 392 | | } |
| | | 393 | | } |
| | | 394 | | |
| | 113 | 395 | | context.AncestorIds.RemoveRange(existingAncestorIds); |
| | | 396 | | } |
| | | 397 | | } |
| | | 398 | | |
| | 113 | 399 | | context.SaveChanges(); |
| | | 400 | | |
| | 113 | 401 | | var folderIds = tuples |
| | 113 | 402 | | .Where(t => t.Item is Folder) |
| | 113 | 403 | | .Select(t => t.Item.Id) |
| | 113 | 404 | | .ToList(); |
| | | 405 | | |
| | 113 | 406 | | var videoIds = tuples |
| | 113 | 407 | | .Where(t => t.Item is Video) |
| | 113 | 408 | | .Select(t => t.Item.Id) |
| | 113 | 409 | | .ToList(); |
| | | 410 | | |
| | 113 | 411 | | var allLinkedChildrenByParent = new Dictionary<Guid, List<LinkedChildEntity>>(); |
| | 113 | 412 | | if (folderIds.Count > 0 || videoIds.Count > 0) |
| | | 413 | | { |
| | 113 | 414 | | var allParentIds = folderIds.Concat(videoIds).Distinct().ToList(); |
| | 113 | 415 | | var allLinkedChildren = context.LinkedChildren |
| | 113 | 416 | | .Where(e => allParentIds.Contains(e.ParentId)) |
| | 113 | 417 | | .ToList(); |
| | | 418 | | |
| | 113 | 419 | | allLinkedChildrenByParent = allLinkedChildren |
| | 113 | 420 | | .GroupBy(e => e.ParentId) |
| | 113 | 421 | | .ToDictionary(g => g.Key, g => g.ToList()); |
| | | 422 | | } |
| | | 423 | | |
| | 452 | 424 | | foreach (var item in tuples) |
| | | 425 | | { |
| | 113 | 426 | | if (item.Item is Folder folder) |
| | | 427 | | { |
| | 113 | 428 | | var existingLinkedChildren = allLinkedChildrenByParent.GetValueOrDefault(item.Item.Id)?.ToList() ?? new |
| | 113 | 429 | | if (folder.LinkedChildren.Length > 0) |
| | | 430 | | { |
| | | 431 | | #pragma warning disable CS0618 // Type or member is obsolete - legacy path resolution for old data |
| | 0 | 432 | | var pathsToResolve = folder.LinkedChildren |
| | 0 | 433 | | .Where(lc => (!lc.ItemId.HasValue || lc.ItemId.Value.IsEmpty()) && !string.IsNullOrEmpty(lc.Path |
| | 0 | 434 | | .Select(lc => lc.Path) |
| | 0 | 435 | | .Distinct() |
| | 0 | 436 | | .ToList(); |
| | | 437 | | |
| | 0 | 438 | | var pathToIdMap = pathsToResolve.Count > 0 |
| | 0 | 439 | | ? context.BaseItems |
| | 0 | 440 | | .Where(e => e.Path != null && pathsToResolve.Contains(e.Path)) |
| | 0 | 441 | | .Select(e => new { e.Path, e.Id }) |
| | 0 | 442 | | .GroupBy(e => e.Path!) |
| | 0 | 443 | | .ToDictionary(g => g.Key, g => g.First().Id) |
| | 0 | 444 | | : []; |
| | | 445 | | |
| | 0 | 446 | | var resolvedChildren = new List<(LinkedChild Child, Guid ChildId)>(); |
| | 0 | 447 | | foreach (var linkedChild in folder.LinkedChildren) |
| | | 448 | | { |
| | 0 | 449 | | var childItemId = linkedChild.ItemId; |
| | 0 | 450 | | if (!childItemId.HasValue || childItemId.Value.IsEmpty()) |
| | | 451 | | { |
| | 0 | 452 | | if (!string.IsNullOrEmpty(linkedChild.Path) && pathToIdMap.TryGetValue(linkedChild.Path, out |
| | | 453 | | { |
| | 0 | 454 | | childItemId = resolvedId; |
| | | 455 | | } |
| | | 456 | | } |
| | | 457 | | #pragma warning restore CS0618 |
| | | 458 | | |
| | 0 | 459 | | if (childItemId.HasValue && !childItemId.Value.IsEmpty()) |
| | | 460 | | { |
| | 0 | 461 | | resolvedChildren.Add((linkedChild, childItemId.Value)); |
| | | 462 | | } |
| | | 463 | | } |
| | | 464 | | |
| | 0 | 465 | | resolvedChildren = resolvedChildren |
| | 0 | 466 | | .GroupBy(c => c.ChildId) |
| | 0 | 467 | | .Select(g => g.Last()) |
| | 0 | 468 | | .ToList(); |
| | | 469 | | |
| | 0 | 470 | | var childIdsToCheck = resolvedChildren.Select(c => c.ChildId).ToList(); |
| | 0 | 471 | | var existingChildIds = childIdsToCheck.Count > 0 |
| | 0 | 472 | | ? context.BaseItems |
| | 0 | 473 | | .Where(e => childIdsToCheck.Contains(e.Id)) |
| | 0 | 474 | | .Select(e => e.Id) |
| | 0 | 475 | | .ToHashSet() |
| | 0 | 476 | | : []; |
| | | 477 | | |
| | 0 | 478 | | var isPlaylist = folder is Playlist; |
| | 0 | 479 | | var sortOrder = 0; |
| | 0 | 480 | | foreach (var (linkedChild, childId) in resolvedChildren) |
| | | 481 | | { |
| | 0 | 482 | | if (!existingChildIds.Contains(childId)) |
| | | 483 | | { |
| | 0 | 484 | | _logger.LogWarning( |
| | 0 | 485 | | "Skipping LinkedChild for parent {ParentName} ({ParentId}): child item {ChildId} does no |
| | 0 | 486 | | item.Item.Name, |
| | 0 | 487 | | item.Item.Id, |
| | 0 | 488 | | childId); |
| | 0 | 489 | | continue; |
| | | 490 | | } |
| | | 491 | | |
| | 0 | 492 | | var existingLink = existingLinkedChildren.FirstOrDefault(e => e.ChildId == childId); |
| | 0 | 493 | | if (existingLink is null) |
| | | 494 | | { |
| | 0 | 495 | | context.LinkedChildren.Add(new LinkedChildEntity() |
| | 0 | 496 | | { |
| | 0 | 497 | | ParentId = item.Item.Id, |
| | 0 | 498 | | ChildId = childId, |
| | 0 | 499 | | ChildType = (DbLinkedChildType)linkedChild.Type, |
| | 0 | 500 | | SortOrder = isPlaylist ? sortOrder : null |
| | 0 | 501 | | }); |
| | | 502 | | } |
| | | 503 | | else |
| | | 504 | | { |
| | 0 | 505 | | existingLink.SortOrder = isPlaylist ? sortOrder : null; |
| | 0 | 506 | | existingLink.ChildType = (DbLinkedChildType)linkedChild.Type; |
| | 0 | 507 | | existingLinkedChildren.Remove(existingLink); |
| | | 508 | | } |
| | | 509 | | |
| | 0 | 510 | | sortOrder++; |
| | | 511 | | } |
| | | 512 | | } |
| | | 513 | | |
| | 113 | 514 | | if (existingLinkedChildren.Count > 0) |
| | | 515 | | { |
| | 0 | 516 | | context.LinkedChildren.RemoveRange(existingLinkedChildren); |
| | | 517 | | } |
| | | 518 | | } |
| | | 519 | | |
| | 113 | 520 | | if (item.Item is Video video) |
| | | 521 | | { |
| | 0 | 522 | | var existingLinkedChildren = (allLinkedChildrenByParent.GetValueOrDefault(video.Id) ?? new List<LinkedCh |
| | 0 | 523 | | .Where(e => (int)e.ChildType == 2 || (int)e.ChildType == 3) |
| | 0 | 524 | | .ToList(); |
| | | 525 | | |
| | 0 | 526 | | var newLinkedChildren = new List<(Guid ChildId, LinkedChildType Type)>(); |
| | | 527 | | |
| | 0 | 528 | | if (video.LocalAlternateVersions.Length > 0) |
| | | 529 | | { |
| | 0 | 530 | | var pathsToResolve = video.LocalAlternateVersions.Where(p => !string.IsNullOrEmpty(p)).ToList(); |
| | 0 | 531 | | if (pathsToResolve.Count > 0) |
| | | 532 | | { |
| | 0 | 533 | | var pathToIdMap = context.BaseItems |
| | 0 | 534 | | .Where(e => e.Path != null && pathsToResolve.Contains(e.Path)) |
| | 0 | 535 | | .Select(e => new { e.Path, e.Id }) |
| | 0 | 536 | | .GroupBy(e => e.Path!) |
| | 0 | 537 | | .ToDictionary(g => g.Key, g => g.First().Id); |
| | | 538 | | |
| | 0 | 539 | | foreach (var path in pathsToResolve) |
| | | 540 | | { |
| | 0 | 541 | | if (pathToIdMap.TryGetValue(path, out var childId)) |
| | | 542 | | { |
| | 0 | 543 | | newLinkedChildren.Add((childId, LinkedChildType.LocalAlternateVersion)); |
| | | 544 | | } |
| | | 545 | | } |
| | | 546 | | } |
| | | 547 | | } |
| | | 548 | | |
| | 0 | 549 | | if (video.LinkedAlternateVersions.Length > 0) |
| | | 550 | | { |
| | 0 | 551 | | foreach (var linkedChild in video.LinkedAlternateVersions) |
| | | 552 | | { |
| | 0 | 553 | | if (linkedChild.ItemId.HasValue && !linkedChild.ItemId.Value.IsEmpty()) |
| | | 554 | | { |
| | 0 | 555 | | newLinkedChildren.Add((linkedChild.ItemId.Value, LinkedChildType.LinkedAlternateVersion)); |
| | | 556 | | } |
| | | 557 | | } |
| | | 558 | | } |
| | | 559 | | |
| | 0 | 560 | | newLinkedChildren = newLinkedChildren |
| | 0 | 561 | | .GroupBy(c => c.ChildId) |
| | 0 | 562 | | .Select(g => g.Last()) |
| | 0 | 563 | | .ToList(); |
| | | 564 | | |
| | 0 | 565 | | var childIdsToCheck = newLinkedChildren.Select(c => c.ChildId).ToList(); |
| | 0 | 566 | | var existingChildIds = childIdsToCheck.Count > 0 |
| | 0 | 567 | | ? context.BaseItems |
| | 0 | 568 | | .Where(e => childIdsToCheck.Contains(e.Id)) |
| | 0 | 569 | | .Select(e => e.Id) |
| | 0 | 570 | | .ToHashSet() |
| | 0 | 571 | | : []; |
| | | 572 | | |
| | 0 | 573 | | int sortOrder = 0; |
| | 0 | 574 | | foreach (var (childId, childType) in newLinkedChildren) |
| | | 575 | | { |
| | 0 | 576 | | if (!existingChildIds.Contains(childId)) |
| | | 577 | | { |
| | 0 | 578 | | _logger.LogWarning( |
| | 0 | 579 | | "Skipping alternate version for video {VideoName} ({VideoId}): child item {ChildId} does not |
| | 0 | 580 | | video.Name, |
| | 0 | 581 | | video.Id, |
| | 0 | 582 | | childId); |
| | 0 | 583 | | continue; |
| | | 584 | | } |
| | | 585 | | |
| | 0 | 586 | | var existingLink = existingLinkedChildren.FirstOrDefault(e => e.ChildId == childId); |
| | 0 | 587 | | if (existingLink is null) |
| | | 588 | | { |
| | 0 | 589 | | context.LinkedChildren.Add(new LinkedChildEntity |
| | 0 | 590 | | { |
| | 0 | 591 | | ParentId = video.Id, |
| | 0 | 592 | | ChildId = childId, |
| | 0 | 593 | | ChildType = (DbLinkedChildType)childType, |
| | 0 | 594 | | SortOrder = sortOrder |
| | 0 | 595 | | }); |
| | | 596 | | } |
| | | 597 | | else |
| | | 598 | | { |
| | 0 | 599 | | existingLink.ChildType = (DbLinkedChildType)childType; |
| | 0 | 600 | | existingLink.SortOrder = sortOrder; |
| | 0 | 601 | | existingLinkedChildren.Remove(existingLink); |
| | | 602 | | } |
| | | 603 | | |
| | 0 | 604 | | sortOrder++; |
| | | 605 | | } |
| | | 606 | | |
| | 0 | 607 | | if (existingLinkedChildren.Count > 0) |
| | | 608 | | { |
| | 0 | 609 | | var orphanedLocalVersionIds = existingLinkedChildren |
| | 0 | 610 | | .Where(e => e.ChildType == DbLinkedChildType.LocalAlternateVersion) |
| | 0 | 611 | | .Select(e => e.ChildId) |
| | 0 | 612 | | .ToList(); |
| | | 613 | | |
| | 0 | 614 | | context.LinkedChildren.RemoveRange(existingLinkedChildren); |
| | | 615 | | |
| | 0 | 616 | | if (orphanedLocalVersionIds.Count > 0) |
| | | 617 | | { |
| | 0 | 618 | | var orphanedItems = context.BaseItems |
| | 0 | 619 | | .Where(e => orphanedLocalVersionIds.Contains(e.Id) && e.OwnerId == video.Id) |
| | 0 | 620 | | .ToList(); |
| | | 621 | | |
| | 0 | 622 | | if (orphanedItems.Count > 0) |
| | | 623 | | { |
| | 0 | 624 | | _logger.LogInformation( |
| | 0 | 625 | | "Deleting {Count} orphaned LocalAlternateVersion items for video {VideoName} ({VideoId}) |
| | 0 | 626 | | orphanedItems.Count, |
| | 0 | 627 | | video.Name, |
| | 0 | 628 | | video.Id); |
| | 0 | 629 | | context.BaseItems.RemoveRange(orphanedItems); |
| | | 630 | | } |
| | | 631 | | } |
| | | 632 | | } |
| | | 633 | | } |
| | | 634 | | } |
| | | 635 | | |
| | 113 | 636 | | context.SaveChanges(); |
| | 113 | 637 | | transaction.Commit(); |
| | 226 | 638 | | } |
| | | 639 | | |
| | | 640 | | private static List<(ItemValueType MagicNumber, string Value)> GetItemValuesToSave(BaseItemDto item, List<string> in |
| | | 641 | | { |
| | 113 | 642 | | var list = new List<(ItemValueType, string)>(); |
| | | 643 | | |
| | 113 | 644 | | if (item is IHasArtist hasArtist) |
| | | 645 | | { |
| | 0 | 646 | | list.AddRange(hasArtist.Artists.Select(i => ((ItemValueType)0, i))); |
| | | 647 | | } |
| | | 648 | | |
| | 113 | 649 | | if (item is IHasAlbumArtist hasAlbumArtist) |
| | | 650 | | { |
| | 0 | 651 | | list.AddRange(hasAlbumArtist.AlbumArtists.Select(i => (ItemValueType.AlbumArtist, i))); |
| | | 652 | | } |
| | | 653 | | |
| | 113 | 654 | | list.AddRange(item.Genres.Select(i => (ItemValueType.Genre, i))); |
| | 113 | 655 | | list.AddRange(item.Studios.Select(i => (ItemValueType.Studios, i))); |
| | 113 | 656 | | list.AddRange(item.Tags.Select(i => (ItemValueType.Tags, i))); |
| | | 657 | | |
| | 113 | 658 | | list.AddRange(inheritedTags.Select(i => (ItemValueType.InheritedTags, i))); |
| | | 659 | | |
| | 113 | 660 | | list.RemoveAll(i => string.IsNullOrWhiteSpace(i.Item2)); |
| | | 661 | | |
| | 113 | 662 | | return list; |
| | | 663 | | } |
| | | 664 | | } |