mirror of
https://github.com/Sonarr/Sonarr.git
synced 2024-12-31 03:11:07 +02:00
Refactored SeriesController. Updated Post title parse.
This commit is contained in:
parent
0ab173dd39
commit
606140832d
@ -23,87 +23,44 @@ namespace NzbDrone.Core.Test
|
||||
public class SeriesTest
|
||||
{
|
||||
[Test]
|
||||
[Ignore("Can't get it to work")]
|
||||
[Description("This test will confirm that a folder will be skipped if it has been resolved to a series already assigned to another folder")]
|
||||
public void skip_same_series_diffrent_folder()
|
||||
public void Map_path_to_series()
|
||||
{
|
||||
var tvDbId = 1234;
|
||||
|
||||
//Arrange
|
||||
var moqData = new Mock<IRepository>();
|
||||
var moqTvdb = new Mock<ITvDbProvider>();
|
||||
|
||||
//setup db to return a fake series
|
||||
Series fakeSeries = Builder<Series>.CreateNew()
|
||||
.With(f => f.TvdbId = tvDbId)
|
||||
.Build();
|
||||
|
||||
moqData.Setup(f => f.Exists<Series>(c => c.TvdbId == tvDbId)).
|
||||
Returns(true);
|
||||
|
||||
//setup tvdb to return the same show,
|
||||
IList<TvdbSearchResult> fakeSearchResult = Builder<TvdbSearchResult>.CreateListOfSize(4).WhereTheFirst(1).Has(f => f.Id = tvDbId).Build();
|
||||
TvdbSeries fakeTvDbSeries = Builder<TvdbSeries>.CreateNew()
|
||||
.With(f => f.Id = tvDbId)
|
||||
.Build();
|
||||
|
||||
moqTvdb.Setup(f => f.GetSeries(It.IsAny<int>(), It.IsAny<TvdbLanguage>())).Returns(fakeTvDbSeries);
|
||||
moqTvdb.Setup(f => f.SearchSeries(It.IsAny<string>())).
|
||||
Returns(fakeSearchResult);
|
||||
|
||||
var kernel = new MockingKernel();
|
||||
kernel.Bind<IRepository>().ToConstant(moqData.Object);
|
||||
kernel.Bind<ITvDbProvider>().ToConstant(moqTvdb.Object);
|
||||
kernel.Bind<IConfigProvider>().ToConstant(MockLib.StandardConfig);
|
||||
kernel.Bind<IDiskProvider>().ToConstant(MockLib.StandardDisk);
|
||||
kernel.Bind<ISeriesProvider>().To<SeriesProvider>();
|
||||
|
||||
|
||||
//Act
|
||||
var seriesController = kernel.Get<ISeriesProvider>();
|
||||
seriesController.SyncSeriesWithDisk();
|
||||
|
||||
//Assert
|
||||
//Verify that the show was added to the database only once.
|
||||
moqData.Verify(c => c.Add(It.IsAny<Series>()), Times.Once());
|
||||
}
|
||||
|
||||
|
||||
[Test]
|
||||
[Row(0)]
|
||||
[Row(1)]
|
||||
[Row(2)]
|
||||
[Row(3)]
|
||||
public void register_series_with_match(int matchPosition)
|
||||
{
|
||||
TvdbSeries fakeSeries = Builder<TvdbSeries>.CreateNew().With(f => f.SeriesName = "The Simpsons").Build();
|
||||
var fakeSearch = Builder<TvdbSearchResult>.CreateListOfSize(4).Build();
|
||||
fakeSearch[matchPosition].Id = fakeSeries.Id;
|
||||
fakeSearch[matchPosition].SeriesName = fakeSeries.SeriesName;
|
||||
var fakeSearch = Builder<TvdbSearchResult>.CreateNew().Build();
|
||||
fakeSearch.Id = fakeSeries.Id;
|
||||
fakeSearch.SeriesName = fakeSeries.SeriesName;
|
||||
|
||||
|
||||
//Arrange
|
||||
var moqData = new Mock<IRepository>();
|
||||
var moqTvdb = new Mock<ITvDbProvider>();
|
||||
|
||||
moqData.Setup(f => f.Exists<Series>(c => c.TvdbId == It.IsAny<long>())).Returns(false);
|
||||
|
||||
moqTvdb.Setup(f => f.SearchSeries(It.IsAny<String>())).Returns(fakeSearch);
|
||||
moqTvdb.Setup(f => f.GetSeries(It.IsAny<String>())).Returns(fakeSearch);
|
||||
moqTvdb.Setup(f => f.GetSeries(fakeSeries.Id, It.IsAny<TvdbLanguage>())).Returns(fakeSeries);
|
||||
|
||||
var kernel = new MockingKernel();
|
||||
kernel.Bind<IRepository>().ToConstant(moqData.Object);
|
||||
kernel.Bind<ITvDbProvider>().ToConstant(moqTvdb.Object);
|
||||
kernel.Bind<ISeriesProvider>().To<SeriesProvider>();
|
||||
|
||||
|
||||
//Act
|
||||
var seriesController = kernel.Get<ISeriesProvider>();
|
||||
seriesController.RegisterSeries(@"D:\TV Shows\The Simpsons");
|
||||
var mappedSeries = seriesController.MapPathToSeries(@"D:\TV Shows\The Simpsons");
|
||||
|
||||
//Assert
|
||||
//Verify that the show was added to the database only once.
|
||||
moqData.Verify(c => c.Add(It.Is<Series>(d => d.TvdbId == fakeSeries.Id)), Times.Once());
|
||||
Assert.AreEqual(fakeSeries, mappedSeries);
|
||||
}
|
||||
|
||||
[Test]
|
||||
[Row(new object[] { "That's Life - 2x03 -The Devil and Miss DeLucca", "That's Life" })]
|
||||
[Row(new object[] { "Van.Duin.Op.Zn.Best.S02E05.DUTCH.WS.PDTV.XViD-DiFFERENT", "Van Duin Op Zn Best" })]
|
||||
[Row(new object[] { "Dollhouse.S02E06.The.Left.Hand.720p.BluRay.x264-SiNNERS", "Dollhouse" })]
|
||||
[Row(new object[] { "Heroes.S02.COMPLETE.German.PROPER.DVDRip.XviD-Prim3time", "Heroes" })]
|
||||
public void Test_Parse_Success(string postTitle, string title)
|
||||
{
|
||||
var result = SeriesProvider.ParseTitle(postTitle);
|
||||
Assert.AreEqual(title, result, postTitle);
|
||||
}
|
||||
|
||||
|
||||
@ -142,7 +99,7 @@ public void tvdbid_is_preserved([RandomNumbers(Minimum = 100, Maximum = 999, Cou
|
||||
[Row(new object[] { "Simpsons The", "Simpsons", true })]
|
||||
public void Name_match_test(string a, string b, bool match)
|
||||
{
|
||||
bool result = SeriesProvider.IsTitleMatch(a, b);
|
||||
bool result = TvDbProvider.IsTitleMatch(a, b);
|
||||
|
||||
Assert.AreEqual(match, result, "{0} , {1}", a, b);
|
||||
}
|
||||
|
@ -12,13 +12,6 @@ public interface ISeriesProvider
|
||||
Series GetSeries(long tvdbId);
|
||||
void SyncSeriesWithDisk();
|
||||
|
||||
/// <summary>
|
||||
/// Parses a post title
|
||||
/// </summary>
|
||||
/// <param name="postTitle">Title of the report</param>
|
||||
/// <returns>TVDB id of the series this report belongs to</returns>
|
||||
long Parse(string postTitle);
|
||||
|
||||
/// <summary>
|
||||
/// Determines if a series is being actively watched.
|
||||
/// </summary>
|
||||
@ -26,7 +19,7 @@ public interface ISeriesProvider
|
||||
/// <returns>Whether or not the show is monitored</returns>
|
||||
bool IsMonitored(long id);
|
||||
|
||||
bool RegisterSeries(string path);
|
||||
TvdbSeries MapPathToSeries(string path);
|
||||
void RegisterSeries(string path, TvdbSeries series);
|
||||
List<String> GetUnmappedFolders();
|
||||
}
|
||||
|
@ -7,5 +7,6 @@ public interface ITvDbProvider
|
||||
{
|
||||
IList<TvdbSearchResult> SearchSeries(string name);
|
||||
TvdbSeries GetSeries(int id, TvdbLanguage language);
|
||||
TvdbSearchResult GetSeries(string title);
|
||||
}
|
||||
}
|
@ -15,7 +15,8 @@ public class SeriesProvider : ISeriesProvider
|
||||
{
|
||||
//TODO: Remove parsing of rest of tv show info we just need the show name
|
||||
|
||||
private static readonly Regex CleanUpRegex = new Regex(@"((\s|^)the(\s|$))|((\s|^)and(\s|$))|[^a-z]", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
||||
//Trims all white spaces and separators from the end of the title.
|
||||
private static readonly Regex CleanTitleRegex = new Regex(@"[\s.][^a-z]*$", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
||||
private static readonly Regex ParseRegex = new Regex(@"(?<showName>.*)
|
||||
(?:
|
||||
s(?<seasonNumber>\d+)e(?<episodeNumber>\d+)-?e(?<episodeNumber2>\d+)
|
||||
@ -38,6 +39,7 @@ public class SeriesProvider : ISeriesProvider
|
||||
private readonly IRepository _sonioRepo;
|
||||
private readonly ITvDbProvider _tvDb;
|
||||
private static readonly Logger Logger = NLog.LogManager.GetCurrentClassLogger();
|
||||
private static readonly Regex CleanUpRegex = new Regex(@"((\s|^)the(\s|$))|((\s|^)and(\s|$))|[^a-z]", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
||||
|
||||
public SeriesProvider(IDiskProvider diskProvider, IConfigProvider configProvider, IRepository dataRepository, ITvDbProvider tvDbProvider)
|
||||
{
|
||||
@ -70,20 +72,18 @@ public bool IsMonitored(long id)
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Parses a post title
|
||||
/// Parses series name out of a post title
|
||||
/// </summary>
|
||||
/// <param name="postTitle">Title of the report</param>
|
||||
/// <returns>TVDB id of the series this report belongs to</returns>
|
||||
public long Parse(string postTitle)
|
||||
/// <returns>Name series this report belongs to</returns>
|
||||
public static string ParseTitle(string postTitle)
|
||||
{
|
||||
var match = ParseRegex.Match(postTitle);
|
||||
|
||||
if (!match.Success)
|
||||
throw new ArgumentException(String.Format("Title doesn't match any know patterns. [{0}]", postTitle));
|
||||
|
||||
//TODO: title should be mapped to a proper Series object. with tvdbId and everything even if it is not in the db or being tracked.
|
||||
|
||||
throw new NotImplementedException();
|
||||
return CleanTitleRegex.Replace(match.Groups["showName"].Value, String.Empty).Replace(".", " ");
|
||||
}
|
||||
|
||||
public void SyncSeriesWithDisk()
|
||||
@ -94,7 +94,23 @@ public void SyncSeriesWithDisk()
|
||||
foreach (string seriesFolder in GetUnmappedFolders())
|
||||
{
|
||||
Logger.Info("Folder '{0}' isn't mapped to a series in the database. Trying to map it.'", seriesFolder);
|
||||
RegisterSeries(seriesFolder);
|
||||
var mappedSeries = MapPathToSeries(seriesFolder);
|
||||
|
||||
if (mappedSeries == null)
|
||||
{
|
||||
Logger.Warn("Unable to find a matching series for '{0}'", seriesFolder);
|
||||
break;
|
||||
}
|
||||
|
||||
if (!_sonioRepo.Exists<Series>(s => s.TvdbId == mappedSeries.Id))
|
||||
{
|
||||
RegisterSeries(seriesFolder, mappedSeries);
|
||||
}
|
||||
else
|
||||
{
|
||||
Logger.Warn("Folder '{0}' mapped to '{1}' which is already another folder assigned to it.'", seriesFolder, mappedSeries.SeriesName);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
@ -113,27 +129,15 @@ public List<String> GetUnmappedFolders()
|
||||
return results;
|
||||
}
|
||||
|
||||
public bool RegisterSeries(string path)
|
||||
public TvdbSeries MapPathToSeries(string path)
|
||||
{
|
||||
var seriesPath = new DirectoryInfo(path);
|
||||
var searchResults = _tvDb.SearchSeries(seriesPath.Name);
|
||||
Logger.Debug("Search for '{0}' returned {1} results", searchResults.Count);
|
||||
var searchResults = _tvDb.GetSeries(seriesPath.Name);
|
||||
|
||||
if (searchResults.Count == 0)
|
||||
return false;
|
||||
if (searchResults == null)
|
||||
return null;
|
||||
|
||||
foreach (var tvdbSearchResult in searchResults)
|
||||
{
|
||||
TvdbSearchResult result = tvdbSearchResult;
|
||||
if (IsTitleMatch(seriesPath.Name, result.SeriesName) && !_sonioRepo.Exists<Series>(c => c.TvdbId == result.Id))
|
||||
{
|
||||
RegisterSeries(path, _tvDb.GetSeries(result.Id, result.Language));
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
Logger.Info("Unable to fine a match for {0}", seriesPath.Name);
|
||||
return false;
|
||||
return _tvDb.GetSeries(searchResults.Id, searchResults.Language);
|
||||
}
|
||||
|
||||
|
||||
@ -157,37 +161,8 @@ public void RegisterSeries(string path, TvdbSeries series)
|
||||
|
||||
#region Static Helpers
|
||||
|
||||
/// <summary>
|
||||
/// Determines whether a title in a search result is equal to the title searched for.
|
||||
/// </summary>
|
||||
/// <param name="directoryName">Name of the directory.</param>
|
||||
/// <param name="tvdbTitle">The TVDB title.</param>
|
||||
/// <returns>
|
||||
/// <c>true</c> if the titles are found to be same; otherwise, <c>false</c>.
|
||||
/// </returns>
|
||||
public static bool IsTitleMatch(string directoryName, string tvdbTitle)
|
||||
{
|
||||
|
||||
|
||||
|
||||
var result = false;
|
||||
|
||||
if (String.IsNullOrEmpty(directoryName))
|
||||
throw new ArgumentException("directoryName");
|
||||
if (String.IsNullOrEmpty(tvdbTitle))
|
||||
throw new ArgumentException("tvdbTitle");
|
||||
|
||||
if (String.Equals(directoryName, tvdbTitle, StringComparison.CurrentCultureIgnoreCase))
|
||||
{
|
||||
result = true;
|
||||
}
|
||||
else if (String.Equals(CleanUpRegex.Replace(directoryName, ""), CleanUpRegex.Replace(tvdbTitle, ""), StringComparison.InvariantCultureIgnoreCase))
|
||||
result = true;
|
||||
|
||||
Logger.Debug("Match between '{0}' and '{1}' was {2}", tvdbTitle, directoryName, result);
|
||||
return result;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
@ -1,5 +1,8 @@
|
||||
using System.Collections.Generic;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Text.RegularExpressions;
|
||||
using NLog;
|
||||
using TvdbLib;
|
||||
using TvdbLib.Cache;
|
||||
using TvdbLib.Data;
|
||||
@ -8,6 +11,9 @@ namespace NzbDrone.Core.Providers
|
||||
{
|
||||
public class TvDbProvider : ITvDbProvider
|
||||
{
|
||||
private static readonly Logger Logger = NLog.LogManager.GetCurrentClassLogger();
|
||||
private static readonly Regex CleanUpRegex = new Regex(@"((\s|^)the(\s|$))|((\s|^)and(\s|$))|[^a-z]", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
||||
|
||||
private const string TVDB_APIKEY = "5D2D188E86E07F4F";
|
||||
private readonly TvdbHandler _handler;
|
||||
|
||||
@ -18,16 +24,66 @@ public TvDbProvider()
|
||||
|
||||
#region ITvDbProvider Members
|
||||
|
||||
public IList<TvdbSearchResult> SearchSeries(string name)
|
||||
public IList<TvdbSearchResult> SearchSeries(string title)
|
||||
{
|
||||
return _handler.SearchSeries(name);
|
||||
Logger.Debug("Searching TVDB for '{0}'", title);
|
||||
var result = new List<TvdbSearchResult>();
|
||||
|
||||
foreach (var tvdbSearchResult in _handler.SearchSeries(title))
|
||||
{
|
||||
if (IsTitleMatch(tvdbSearchResult.SeriesName, title))
|
||||
{
|
||||
result.Add(tvdbSearchResult);
|
||||
}
|
||||
}
|
||||
|
||||
Logger.Debug("Search for '{0}' returned {1} results", title);
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
public TvdbSearchResult GetSeries(string title)
|
||||
{
|
||||
return SearchSeries(title)[0];
|
||||
}
|
||||
|
||||
public TvdbSeries GetSeries(int id, TvdbLanguage language)
|
||||
{
|
||||
Logger.Debug("Fetching seriesId'{0}' - '{1}' from tvdb", id, language);
|
||||
return _handler.GetSeries(id, language, true, false, false);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Determines whether a title in a search result is equal to the title searched for.
|
||||
/// </summary>
|
||||
/// <param name="directoryName">Name of the directory.</param>
|
||||
/// <param name="tvdbTitle">The TVDB title.</param>
|
||||
/// <returns>
|
||||
/// <c>true</c> if the titles are found to be same; otherwise, <c>false</c>.
|
||||
/// </returns>
|
||||
public static bool IsTitleMatch(string directoryName, string tvdbTitle)
|
||||
{
|
||||
Logger.Debug("Trying to match '{0}' and '{1}'", tvdbTitle, directoryName);
|
||||
|
||||
var result = false;
|
||||
|
||||
if (String.IsNullOrEmpty(directoryName))
|
||||
throw new ArgumentException("directoryName");
|
||||
if (String.IsNullOrEmpty(tvdbTitle))
|
||||
throw new ArgumentException("tvdbTitle");
|
||||
|
||||
if (String.Equals(directoryName, tvdbTitle, StringComparison.CurrentCultureIgnoreCase))
|
||||
{
|
||||
result = true;
|
||||
}
|
||||
else if (String.Equals(CleanUpRegex.Replace(directoryName, ""), CleanUpRegex.Replace(tvdbTitle, ""), StringComparison.InvariantCultureIgnoreCase))
|
||||
result = true;
|
||||
|
||||
Logger.Debug("Match between '{0}' and '{1}' was {2}", tvdbTitle, directoryName, result);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
@ -12,6 +12,7 @@ body
|
||||
font-family: Segoe UI, Tahoma, Geneva, sans-serif;
|
||||
font-size: 11px;
|
||||
color: #3C3C3C;
|
||||
background-attachment: fixed;
|
||||
}
|
||||
|
||||
h1, h2, h3
|
||||
|
@ -32,7 +32,6 @@ public ActionResult Index(SettingsModel model)
|
||||
_configProvider.SeriesRoot = model.TvFolder;
|
||||
//return RedirectToAction("index");
|
||||
}
|
||||
|
||||
return View(model);
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user