using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Text ;
using System.Threading.Tasks ;
using NLog ;
using NzbDrone.Core.Download ;
using NzbDrone.Core.Indexers ;
using NzbDrone.Core.Tv ;
using NzbDrone.Core.Model ;
using NzbDrone.Core.Model.Notification ;
using NzbDrone.Core.DecisionEngine ;
using NzbDrone.Core.Repository ;
using NzbDrone.Core.Repository.Search ;
namespace NzbDrone.Core.Providers.Search
{
public class PartialSeasonSearch : SearchBase
{
private static readonly Logger logger = LogManager . GetCurrentClassLogger ( ) ;
public PartialSeasonSearch ( ISeriesService seriesService , IEpisodeService episodeService , DownloadProvider downloadProvider , IIndexerService indexerService ,
SceneMappingProvider sceneMappingProvider , AllowedDownloadSpecification allowedDownloadSpecification ,
SearchHistoryProvider searchHistoryProvider , ISeriesRepository seriesRepository )
: base ( seriesService , seriesRepository , episodeService , downloadProvider , indexerService , sceneMappingProvider ,
allowedDownloadSpecification , searchHistoryProvider )
{
}
public PartialSeasonSearch ( )
{
}
public override List < EpisodeParseResult > PerformSearch ( Series series , dynamic options , ProgressNotification notification )
{
if ( options . SeasonNumber = = null | | options . SeasonNumber < 0 )
throw new ArgumentException ( "SeasonNumber is invalid" ) ;
if ( options . Episodes = = null )
throw new ArgumentException ( "Episodes were not provided" ) ;
List < Episode > episodes = options . Episodes ;
if ( ! episodes . Any ( ) )
throw new ArgumentException ( "Episodes were not provided" ) ;
notification . CurrentMessage = String . Format ( "Looking for {0} - Season {1}" , series . Title , options . SeasonNumber ) ;
var reports = new List < EpisodeParseResult > ( ) ;
object reportsLock = new object ( ) ;
var title = GetSearchTitle ( series ) ;
var prefixes = GetEpisodeNumberPrefixes ( episodes . Select ( e = > e . EpisodeNumber ) ) ;
foreach ( var p in prefixes )
{
var prefix = p ;
Parallel . ForEach ( _indexerService . GetEnabledIndexers ( ) , indexer = >
{
try
{
lock ( reportsLock )
{
reports . AddRange ( indexer . FetchPartialSeason ( title , options . SeasonNumber , prefix ) ) ;
}
}
catch ( Exception e )
{
logger . ErrorException (
String . Format (
"An error has occurred while searching for {0} Season {1:00} Prefix: {2} from: {3}" ,
series . Title , options . SeasonNumber , prefix , indexer . Name ) ,
e ) ;
}
} ) ;
}
return reports ;
}
public override SearchHistoryItem CheckReport ( Series series , dynamic options , EpisodeParseResult episodeParseResult ,
SearchHistoryItem item )
{
if ( options . SeasonNumber ! = episodeParseResult . SeasonNumber )
{
logger . Trace ( "Season number does not match searched season number, skipping." ) ;
item . SearchError = ReportRejectionType . WrongSeason ;
return item ;
}
return item ;
}
protected override void FinalizeSearch ( Series series , dynamic options , Boolean reportsFound , ProgressNotification notification )
{
logger . Warn ( "Unable to find {0} - Season {1} in any of indexers." , series . Title , options . SeasonNumber ) ;
notification . CurrentMessage = reportsFound ? String . Format ( "Sorry, couldn't find {0} Season {1:00}, that matches your preferences." , series . Title , options . SeasonNumber )
: String . Format ( "Sorry, couldn't find {0} Season {1:00} in any of indexers." , series . Title , options . SeasonNumber ) ;
}
private List < int > GetEpisodeNumberPrefixes ( IEnumerable < int > episodeNumbers )
{
var results = new List < int > ( ) ;
foreach ( var i in episodeNumbers )
{
results . Add ( i / 10 ) ;
}
return results . Distinct ( ) . ToList ( ) ;
}
}
}