AudioResolver.cs 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. #nullable disable
  2. #pragma warning disable CS1591
  3. using System;
  4. using System.Collections.Generic;
  5. using System.IO;
  6. using System.Linq;
  7. using Emby.Naming.Audio;
  8. using Emby.Naming.AudioBook;
  9. using Emby.Naming.Common;
  10. using Emby.Naming.Video;
  11. using Jellyfin.Data.Enums;
  12. using MediaBrowser.Controller.Entities;
  13. using MediaBrowser.Controller.Library;
  14. using MediaBrowser.Controller.Providers;
  15. using MediaBrowser.Controller.Resolvers;
  16. using MediaBrowser.Model.IO;
  17. namespace Emby.Server.Implementations.Library.Resolvers.Audio
  18. {
  19. /// <summary>
  20. /// Class AudioResolver.
  21. /// </summary>
  22. public class AudioResolver : ItemResolver<MediaBrowser.Controller.Entities.Audio.Audio>, IMultiItemResolver
  23. {
  24. private readonly NamingOptions _namingOptions;
  25. public AudioResolver(NamingOptions namingOptions)
  26. {
  27. _namingOptions = namingOptions;
  28. }
  29. /// <summary>
  30. /// Gets the priority.
  31. /// </summary>
  32. /// <value>The priority.</value>
  33. public override ResolverPriority Priority => ResolverPriority.Fifth;
  34. public MultiItemResolverResult ResolveMultiple(
  35. Folder parent,
  36. List<FileSystemMetadata> files,
  37. CollectionType? collectionType,
  38. IDirectoryService directoryService)
  39. {
  40. var result = ResolveMultipleInternal(parent, files, collectionType);
  41. if (result is not null)
  42. {
  43. foreach (var item in result.Items)
  44. {
  45. SetInitialItemValues((MediaBrowser.Controller.Entities.Audio.Audio)item, null);
  46. }
  47. }
  48. return result;
  49. }
  50. private MultiItemResolverResult ResolveMultipleInternal(
  51. Folder parent,
  52. List<FileSystemMetadata> files,
  53. CollectionType? collectionType)
  54. {
  55. if (collectionType == CollectionType.books)
  56. {
  57. return ResolveMultipleAudio(parent, files, true);
  58. }
  59. return null;
  60. }
  61. /// <summary>
  62. /// Resolves the specified args.
  63. /// </summary>
  64. /// <param name="args">The args.</param>
  65. /// <returns>Entities.Audio.Audio.</returns>
  66. protected override MediaBrowser.Controller.Entities.Audio.Audio Resolve(ItemResolveArgs args)
  67. {
  68. // Return audio if the path is a file and has a matching extension
  69. var collectionType = args.GetCollectionType();
  70. var isBooksCollectionType = collectionType == CollectionType.books;
  71. if (args.IsDirectory)
  72. {
  73. if (!isBooksCollectionType)
  74. {
  75. return null;
  76. }
  77. return FindAudioBook(args, false);
  78. }
  79. if (AudioFileParser.IsAudioFile(args.Path, _namingOptions))
  80. {
  81. var extension = Path.GetExtension(args.Path.AsSpan());
  82. if (extension.Equals(".cue", StringComparison.OrdinalIgnoreCase))
  83. {
  84. // if audio file exists of same name, return null
  85. return null;
  86. }
  87. var isMixedCollectionType = collectionType is null;
  88. // For conflicting extensions, give priority to videos
  89. if (isMixedCollectionType && VideoResolver.IsVideoFile(args.Path, _namingOptions))
  90. {
  91. return null;
  92. }
  93. MediaBrowser.Controller.Entities.Audio.Audio item = null;
  94. var isMusicCollectionType = collectionType == CollectionType.music;
  95. // Use regular audio type for mixed libraries, owned items and music
  96. if (isMixedCollectionType ||
  97. args.Parent is null ||
  98. isMusicCollectionType)
  99. {
  100. item = new MediaBrowser.Controller.Entities.Audio.Audio();
  101. }
  102. else if (isBooksCollectionType)
  103. {
  104. item = new AudioBook();
  105. }
  106. if (item is not null)
  107. {
  108. item.IsShortcut = extension.Equals(".strm", StringComparison.OrdinalIgnoreCase);
  109. item.IsInMixedFolder = true;
  110. }
  111. return item;
  112. }
  113. return null;
  114. }
  115. private AudioBook FindAudioBook(ItemResolveArgs args, bool parseName)
  116. {
  117. // TODO: Allow GetMultiDiscMovie in here
  118. var result = ResolveMultipleAudio(args.Parent, args.GetActualFileSystemChildren(), parseName);
  119. if (result is null || result.Items.Count != 1 || result.Items[0] is not AudioBook item)
  120. {
  121. return null;
  122. }
  123. // If we were supporting this we'd be checking filesFromOtherItems
  124. item.IsInMixedFolder = false;
  125. item.Name = Path.GetFileName(item.ContainingFolderPath);
  126. return item;
  127. }
  128. private MultiItemResolverResult ResolveMultipleAudio(Folder parent, IEnumerable<FileSystemMetadata> fileSystemEntries, bool parseName)
  129. {
  130. var files = new List<FileSystemMetadata>();
  131. var leftOver = new List<FileSystemMetadata>();
  132. // Loop through each child file/folder and see if we find a video
  133. foreach (var child in fileSystemEntries)
  134. {
  135. if (child.IsDirectory)
  136. {
  137. leftOver.Add(child);
  138. }
  139. else
  140. {
  141. files.Add(child);
  142. }
  143. }
  144. var resolver = new AudioBookListResolver(_namingOptions);
  145. var resolverResult = resolver.Resolve(files).ToList();
  146. var result = new MultiItemResolverResult
  147. {
  148. ExtraFiles = leftOver,
  149. Items = new List<BaseItem>()
  150. };
  151. var isInMixedFolder = resolverResult.Count > 1 || (parent is not null && parent.IsTopParent);
  152. foreach (var resolvedItem in resolverResult)
  153. {
  154. if (resolvedItem.Files.Count > 1)
  155. {
  156. // For now, until we sort out naming for multi-part books
  157. continue;
  158. }
  159. // Until multi-part books are handled letting files stack hides them from browsing in the client
  160. if (resolvedItem.Files.Count == 0 || resolvedItem.Extras.Count > 0 || resolvedItem.AlternateVersions.Count > 0)
  161. {
  162. continue;
  163. }
  164. var firstMedia = resolvedItem.Files[0];
  165. var libraryItem = new AudioBook
  166. {
  167. Path = firstMedia.Path,
  168. IsInMixedFolder = isInMixedFolder,
  169. ProductionYear = resolvedItem.Year,
  170. Name = parseName ?
  171. resolvedItem.Name :
  172. Path.GetFileNameWithoutExtension(firstMedia.Path),
  173. // AdditionalParts = resolvedItem.Files.Skip(1).Select(i => i.Path).ToArray(),
  174. // LocalAlternateVersions = resolvedItem.AlternateVersions.Select(i => i.Path).ToArray()
  175. };
  176. result.Items.Add(libraryItem);
  177. }
  178. result.ExtraFiles.AddRange(files.Where(i => !ContainsFile(resolverResult, i)));
  179. return result;
  180. }
  181. private static bool ContainsFile(IEnumerable<AudioBookInfo> result, FileSystemMetadata file)
  182. {
  183. return result.Any(i => ContainsFile(i, file));
  184. }
  185. private static bool ContainsFile(AudioBookInfo result, FileSystemMetadata file)
  186. {
  187. return result.Files.Any(i => ContainsFile(i, file)) ||
  188. result.AlternateVersions.Any(i => ContainsFile(i, file)) ||
  189. result.Extras.Any(i => ContainsFile(i, file));
  190. }
  191. private static bool ContainsFile(AudioBookFileInfo result, FileSystemMetadata file)
  192. {
  193. return string.Equals(result.Path, file.FullName, StringComparison.OrdinalIgnoreCase);
  194. }
  195. }
  196. }