123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362 |
- using System;
- using System.Collections.Generic;
- using System.IO;
- using System.Linq;
- using System.Threading;
- using System.Threading.Tasks;
- using Jellyfin.Data.Entities;
- using MediaBrowser.Common.Configuration;
- using MediaBrowser.Controller.Collections;
- using MediaBrowser.Controller.Entities;
- using MediaBrowser.Controller.Entities.Movies;
- using MediaBrowser.Controller.Library;
- using MediaBrowser.Controller.Providers;
- using MediaBrowser.Model.Configuration;
- using MediaBrowser.Model.Entities;
- using MediaBrowser.Model.Globalization;
- using MediaBrowser.Model.IO;
- using Microsoft.Extensions.Logging;
- namespace Emby.Server.Implementations.Collections
- {
- /// <summary>
- /// The collection manager.
- /// </summary>
- public class CollectionManager : ICollectionManager
- {
- private readonly ILibraryManager _libraryManager;
- private readonly IFileSystem _fileSystem;
- private readonly ILibraryMonitor _iLibraryMonitor;
- private readonly ILogger<CollectionManager> _logger;
- private readonly IProviderManager _providerManager;
- private readonly ILocalizationManager _localizationManager;
- private readonly IApplicationPaths _appPaths;
- /// <summary>
- /// Initializes a new instance of the <see cref="CollectionManager"/> class.
- /// </summary>
- /// <param name="libraryManager">The library manager.</param>
- /// <param name="appPaths">The application paths.</param>
- /// <param name="localizationManager">The localization manager.</param>
- /// <param name="fileSystem">The filesystem.</param>
- /// <param name="iLibraryMonitor">The library monitor.</param>
- /// <param name="loggerFactory">The logger factory.</param>
- /// <param name="providerManager">The provider manager.</param>
- public CollectionManager(
- ILibraryManager libraryManager,
- IApplicationPaths appPaths,
- ILocalizationManager localizationManager,
- IFileSystem fileSystem,
- ILibraryMonitor iLibraryMonitor,
- ILoggerFactory loggerFactory,
- IProviderManager providerManager)
- {
- _libraryManager = libraryManager;
- _fileSystem = fileSystem;
- _iLibraryMonitor = iLibraryMonitor;
- _logger = loggerFactory.CreateLogger<CollectionManager>();
- _providerManager = providerManager;
- _localizationManager = localizationManager;
- _appPaths = appPaths;
- }
- /// <inheritdoc />
- public event EventHandler<CollectionCreatedEventArgs>? CollectionCreated;
- /// <inheritdoc />
- public event EventHandler<CollectionModifiedEventArgs>? ItemsAddedToCollection;
- /// <inheritdoc />
- public event EventHandler<CollectionModifiedEventArgs>? ItemsRemovedFromCollection;
- private IEnumerable<Folder> FindFolders(string path)
- {
- return _libraryManager
- .RootFolder
- .Children
- .OfType<Folder>()
- .Where(i => _fileSystem.AreEqual(path, i.Path) || _fileSystem.ContainsSubPath(i.Path, path));
- }
- internal async Task<Folder?> EnsureLibraryFolder(string path, bool createIfNeeded)
- {
- var existingFolder = FindFolders(path).FirstOrDefault();
- if (existingFolder != null)
- {
- return existingFolder;
- }
- if (!createIfNeeded)
- {
- return null;
- }
- Directory.CreateDirectory(path);
- var libraryOptions = new LibraryOptions
- {
- PathInfos = new[] { new MediaPathInfo(path) },
- EnableRealtimeMonitor = false,
- SaveLocalMetadata = true
- };
- var name = _localizationManager.GetLocalizedString("Collections");
- await _libraryManager.AddVirtualFolder(name, CollectionTypeOptions.BoxSets, libraryOptions, true).ConfigureAwait(false);
- return FindFolders(path).First();
- }
- internal string GetCollectionsFolderPath()
- {
- return Path.Combine(_appPaths.DataPath, "collections");
- }
- private Task<Folder?> GetCollectionsFolder(bool createIfNeeded)
- {
- return EnsureLibraryFolder(GetCollectionsFolderPath(), createIfNeeded);
- }
- private IEnumerable<BoxSet> GetCollections(User user)
- {
- var folder = GetCollectionsFolder(false).GetAwaiter().GetResult();
- return folder == null
- ? Enumerable.Empty<BoxSet>()
- : folder.GetChildren(user, true).OfType<BoxSet>();
- }
- /// <inheritdoc />
- public async Task<BoxSet> CreateCollectionAsync(CollectionCreationOptions options)
- {
- var name = options.Name;
- // Need to use the [boxset] suffix
- // If internet metadata is not found, or if xml saving is off there will be no collection.xml
- // This could cause it to get re-resolved as a plain folder
- var folderName = _fileSystem.GetValidFilename(name) + " [boxset]";
- var parentFolder = await GetCollectionsFolder(true).ConfigureAwait(false);
- if (parentFolder == null)
- {
- throw new ArgumentException(nameof(parentFolder));
- }
- var path = Path.Combine(parentFolder.Path, folderName);
- _iLibraryMonitor.ReportFileSystemChangeBeginning(path);
- try
- {
- Directory.CreateDirectory(path);
- var collection = new BoxSet
- {
- Name = name,
- Path = path,
- IsLocked = options.IsLocked,
- ProviderIds = options.ProviderIds,
- DateCreated = DateTime.UtcNow
- };
- parentFolder.AddChild(collection);
- if (options.ItemIdList.Count > 0)
- {
- await AddToCollectionAsync(
- collection.Id,
- options.ItemIdList.Select(x => new Guid(x)),
- false,
- new MetadataRefreshOptions(new DirectoryService(_fileSystem))
- {
- // The initial adding of items is going to create a local metadata file
- // This will cause internet metadata to be skipped as a result
- MetadataRefreshMode = MetadataRefreshMode.FullRefresh
- }).ConfigureAwait(false);
- }
- else
- {
- _providerManager.QueueRefresh(collection.Id, new MetadataRefreshOptions(new DirectoryService(_fileSystem)), RefreshPriority.High);
- }
- CollectionCreated?.Invoke(this, new CollectionCreatedEventArgs
- {
- Collection = collection,
- Options = options
- });
- return collection;
- }
- finally
- {
- // Refresh handled internally
- _iLibraryMonitor.ReportFileSystemChangeComplete(path, false);
- }
- }
- /// <inheritdoc />
- public Task AddToCollectionAsync(Guid collectionId, IEnumerable<Guid> itemIds)
- => AddToCollectionAsync(collectionId, itemIds, true, new MetadataRefreshOptions(new DirectoryService(_fileSystem)));
- private async Task AddToCollectionAsync(Guid collectionId, IEnumerable<Guid> ids, bool fireEvent, MetadataRefreshOptions refreshOptions)
- {
- if (_libraryManager.GetItemById(collectionId) is not BoxSet collection)
- {
- throw new ArgumentException("No collection exists with the supplied Id");
- }
- var list = new List<LinkedChild>();
- var itemList = new List<BaseItem>();
- var linkedChildrenList = collection.GetLinkedChildren();
- var currentLinkedChildrenIds = linkedChildrenList.Select(i => i.Id).ToList();
- foreach (var id in ids)
- {
- var item = _libraryManager.GetItemById(id);
- if (item == null)
- {
- throw new ArgumentException("No item exists with the supplied Id");
- }
- if (!currentLinkedChildrenIds.Contains(id))
- {
- itemList.Add(item);
- list.Add(LinkedChild.Create(item));
- linkedChildrenList.Add(item);
- }
- }
- if (list.Count > 0)
- {
- var newList = collection.LinkedChildren.ToList();
- newList.AddRange(list);
- collection.LinkedChildren = newList.ToArray();
- collection.UpdateRatingToItems(linkedChildrenList);
- await collection.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
- refreshOptions.ForceSave = true;
- _providerManager.QueueRefresh(collection.Id, refreshOptions, RefreshPriority.High);
- if (fireEvent)
- {
- ItemsAddedToCollection?.Invoke(this, new CollectionModifiedEventArgs(collection, itemList));
- }
- }
- }
- /// <inheritdoc />
- public async Task RemoveFromCollectionAsync(Guid collectionId, IEnumerable<Guid> itemIds)
- {
- if (_libraryManager.GetItemById(collectionId) is not BoxSet collection)
- {
- throw new ArgumentException("No collection exists with the supplied Id");
- }
- var list = new List<LinkedChild>();
- var itemList = new List<BaseItem>();
- foreach (var guidId in itemIds)
- {
- var childItem = _libraryManager.GetItemById(guidId);
- var child = collection.LinkedChildren.FirstOrDefault(i => (i.ItemId.HasValue && i.ItemId.Value.Equals(guidId)) || (childItem != null && string.Equals(childItem.Path, i.Path, StringComparison.OrdinalIgnoreCase)));
- if (child == null)
- {
- _logger.LogWarning("No collection title exists with the supplied Id");
- continue;
- }
- list.Add(child);
- if (childItem != null)
- {
- itemList.Add(childItem);
- }
- }
- if (list.Count > 0)
- {
- collection.LinkedChildren = collection.LinkedChildren.Except(list).ToArray();
- }
- await collection.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
- _providerManager.QueueRefresh(
- collection.Id,
- new MetadataRefreshOptions(new DirectoryService(_fileSystem))
- {
- ForceSave = true
- },
- RefreshPriority.High);
- ItemsRemovedFromCollection?.Invoke(this, new CollectionModifiedEventArgs(collection, itemList));
- }
- /// <inheritdoc />
- public IEnumerable<BaseItem> CollapseItemsWithinBoxSets(IEnumerable<BaseItem> items, User user)
- {
- var results = new Dictionary<Guid, BaseItem>();
- var allBoxSets = GetCollections(user).ToList();
- foreach (var item in items)
- {
- if (item is ISupportsBoxSetGrouping)
- {
- var itemId = item.Id;
- var itemIsInBoxSet = false;
- foreach (var boxSet in allBoxSets)
- {
- if (!boxSet.ContainsLinkedChildByItemId(itemId))
- {
- continue;
- }
- itemIsInBoxSet = true;
- results.TryAdd(boxSet.Id, boxSet);
- }
- // skip any item that is in a box set
- if (itemIsInBoxSet)
- {
- continue;
- }
- var alreadyInResults = false;
- // this is kind of a performance hack because only Video has alternate versions that should be in a box set?
- if (item is Video video)
- {
- foreach (var childId in video.GetLocalAlternateVersionIds())
- {
- if (!results.ContainsKey(childId))
- {
- continue;
- }
- alreadyInResults = true;
- break;
- }
- }
- if (alreadyInResults)
- {
- continue;
- }
- }
- results[item.Id] = item;
- }
- return results.Values;
- }
- }
- }
|