jellyfin/Emby.Server.Implementations/Collections/CollectionManager.cs

368 lines
13 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
2014-03-06 00:17:13 -05:00
using System.IO;
using System.Linq;
2014-03-06 00:17:13 -05:00
using System.Threading;
using System.Threading.Tasks;
2020-05-20 13:07:53 -04:00
using Jellyfin.Data.Entities;
2018-09-12 13:26:21 -04:00
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;
2018-09-12 13:26:21 -04:00
using MediaBrowser.Model.Globalization;
using MediaBrowser.Model.IO;
using Microsoft.Extensions.Logging;
2014-03-06 00:17:13 -05:00
namespace Emby.Server.Implementations.Collections
2014-03-06 00:17:13 -05:00
{
/// <summary>
/// The collection manager.
/// </summary>
2014-03-06 00:17:13 -05:00
public class CollectionManager : ICollectionManager
{
private readonly ILibraryManager _libraryManager;
private readonly IFileSystem _fileSystem;
private readonly ILibraryMonitor _iLibraryMonitor;
2020-06-05 20:15:56 -04:00
private readonly ILogger<CollectionManager> _logger;
2015-09-29 12:29:06 -04:00
private readonly IProviderManager _providerManager;
2018-09-12 13:26:21 -04:00
private readonly ILocalizationManager _localizationManager;
2019-12-30 10:03:20 -05:00
private readonly IApplicationPaths _appPaths;
2014-07-07 21:41:03 -04:00
/// <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)
2014-03-06 00:17:13 -05:00
{
_libraryManager = libraryManager;
_fileSystem = fileSystem;
_iLibraryMonitor = iLibraryMonitor;
2020-06-05 20:15:56 -04:00
_logger = loggerFactory.CreateLogger<CollectionManager>();
2015-09-29 12:29:06 -04:00
_providerManager = providerManager;
2018-09-12 13:26:21 -04:00
_localizationManager = localizationManager;
_appPaths = appPaths;
2014-03-06 00:17:13 -05:00
}
/// <inheritdoc />
2021-08-15 11:20:07 -04:00
public event EventHandler<CollectionCreatedEventArgs>? CollectionCreated;
/// <inheritdoc />
2021-08-15 11:20:07 -04:00
public event EventHandler<CollectionModifiedEventArgs>? ItemsAddedToCollection;
/// <inheritdoc />
2021-08-15 11:20:07 -04:00
public event EventHandler<CollectionModifiedEventArgs>? ItemsRemovedFromCollection;
2019-12-30 10:03:20 -05:00
2018-09-12 13:26:21 -04:00
private IEnumerable<Folder> FindFolders(string path)
2014-07-01 00:06:28 -04:00
{
2018-09-12 13:26:21 -04:00
return _libraryManager
.RootFolder
.Children
.OfType<Folder>()
.Where(i => _fileSystem.AreEqual(path, i.Path) || _fileSystem.ContainsSubPath(i.Path, path));
2014-07-01 00:06:28 -04:00
}
2021-08-15 11:20:07 -04:00
internal async Task<Folder?> EnsureLibraryFolder(string path, bool createIfNeeded)
2015-01-24 14:03:55 -05:00
{
var existingFolder = FindFolders(path).FirstOrDefault();
2022-12-05 09:01:13 -05:00
if (existingFolder is not null)
2018-09-12 13:26:21 -04:00
{
return existingFolder;
2018-09-12 13:26:21 -04:00
}
if (!createIfNeeded)
{
return null;
}
Directory.CreateDirectory(path);
2018-09-12 13:26:21 -04:00
var libraryOptions = new LibraryOptions
{
2021-08-28 11:32:09 -04:00
PathInfos = new[] { new MediaPathInfo(path) },
2018-09-12 13:26:21 -04:00
EnableRealtimeMonitor = false,
SaveLocalMetadata = true
};
var name = _localizationManager.GetLocalizedString("Collections");
2021-02-23 20:34:50 -05:00
await _libraryManager.AddVirtualFolder(name, CollectionTypeOptions.BoxSets, libraryOptions, true).ConfigureAwait(false);
2018-09-12 13:26:21 -04:00
return FindFolders(path).First();
}
internal string GetCollectionsFolderPath()
{
return Path.Combine(_appPaths.DataPath, "collections");
}
2023-02-01 13:34:58 -05:00
/// <inheritdoc />
public Task<Folder?> GetCollectionsFolder(bool createIfNeeded)
2018-09-12 13:26:21 -04:00
{
return EnsureLibraryFolder(GetCollectionsFolderPath(), createIfNeeded);
}
2020-05-20 13:07:53 -04:00
private IEnumerable<BoxSet> GetCollections(User user)
2018-09-12 13:26:21 -04:00
{
var folder = GetCollectionsFolder(false).GetAwaiter().GetResult();
2018-09-12 13:26:21 -04:00
2022-12-05 09:00:20 -05:00
return folder is null
2020-04-11 06:03:10 -04:00
? Enumerable.Empty<BoxSet>()
: folder.GetChildren(user, true).OfType<BoxSet>();
2015-01-24 14:03:55 -05:00
}
/// <inheritdoc />
2020-08-21 16:01:19 -04:00
public async Task<BoxSet> CreateCollectionAsync(CollectionCreationOptions options)
2014-03-06 00:17:13 -05:00
{
var name = options.Name;
2014-03-07 23:20:31 -05:00
// 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]";
2014-03-06 00:17:13 -05:00
2020-08-21 16:01:19 -04:00
var parentFolder = await GetCollectionsFolder(true).ConfigureAwait(false);
2014-03-06 00:17:13 -05:00
2022-12-05 09:00:20 -05:00
if (parentFolder is null)
2014-03-06 00:17:13 -05:00
{
2021-12-24 16:18:24 -05:00
throw new ArgumentException(nameof(parentFolder));
2014-03-06 00:17:13 -05:00
}
var path = Path.Combine(parentFolder.Path, folderName);
_iLibraryMonitor.ReportFileSystemChangeBeginning(path);
try
{
Directory.CreateDirectory(path);
2014-03-06 00:17:13 -05:00
var collection = new BoxSet
{
Name = name,
Path = path,
2014-04-26 23:42:05 -04:00
IsLocked = options.IsLocked,
2014-12-12 22:56:30 -05:00
ProviderIds = options.ProviderIds,
2018-09-12 13:26:21 -04:00
DateCreated = DateTime.UtcNow
2014-03-06 00:17:13 -05:00
};
parentFolder.AddChild(collection);
2014-03-06 00:17:13 -05:00
if (options.ItemIdList.Count > 0)
2014-03-15 11:17:46 -04:00
{
2020-08-21 16:01:19 -04:00
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);
2014-03-15 11:17:46 -04:00
}
2015-09-29 12:29:06 -04:00
else
{
2019-09-10 16:37:53 -04:00
_providerManager.QueueRefresh(collection.Id, new MetadataRefreshOptions(new DirectoryService(_fileSystem)), RefreshPriority.High);
2015-09-29 12:29:06 -04:00
}
2014-03-15 11:17:46 -04:00
CollectionCreated?.Invoke(this, new CollectionCreatedEventArgs
2014-07-07 21:41:03 -04:00
{
Collection = collection,
Options = options
});
2014-07-07 21:41:03 -04:00
2014-03-15 11:17:46 -04:00
return collection;
2014-03-06 00:17:13 -05:00
}
finally
{
// Refresh handled internally
_iLibraryMonitor.ReportFileSystemChangeComplete(path, false);
}
}
/// <inheritdoc />
2021-09-03 12:46:34 -04:00
public Task AddToCollectionAsync(Guid collectionId, IEnumerable<Guid> itemIds)
=> AddToCollectionAsync(collectionId, itemIds, true, new MetadataRefreshOptions(new DirectoryService(_fileSystem)));
2017-08-20 17:07:47 -04:00
2020-08-21 16:01:19 -04:00
private async Task AddToCollectionAsync(Guid collectionId, IEnumerable<Guid> ids, bool fireEvent, MetadataRefreshOptions refreshOptions)
2014-03-06 00:17:13 -05:00
{
2021-08-15 11:20:07 -04:00
if (_libraryManager.GetItemById(collectionId) is not BoxSet collection)
{
throw new ArgumentException("No collection exists with the supplied Id");
}
2023-02-28 18:44:57 -05:00
List<BaseItem>? itemList = null;
2018-09-12 13:26:21 -04:00
var linkedChildrenList = collection.GetLinkedChildren();
2022-01-04 11:57:19 -05:00
var currentLinkedChildrenIds = linkedChildrenList.Select(i => i.Id).ToList();
2017-08-20 17:07:47 -04:00
foreach (var id in ids)
{
2020-08-21 16:01:19 -04:00
var item = _libraryManager.GetItemById(id);
2022-12-05 09:00:20 -05:00
if (item is null)
{
throw new ArgumentException("No item exists with the supplied Id");
}
2020-08-21 16:01:19 -04:00
if (!currentLinkedChildrenIds.Contains(id))
{
2023-02-28 18:44:57 -05:00
(itemList ??= new()).Add(item);
2018-09-12 13:26:21 -04:00
linkedChildrenList.Add(item);
}
}
2023-02-28 18:44:57 -05:00
if (itemList is not null)
2015-10-07 17:42:29 -04:00
{
2023-02-28 18:44:57 -05:00
var originalLen = collection.LinkedChildren.Length;
var newItemCount = itemList.Count;
LinkedChild[] newChildren = new LinkedChild[originalLen + newItemCount];
2022-10-24 20:40:47 -04:00
collection.LinkedChildren.CopyTo(newChildren, 0);
2023-02-28 18:44:57 -05:00
for (int i = 0; i < newItemCount; i++)
{
newChildren[originalLen + i] = LinkedChild.Create(itemList[i]);
}
2022-10-24 20:40:47 -04:00
collection.LinkedChildren = newChildren;
2018-09-12 13:26:21 -04:00
collection.UpdateRatingToItems(linkedChildrenList);
2020-08-21 16:01:19 -04:00
await collection.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
2015-09-29 12:29:06 -04:00
2018-09-12 13:26:21 -04:00
refreshOptions.ForceSave = true;
2017-04-29 22:37:51 -04:00
_providerManager.QueueRefresh(collection.Id, refreshOptions, RefreshPriority.High);
2014-07-07 21:41:03 -04:00
2015-10-07 17:42:29 -04:00
if (fireEvent)
2014-07-07 21:41:03 -04:00
{
ItemsAddedToCollection?.Invoke(this, new CollectionModifiedEventArgs(collection, itemList));
2015-10-07 17:42:29 -04:00
}
2014-07-07 21:41:03 -04:00
}
}
/// <inheritdoc />
2020-08-21 16:01:19 -04:00
public async Task RemoveFromCollectionAsync(Guid collectionId, IEnumerable<Guid> itemIds)
{
2021-08-15 11:20:07 -04:00
if (_libraryManager.GetItemById(collectionId) is not BoxSet collection)
{
throw new ArgumentException("No collection exists with the supplied Id");
}
var list = new List<LinkedChild>();
2014-07-07 21:41:03 -04:00
var itemList = new List<BaseItem>();
2017-08-20 17:07:47 -04:00
foreach (var guidId in itemIds)
{
2017-08-19 15:43:35 -04:00
var childItem = _libraryManager.GetItemById(guidId);
2016-10-10 14:18:28 -04:00
2022-12-05 09:01:13 -05:00
var child = collection.LinkedChildren.FirstOrDefault(i => (i.ItemId.HasValue && i.ItemId.Value.Equals(guidId)) || (childItem is not null && string.Equals(childItem.Path, i.Path, StringComparison.OrdinalIgnoreCase)));
2022-12-05 09:00:20 -05:00
if (child is null)
{
_logger.LogWarning("No collection title exists with the supplied Id");
2018-09-12 13:26:21 -04:00
continue;
}
list.Add(child);
2014-03-15 00:14:07 -04:00
2022-12-05 09:01:13 -05:00
if (childItem is not null)
2014-07-07 21:41:03 -04:00
{
itemList.Add(childItem);
}
}
2017-08-10 14:01:31 -04:00
if (list.Count > 0)
{
2017-08-10 14:01:31 -04:00
collection.LinkedChildren = collection.LinkedChildren.Except(list).ToArray();
}
2020-08-21 16:01:19 -04:00
await collection.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
2020-04-11 06:03:10 -04:00
_providerManager.QueueRefresh(
collection.Id,
new MetadataRefreshOptions(new DirectoryService(_fileSystem))
{
ForceSave = true
},
RefreshPriority.High);
2014-07-07 21:41:03 -04:00
ItemsRemovedFromCollection?.Invoke(this, new CollectionModifiedEventArgs(collection, itemList));
2014-03-06 00:17:13 -05:00
}
2014-04-22 13:25:54 -04:00
/// <inheritdoc />
2020-05-20 13:07:53 -04:00
public IEnumerable<BaseItem> CollapseItemsWithinBoxSets(IEnumerable<BaseItem> items, User user)
2014-04-22 13:25:54 -04:00
{
2015-01-22 11:41:34 -05:00
var results = new Dictionary<Guid, BaseItem>();
2015-01-24 14:03:55 -05:00
2022-01-04 11:57:19 -05:00
var allBoxSets = GetCollections(user).ToList();
2014-04-22 13:25:54 -04:00
2015-01-22 11:41:34 -05:00
foreach (var item in items)
2014-04-22 13:25:54 -04:00
{
2021-08-15 11:20:07 -04:00
if (item is ISupportsBoxSetGrouping)
2014-04-22 13:25:54 -04:00
{
2015-01-22 11:41:34 -05:00
var itemId = item.Id;
var itemIsInBoxSet = false;
foreach (var boxSet in allBoxSets)
2015-01-22 11:41:34 -05:00
{
if (!boxSet.ContainsLinkedChildByItemId(itemId))
2015-01-22 11:41:34 -05:00
{
continue;
2015-01-22 11:41:34 -05:00
}
itemIsInBoxSet = true;
2021-04-13 14:12:50 -04:00
results.TryAdd(boxSet.Id, boxSet);
2015-01-22 11:41:34 -05:00
}
// skip any item that is in a box set
if (itemIsInBoxSet)
2015-01-22 11:41:34 -05:00
{
continue;
2015-01-22 11:41:34 -05:00
}
var alreadyInResults = false;
2021-08-15 11:20:07 -04:00
// 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)
2015-01-22 11:41:34 -05:00
{
foreach (var childId in video.GetLocalAlternateVersionIds())
{
if (!results.ContainsKey(childId))
{
continue;
}
2021-03-13 14:33:05 -05:00
alreadyInResults = true;
break;
}
2015-01-22 11:41:34 -05:00
}
2021-08-15 11:20:07 -04:00
if (alreadyInResults)
{
2021-08-15 11:20:07 -04:00
continue;
}
2014-04-22 13:25:54 -04:00
}
2021-08-15 11:20:07 -04:00
results[item.Id] = item;
2014-04-22 13:25:54 -04:00
}
2015-01-22 11:41:34 -05:00
return results.Values;
2014-04-22 13:25:54 -04:00
}
2014-03-06 00:17:13 -05:00
}
}