2
0

CollectionManager.cs 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. using MediaBrowser.Common.IO;
  2. using MediaBrowser.Controller.Collections;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Movies;
  5. using MediaBrowser.Controller.Library;
  6. using MediaBrowser.Controller.Providers;
  7. using System;
  8. using System.Collections.Generic;
  9. using System.IO;
  10. using System.Linq;
  11. using System.Threading;
  12. using System.Threading.Tasks;
  13. namespace MediaBrowser.Server.Implementations.Collections
  14. {
  15. public class CollectionManager : ICollectionManager
  16. {
  17. private readonly ILibraryManager _libraryManager;
  18. private readonly IFileSystem _fileSystem;
  19. private readonly ILibraryMonitor _iLibraryMonitor;
  20. public CollectionManager(ILibraryManager libraryManager, IFileSystem fileSystem, ILibraryMonitor iLibraryMonitor)
  21. {
  22. _libraryManager = libraryManager;
  23. _fileSystem = fileSystem;
  24. _iLibraryMonitor = iLibraryMonitor;
  25. }
  26. public async Task CreateCollection(CollectionCreationOptions options)
  27. {
  28. var name = options.Name;
  29. // Need to use the [boxset] suffix
  30. // If internet metadata is not found, or if xml saving is off there will be no collection.xml
  31. // This could cause it to get re-resolved as a plain folder
  32. var folderName = _fileSystem.GetValidFilename(name) + " [boxset]";
  33. var parentFolder = GetParentFolder(options.ParentId);
  34. if (parentFolder == null)
  35. {
  36. throw new ArgumentException();
  37. }
  38. var path = Path.Combine(parentFolder.Path, folderName);
  39. _iLibraryMonitor.ReportFileSystemChangeBeginning(path);
  40. try
  41. {
  42. Directory.CreateDirectory(path);
  43. var collection = new BoxSet
  44. {
  45. Name = name,
  46. Parent = parentFolder,
  47. DisplayMediaType = "Collection",
  48. Path = path,
  49. DontFetchMeta = options.IsLocked,
  50. ProviderIds = options.ProviderIds
  51. };
  52. await parentFolder.AddChild(collection, CancellationToken.None).ConfigureAwait(false);
  53. await collection.RefreshMetadata(new MetadataRefreshOptions(), CancellationToken.None)
  54. .ConfigureAwait(false);
  55. }
  56. finally
  57. {
  58. // Refresh handled internally
  59. _iLibraryMonitor.ReportFileSystemChangeComplete(path, false);
  60. }
  61. }
  62. private Folder GetParentFolder(Guid? parentId)
  63. {
  64. if (parentId.HasValue)
  65. {
  66. if (parentId.Value == Guid.Empty)
  67. {
  68. throw new ArgumentNullException("parentId");
  69. }
  70. var folder = _libraryManager.GetItemById(parentId.Value) as Folder;
  71. // Find an actual physical folder
  72. if (folder is CollectionFolder)
  73. {
  74. return _libraryManager.RootFolder.Children.OfType<Folder>().First(i => folder.PhysicalLocations.Contains(i.Path, StringComparer.OrdinalIgnoreCase));
  75. }
  76. }
  77. return _libraryManager.RootFolder.Children.OfType<ManualCollectionsFolder>().FirstOrDefault() ??
  78. _libraryManager.RootFolder.GetHiddenChildren().OfType<ManualCollectionsFolder>().FirstOrDefault();
  79. }
  80. public async Task AddToCollection(Guid collectionId, IEnumerable<Guid> ids)
  81. {
  82. var collection = _libraryManager.GetItemById(collectionId) as BoxSet;
  83. if (collection == null)
  84. {
  85. throw new ArgumentException("No collection exists with the supplied Id");
  86. }
  87. var list = new List<LinkedChild>();
  88. foreach (var itemId in ids)
  89. {
  90. var item = _libraryManager.GetItemById(itemId);
  91. if (item == null)
  92. {
  93. throw new ArgumentException("No item exists with the supplied Id");
  94. }
  95. if (collection.LinkedChildren.Any(i => i.ItemId.HasValue && i.ItemId == itemId))
  96. {
  97. throw new ArgumentException("Item already exists in collection");
  98. }
  99. list.Add(new LinkedChild
  100. {
  101. ItemName = item.Name,
  102. ItemYear = item.ProductionYear,
  103. ItemType = item.GetType().Name,
  104. Type = LinkedChildType.Manual
  105. });
  106. }
  107. collection.LinkedChildren.AddRange(list);
  108. await collection.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
  109. await collection.RefreshMetadata(CancellationToken.None).ConfigureAwait(false);
  110. }
  111. public async Task RemoveFromCollection(Guid collectionId, IEnumerable<Guid> itemIds)
  112. {
  113. var collection = _libraryManager.GetItemById(collectionId) as BoxSet;
  114. if (collection == null)
  115. {
  116. throw new ArgumentException("No collection exists with the supplied Id");
  117. }
  118. var list = new List<LinkedChild>();
  119. foreach (var itemId in itemIds)
  120. {
  121. var child = collection.LinkedChildren.FirstOrDefault(i => i.ItemId.HasValue && i.ItemId.Value == itemId);
  122. if (child == null)
  123. {
  124. throw new ArgumentException("No collection title exists with the supplied Id");
  125. }
  126. list.Add(child);
  127. }
  128. var shortcutFiles = Directory
  129. .EnumerateFiles(collection.Path, "*", SearchOption.TopDirectoryOnly)
  130. .Where(i => _fileSystem.IsShortcut(i))
  131. .ToList();
  132. var shortcutFilesToDelete = list.Where(child => !string.IsNullOrWhiteSpace(child.Path) && child.Type == LinkedChildType.Shortcut)
  133. .Select(child => shortcutFiles.FirstOrDefault(i => string.Equals(child.Path, _fileSystem.ResolveShortcut(i), StringComparison.OrdinalIgnoreCase)))
  134. .Where(i => !string.IsNullOrWhiteSpace(i))
  135. .ToList();
  136. foreach (var file in shortcutFilesToDelete)
  137. {
  138. _iLibraryMonitor.ReportFileSystemChangeBeginning(file);
  139. }
  140. try
  141. {
  142. foreach (var file in shortcutFilesToDelete)
  143. {
  144. File.Delete(file);
  145. }
  146. foreach (var child in list)
  147. {
  148. collection.LinkedChildren.Remove(child);
  149. }
  150. }
  151. finally
  152. {
  153. foreach (var file in shortcutFilesToDelete)
  154. {
  155. _iLibraryMonitor.ReportFileSystemChangeComplete(file, false);
  156. }
  157. }
  158. await collection.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
  159. await collection.RefreshMetadata(CancellationToken.None).ConfigureAwait(false);
  160. }
  161. }
  162. }