CollectionManager.cs 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  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<BoxSet> 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. if (options.ItemIdList.Count > 0)
  56. {
  57. await AddToCollection(collection.Id, options.ItemIdList);
  58. }
  59. return collection;
  60. }
  61. finally
  62. {
  63. // Refresh handled internally
  64. _iLibraryMonitor.ReportFileSystemChangeComplete(path, false);
  65. }
  66. }
  67. private Folder GetParentFolder(Guid? parentId)
  68. {
  69. if (parentId.HasValue)
  70. {
  71. if (parentId.Value == Guid.Empty)
  72. {
  73. throw new ArgumentNullException("parentId");
  74. }
  75. var folder = _libraryManager.GetItemById(parentId.Value) as Folder;
  76. // Find an actual physical folder
  77. if (folder is CollectionFolder)
  78. {
  79. return _libraryManager.RootFolder.Children.OfType<Folder>().First(i => folder.PhysicalLocations.Contains(i.Path, StringComparer.OrdinalIgnoreCase));
  80. }
  81. }
  82. return _libraryManager.RootFolder.Children.OfType<ManualCollectionsFolder>().FirstOrDefault() ??
  83. _libraryManager.RootFolder.GetHiddenChildren().OfType<ManualCollectionsFolder>().FirstOrDefault();
  84. }
  85. public async Task AddToCollection(Guid collectionId, IEnumerable<Guid> ids)
  86. {
  87. var collection = _libraryManager.GetItemById(collectionId) as BoxSet;
  88. if (collection == null)
  89. {
  90. throw new ArgumentException("No collection exists with the supplied Id");
  91. }
  92. var list = new List<LinkedChild>();
  93. foreach (var itemId in ids)
  94. {
  95. var item = _libraryManager.GetItemById(itemId);
  96. if (item == null)
  97. {
  98. throw new ArgumentException("No item exists with the supplied Id");
  99. }
  100. if (collection.LinkedChildren.Any(i => i.ItemId.HasValue && i.ItemId == itemId))
  101. {
  102. throw new ArgumentException("Item already exists in collection");
  103. }
  104. list.Add(new LinkedChild
  105. {
  106. ItemName = item.Name,
  107. ItemYear = item.ProductionYear,
  108. ItemType = item.GetType().Name,
  109. Type = LinkedChildType.Manual
  110. });
  111. var supportsGrouping = item as ISupportsBoxSetGrouping;
  112. if (supportsGrouping != null)
  113. {
  114. var boxsetIdList = supportsGrouping.BoxSetIdList.ToList();
  115. if (!boxsetIdList.Contains(collectionId))
  116. {
  117. boxsetIdList.Add(collectionId);
  118. }
  119. supportsGrouping.BoxSetIdList = boxsetIdList;
  120. }
  121. }
  122. collection.LinkedChildren.AddRange(list);
  123. await collection.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
  124. await collection.RefreshMetadata(CancellationToken.None).ConfigureAwait(false);
  125. }
  126. public async Task RemoveFromCollection(Guid collectionId, IEnumerable<Guid> itemIds)
  127. {
  128. var collection = _libraryManager.GetItemById(collectionId) as BoxSet;
  129. if (collection == null)
  130. {
  131. throw new ArgumentException("No collection exists with the supplied Id");
  132. }
  133. var list = new List<LinkedChild>();
  134. foreach (var itemId in itemIds)
  135. {
  136. var child = collection.LinkedChildren.FirstOrDefault(i => i.ItemId.HasValue && i.ItemId.Value == itemId);
  137. if (child == null)
  138. {
  139. throw new ArgumentException("No collection title exists with the supplied Id");
  140. }
  141. list.Add(child);
  142. var childItem = _libraryManager.GetItemById(itemId);
  143. var supportsGrouping = childItem as ISupportsBoxSetGrouping;
  144. if (supportsGrouping != null)
  145. {
  146. var boxsetIdList = supportsGrouping.BoxSetIdList.ToList();
  147. boxsetIdList.Remove(collectionId);
  148. supportsGrouping.BoxSetIdList = boxsetIdList;
  149. }
  150. }
  151. var shortcutFiles = Directory
  152. .EnumerateFiles(collection.Path, "*", SearchOption.TopDirectoryOnly)
  153. .Where(i => _fileSystem.IsShortcut(i))
  154. .ToList();
  155. var shortcutFilesToDelete = list.Where(child => !string.IsNullOrWhiteSpace(child.Path) && child.Type == LinkedChildType.Shortcut)
  156. .Select(child => shortcutFiles.FirstOrDefault(i => string.Equals(child.Path, _fileSystem.ResolveShortcut(i), StringComparison.OrdinalIgnoreCase)))
  157. .Where(i => !string.IsNullOrWhiteSpace(i))
  158. .ToList();
  159. foreach (var file in shortcutFilesToDelete)
  160. {
  161. _iLibraryMonitor.ReportFileSystemChangeBeginning(file);
  162. }
  163. try
  164. {
  165. foreach (var file in shortcutFilesToDelete)
  166. {
  167. File.Delete(file);
  168. }
  169. foreach (var child in list)
  170. {
  171. collection.LinkedChildren.Remove(child);
  172. }
  173. }
  174. finally
  175. {
  176. foreach (var file in shortcutFilesToDelete)
  177. {
  178. _iLibraryMonitor.ReportFileSystemChangeComplete(file, false);
  179. }
  180. }
  181. await collection.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);
  182. await collection.RefreshMetadata(CancellationToken.None).ConfigureAwait(false);
  183. }
  184. }
  185. }