jellyfin/Emby.Server.Implementations/Library/Resolvers/Audio/AudioResolver.cs

241 lines
8.2 KiB
C#
Raw Normal View History

#nullable disable
2019-11-01 13:38:54 -04:00
#pragma warning disable CS1591
2014-02-23 15:35:58 -05:00
using System;
using System.Collections.Generic;
2017-06-15 13:22:05 -04:00
using System.IO;
2017-11-16 16:25:18 -05:00
using System.Linq;
using Emby.Naming.Audio;
using Emby.Naming.AudioBook;
using Emby.Naming.Common;
using Emby.Naming.Video;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
2017-11-16 16:25:18 -05:00
using MediaBrowser.Controller.Providers;
using MediaBrowser.Controller.Resolvers;
using MediaBrowser.Model.Entities;
2017-11-16 16:25:18 -05:00
using MediaBrowser.Model.IO;
namespace Emby.Server.Implementations.Library.Resolvers.Audio
{
/// <summary>
2019-11-01 13:38:54 -04:00
/// Class AudioResolver.
/// </summary>
2017-11-16 16:25:18 -05:00
public class AudioResolver : ItemResolver<MediaBrowser.Controller.Entities.Audio.Audio>, IMultiItemResolver
{
private readonly NamingOptions _namingOptions;
2014-11-16 15:44:08 -05:00
public AudioResolver(NamingOptions namingOptions)
2014-11-16 15:44:08 -05:00
{
_namingOptions = namingOptions;
2014-11-16 15:44:08 -05:00
}
/// <summary>
/// Gets the priority.
/// </summary>
/// <value>The priority.</value>
public override ResolverPriority Priority => ResolverPriority.Fifth;
2020-10-12 14:05:11 -04:00
public MultiItemResolverResult ResolveMultiple(
Folder parent,
2017-11-16 16:25:18 -05:00
List<FileSystemMetadata> files,
string collectionType,
IDirectoryService directoryService)
{
var result = ResolveMultipleInternal(parent, files, collectionType);
2017-11-16 16:25:18 -05:00
2022-12-05 09:01:13 -05:00
if (result is not null)
2017-11-16 16:25:18 -05:00
{
foreach (var item in result.Items)
{
SetInitialItemValues((MediaBrowser.Controller.Entities.Audio.Audio)item, null);
}
}
return result;
}
2020-10-12 14:05:11 -04:00
private MultiItemResolverResult ResolveMultipleInternal(
Folder parent,
2017-11-16 16:25:18 -05:00
List<FileSystemMetadata> files,
string collectionType)
2017-11-16 16:25:18 -05:00
{
if (string.Equals(collectionType, CollectionType.Books, StringComparison.OrdinalIgnoreCase))
{
return ResolveMultipleAudio(parent, files, true);
2017-11-16 16:25:18 -05:00
}
return null;
}
/// <summary>
/// Resolves the specified args.
/// </summary>
/// <param name="args">The args.</param>
/// <returns>Entities.Audio.Audio.</returns>
protected override MediaBrowser.Controller.Entities.Audio.Audio Resolve(ItemResolveArgs args)
{
// Return audio if the path is a file and has a matching extension
2017-11-16 16:25:18 -05:00
var collectionType = args.GetCollectionType();
var isBooksCollectionType = string.Equals(collectionType, CollectionType.Books, StringComparison.OrdinalIgnoreCase);
if (args.IsDirectory)
{
2017-11-16 16:25:18 -05:00
if (!isBooksCollectionType)
{
return null;
}
return FindAudioBook(args, false);
2017-11-16 16:25:18 -05:00
}
if (AudioFileParser.IsAudioFile(args.Path, _namingOptions))
2017-11-16 16:25:18 -05:00
{
2018-09-12 13:26:21 -04:00
var extension = Path.GetExtension(args.Path);
if (string.Equals(extension, ".cue", StringComparison.OrdinalIgnoreCase))
{
2017-11-16 16:25:18 -05:00
// if audio file exists of same name, return null
return null;
}
2017-06-15 13:22:05 -04:00
2018-09-12 13:26:21 -04:00
var isMixedCollectionType = string.IsNullOrEmpty(collectionType);
2017-06-15 13:22:05 -04:00
2017-11-16 16:25:18 -05:00
// For conflicting extensions, give priority to videos
if (isMixedCollectionType && VideoResolver.IsVideoFile(args.Path, _namingOptions))
2017-11-16 16:25:18 -05:00
{
return null;
}
2014-02-23 15:35:58 -05:00
2017-11-16 16:25:18 -05:00
MediaBrowser.Controller.Entities.Audio.Audio item = null;
2014-12-11 01:20:28 -05:00
2017-11-16 16:25:18 -05:00
var isMusicCollectionType = string.Equals(collectionType, CollectionType.Music, StringComparison.OrdinalIgnoreCase);
2014-12-11 01:20:28 -05:00
2017-11-16 16:25:18 -05:00
// Use regular audio type for mixed libraries, owned items and music
if (isMixedCollectionType ||
2022-12-05 09:00:20 -05:00
args.Parent is null ||
2017-11-16 16:25:18 -05:00
isMusicCollectionType)
{
item = new MediaBrowser.Controller.Entities.Audio.Audio();
}
else if (isBooksCollectionType)
{
item = new AudioBook();
}
2016-12-12 00:49:19 -05:00
2022-12-05 09:01:13 -05:00
if (item is not null)
2017-11-16 16:25:18 -05:00
{
2018-09-12 13:26:21 -04:00
item.IsShortcut = string.Equals(extension, ".strm", StringComparison.OrdinalIgnoreCase);
2017-11-16 16:25:18 -05:00
item.IsInMixedFolder = true;
}
2017-11-16 16:25:18 -05:00
return item;
}
return null;
}
2017-11-16 16:25:18 -05:00
private AudioBook FindAudioBook(ItemResolveArgs args, bool parseName)
2017-11-16 16:25:18 -05:00
{
// TODO: Allow GetMultiDiscMovie in here
var result = ResolveMultipleAudio(args.Parent, args.GetActualFileSystemChildren(), parseName);
2017-11-16 16:25:18 -05:00
2022-12-05 09:00:20 -05:00
if (result is null || result.Items.Count != 1 || result.Items[0] is not AudioBook item)
2017-11-16 16:25:18 -05:00
{
return null;
2017-11-16 16:25:18 -05:00
}
// If we were supporting this we'd be checking filesFromOtherItems
item.IsInMixedFolder = false;
item.Name = Path.GetFileName(item.ContainingFolderPath);
return item;
2017-11-16 16:25:18 -05:00
}
private MultiItemResolverResult ResolveMultipleAudio(Folder parent, IEnumerable<FileSystemMetadata> fileSystemEntries, bool parseName)
2017-11-16 16:25:18 -05:00
{
var files = new List<FileSystemMetadata>();
var leftOver = new List<FileSystemMetadata>();
// Loop through each child file/folder and see if we find a video
foreach (var child in fileSystemEntries)
{
if (child.IsDirectory)
{
leftOver.Add(child);
}
else
2017-11-16 16:25:18 -05:00
{
files.Add(child);
}
}
var resolver = new AudioBookListResolver(_namingOptions);
2017-11-16 16:25:18 -05:00
var resolverResult = resolver.Resolve(files).ToList();
var result = new MultiItemResolverResult
{
ExtraFiles = leftOver,
2023-02-28 18:44:57 -05:00
Items = new List<BaseItem>()
2017-11-16 16:25:18 -05:00
};
2022-12-05 09:01:13 -05:00
var isInMixedFolder = resolverResult.Count > 1 || (parent is not null && parent.IsTopParent);
2017-11-16 16:25:18 -05:00
foreach (var resolvedItem in resolverResult)
{
2017-11-19 19:20:12 -05:00
if (resolvedItem.Files.Count > 1)
{
// For now, until we sort out naming for multi-part books
continue;
}
// Until multi-part books are handled letting files stack hides them from browsing in the client
if (resolvedItem.Files.Count == 0 || resolvedItem.Extras.Count > 0 || resolvedItem.AlternateVersions.Count > 0)
{
continue;
}
2020-11-14 10:28:49 -05:00
var firstMedia = resolvedItem.Files[0];
2019-01-07 18:27:46 -05:00
var libraryItem = new AudioBook
2017-11-16 16:25:18 -05:00
{
Path = firstMedia.Path,
IsInMixedFolder = isInMixedFolder,
2017-11-19 19:20:12 -05:00
ProductionYear = resolvedItem.Year,
2017-11-16 16:25:18 -05:00
Name = parseName ?
resolvedItem.Name :
Path.GetFileNameWithoutExtension(firstMedia.Path),
2020-06-14 05:11:11 -04:00
// AdditionalParts = resolvedItem.Files.Skip(1).Select(i => i.Path).ToArray(),
// LocalAlternateVersions = resolvedItem.AlternateVersions.Select(i => i.Path).ToArray()
2017-11-16 16:25:18 -05:00
};
result.Items.Add(libraryItem);
}
result.ExtraFiles.AddRange(files.Where(i => !ContainsFile(resolverResult, i)));
return result;
}
private static bool ContainsFile(IEnumerable<AudioBookInfo> result, FileSystemMetadata file)
2017-11-16 16:25:18 -05:00
{
return result.Any(i => ContainsFile(i, file));
}
private static bool ContainsFile(AudioBookInfo result, FileSystemMetadata file)
2017-11-16 16:25:18 -05:00
{
return result.Files.Any(i => ContainsFile(i, file)) ||
result.AlternateVersions.Any(i => ContainsFile(i, file)) ||
result.Extras.Any(i => ContainsFile(i, file));
}
private static bool ContainsFile(AudioBookFileInfo result, FileSystemMetadata file)
2017-11-16 16:25:18 -05:00
{
return string.Equals(result.Path, file.FullName, StringComparison.OrdinalIgnoreCase);
}
}
}