mirror of
https://github.com/Readarr/Readarr
synced 2025-12-28 11:13:22 +01:00
Fixed: Support nested file naming formats
This commit is contained in:
parent
eb431f09fd
commit
dc1fbb3a7e
2 changed files with 163 additions and 9 deletions
|
|
@ -0,0 +1,127 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using FizzWare.NBuilder;
|
||||
using FluentAssertions;
|
||||
using NUnit.Framework;
|
||||
using NzbDrone.Core.Books;
|
||||
using NzbDrone.Core.MediaFiles;
|
||||
using NzbDrone.Core.Organizer;
|
||||
using NzbDrone.Core.Qualities;
|
||||
using NzbDrone.Core.Test.Framework;
|
||||
using NzbDrone.Test.Common;
|
||||
|
||||
namespace NzbDrone.Core.Test.OrganizerTests.FileNameBuilderTests
|
||||
{
|
||||
[TestFixture]
|
||||
public class NestedFileNameBuilderFixture : CoreTest<FileNameBuilder>
|
||||
{
|
||||
private Author _artist;
|
||||
private Book _album;
|
||||
private Edition _release;
|
||||
private BookFile _trackFile;
|
||||
private NamingConfig _namingConfig;
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
_artist = Builder<Author>
|
||||
.CreateNew()
|
||||
.With(s => s.Name = "AuthorName")
|
||||
.With(s => s.Metadata = new AuthorMetadata
|
||||
{
|
||||
Disambiguation = "US Author",
|
||||
Name = "AuthorName"
|
||||
})
|
||||
.Build();
|
||||
|
||||
_album = Builder<Book>
|
||||
.CreateNew()
|
||||
.With(s => s.Author = _artist)
|
||||
.With(s => s.AuthorMetadata = _artist.Metadata.Value)
|
||||
.With(s => s.Title = "A Novel")
|
||||
.With(s => s.ReleaseDate = new DateTime(2020, 1, 15))
|
||||
.With(s => s.SeriesLinks = new List<SeriesBookLink>())
|
||||
.Build();
|
||||
|
||||
_release = Builder<Edition>
|
||||
.CreateNew()
|
||||
.With(s => s.Monitored = true)
|
||||
.With(s => s.Book = _album)
|
||||
.With(s => s.Title = "A Novel")
|
||||
.With(s => s.ReleaseDate = new DateTime(2020, 1, 15))
|
||||
.Build();
|
||||
|
||||
_namingConfig = NamingConfig.Default;
|
||||
_namingConfig.RenameBooks = true;
|
||||
|
||||
Mocker.GetMock<INamingConfigService>()
|
||||
.Setup(c => c.GetConfig()).Returns(_namingConfig);
|
||||
|
||||
_trackFile = Builder<BookFile>.CreateNew()
|
||||
.With(e => e.Quality = new QualityModel(Quality.MOBI))
|
||||
.With(e => e.ReleaseGroup = "ReadarrTest")
|
||||
.Build();
|
||||
|
||||
Mocker.GetMock<IQualityDefinitionService>()
|
||||
.Setup(v => v.Get(Moq.It.IsAny<Quality>()))
|
||||
.Returns<Quality>(v => Quality.DefaultQualityDefinitions.First(c => c.Quality == v));
|
||||
}
|
||||
|
||||
private void WithSeries()
|
||||
{
|
||||
_album.SeriesLinks = new List<SeriesBookLink>
|
||||
{
|
||||
new SeriesBookLink
|
||||
{
|
||||
Series = new Series
|
||||
{
|
||||
Title = "A Series",
|
||||
},
|
||||
Position = "2-3",
|
||||
SeriesPosition = 1
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
[Test]
|
||||
public void should_build_nested_standard_track_filename_with_forward_slash()
|
||||
{
|
||||
WithSeries();
|
||||
|
||||
_namingConfig.StandardBookFormat = "{Book Series}/{Book SeriesTitle - }{Book Title} {(Release Year)}";
|
||||
|
||||
var name = Subject.BuildBookFileName(_artist, _release, _trackFile)
|
||||
.Should().Be("A Series\\A Series #2-3 - A Novel (2020)".AsOsAgnostic());
|
||||
}
|
||||
|
||||
[Test]
|
||||
public void should_build_standard_track_filename_with_forward_slash()
|
||||
{
|
||||
_namingConfig.StandardBookFormat = "{Book Series}/{Book SeriesTitle - }{Book Title} {(Release Year)}";
|
||||
|
||||
Subject.BuildBookFileName(_artist, _release, _trackFile)
|
||||
.Should().Be("A Novel (2020)".AsOsAgnostic());
|
||||
}
|
||||
|
||||
[Test]
|
||||
public void should_build_nested_standard_track_filename_with_back_slash()
|
||||
{
|
||||
WithSeries();
|
||||
|
||||
_namingConfig.StandardBookFormat = "{Book Series}\\{Book SeriesTitle - }{Book Title} {(Release Year)}";
|
||||
|
||||
Subject.BuildBookFileName(_artist, _release, _trackFile)
|
||||
.Should().Be("A Series\\A Series #2-3 - A Novel (2020)".AsOsAgnostic());
|
||||
}
|
||||
|
||||
[Test]
|
||||
public void should_build_standard_track_filename_with_back_slash()
|
||||
{
|
||||
_namingConfig.StandardBookFormat = "{Book Series}\\{Book SeriesTitle - }{Book Title} {(Release Year)}";
|
||||
|
||||
Subject.BuildBookFileName(_artist, _release, _trackFile)
|
||||
.Should().Be("A Novel (2020)".AsOsAgnostic());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -88,9 +88,6 @@ public string BuildBookFileName(Author author, Edition edition, BookFile bookFil
|
|||
|
||||
var pattern = namingConfig.StandardBookFormat;
|
||||
|
||||
var subFolders = pattern.Split(new[] { '/', '\\' }, StringSplitOptions.RemoveEmptyEntries);
|
||||
var safePattern = subFolders.Aggregate("", (current, folderLevel) => Path.Combine(current, folderLevel));
|
||||
|
||||
var tokenHandlers = new Dictionary<string, Func<TokenMatch, string>>(FileNameBuilderTokenEqualityComparer.Instance);
|
||||
|
||||
AddAuthorTokens(tokenHandlers, author);
|
||||
|
|
@ -100,13 +97,26 @@ public string BuildBookFileName(Author author, Edition edition, BookFile bookFil
|
|||
AddMediaInfoTokens(tokenHandlers, bookFile);
|
||||
AddPreferredWords(tokenHandlers, author, bookFile, preferredWords);
|
||||
|
||||
var fileName = ReplacePartTokens(safePattern, tokenHandlers, namingConfig);
|
||||
fileName = ReplaceTokens(fileName, tokenHandlers, namingConfig).Trim();
|
||||
var splitPatterns = pattern.Split(new char[] { '\\', '/' }, StringSplitOptions.RemoveEmptyEntries);
|
||||
var components = new List<string>();
|
||||
|
||||
fileName = FileNameCleanupRegex.Replace(fileName, match => match.Captures[0].Value[0].ToString());
|
||||
fileName = TrimSeparatorsRegex.Replace(fileName, string.Empty);
|
||||
foreach (var s in splitPatterns)
|
||||
{
|
||||
var splitPattern = s;
|
||||
|
||||
return fileName;
|
||||
var component = ReplacePartTokens(splitPattern, tokenHandlers, namingConfig).Trim();
|
||||
component = ReplaceTokens(component, tokenHandlers, namingConfig).Trim();
|
||||
|
||||
component = FileNameCleanupRegex.Replace(component, match => match.Captures[0].Value[0].ToString());
|
||||
component = TrimSeparatorsRegex.Replace(component, string.Empty);
|
||||
|
||||
if (component.IsNotNullOrWhiteSpace())
|
||||
{
|
||||
components.Add(component);
|
||||
}
|
||||
}
|
||||
|
||||
return Path.Combine(components.ToArray());
|
||||
}
|
||||
|
||||
public string BuildBookFilePath(Author author, Edition edition, string fileName, string extension)
|
||||
|
|
@ -175,11 +185,28 @@ public string GetAuthorFolder(Author author, NamingConfig namingConfig = null)
|
|||
namingConfig = _namingConfigService.GetConfig();
|
||||
}
|
||||
|
||||
var pattern = namingConfig.AuthorFolderFormat;
|
||||
var tokenHandlers = new Dictionary<string, Func<TokenMatch, string>>(FileNameBuilderTokenEqualityComparer.Instance);
|
||||
|
||||
AddAuthorTokens(tokenHandlers, author);
|
||||
|
||||
return CleanFolderName(ReplaceTokens(namingConfig.AuthorFolderFormat, tokenHandlers, namingConfig));
|
||||
var splitPatterns = pattern.Split(new char[] { '\\', '/' }, StringSplitOptions.RemoveEmptyEntries);
|
||||
var components = new List<string>();
|
||||
|
||||
foreach (var s in splitPatterns)
|
||||
{
|
||||
var splitPattern = s;
|
||||
|
||||
var component = ReplaceTokens(splitPattern, tokenHandlers, namingConfig);
|
||||
component = CleanFolderName(component);
|
||||
|
||||
if (component.IsNotNullOrWhiteSpace())
|
||||
{
|
||||
components.Add(component);
|
||||
}
|
||||
}
|
||||
|
||||
return Path.Combine(components.ToArray());
|
||||
}
|
||||
|
||||
public static string CleanTitle(string title)
|
||||
|
|
|
|||
Loading…
Reference in a new issue