2010-09-28 19:32:19 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
using System.Linq;
|
2010-09-28 19:32:19 +00:00
|
|
|
|
using System.Text.RegularExpressions;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
using log4net;
|
2010-10-02 19:01:43 +00:00
|
|
|
|
using NLog;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
using NzbDrone.Core.Repository;
|
|
|
|
|
using SubSonic.Repository;
|
|
|
|
|
using TvdbLib.Data;
|
|
|
|
|
|
2010-09-28 04:25:41 +00:00
|
|
|
|
namespace NzbDrone.Core.Providers
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-09-28 04:25:41 +00:00
|
|
|
|
public class SeriesProvider : ISeriesProvider
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-09-28 19:32:19 +00:00
|
|
|
|
//TODO: Remove parsing of rest of tv show info we just need the show name
|
2010-10-02 19:01:43 +00:00
|
|
|
|
|
|
|
|
|
private static readonly Regex CleanUpRegex = new Regex(@"((\s|^)the(\s|$))|((\s|^)and(\s|$))|[^a-z]", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
2010-09-28 19:32:19 +00:00
|
|
|
|
private static readonly Regex ParseRegex = new Regex(@"(?<showName>.*)
|
|
|
|
|
(?:
|
|
|
|
|
s(?<seasonNumber>\d+)e(?<episodeNumber>\d+)-?e(?<episodeNumber2>\d+)
|
|
|
|
|
| s(?<seasonNumber>\d+)e(?<episodeNumber>\d+)
|
|
|
|
|
| (?<seasonNumber>\d+)x(?<episodeNumber>\d+)
|
|
|
|
|
| (?<airDate>\d{4}.\d{2}.\d{2})
|
|
|
|
|
)
|
|
|
|
|
(?:
|
|
|
|
|
(?<episodeName>.*?)
|
|
|
|
|
(?<release>
|
|
|
|
|
(?:hdtv|pdtv|xvid|ws|720p|x264|bdrip|dvdrip|dsr|proper)
|
|
|
|
|
.*)
|
|
|
|
|
| (?<episodeName>.*)
|
|
|
|
|
)", RegexOptions.IgnoreCase | RegexOptions.Compiled | RegexOptions.IgnorePatternWhitespace);
|
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
|
|
|
|
|
|
2010-09-28 04:25:41 +00:00
|
|
|
|
private readonly IConfigProvider _config;
|
|
|
|
|
private readonly IDiskProvider _diskProvider;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
private readonly IRepository _sonioRepo;
|
2010-09-28 04:25:41 +00:00
|
|
|
|
private readonly ITvDbProvider _tvDb;
|
2010-10-02 19:01:43 +00:00
|
|
|
|
private static readonly Logger Logger = NLog.LogManager.GetCurrentClassLogger();
|
2010-09-23 03:19:47 +00:00
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
public SeriesProvider(IDiskProvider diskProvider, IConfigProvider configProvider, IRepository dataRepository, ITvDbProvider tvDbProvider)
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-09-28 04:25:41 +00:00
|
|
|
|
_diskProvider = diskProvider;
|
|
|
|
|
_config = configProvider;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
_sonioRepo = dataRepository;
|
2010-09-28 04:25:41 +00:00
|
|
|
|
_tvDb = tvDbProvider;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-09-28 04:25:41 +00:00
|
|
|
|
#region ISeriesProvider Members
|
2010-09-23 03:19:47 +00:00
|
|
|
|
|
|
|
|
|
public IQueryable<Series> GetSeries()
|
|
|
|
|
{
|
|
|
|
|
return _sonioRepo.All<Series>();
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-28 19:32:19 +00:00
|
|
|
|
public Series GetSeries(long tvdbId)
|
2010-09-24 07:14:42 +00:00
|
|
|
|
{
|
2010-10-01 00:09:22 +00:00
|
|
|
|
return _sonioRepo.Single<Series>(s => s.TvdbId == tvdbId);
|
2010-09-24 07:14:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Determines if a series is being actively watched.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="id">The TVDB ID of the series</param>
|
|
|
|
|
/// <returns>Whether or not the show is monitored</returns>
|
|
|
|
|
public bool IsMonitored(long id)
|
2010-09-28 20:44:33 +00:00
|
|
|
|
{
|
2010-10-02 19:01:43 +00:00
|
|
|
|
return _sonioRepo.Exists<Series>(c => c.TvdbId == id && c.Monitored);
|
2010-09-28 20:44:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
/// <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>
|
|
|
|
|
public long Parse(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();
|
|
|
|
|
}
|
2010-10-01 00:09:22 +00:00
|
|
|
|
|
2010-09-23 03:19:47 +00:00
|
|
|
|
public void SyncSeriesWithDisk()
|
|
|
|
|
{
|
2010-10-02 19:01:43 +00:00
|
|
|
|
if (String.IsNullOrEmpty(_config.SeriesRoot))
|
|
|
|
|
throw new InvalidOperationException("TV Series folder is not configured yet.");
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<String> GetUnmappedFolders()
|
|
|
|
|
{
|
|
|
|
|
var results = new List<String>();
|
2010-09-28 04:25:41 +00:00
|
|
|
|
foreach (string seriesFolder in _diskProvider.GetDirectories(_config.SeriesRoot))
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-09-28 05:01:54 +00:00
|
|
|
|
var cleanPath = DiskProvider.CleanPath(new DirectoryInfo(seriesFolder).FullName);
|
2010-09-24 07:14:42 +00:00
|
|
|
|
if (!_sonioRepo.Exists<Series>(s => s.Path == cleanPath))
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-10-02 19:01:43 +00:00
|
|
|
|
results.Add(cleanPath);
|
2010-09-23 03:19:47 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
return results;
|
|
|
|
|
}
|
2010-09-23 03:19:47 +00:00
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
public bool RegisterSeries(string path)
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-10-02 19:01:43 +00:00
|
|
|
|
var seriesPath = new DirectoryInfo(path);
|
|
|
|
|
var searchResults = _tvDb.SearchSeries(seriesPath.Name);
|
|
|
|
|
Logger.Debug("Search for '{0}' returned {1} results", searchResults.Count);
|
|
|
|
|
|
|
|
|
|
if (searchResults.Count == 0)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
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;
|
2010-09-23 03:19:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
|
|
|
|
|
public void RegisterSeries(string path, TvdbSeries series)
|
2010-09-23 03:19:47 +00:00
|
|
|
|
{
|
2010-10-02 19:01:43 +00:00
|
|
|
|
Logger.Info("registering '{0}' with [{1}]-{2}", path, series.Id, series.SeriesName);
|
2010-09-28 03:04:39 +00:00
|
|
|
|
var repoSeries = new Series();
|
2010-10-01 00:09:22 +00:00
|
|
|
|
repoSeries.TvdbId = series.Id;
|
2010-10-02 19:01:43 +00:00
|
|
|
|
repoSeries.Title = series.SeriesName;
|
2010-09-28 03:04:39 +00:00
|
|
|
|
repoSeries.AirTimes = series.AirsTime;
|
|
|
|
|
repoSeries.AirsDayOfWeek = series.AirsDayOfWeek;
|
|
|
|
|
repoSeries.Overview = series.Overview;
|
|
|
|
|
repoSeries.Status = series.Status;
|
|
|
|
|
repoSeries.Language = series.Language != null ? series.Language.Abbriviation : string.Empty;
|
|
|
|
|
repoSeries.Path = path;
|
2010-10-02 19:01:43 +00:00
|
|
|
|
repoSeries.CleanTitle = CleanUpRegex.Replace(series.SeriesName, "").ToLower();
|
2010-09-28 03:04:39 +00:00
|
|
|
|
_sonioRepo.Add(repoSeries);
|
2010-09-23 03:19:47 +00:00
|
|
|
|
}
|
2010-09-28 19:32:19 +00:00
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region Static Helpers
|
|
|
|
|
|
2010-09-28 19:32:19 +00:00
|
|
|
|
/// <summary>
|
2010-10-02 19:01:43 +00:00
|
|
|
|
/// Determines whether a title in a search result is equal to the title searched for.
|
2010-09-28 19:32:19 +00:00
|
|
|
|
/// </summary>
|
2010-10-02 19:01:43 +00:00
|
|
|
|
/// <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)
|
2010-09-28 19:32:19 +00:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
var result = false;
|
2010-09-28 19:32:19 +00:00
|
|
|
|
|
2010-10-02 19:01:43 +00:00
|
|
|
|
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;
|
2010-09-28 19:32:19 +00:00
|
|
|
|
}
|
2010-10-02 19:01:43 +00:00
|
|
|
|
|
|
|
|
|
#endregion
|
2010-09-23 03:19:47 +00:00
|
|
|
|
}
|
|
|
|
|
}
|