@ -9,22 +9,70 @@ namespace NzbDrone.Core.Indexers.Torznab
{
{
public class TorznabRequestGenerator : IIndexerRequestGenerator
public class TorznabRequestGenerator : IIndexerRequestGenerator
{
{
public Int32 MaxPages { get ; set ; }
private readonly ITorznabCapabilitiesProvider _capabilitiesProvider ;
public Int32 PageSize { get ; set ; }
public int MaxPages { get ; set ; }
public int PageSize { get ; set ; }
public TorznabSettings Settings { get ; set ; }
public TorznabSettings Settings { get ; set ; }
public TorznabRequestGenerator ( )
public TorznabRequestGenerator ( ITorznabCapabilitiesProvider capabilitiesProvider )
{
{
_capabilitiesProvider = capabilitiesProvider ;
MaxPages = 30 ;
MaxPages = 30 ;
PageSize = 100 ;
PageSize = 100 ;
}
}
private bool SupportsSearch
{
get
{
var capabilities = _capabilitiesProvider . GetCapabilities ( Settings ) ;
return capabilities . SupportedSearchParameters ! = null
& & capabilities . SupportedSearchParameters . Contains ( "q" ) ;
}
}
private bool SupportsTvSearch
{
get
{
var capabilities = _capabilitiesProvider . GetCapabilities ( Settings ) ;
return capabilities . SupportedTvSearchParameters ! = null
& & capabilities . SupportedTvSearchParameters . Contains ( "q" )
& & capabilities . SupportedTvSearchParameters . Contains ( "season" )
& & capabilities . SupportedTvSearchParameters . Contains ( "ep" ) ;
}
}
private bool SupportsTvRageSearch
{
get
{
var capabilities = _capabilitiesProvider . GetCapabilities ( Settings ) ;
return capabilities . SupportedTvSearchParameters ! = null
& & capabilities . SupportedTvSearchParameters . Contains ( "rid" )
& & capabilities . SupportedTvSearchParameters . Contains ( "season" )
& & capabilities . SupportedTvSearchParameters . Contains ( "ep" )
& & Settings . EnableRageIDLookup ;
}
}
public virtual IList < IEnumerable < IndexerRequest > > GetRecentRequests ( )
public virtual IList < IEnumerable < IndexerRequest > > GetRecentRequests ( )
{
{
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var capabilities = _capabilitiesProvider . GetCapabilities ( Settings ) ;
if ( capabilities . SupportedTvSearchParameters ! = null )
{
// TODO: We might consider getting multiple pages in the future, but atm we limit it to 1 page.
// TODO: We might consider getting multiple pages in the future, but atm we limit it to 1 page.
pageableRequests . AddIfNotNull ( GetPagedRequests ( 1 , Settings . Categories . Concat ( Settings . AnimeCategories ) , "tvsearch" , "" ) ) ;
pageableRequests . AddIfNotNull ( GetPagedRequests ( 1 , Settings . Categories . Concat ( Settings . AnimeCategories ) , "tvsearch" , "" ) ) ;
}
return pageableRequests ;
return pageableRequests ;
}
}
@ -33,20 +81,20 @@ namespace NzbDrone.Core.Indexers.Torznab
{
{
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
if ( searchCriteria . Series . TvRageId > 0 & & S ettings. EnableRageIDLookup )
if ( searchCriteria . Series . TvRageId > 0 & & S upportsTvRageSearch )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
S tring. Format ( "&rid={0}&season={1}&ep={2}" ,
s tring. Format ( "&rid={0}&season={1}&ep={2}" ,
searchCriteria . Series . TvRageId ,
searchCriteria . Series . TvRageId ,
searchCriteria . SeasonNumber ,
searchCriteria . SeasonNumber ,
searchCriteria . EpisodeNumber ) ) ) ;
searchCriteria . EpisodeNumber ) ) ) ;
}
}
else
else if ( SupportsTvSearch )
{
{
foreach ( var queryTitle in searchCriteria . QueryTitles )
foreach ( var queryTitle in searchCriteria . QueryTitles )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
S tring. Format ( "&q={0}&season={1}&ep={2}" ,
s tring. Format ( "&q={0}&season={1}&ep={2}" ,
NewsnabifyTitle ( queryTitle ) ,
NewsnabifyTitle ( queryTitle ) ,
searchCriteria . SeasonNumber ,
searchCriteria . SeasonNumber ,
searchCriteria . EpisodeNumber ) ) ) ;
searchCriteria . EpisodeNumber ) ) ) ;
@ -60,19 +108,19 @@ namespace NzbDrone.Core.Indexers.Torznab
{
{
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
if ( searchCriteria . Series . TvRageId > 0 & & S ettings. EnableRageIDLookup )
if ( searchCriteria . Series . TvRageId > 0 & & S upportsTvRageSearch )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
S tring. Format ( "&rid={0}&season={1}" ,
s tring. Format ( "&rid={0}&season={1}" ,
searchCriteria . Series . TvRageId ,
searchCriteria . Series . TvRageId ,
searchCriteria . SeasonNumber ) ) ) ;
searchCriteria . SeasonNumber ) ) ) ;
}
}
else
else if ( SupportsTvSearch )
{
{
foreach ( var queryTitle in searchCriteria . QueryTitles )
foreach ( var queryTitle in searchCriteria . QueryTitles )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
S tring. Format ( "&q={0}&season={1}" ,
s tring. Format ( "&q={0}&season={1}" ,
NewsnabifyTitle ( queryTitle ) ,
NewsnabifyTitle ( queryTitle ) ,
searchCriteria . SeasonNumber ) ) ) ;
searchCriteria . SeasonNumber ) ) ) ;
}
}
@ -85,19 +133,19 @@ namespace NzbDrone.Core.Indexers.Torznab
{
{
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
if ( searchCriteria . Series . TvRageId > 0 & & S ettings. EnableRageIDLookup )
if ( searchCriteria . Series . TvRageId > 0 & & S upportsTvRageSearch )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
S tring. Format ( "&rid={0}&season={1:yyyy}&ep={1:MM}/{1:dd}" ,
s tring. Format ( "&rid={0}&season={1:yyyy}&ep={1:MM}/{1:dd}" ,
searchCriteria . Series . TvRageId ,
searchCriteria . Series . TvRageId ,
searchCriteria . AirDate ) ) ) ;
searchCriteria . AirDate ) ) ) ;
}
}
else
else if ( SupportsTvSearch )
{
{
foreach ( var queryTitle in searchCriteria . QueryTitles )
foreach ( var queryTitle in searchCriteria . QueryTitles )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories , "tvsearch" ,
S tring. Format ( "&q={0}&season={1:yyyy}&ep={1:MM}/{1:dd}" ,
s tring. Format ( "&q={0}&season={1:yyyy}&ep={1:MM}/{1:dd}" ,
NewsnabifyTitle ( queryTitle ) ,
NewsnabifyTitle ( queryTitle ) ,
searchCriteria . AirDate ) ) ) ;
searchCriteria . AirDate ) ) ) ;
}
}
@ -110,13 +158,16 @@ namespace NzbDrone.Core.Indexers.Torznab
{
{
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
if ( SupportsSearch )
{
foreach ( var queryTitle in searchCriteria . QueryTitles )
foreach ( var queryTitle in searchCriteria . QueryTitles )
{
{
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . AnimeCategories , "search" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . AnimeCategories , "search" ,
S tring. Format ( "&q={0}+{1:00}" ,
s tring. Format ( "&q={0}+{1:00}" ,
NewsnabifyTitle ( queryTitle ) ,
NewsnabifyTitle ( queryTitle ) ,
searchCriteria . AbsoluteEpisodeNumber ) ) ) ;
searchCriteria . AbsoluteEpisodeNumber ) ) ) ;
}
}
}
return pageableRequests ;
return pageableRequests ;
}
}
@ -125,29 +176,32 @@ namespace NzbDrone.Core.Indexers.Torznab
{
{
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
var pageableRequests = new List < IEnumerable < IndexerRequest > > ( ) ;
if ( SupportsSearch )
{
foreach ( var queryTitle in searchCriteria . EpisodeQueryTitles )
foreach ( var queryTitle in searchCriteria . EpisodeQueryTitles )
{
{
var query = queryTitle . Replace ( '+' , ' ' ) ;
var query = queryTitle . Replace ( '+' , ' ' ) ;
query = System . Web . HttpUtility . UrlEncode ( query ) ;
query = System . Web . HttpUtility . UrlEncode ( query ) ;
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories . Concat ( Settings . AnimeCategories ) , "search" ,
pageableRequests . AddIfNotNull ( GetPagedRequests ( MaxPages , Settings . Categories . Concat ( Settings . AnimeCategories ) , "search" ,
S tring. Format ( "&q={0}" ,
s tring. Format ( "&q={0}" ,
query ) ) ) ;
query ) ) ) ;
}
}
}
return pageableRequests ;
return pageableRequests ;
}
}
private IEnumerable < IndexerRequest > GetPagedRequests ( Int32 maxPages , IEnumerable < Int32 > categories , String searchType , S tring parameters )
private IEnumerable < IndexerRequest > GetPagedRequests ( int maxPages , IEnumerable < int > categories , string searchType , s tring parameters )
{
{
if ( categories . Empty ( ) )
if ( categories . Empty ( ) )
{
{
yield break ;
yield break ;
}
}
var categoriesQuery = S tring. Join ( "," , categories . Distinct ( ) ) ;
var categoriesQuery = s tring. Join ( "," , categories . Distinct ( ) ) ;
var baseUrl = S tring. Format ( "{0}/api?t={1}&cat={2}&extended=1{3}" , Settings . Url . TrimEnd ( '/' ) , searchType , categoriesQuery , Settings . AdditionalParameters ) ;
var baseUrl = s tring. Format ( "{0}/api?t={1}&cat={2}&extended=1{3}" , Settings . Url . TrimEnd ( '/' ) , searchType , categoriesQuery , Settings . AdditionalParameters ) ;
if ( Settings . ApiKey . IsNotNullOrWhiteSpace ( ) )
if ( Settings . ApiKey . IsNotNullOrWhiteSpace ( ) )
{
{
@ -156,18 +210,18 @@ namespace NzbDrone.Core.Indexers.Torznab
if ( PageSize = = 0 )
if ( PageSize = = 0 )
{
{
yield return new IndexerRequest ( S tring. Format ( "{0}{1}" , baseUrl , parameters ) , HttpAccept . Rss ) ;
yield return new IndexerRequest ( s tring. Format ( "{0}{1}" , baseUrl , parameters ) , HttpAccept . Rss ) ;
}
}
else
else
{
{
for ( var page = 0 ; page < maxPages ; page + + )
for ( var page = 0 ; page < maxPages ; page + + )
{
{
yield return new IndexerRequest ( S tring. Format ( "{0}&offset={1}&limit={2}{3}" , baseUrl , page * PageSize , PageSize , parameters ) , HttpAccept . Rss ) ;
yield return new IndexerRequest ( s tring. Format ( "{0}&offset={1}&limit={2}{3}" , baseUrl , page * PageSize , PageSize , parameters ) , HttpAccept . Rss ) ;
}
}
}
}
}
}
private static String NewsnabifyTitle ( S tring title )
private static string NewsnabifyTitle ( s tring title )
{
{
return title . Replace ( "+" , "%20" ) ;
return title . Replace ( "+" , "%20" ) ;
}
}