|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using Ninject;
|
|
|
|
|
using NLog;
|
|
|
|
|
using NzbDrone.Core.Model;
|
|
|
|
|
using NzbDrone.Core.Repository;
|
|
|
|
|
using NzbDrone.Core.Repository.Quality;
|
|
|
|
|
|
|
|
|
|
namespace NzbDrone.Core.Providers
|
|
|
|
|
{
|
|
|
|
|
public class InventoryProvider
|
|
|
|
|
{
|
|
|
|
|
private readonly SeriesProvider _seriesProvider;
|
|
|
|
|
private readonly EpisodeProvider _episodeProvider;
|
|
|
|
|
private readonly HistoryProvider _historyProvider;
|
|
|
|
|
private readonly QualityTypeProvider _qualityTypeProvider;
|
|
|
|
|
private readonly QualityProvider _qualityProvider;
|
|
|
|
|
|
|
|
|
|
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
|
|
|
|
|
|
|
|
|
|
[Inject]
|
|
|
|
|
public InventoryProvider(SeriesProvider seriesProvider, EpisodeProvider episodeProvider,
|
|
|
|
|
HistoryProvider historyProvider, QualityTypeProvider qualityTypeProvider,
|
|
|
|
|
QualityProvider qualityProvider)
|
|
|
|
|
{
|
|
|
|
|
_seriesProvider = seriesProvider;
|
|
|
|
|
_episodeProvider = episodeProvider;
|
|
|
|
|
_historyProvider = historyProvider;
|
|
|
|
|
_qualityTypeProvider = qualityTypeProvider;
|
|
|
|
|
_qualityProvider = qualityProvider;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public InventoryProvider()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public virtual bool IsMonitored(EpisodeParseResult parseResult)
|
|
|
|
|
{
|
|
|
|
|
var series = _seriesProvider.FindSeries(parseResult.CleanTitle);
|
|
|
|
|
|
|
|
|
|
if (series == null)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("{0} is not mapped to any series in DB. skipping", parseResult.CleanTitle);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
parseResult.Series = series;
|
|
|
|
|
|
|
|
|
|
if (!series.Monitored)
|
|
|
|
|
{
|
|
|
|
|
Logger.Debug("{0} is present in the DB but not tracked. skipping.", parseResult.CleanTitle);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var episodes = _episodeProvider.GetEpisodesByParseResult(parseResult, true);
|
|
|
|
|
|
|
|
|
|
//return monitored if any of the episodes are monitored
|
|
|
|
|
if (episodes.Any(episode => !episode.Ignored))
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Logger.Debug("All episodes are ignored. skipping.");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Comprehensive check on whether or not this episode is needed.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name = "parsedReport">Episode that needs to be checked</param>
|
|
|
|
|
/// <param name="skipHistory">False unless called by a manual search job</param>
|
|
|
|
|
/// <returns>Whether or not the file quality meets the requirements </returns>
|
|
|
|
|
/// <remarks>for multi episode files, all episodes need to meet the requirement
|
|
|
|
|
/// before the report is downloaded</remarks>
|
|
|
|
|
public virtual bool IsQualityNeeded(EpisodeParseResult parsedReport, bool skipHistory = false)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("Checking if report meets quality requirements. {0}", parsedReport.Quality);
|
|
|
|
|
if (!parsedReport.Series.QualityProfile.Allowed.Contains(parsedReport.Quality.QualityType))
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("Quality {0} rejected by Series' quality profile", parsedReport.Quality);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var cutoff = parsedReport.Series.QualityProfile.Cutoff;
|
|
|
|
|
|
|
|
|
|
if (!IsAcceptableSize(parsedReport))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
foreach (var episode in _episodeProvider.GetEpisodesByParseResult(parsedReport, true))
|
|
|
|
|
{
|
|
|
|
|
//Checking File
|
|
|
|
|
var file = episode.EpisodeFile;
|
|
|
|
|
if (file != null)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("Comparing file quality with report. Existing file is {0} proper:{1}", file.Quality, file.Proper);
|
|
|
|
|
if (!IsUpgrade(new Quality { QualityType = file.Quality, Proper = file.Proper }, parsedReport.Quality, cutoff))
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//Checking History (If not a manual search)
|
|
|
|
|
if (!skipHistory)
|
|
|
|
|
{
|
|
|
|
|
var bestQualityInHistory = _historyProvider.GetBestQualityInHistory(episode.EpisodeId);
|
|
|
|
|
if(bestQualityInHistory != null)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("Comparing history quality with report. History is {0}", bestQualityInHistory);
|
|
|
|
|
if(!IsUpgrade(bestQualityInHistory, parsedReport.Quality, cutoff))
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Logger.Debug("Episode {0} is needed", parsedReport);
|
|
|
|
|
return true; //If we get to this point and the file has not yet been rejected then accept it
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static bool IsUpgrade(Quality currentQuality, Quality newQuality, QualityTypes cutOff)
|
|
|
|
|
{
|
|
|
|
|
if (currentQuality.QualityType >= cutOff)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("Existing item meets cut-off. skipping.");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (newQuality > currentQuality)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
if (currentQuality > newQuality)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("existing item has better quality. skipping");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (currentQuality == newQuality && !newQuality.Proper)
|
|
|
|
|
{
|
|
|
|
|
Logger.Trace("same quality. not proper skipping");
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Logger.Debug("New item has better quality than existing item");
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public virtual bool IsAcceptableSize(EpisodeParseResult parseResult)
|
|
|
|
|
{
|
|
|
|
|
var qualityType = _qualityTypeProvider.Get((int) parseResult.Quality.QualityType);
|
|
|
|
|
|
|
|
|
|
//Need to determine if this is a 30 or 60 minute episode
|
|
|
|
|
//Is it a multi-episode release?
|
|
|
|
|
//Is it the first or last series of a season?
|
|
|
|
|
|
|
|
|
|
//0 will be treated as unlimited
|
|
|
|
|
if (qualityType.MaxSize == 0)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
var maxSize = qualityType.MaxSize.Megabytes();
|
|
|
|
|
var series = parseResult.Series;
|
|
|
|
|
|
|
|
|
|
//Multiply maxSize by Series.Runtime
|
|
|
|
|
maxSize = maxSize * series.Runtime;
|
|
|
|
|
|
|
|
|
|
//Multiply maxSize by the number of episodes parsed (if EpisodeNumbers is null it will be treated as a single episode)
|
|
|
|
|
if (parseResult.EpisodeNumbers != null)
|
|
|
|
|
maxSize = maxSize * parseResult.EpisodeNumbers.Count;
|
|
|
|
|
|
|
|
|
|
//Check if there was only one episode parsed
|
|
|
|
|
//and it is the first or last episode of the season
|
|
|
|
|
if (parseResult.EpisodeNumbers != null && parseResult.EpisodeNumbers.Count == 1 &&
|
|
|
|
|
_episodeProvider.IsFirstOrLastEpisodeOfSeason(series.SeriesId,
|
|
|
|
|
parseResult.SeasonNumber, parseResult.EpisodeNumbers[0]))
|
|
|
|
|
{
|
|
|
|
|
maxSize = maxSize * 2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//If the parsed size is greater than maxSize we don't want it
|
|
|
|
|
if (parseResult.Size > maxSize)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public virtual bool IsUpgradePossible(Episode episode)
|
|
|
|
|
{
|
|
|
|
|
//Used to check if the existing episode can be upgraded by searching (Before we search)
|
|
|
|
|
|
|
|
|
|
//If no episode file exists on disk, then an upgrade is possible
|
|
|
|
|
if (episode.EpisodeFileId == 0)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
//Get the quality profile for the series
|
|
|
|
|
var profile = _qualityProvider.Get(episode.Series.QualityProfileId);
|
|
|
|
|
|
|
|
|
|
//If the current episode file meets or exceeds the cutoff, do not attempt upgrade
|
|
|
|
|
if (episode.EpisodeFile.Quality >= profile.Cutoff)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|