sonarr-repo-only/NzbDrone.Core/Providers/SeriesProvider.cs

209 lines
8.2 KiB
C#
Raw Normal View History

using System;
2011-05-27 06:03:57 +00:00
using System.Collections.Generic;
using System.IO;
2010-09-23 03:19:47 +00:00
using System.Linq;
using System.Text.RegularExpressions;
2010-10-02 19:01:43 +00:00
using NLog;
2011-04-04 03:50:12 +00:00
using NzbDrone.Core.Providers.Core;
using NzbDrone.Core.Repository;
using NzbDrone.Core.Repository.Quality;
2011-06-15 02:31:41 +00:00
using PetaPoco;
2010-09-23 03:19:47 +00:00
using TvdbLib.Data;
namespace NzbDrone.Core.Providers
2010-09-23 03:19:47 +00:00
{
2011-04-08 23:55:23 +00:00
public class SeriesProvider
2010-09-23 03:19:47 +00:00
{
2011-04-10 02:44:01 +00:00
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
private readonly ConfigProvider _configProvider;
private readonly TvDbProvider _tvDbProvider;
2011-06-15 02:31:41 +00:00
private readonly IDatabase _database;
private readonly SceneMappingProvider _sceneNameMappingProvider;
private static readonly Regex TimeRegex = new Regex(@"^(?<time>\d+:?\d*)\W*(?<meridiem>am|pm)?", RegexOptions.IgnoreCase | RegexOptions.Compiled);
2010-09-23 03:19:47 +00:00
public SeriesProvider(IDatabase database, ConfigProvider configProviderProvider,
TvDbProvider tvDbProviderProvider, SceneMappingProvider sceneNameMappingProvider)
2010-09-23 03:19:47 +00:00
{
_database = database;
_configProvider = configProviderProvider;
_tvDbProvider = tvDbProviderProvider;
2011-06-16 06:53:23 +00:00
_sceneNameMappingProvider = sceneNameMappingProvider;
2010-09-23 03:19:47 +00:00
}
public SeriesProvider()
{
}
2011-05-27 06:03:57 +00:00
public virtual IList<Series> GetAllSeries()
2010-09-23 03:19:47 +00:00
{
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId");
return series;
}
public virtual IList<Series> GetAllSeriesWithEpisodeCount(bool ignoreSpecialsInSeasonCount)
{
var seasonNumber = 0;
if (!ignoreSpecialsInSeasonCount)
seasonNumber = -1;
var series = _database.Fetch<Series, QualityProfile>(@"SELECT Series.*, COUNT (NULLIF(Ignored, 1)) AS EpisodeCount,
SUM(CASE WHEN Ignored = 0 AND EpisodeFileId > 0 THEN 1 ELSE 0 END) as EpisodeFileCount,
COUNT (DISTINCT(NULLIF(SeasonNumber, @0))) as SeasonCount,
QualityProfiles.*
FROM Series
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
JOIN Episodes ON Series.SeriesId = Episodes.SeriesId
GROUP BY seriesId", seasonNumber);
2011-06-15 02:31:41 +00:00
return series;
2010-09-23 03:19:47 +00:00
}
2011-04-07 02:25:52 +00:00
public virtual Series GetSeries(int seriesId)
2010-09-24 07:14:42 +00:00
{
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
WHERE seriesId= @0", seriesId).Single();
return series;
2010-09-24 07:14:42 +00:00
}
2010-10-02 19:01:43 +00:00
/// <summary>
2011-04-10 02:44:01 +00:00
/// Determines if a series is being actively watched.
2010-10-02 19:01:43 +00:00
/// </summary>
2011-04-10 02:44:01 +00:00
/// <param name = "id">The TVDB ID of the series</param>
2010-10-02 19:01:43 +00:00
/// <returns>Whether or not the show is monitored</returns>
2011-04-07 02:25:52 +00:00
public virtual bool IsMonitored(long id)
{
2011-06-15 02:31:41 +00:00
return GetAllSeries().Any(c => c.SeriesId == id && c.Monitored);
}
2011-04-07 02:25:52 +00:00
public virtual Series UpdateSeriesInfo(int seriesId)
{
var tvDbSeries = _tvDbProvider.GetSeries(seriesId, true);
var series = GetSeries(seriesId);
series.SeriesId = tvDbSeries.Id;
series.Title = tvDbSeries.SeriesName;
series.AirTimes = CleanAirsTime(tvDbSeries.AirsTime);
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;
UpdateSeries(series);
return series;
}
2011-04-07 02:25:52 +00:00
public virtual void AddSeries(string path, int tvDbSeriesId, int qualityProfileId)
2010-09-23 03:19:47 +00:00
{
Logger.Info("Adding Series [{0}] Path: [{1}]", tvDbSeriesId, path);
var repoSeries = new Series();
repoSeries.SeriesId = tvDbSeriesId;
repoSeries.Path = path;
repoSeries.Monitored = true; //New shows should be monitored
repoSeries.QualityProfileId = qualityProfileId;
if (qualityProfileId == 0)
2011-06-17 02:27:10 +00:00
repoSeries.QualityProfileId = Convert.ToInt32(_configProvider.GetValue("DefaultQualityProfile", "1"));
repoSeries.SeasonFolder = _configProvider.UseSeasonFolder;
2011-06-15 02:31:41 +00:00
_database.Insert(repoSeries);
2010-09-23 03:19:47 +00:00
}
2011-04-07 02:25:52 +00:00
public virtual Series FindSeries(string title)
{
try
{
var normalizeTitle = Parser.NormalizeTitle(title);
var seriesId = _sceneNameMappingProvider.GetSeriesId(normalizeTitle);
if (seriesId != null)
{
return GetSeries(seriesId.Value);
}
var series = _database.Fetch<Series, QualityProfile>(@"SELECT * FROM Series
INNER JOIN QualityProfiles ON Series.QualityProfileId = QualityProfiles.QualityProfileId
WHERE CleanTitle = @0", normalizeTitle).FirstOrDefault();
return series;
}
2011-06-22 01:22:52 +00:00
catch (InvalidOperationException)
{
2011-06-22 01:22:52 +00: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
return null;
}
}
2011-04-07 02:25:52 +00:00
public virtual void UpdateSeries(Series series)
{
2011-06-15 02:31:41 +00:00
_database.Update(series);
}
2011-04-07 02:25:52 +00:00
public virtual void DeleteSeries(int seriesId)
{
2011-06-17 02:27:10 +00:00
var series = GetSeries(seriesId);
Logger.Warn("Deleting Series [{0}]", series.Title);
2011-06-17 02:27:10 +00:00
using (var tran = _database.GetTransaction())
{
//Delete History, Files, Episodes, Seasons then the Series
2011-06-17 02:27:10 +00:00
Logger.Debug("Deleting History Items from DB for Series: {0}", series.Title);
_database.Delete<History>("WHERE SeriesId=@0", seriesId);
2011-06-17 02:27:10 +00:00
Logger.Debug("Deleting EpisodeFiles from DB for Series: {0}", series.Title);
_database.Delete<EpisodeFile>("WHERE SeriesId=@0", seriesId);
2011-06-17 02:27:10 +00:00
Logger.Debug("Deleting Episodes from DB for Series: {0}", series.Title);
_database.Delete<Episode>("WHERE SeriesId=@0", seriesId);
2011-06-17 02:27:10 +00:00
Logger.Debug("Deleting Series from DB {0}", series.Title);
_database.Delete<Series>("WHERE SeriesId=@0", seriesId);
2011-06-17 02:27:10 +00:00
Logger.Info("Successfully deleted Series [{0}]", series.Title);
2011-06-17 02:27:10 +00:00
tran.Complete();
}
}
2011-04-07 02:25:52 +00:00
public virtual bool SeriesPathExists(string cleanPath)
{
2011-06-15 02:31:41 +00:00
if (GetAllSeries().Any(s => s.Path == cleanPath))
return true;
return false;
}
/// <summary>
/// Cleans up the AirsTime Component from TheTVDB since it can be garbage that comes in.
/// </summary>
/// <param name = "rawTime">The TVDB AirsTime</param>
/// <returns>String that contains the AirTimes</returns>
private static string CleanAirsTime(string rawTime)
{
var match = TimeRegex.Match(rawTime);
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 16:47:53 +00:00
var dateTime = DateTime.Parse(time + " " + meridiem.ToUpper());
return dateTime.ToString("hh:mm tt");
}
2010-09-23 03:19:47 +00:00
}
}