2010-09-28 12:32:19 -07:00
|
|
|
|
using System;
|
2011-05-26 23:03:57 -07:00
|
|
|
|
using System.Collections.Generic;
|
2010-09-22 20:19:47 -07:00
|
|
|
|
using System.Linq;
|
2011-04-26 19:47:29 -07:00
|
|
|
|
using System.Text.RegularExpressions;
|
2010-10-02 12:01:43 -07:00
|
|
|
|
using NLog;
|
2011-12-10 10:54:16 -08:00
|
|
|
|
using NzbDrone.Common;
|
2011-04-03 20:50:12 -07:00
|
|
|
|
using NzbDrone.Core.Providers.Core;
|
2010-10-20 18:49:23 -07:00
|
|
|
|
using NzbDrone.Core.Repository;
|
2011-06-20 00:13:17 -07:00
|
|
|
|
using NzbDrone.Core.Repository.Quality;
|
2011-06-14 19:31:41 -07:00
|
|
|
|
using PetaPoco;
|
2010-09-22 20:19:47 -07:00
|
|
|
|
|
2010-09-27 21:25:41 -07:00
|
|
|
|
namespace NzbDrone.Core.Providers
|
2010-09-22 20:19:47 -07:00
|
|
|
|
{
|
2011-04-08 16:55:23 -07:00
|
|
|
|
public class SeriesProvider
|
2010-09-22 20:19:47 -07:00
|
|
|
|
{
|
2011-04-09 19:44:01 -07:00
|
|
|
|
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
|
2011-04-25 17:28:33 -07:00
|
|
|
|
private readonly ConfigProvider _configProvider;
|
|
|
|
|
private readonly TvDbProvider _tvDbProvider;
|
2011-06-14 19:31:41 -07:00
|
|
|
|
private readonly IDatabase _database;
|
2011-06-16 23:04:01 -07:00
|
|
|
|
private readonly SceneMappingProvider _sceneNameMappingProvider;
|
2011-06-16 19:48:24 -07:00
|
|
|
|
private static readonly Regex TimeRegex = new Regex(@"^(?<time>\d+:?\d*)\W*(?<meridiem>am|pm)?", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
2010-09-22 20:19:47 -07:00
|
|
|
|
|
2011-06-20 00:13:17 -07:00
|
|
|
|
public SeriesProvider(IDatabase database, ConfigProvider configProviderProvider,
|
2011-06-16 23:04:01 -07:00
|
|
|
|
TvDbProvider tvDbProviderProvider, SceneMappingProvider sceneNameMappingProvider)
|
2010-09-22 20:19:47 -07:00
|
|
|
|
{
|
2011-06-16 19:48:24 -07:00
|
|
|
|
_database = database;
|
2011-04-25 17:28:33 -07:00
|
|
|
|
_configProvider = configProviderProvider;
|
|
|
|
|
_tvDbProvider = tvDbProviderProvider;
|
2011-06-15 23:53:23 -07:00
|
|
|
|
_sceneNameMappingProvider = sceneNameMappingProvider;
|
2010-09-22 20:19:47 -07:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 17:14:51 -07:00
|
|
|
|
public SeriesProvider()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-26 23:03:57 -07:00
|
|
|
|
public virtual IList<Series> GetAllSeries()
|
2010-09-22 20:19:47 -07:00
|
|
|
|
{
|
2011-06-20 00:13:17 -07:00
|
|
|
|
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
|
|
|
|
|
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId");
|
|
|
|
|
|
|
|
|
|
return series;
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-23 11:37:17 -07:00
|
|
|
|
public virtual IList<Series> GetAllSeriesWithEpisodeCount()
|
2011-06-20 00:13:17 -07:00
|
|
|
|
{
|
2011-06-22 23:56:17 -07:00
|
|
|
|
var series = _database
|
2011-06-23 11:37:17 -07:00
|
|
|
|
.Fetch<Series, QualityProfile>(@"SELECT Series.SeriesId, Series.Title, Series.CleanTitle, Series.Status, Series.Overview, Series.AirsDayOfWeek,Series.AirTimes,
|
|
|
|
|
Series.Language, Series.Path, Series.Monitored, Series.QualityProfileId, Series.SeasonFolder,
|
2011-07-09 19:45:31 -07:00
|
|
|
|
SUM(CASE WHEN Ignored = 0 AND Airdate <= @0 THEN 1 ELSE 0 END) AS EpisodeCount,
|
|
|
|
|
SUM(CASE WHEN Episodes.Ignored = 0 AND Episodes.EpisodeFileId > 0 AND Episodes.AirDate <= @0 THEN 1 ELSE 0 END) as EpisodeFileCount,
|
2011-10-03 16:38:22 -07:00
|
|
|
|
MAX(Episodes.SeasonNumber) as SeasonCount, MIN(CASE WHEN AirDate < @0 OR Ignored = 1 THEN NULL ELSE AirDate END) as NextAiring,
|
2011-06-23 11:37:17 -07:00
|
|
|
|
QualityProfiles.QualityProfileId, QualityProfiles.Name, QualityProfiles.Cutoff, QualityProfiles.SonicAllowed
|
|
|
|
|
FROM Series
|
|
|
|
|
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
|
|
|
|
|
LEFT JOIN Episodes ON Series.SeriesId = Episodes.SeriesId
|
|
|
|
|
GROUP BY Series.SeriesId, Series.Title, Series.CleanTitle, Series.Status, Series.Overview, Series.AirsDayOfWeek,Series.AirTimes,
|
|
|
|
|
Series.Language, Series.Path, Series.Monitored, Series.QualityProfileId, Series.SeasonFolder,
|
2011-07-09 19:45:31 -07:00
|
|
|
|
QualityProfiles.QualityProfileId, QualityProfiles.Name, QualityProfiles.Cutoff, QualityProfiles.SonicAllowed", DateTime.Today);
|
2011-06-20 00:13:17 -07:00
|
|
|
|
|
2011-06-14 19:31:41 -07:00
|
|
|
|
return series;
|
2010-09-22 20:19:47 -07:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual Series GetSeries(int seriesId)
|
2010-09-24 00:14:42 -07:00
|
|
|
|
{
|
2011-06-20 00:13:17 -07:00
|
|
|
|
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
|
|
|
|
|
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
|
|
|
|
|
WHERE seriesId= @0", seriesId).Single();
|
2011-06-18 22:56:52 -07:00
|
|
|
|
|
2011-06-17 21:39:02 -07:00
|
|
|
|
return series;
|
2010-09-24 00:14:42 -07:00
|
|
|
|
}
|
|
|
|
|
|
2010-10-02 12:01:43 -07:00
|
|
|
|
/// <summary>
|
2011-04-09 19:44:01 -07:00
|
|
|
|
/// Determines if a series is being actively watched.
|
2010-10-02 12:01:43 -07:00
|
|
|
|
/// </summary>
|
2011-04-09 19:44:01 -07:00
|
|
|
|
/// <param name = "id">The TVDB ID of the series</param>
|
2010-10-02 12:01:43 -07:00
|
|
|
|
/// <returns>Whether or not the show is monitored</returns>
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual bool IsMonitored(long id)
|
2010-09-28 13:44:33 -07:00
|
|
|
|
{
|
2011-06-14 19:31:41 -07:00
|
|
|
|
return GetAllSeries().Any(c => c.SeriesId == id && c.Monitored);
|
2010-09-28 13:44:33 -07:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual Series UpdateSeriesInfo(int seriesId)
|
2011-03-09 23:49:59 -08:00
|
|
|
|
{
|
2011-09-29 21:40:00 -07:00
|
|
|
|
var tvDbSeries = _tvDbProvider.GetSeries(seriesId, false);
|
2011-03-31 23:36:34 -07:00
|
|
|
|
var series = GetSeries(seriesId);
|
|
|
|
|
|
|
|
|
|
series.SeriesId = tvDbSeries.Id;
|
|
|
|
|
series.Title = tvDbSeries.SeriesName;
|
2011-04-26 19:47:29 -07:00
|
|
|
|
series.AirTimes = CleanAirsTime(tvDbSeries.AirsTime);
|
2011-03-31 23:36:34 -07:00
|
|
|
|
series.AirsDayOfWeek = tvDbSeries.AirsDayOfWeek;
|
|
|
|
|
series.Overview = tvDbSeries.Overview;
|
|
|
|
|
series.Status = tvDbSeries.Status;
|
|
|
|
|
series.Language = tvDbSeries.Language != null ? tvDbSeries.Language.Abbriviation : string.Empty;
|
|
|
|
|
series.CleanTitle = Parser.NormalizeTitle(tvDbSeries.SeriesName);
|
|
|
|
|
series.LastInfoSync = DateTime.Now;
|
2011-09-10 01:42:05 -07:00
|
|
|
|
series.Runtime = (int)tvDbSeries.Runtime;
|
|
|
|
|
series.BannerUrl = tvDbSeries.BannerPath;
|
2011-03-31 23:36:34 -07:00
|
|
|
|
|
|
|
|
|
UpdateSeries(series);
|
|
|
|
|
return series;
|
2011-03-09 23:49:59 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual void AddSeries(string path, int tvDbSeriesId, int qualityProfileId)
|
2010-09-22 20:19:47 -07:00
|
|
|
|
{
|
2011-03-31 23:36:34 -07:00
|
|
|
|
Logger.Info("Adding Series [{0}] Path: [{1}]", tvDbSeriesId, path);
|
|
|
|
|
|
2010-09-27 20:04:39 -07:00
|
|
|
|
var repoSeries = new Series();
|
2011-03-31 23:36:34 -07:00
|
|
|
|
repoSeries.SeriesId = tvDbSeriesId;
|
2010-09-27 20:04:39 -07:00
|
|
|
|
repoSeries.Path = path;
|
2011-02-14 19:48:39 -08:00
|
|
|
|
repoSeries.Monitored = true; //New shows should be monitored
|
2011-03-28 13:22:12 -07:00
|
|
|
|
repoSeries.QualityProfileId = qualityProfileId;
|
|
|
|
|
if (qualityProfileId == 0)
|
2011-07-07 20:36:02 -07:00
|
|
|
|
repoSeries.QualityProfileId = _configProvider.DefaultQualityProfile;
|
2011-02-24 23:20:24 -08:00
|
|
|
|
|
2011-04-25 17:28:33 -07:00
|
|
|
|
repoSeries.SeasonFolder = _configProvider.UseSeasonFolder;
|
2011-02-24 23:20:24 -08:00
|
|
|
|
|
2011-06-14 19:31:41 -07:00
|
|
|
|
_database.Insert(repoSeries);
|
2010-09-22 20:19:47 -07:00
|
|
|
|
}
|
2010-09-28 12:32:19 -07:00
|
|
|
|
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual Series FindSeries(string title)
|
2011-01-28 22:10:22 -08:00
|
|
|
|
{
|
2011-06-21 18:10:31 -07:00
|
|
|
|
try
|
2011-05-19 20:47:07 -07:00
|
|
|
|
{
|
2011-06-21 18:10:31 -07:00
|
|
|
|
var normalizeTitle = Parser.NormalizeTitle(title);
|
2011-05-19 20:47:07 -07:00
|
|
|
|
|
2011-06-21 18:10:31 -07:00
|
|
|
|
var seriesId = _sceneNameMappingProvider.GetSeriesId(normalizeTitle);
|
|
|
|
|
if (seriesId != null)
|
|
|
|
|
{
|
|
|
|
|
return GetSeries(seriesId.Value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
|
2011-06-20 00:13:17 -07:00
|
|
|
|
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
|
2011-07-27 15:59:48 -07:00
|
|
|
|
WHERE CleanTitle = @0", normalizeTitle).SingleOrDefault();
|
2011-06-18 00:17:47 -07:00
|
|
|
|
|
2011-06-21 18:10:31 -07:00
|
|
|
|
return series;
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-21 18:22:52 -07:00
|
|
|
|
|
|
|
|
|
catch (InvalidOperationException)
|
2011-06-21 18:10:31 -07:00
|
|
|
|
{
|
2011-06-21 18:22:52 -07:00
|
|
|
|
//This will catch InvalidOperationExceptions(Sequence contains no element)
|
|
|
|
|
//that may be thrown for GetSeries due to the series being in SceneMapping, but not in the users Database
|
2011-06-21 18:10:31 -07:00
|
|
|
|
return null;
|
|
|
|
|
}
|
2011-01-28 22:10:22 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual void UpdateSeries(Series series)
|
2011-02-17 18:50:40 -08:00
|
|
|
|
{
|
2011-06-14 19:31:41 -07:00
|
|
|
|
_database.Update(series);
|
2011-02-17 18:50:40 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 19:25:52 -07:00
|
|
|
|
public virtual void DeleteSeries(int seriesId)
|
2011-02-17 22:49:23 -08:00
|
|
|
|
{
|
2011-06-16 19:27:10 -07:00
|
|
|
|
var series = GetSeries(seriesId);
|
|
|
|
|
Logger.Warn("Deleting Series [{0}]", series.Title);
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
using (var tran = _database.GetTransaction())
|
|
|
|
|
{
|
|
|
|
|
//Delete History, Files, Episodes, Seasons then the Series
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
Logger.Debug("Deleting History Items from DB for Series: {0}", series.Title);
|
|
|
|
|
_database.Delete<History>("WHERE SeriesId=@0", seriesId);
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
Logger.Debug("Deleting EpisodeFiles from DB for Series: {0}", series.Title);
|
|
|
|
|
_database.Delete<EpisodeFile>("WHERE SeriesId=@0", seriesId);
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
Logger.Debug("Deleting Episodes from DB for Series: {0}", series.Title);
|
|
|
|
|
_database.Delete<Episode>("WHERE SeriesId=@0", seriesId);
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
Logger.Debug("Deleting Series from DB {0}", series.Title);
|
|
|
|
|
_database.Delete<Series>("WHERE SeriesId=@0", seriesId);
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
Logger.Info("Successfully deleted Series [{0}]", series.Title);
|
2011-05-15 16:35:45 -07:00
|
|
|
|
|
2011-06-16 19:27:10 -07:00
|
|
|
|
tran.Complete();
|
|
|
|
|
}
|
2011-02-17 22:49:23 -08:00
|
|
|
|
}
|
|
|
|
|
|
2011-12-10 10:54:16 -08:00
|
|
|
|
public virtual bool SeriesPathExists(string path)
|
2011-03-08 23:40:48 -08:00
|
|
|
|
{
|
2011-12-10 11:22:47 -08:00
|
|
|
|
return GetAllSeries().Any(s => DiskProvider.PathEquals(s.Path, path));
|
2011-03-08 23:40:48 -08:00
|
|
|
|
}
|
2011-04-26 19:47:29 -07:00
|
|
|
|
|
2011-10-20 16:36:47 -07:00
|
|
|
|
public virtual List<Series> SearchForSeries(string title)
|
|
|
|
|
{
|
2011-12-08 00:44:22 -08:00
|
|
|
|
var query = String.Format("%{0}%", title);
|
2011-10-20 16:36:47 -07:00
|
|
|
|
|
2011-12-08 00:44:22 -08:00
|
|
|
|
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
|
|
|
|
|
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
|
|
|
|
|
WHERE Title LIKE @0", query);
|
2011-10-20 16:36:47 -07:00
|
|
|
|
|
|
|
|
|
return series;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-26 19:47:29 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Cleans up the AirsTime Component from TheTVDB since it can be garbage that comes in.
|
|
|
|
|
/// </summary>
|
2011-06-16 19:48:24 -07:00
|
|
|
|
/// <param name = "rawTime">The TVDB AirsTime</param>
|
2011-04-26 19:47:29 -07:00
|
|
|
|
/// <returns>String that contains the AirTimes</returns>
|
2011-06-16 19:48:24 -07:00
|
|
|
|
private static string CleanAirsTime(string rawTime)
|
2011-04-26 19:47:29 -07:00
|
|
|
|
{
|
2011-06-16 19:48:24 -07:00
|
|
|
|
var match = TimeRegex.Match(rawTime);
|
2011-04-26 19:47:29 -07:00
|
|
|
|
var time = match.Groups["time"].Value;
|
|
|
|
|
var meridiem = match.Groups["meridiem"].Value;
|
|
|
|
|
|
|
|
|
|
//Lets assume that a string that doesn't contain a Merideim is aired at night... So we'll add it
|
|
|
|
|
if (String.IsNullOrEmpty(meridiem))
|
|
|
|
|
meridiem = "PM";
|
|
|
|
|
|
|
|
|
|
if (String.IsNullOrEmpty(time))
|
|
|
|
|
return String.Empty;
|
|
|
|
|
|
2011-04-27 09:47:53 -07:00
|
|
|
|
var dateTime = DateTime.Parse(time + " " + meridiem.ToUpper());
|
2011-04-26 19:47:29 -07:00
|
|
|
|
return dateTime.ToString("hh:mm tt");
|
|
|
|
|
}
|
2010-09-22 20:19:47 -07:00
|
|
|
|
}
|
|
|
|
|
}
|