2010-09-28 22:32:19 +03:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.IO;
|
2010-09-23 06:19:47 +03:00
|
|
|
using System.Linq;
|
2010-09-28 22:32:19 +03:00
|
|
|
using System.Text.RegularExpressions;
|
2010-10-02 22:01:43 +03:00
|
|
|
using NLog;
|
2010-10-08 06:35:04 +03:00
|
|
|
using NzbDrone.Core.Entities;
|
|
|
|
using NzbDrone.Core.Entities.Notification;
|
2010-09-23 06:19:47 +03:00
|
|
|
using SubSonic.Repository;
|
|
|
|
using TvdbLib.Data;
|
|
|
|
|
2010-09-28 07:25:41 +03:00
|
|
|
namespace NzbDrone.Core.Providers
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-09-28 07:25:41 +03:00
|
|
|
public class SeriesProvider : ISeriesProvider
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-09-28 22:32:19 +03:00
|
|
|
//TODO: Remove parsing of rest of tv show info we just need the show name
|
2010-10-02 22:01:43 +03:00
|
|
|
|
2010-10-04 04:00:50 +03:00
|
|
|
//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);
|
2010-09-28 22:32:19 +03: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 22:01:43 +03:00
|
|
|
|
2010-10-08 06:35:04 +03:00
|
|
|
private readonly INotificationProvider _notificationProvider;
|
2010-09-28 07:25:41 +03:00
|
|
|
private readonly IConfigProvider _config;
|
|
|
|
private readonly IDiskProvider _diskProvider;
|
2010-09-23 06:19:47 +03:00
|
|
|
private readonly IRepository _sonioRepo;
|
2010-09-28 07:25:41 +03:00
|
|
|
private readonly ITvDbProvider _tvDb;
|
2010-10-05 09:21:18 +03:00
|
|
|
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
|
2010-10-04 04:00:50 +03:00
|
|
|
private static readonly Regex CleanUpRegex = new Regex(@"((\s|^)the(\s|$))|((\s|^)and(\s|$))|[^a-z]", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
2010-09-23 06:19:47 +03:00
|
|
|
|
2010-10-08 06:35:04 +03:00
|
|
|
private ProgressNotification _progress;
|
|
|
|
|
|
|
|
public SeriesProvider(INotificationProvider notificationProvider, IDiskProvider diskProvider, IConfigProvider configProvider, IRepository dataRepository, ITvDbProvider tvDbProvider)
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-10-08 06:35:04 +03:00
|
|
|
_notificationProvider = notificationProvider;
|
2010-09-28 07:25:41 +03:00
|
|
|
_diskProvider = diskProvider;
|
|
|
|
_config = configProvider;
|
2010-09-23 06:19:47 +03:00
|
|
|
_sonioRepo = dataRepository;
|
2010-09-28 07:25:41 +03:00
|
|
|
_tvDb = tvDbProvider;
|
2010-09-23 06:19:47 +03:00
|
|
|
}
|
|
|
|
|
2010-09-28 07:25:41 +03:00
|
|
|
#region ISeriesProvider Members
|
2010-09-23 06:19:47 +03:00
|
|
|
|
|
|
|
public IQueryable<Series> GetSeries()
|
|
|
|
{
|
|
|
|
return _sonioRepo.All<Series>();
|
|
|
|
}
|
|
|
|
|
2010-10-05 09:21:18 +03:00
|
|
|
public Series GetSeries(int seriesId)
|
2010-09-24 10:14:42 +03:00
|
|
|
{
|
2010-10-05 09:21:18 +03:00
|
|
|
return _sonioRepo.Single<Series>(s => s.SeriesId == seriesId);
|
2010-09-24 10:14:42 +03:00
|
|
|
}
|
|
|
|
|
2010-10-02 22:01:43 +03: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 23:44:33 +03:00
|
|
|
{
|
2010-10-05 09:21:18 +03:00
|
|
|
return _sonioRepo.Exists<Series>(c => c.SeriesId == id && c.Monitored);
|
2010-09-28 23:44:33 +03:00
|
|
|
}
|
|
|
|
|
2010-10-02 22:01:43 +03:00
|
|
|
/// <summary>
|
2010-10-04 04:00:50 +03:00
|
|
|
/// Parses series name out of a post title
|
2010-10-02 22:01:43 +03:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="postTitle">Title of the report</param>
|
2010-10-04 04:00:50 +03:00
|
|
|
/// <returns>Name series this report belongs to</returns>
|
|
|
|
public static string ParseTitle(string postTitle)
|
2010-10-02 22:01:43 +03:00
|
|
|
{
|
|
|
|
var match = ParseRegex.Match(postTitle);
|
|
|
|
|
|
|
|
if (!match.Success)
|
|
|
|
throw new ArgumentException(String.Format("Title doesn't match any know patterns. [{0}]", postTitle));
|
|
|
|
|
2010-10-04 04:00:50 +03:00
|
|
|
return CleanTitleRegex.Replace(match.Groups["showName"].Value, String.Empty).Replace(".", " ");
|
2010-10-02 22:01:43 +03:00
|
|
|
}
|
2010-10-01 03:09:22 +03:00
|
|
|
|
2010-09-23 06:19:47 +03:00
|
|
|
public void SyncSeriesWithDisk()
|
|
|
|
{
|
2010-10-08 06:35:04 +03:00
|
|
|
if (_progress != null && _progress.Status == NotificationStatus.InProgress)
|
|
|
|
throw new InvalidOperationException("Another Task is already in progress. " + _progress.Title);
|
|
|
|
|
2010-10-02 22:01:43 +03:00
|
|
|
if (String.IsNullOrEmpty(_config.SeriesRoot))
|
|
|
|
throw new InvalidOperationException("TV Series folder is not configured yet.");
|
|
|
|
|
2010-10-08 06:35:04 +03:00
|
|
|
using (_progress = new ProgressNotification("Updating Series From Disk"))
|
2010-10-02 22:01:43 +03:00
|
|
|
{
|
2010-10-08 06:35:04 +03:00
|
|
|
_notificationProvider.Register(_progress);
|
2010-10-04 04:00:50 +03:00
|
|
|
|
2010-10-08 06:35:04 +03:00
|
|
|
var unmappedFolders = GetUnmappedFolders();
|
|
|
|
_progress.ProgressMax = unmappedFolders.Count;
|
|
|
|
|
|
|
|
foreach (string seriesFolder in unmappedFolders)
|
2010-10-04 04:00:50 +03:00
|
|
|
{
|
2010-10-08 06:35:04 +03:00
|
|
|
_progress.CurrentStatus = String.Format("Mapping folder {0}", seriesFolder);
|
|
|
|
|
|
|
|
Logger.Info("Folder '{0}' isn't mapped to a series in the database. Trying to map it.'", seriesFolder);
|
|
|
|
var mappedSeries = MapPathToSeries(seriesFolder);
|
2010-10-05 09:21:18 +03:00
|
|
|
|
2010-10-08 06:35:04 +03:00
|
|
|
if (mappedSeries == null)
|
2010-10-05 09:21:18 +03:00
|
|
|
{
|
2010-10-08 06:35:04 +03:00
|
|
|
Logger.Warn("Unable to find a matching series for '{0}'", seriesFolder);
|
2010-10-05 09:21:18 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-10-08 06:35:04 +03:00
|
|
|
if (!_sonioRepo.Exists<Series>(s => s.SeriesId == mappedSeries.Id))
|
|
|
|
{
|
|
|
|
RegisterSeries(seriesFolder, mappedSeries);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Logger.Warn("Folder '{0}' mapped to '{1}' which is already another folder assigned to it.'", seriesFolder, mappedSeries.SeriesName);
|
|
|
|
}
|
2010-10-05 09:21:18 +03:00
|
|
|
}
|
2010-10-08 06:35:04 +03:00
|
|
|
_progress.ProgressValue++;
|
2010-10-04 04:00:50 +03:00
|
|
|
}
|
|
|
|
|
2010-10-08 06:35:04 +03:00
|
|
|
_progress.Status = NotificationStatus.Completed;
|
2010-10-02 22:01:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public List<String> GetUnmappedFolders()
|
|
|
|
{
|
|
|
|
var results = new List<String>();
|
2010-09-28 07:25:41 +03:00
|
|
|
foreach (string seriesFolder in _diskProvider.GetDirectories(_config.SeriesRoot))
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-09-28 08:01:54 +03:00
|
|
|
var cleanPath = DiskProvider.CleanPath(new DirectoryInfo(seriesFolder).FullName);
|
2010-09-24 10:14:42 +03:00
|
|
|
if (!_sonioRepo.Exists<Series>(s => s.Path == cleanPath))
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-10-02 22:01:43 +03:00
|
|
|
results.Add(cleanPath);
|
2010-09-23 06:19:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-02 22:01:43 +03:00
|
|
|
return results;
|
|
|
|
}
|
2010-09-23 06:19:47 +03:00
|
|
|
|
2010-10-04 04:00:50 +03:00
|
|
|
public TvdbSeries MapPathToSeries(string path)
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-10-02 22:01:43 +03:00
|
|
|
var seriesPath = new DirectoryInfo(path);
|
2010-10-04 04:00:50 +03:00
|
|
|
var searchResults = _tvDb.GetSeries(seriesPath.Name);
|
2010-10-02 22:01:43 +03:00
|
|
|
|
2010-10-04 04:00:50 +03:00
|
|
|
if (searchResults == null)
|
|
|
|
return null;
|
2010-10-02 22:01:43 +03:00
|
|
|
|
2010-10-05 09:21:18 +03:00
|
|
|
return _tvDb.GetSeries(searchResults.Id, false);
|
2010-09-23 06:19:47 +03:00
|
|
|
}
|
|
|
|
|
2010-10-02 22:01:43 +03:00
|
|
|
|
|
|
|
public void RegisterSeries(string path, TvdbSeries series)
|
2010-09-23 06:19:47 +03:00
|
|
|
{
|
2010-10-02 22:01:43 +03:00
|
|
|
Logger.Info("registering '{0}' with [{1}]-{2}", path, series.Id, series.SeriesName);
|
2010-09-28 06:04:39 +03:00
|
|
|
var repoSeries = new Series();
|
2010-10-05 09:21:18 +03:00
|
|
|
repoSeries.SeriesId = series.Id;
|
2010-10-02 22:01:43 +03:00
|
|
|
repoSeries.Title = series.SeriesName;
|
2010-09-28 06:04:39 +03: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 22:01:43 +03:00
|
|
|
repoSeries.CleanTitle = CleanUpRegex.Replace(series.SeriesName, "").ToLower();
|
2010-09-28 06:04:39 +03:00
|
|
|
_sonioRepo.Add(repoSeries);
|
2010-09-23 06:19:47 +03:00
|
|
|
}
|
2010-09-28 22:32:19 +03:00
|
|
|
|
2010-10-02 22:01:43 +03:00
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region Static Helpers
|
|
|
|
|
2010-09-28 22:32:19 +03:00
|
|
|
|
|
|
|
|
2010-10-02 22:01:43 +03:00
|
|
|
#endregion
|
2010-09-23 06:19:47 +03:00
|
|
|
}
|
|
|
|
}
|