Skip to content

Commit

Permalink
New: Add option to search for anime using standard episode numbers
Browse files Browse the repository at this point in the history
  • Loading branch information
Bakeneko authored and markus101 committed Jan 23, 2022
1 parent 05b1581 commit cee1748
Show file tree
Hide file tree
Showing 12 changed files with 281 additions and 26 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
using System.Collections.Generic;
using System.Linq;
using FluentAssertions;
using NUnit.Framework;
using NzbDrone.Core.Indexers.Fanzub;
using NzbDrone.Core.IndexerSearch.Definitions;
using NzbDrone.Core.Test.Framework;

namespace NzbDrone.Core.Test.IndexerTests.FanzubTests
{
public class FanzubRequestGeneratorFixture : CoreTest<FanzubRequestGenerator>
{
private SeasonSearchCriteria _seasonSearchCriteria;
private AnimeEpisodeSearchCriteria _animeSearchCriteria;

[SetUp]
public void SetUp()
{
Subject.Settings = new FanzubSettings()
{
BaseUrl = "http://127.0.0.1:1234/",
};

_seasonSearchCriteria = new SeasonSearchCriteria()
{
SceneTitles = new List<string>() { "Naruto Shippuuden" },
SeasonNumber = 1,
};

_animeSearchCriteria = new AnimeEpisodeSearchCriteria()
{
SceneTitles = new List<string>() { "Naruto Shippuuden" },
AbsoluteEpisodeNumber = 9,
SeasonNumber = 1,
EpisodeNumber = 9
};
}

[Test]
public void should_not_search_season()
{
var results = Subject.GetSearchRequests(_seasonSearchCriteria);

results.GetAllTiers().Should().HaveCount(0);
}

[Test]
public void should_search_season()
{
Subject.Settings.AnimeStandardFormatSearch = true;
var results = Subject.GetSearchRequests(_seasonSearchCriteria);

results.GetAllTiers().Should().HaveCount(1);

var page = results.GetAllTiers().First().First();

page.Url.FullUri.Should().Contain("q=\"Naruto+Shippuuden%20S01\"|\"Naruto+Shippuuden%20-%20S01\"");
}

[Test]
public void should_use_only_absolute_numbering_for_anime_search()
{
var results = Subject.GetSearchRequests(_animeSearchCriteria);

results.GetAllTiers().Should().HaveCount(1);

var page = results.GetAllTiers().First().First();

page.Url.FullUri.Should().Contain("q=\"Naruto+Shippuuden%2009\"|\"Naruto+Shippuuden%20-%2009\"");
}

[Test]
public void should_also_use_standard_numbering_for_anime_search()
{
Subject.Settings.AnimeStandardFormatSearch = true;
var results = Subject.GetSearchRequests(_animeSearchCriteria);

results.GetAllTiers().Should().HaveCount(1);

var page = results.GetAllTiers().First().First();

page.Url.FullUri.Should().Contain("q=\"Naruto+Shippuuden%2009\"|\"Naruto+Shippuuden%20-%2009\"|\"Naruto+Shippuuden%20S01E09\"|\"Naruto+Shippuuden%20-%20S01E09\"");
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,9 @@ public void SetUp()
_animeSearchCriteria = new AnimeEpisodeSearchCriteria()
{
SceneTitles = new List<string>() { "Monkey+Island" },
AbsoluteEpisodeNumber = 100
AbsoluteEpisodeNumber = 100,
SeasonNumber = 5,
EpisodeNumber = 4
};

_capabilities = new NewznabCapabilities();
Expand Down Expand Up @@ -123,6 +125,31 @@ public void should_not_get_unlimited_pages()
pages.Count.Should().BeLessThan(500);
}

[Test]
public void should_use_only_absolute_numbering_for_anime_search()
{
var results = Subject.GetSearchRequests(_animeSearchCriteria);

results.GetAllTiers().Should().HaveCount(1);

var page = results.GetAllTiers().First().First();

page.Url.FullUri.Should().Contain("q=Monkey%20Island+100");
}

[Test]
public void should_also_use_standard_numbering_for_anime_search()
{
Subject.Settings.AnimeStandardFormatSearch = true;
var results = Subject.GetSearchRequests(_animeSearchCriteria);

results.GetTier(0).Should().HaveCount(2);
var pages = results.GetTier(0).Take(2).Select(t => t.First()).ToList();

pages[0].Url.FullUri.Should().Contain("q=Monkey%20Island+100");
pages[1].Url.FullUri.Should().Contain("q=Monkey%20Island+s05e04");
}

[Test]
public void should_not_search_by_rid_if_not_supported()
{
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
using System.Collections.Generic;
using System.Linq;
using FluentAssertions;
using NUnit.Framework;
using NzbDrone.Core.Indexers.Nyaa;
using NzbDrone.Core.IndexerSearch.Definitions;
using NzbDrone.Core.Test.Framework;

namespace NzbDrone.Core.Test.IndexerTests.NyaaTests
{
public class NyaaRequestGeneratorFixture : CoreTest<NyaaRequestGenerator>
{
private SeasonSearchCriteria _seasonSearchCriteria;
private AnimeEpisodeSearchCriteria _animeSearchCriteria;

[SetUp]
public void SetUp()
{
Subject.Settings = new NyaaSettings()
{
BaseUrl = "http://127.0.0.1:1234/",
};

_seasonSearchCriteria = new SeasonSearchCriteria()
{
SceneTitles = new List<string>() { "Naruto Shippuuden" },
SeasonNumber = 1,
};

_animeSearchCriteria = new AnimeEpisodeSearchCriteria()
{
SceneTitles = new List<string>() { "Naruto Shippuuden" },
AbsoluteEpisodeNumber = 9,
SeasonNumber = 1,
EpisodeNumber = 9
};
}

[Test]
public void should_not_search_season()
{
var results = Subject.GetSearchRequests(_seasonSearchCriteria);

results.GetAllTiers().Should().HaveCount(0);
}

[Test]
public void should_search_season()
{
Subject.Settings.AnimeStandardFormatSearch = true;
var results = Subject.GetSearchRequests(_seasonSearchCriteria);

results.GetAllTiers().Should().HaveCount(1);

var page = results.GetAllTiers().First().First();

page.Url.FullUri.Should().Contain("term=Naruto+Shippuuden+s01");
}

[Test]
public void should_use_only_absolute_numbering_for_anime_search()
{
var results = Subject.GetSearchRequests(_animeSearchCriteria);

results.GetTier(0).Should().HaveCount(2);
var pages = results.GetTier(0).Take(2).Select(t => t.First()).ToList();

pages[0].Url.FullUri.Should().Contain("term=Naruto+Shippuuden+9");
pages[1].Url.FullUri.Should().Contain("term=Naruto+Shippuuden+09");
}

[Test]
public void should_also_use_standard_numbering_for_anime_search()
{
Subject.Settings.AnimeStandardFormatSearch = true;
var results = Subject.GetSearchRequests(_animeSearchCriteria);

results.GetTier(0).Should().HaveCount(3);
var pages = results.GetTier(0).Take(3).Select(t => t.First()).ToList();

pages[0].Url.FullUri.Should().Contain("term=Naruto+Shippuuden+9");
pages[1].Url.FullUri.Should().Contain("term=Naruto+Shippuuden+09");
pages[2].Url.FullUri.Should().Contain("term=Naruto+Shippuuden+s01e09");
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,11 +3,13 @@
public class AnimeEpisodeSearchCriteria : SearchCriteriaBase
{
public int AbsoluteEpisodeNumber { get; set; }
public int EpisodeNumber { get; set; }
public int SeasonNumber { get; set; }
public bool IsSeasonSearch { get; set; }

public override string ToString()
{
return string.Format("[{0} : {1:00}]", Series.Title, AbsoluteEpisodeNumber);
return $"[{Series.Title} : S{SeasonNumber:00}E{EpisodeNumber:00} ({AbsoluteEpisodeNumber:00})]";
}
}
}
16 changes: 3 additions & 13 deletions src/NzbDrone.Core/IndexerSearch/ReleaseSearchService.cs
Original file line number Diff line number Diff line change
Expand Up @@ -349,19 +349,9 @@ private List<DownloadDecision> SearchAnime(Series series, Episode episode, bool

searchSpec.IsSeasonSearch = isSeasonSearch;

if (episode.SceneAbsoluteEpisodeNumber.HasValue)
{
searchSpec.AbsoluteEpisodeNumber = episode.SceneAbsoluteEpisodeNumber.Value;
}
else if (episode.AbsoluteEpisodeNumber.HasValue)
{
searchSpec.AbsoluteEpisodeNumber = episode.AbsoluteEpisodeNumber.Value;
}
else
{
_logger.Error($"Can not search for {series.Title} - S{episode.SeasonNumber:00}E{episode.EpisodeNumber:00} it does not have an absolute episode number");
throw new SearchFailedException("Absolute episode number is missing");
}
searchSpec.SeasonNumber = episode.SceneSeasonNumber ?? episode.SeasonNumber;
searchSpec.EpisodeNumber = episode.SceneEpisodeNumber ?? episode.EpisodeNumber;
searchSpec.AbsoluteEpisodeNumber = episode.SceneAbsoluteEpisodeNumber ?? episode.AbsoluteEpisodeNumber ?? 0;

return Dispatch(indexer => indexer.Fetch(searchSpec), searchSpec);
}
Expand Down
29 changes: 28 additions & 1 deletion src/NzbDrone.Core/Indexers/Fanzub/FanzubRequestGenerator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,15 @@ public virtual IndexerPageableRequestChain GetSearchRequests(SingleEpisodeSearch

public virtual IndexerPageableRequestChain GetSearchRequests(SeasonSearchCriteria searchCriteria)
{
return new IndexerPageableRequestChain();
var pageableRequests = new IndexerPageableRequestChain();

if (Settings.AnimeStandardFormatSearch && searchCriteria.SeasonNumber > 0)
{
var searchTitles = searchCriteria.CleanSceneTitles.SelectMany(v => GetSeasonSearchStrings(v, searchCriteria.SeasonNumber)).ToList();
pageableRequests.Add(GetPagedRequests(string.Join("|", searchTitles)));
}

return pageableRequests;
}

public virtual IndexerPageableRequestChain GetSearchRequests(DailyEpisodeSearchCriteria searchCriteria)
Expand All @@ -55,6 +63,11 @@ public virtual IndexerPageableRequestChain GetSearchRequests(AnimeEpisodeSearchC

var searchTitles = searchCriteria.CleanSceneTitles.SelectMany(v => GetTitleSearchStrings(v, searchCriteria.AbsoluteEpisodeNumber)).ToList();

if (Settings.AnimeStandardFormatSearch && searchCriteria.SeasonNumber > 0 && searchCriteria.EpisodeNumber > 0)
{
searchTitles.AddRange(searchCriteria.CleanSceneTitles.SelectMany(v => GetTitleSearchStrings(v, searchCriteria.SeasonNumber, searchCriteria.EpisodeNumber)).ToList());
}

pageableRequests.Add(GetPagedRequests(string.Join("|", searchTitles)));

return pageableRequests;
Expand All @@ -78,13 +91,27 @@ private IEnumerable<IndexerRequest> GetPagedRequests(string query)
yield return new IndexerRequest(url.ToString(), HttpAccept.Rss);
}

private IEnumerable<string> GetSeasonSearchStrings(string title, int seasonNumber)
{
var formats = new[] { "{0}%20S{1:00}", "{0}%20-%20S{1:00}" };

return formats.Select(s => "\"" + string.Format(s, CleanTitle(title), seasonNumber) + "\"");
}

private IEnumerable<string> GetTitleSearchStrings(string title, int absoluteEpisodeNumber)
{
var formats = new[] { "{0}%20{1:00}", "{0}%20-%20{1:00}" };

return formats.Select(s => "\"" + string.Format(s, CleanTitle(title), absoluteEpisodeNumber) + "\"");
}

private IEnumerable<string> GetTitleSearchStrings(string title, int seasonNumber, int episodeNumber)
{
var formats = new[] { "{0}%20S{1:00}E{2:00}", "{0}%20-%20S{1:00}E{2:00}" };

return formats.Select(s => "\"" + string.Format(s, CleanTitle(title), seasonNumber, episodeNumber) + "\"");
}

private string CleanTitle(string title)
{
return RemoveCharactersRegex.Replace(title, "");
Expand Down
3 changes: 3 additions & 0 deletions src/NzbDrone.Core/Indexers/Fanzub/FanzubSettings.cs
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,9 @@ public FanzubSettings()
[FieldDefinition(0, Label = "Rss URL", HelpText = "Enter to URL to an Fanzub compatible RSS feed")]
public string BaseUrl { get; set; }

[FieldDefinition(1, Label = "Anime Standard Format Search", Type = FieldType.Checkbox, HelpText = "Also search for anime using the standard numbering")]
public bool AnimeStandardFormatSearch { get; set; }

public NzbDroneValidationResult Validate()
{
return new NzbDroneValidationResult(Validator.Validate(this));
Expand Down
9 changes: 9 additions & 0 deletions src/NzbDrone.Core/Indexers/Newznab/NewznabRequestGenerator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -321,6 +321,15 @@ public virtual IndexerPageableRequestChain GetSearchRequests(AnimeEpisodeSearchC
string.Format("&q={0}+{1:00}",
NewsnabifyTitle(queryTitle),
searchCriteria.AbsoluteEpisodeNumber)));

if (Settings.AnimeStandardFormatSearch && searchCriteria.SeasonNumber > 0 && searchCriteria.EpisodeNumber > 0)
{
pageableRequests.Add(GetPagedRequests(MaxPages, Settings.AnimeCategories, "search",
string.Format("&q={0}+s{1:00}e{2:00}",
NewsnabifyTitle(queryTitle),
searchCriteria.SeasonNumber,
searchCriteria.EpisodeNumber)));
}
}
}

Expand Down
5 changes: 4 additions & 1 deletion src/NzbDrone.Core/Indexers/Newznab/NewznabSettings.cs
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,10 @@ public NewznabSettings()
[FieldDefinition(4, Label = "Anime Categories", Type = FieldType.Select, SelectOptionsProviderAction = "newznabCategories", HelpText = "Drop down list, leave blank to disable anime")]
public IEnumerable<int> AnimeCategories { get; set; }

[FieldDefinition(5, Label = "Additional Parameters", HelpText = "Additional Newznab parameters", Advanced = true)]
[FieldDefinition(5, Label = "Anime Standard Format Search", Type = FieldType.Checkbox, HelpText = "Also search for anime using the standard numbering")]
public bool AnimeStandardFormatSearch { get; set; }

[FieldDefinition(6, Label = "Additional Parameters", HelpText = "Additional Newznab parameters", Advanced = true)]
public string AdditionalParameters { get; set; }

// Field 6 is used by TorznabSettings MinimumSeeders
Expand Down
28 changes: 24 additions & 4 deletions src/NzbDrone.Core/Indexers/Nyaa/NyaaRequestGenerator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,19 @@ public virtual IndexerPageableRequestChain GetSearchRequests(SingleEpisodeSearch

public virtual IndexerPageableRequestChain GetSearchRequests(SeasonSearchCriteria searchCriteria)
{
return new IndexerPageableRequestChain();
var pageableRequests = new IndexerPageableRequestChain();

if (Settings.AnimeStandardFormatSearch && searchCriteria.SeasonNumber > 0)
{
foreach (var queryTitle in searchCriteria.SceneTitles)
{
var searchTitle = PrepareQuery(queryTitle);

pageableRequests.Add(GetPagedRequests(MaxPages, $"{searchTitle}+s{searchCriteria.SeasonNumber:00}"));
}
}

return pageableRequests;
}

public virtual IndexerPageableRequestChain GetSearchRequests(DailyEpisodeSearchCriteria searchCriteria)
Expand All @@ -54,11 +66,19 @@ public virtual IndexerPageableRequestChain GetSearchRequests(AnimeEpisodeSearchC
{
var searchTitle = PrepareQuery(queryTitle);

pageableRequests.Add(GetPagedRequests(MaxPages, $"{searchTitle}+{searchCriteria.AbsoluteEpisodeNumber:0}"));
if (searchCriteria.AbsoluteEpisodeNumber > 0)
{
pageableRequests.Add(GetPagedRequests(MaxPages, $"{searchTitle}+{searchCriteria.AbsoluteEpisodeNumber:0}"));

if (searchCriteria.AbsoluteEpisodeNumber < 10)
{
pageableRequests.Add(GetPagedRequests(MaxPages, $"{searchTitle}+{searchCriteria.AbsoluteEpisodeNumber:00}"));
}
}

if (searchCriteria.AbsoluteEpisodeNumber < 10)
if (Settings.AnimeStandardFormatSearch && searchCriteria.SeasonNumber > 0 && searchCriteria.EpisodeNumber > 0)
{
pageableRequests.Add(GetPagedRequests(MaxPages, $"{searchTitle}+{searchCriteria.AbsoluteEpisodeNumber:00}"));
pageableRequests.Add(GetPagedRequests(MaxPages, $"{searchTitle}+s{searchCriteria.SeasonNumber:00}e{searchCriteria.EpisodeNumber:00}"));
}
}

Expand Down
Loading

0 comments on commit cee1748

Please sign in to comment.