|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Net;
|
|
|
|
using System.Xml;
|
|
|
|
using System.Xml.Linq;
|
|
|
|
using NLog;
|
|
|
|
using NzbDrone.Common.Cache;
|
|
|
|
using NzbDrone.Common.Extensions;
|
|
|
|
using NzbDrone.Common.Http;
|
|
|
|
using NzbDrone.Common.Serializer;
|
|
|
|
|
|
|
|
namespace NzbDrone.Core.Indexers.Newznab
|
|
|
|
{
|
|
|
|
public interface INewznabCapabilitiesProvider
|
|
|
|
{
|
|
|
|
NewznabCapabilities GetCapabilities(NewznabSettings settings);
|
|
|
|
}
|
|
|
|
|
|
|
|
public class NewznabCapabilitiesProvider : INewznabCapabilitiesProvider
|
|
|
|
{
|
|
|
|
private readonly ICached<NewznabCapabilities> _capabilitiesCache;
|
|
|
|
private readonly IHttpClient _httpClient;
|
|
|
|
private readonly Logger _logger;
|
|
|
|
|
|
|
|
public NewznabCapabilitiesProvider(ICacheManager cacheManager, IHttpClient httpClient, Logger logger)
|
|
|
|
{
|
|
|
|
_capabilitiesCache = cacheManager.GetCache<NewznabCapabilities>(GetType());
|
|
|
|
_httpClient = httpClient;
|
|
|
|
_logger = logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
public NewznabCapabilities GetCapabilities(NewznabSettings indexerSettings)
|
|
|
|
{
|
|
|
|
var key = indexerSettings.ToJson();
|
|
|
|
var capabilities = _capabilitiesCache.Get(key, () => FetchCapabilities(indexerSettings), TimeSpan.FromDays(7));
|
|
|
|
|
|
|
|
return capabilities;
|
|
|
|
}
|
|
|
|
|
|
|
|
private NewznabCapabilities FetchCapabilities(NewznabSettings indexerSettings)
|
|
|
|
{
|
|
|
|
var capabilities = new NewznabCapabilities();
|
|
|
|
|
|
|
|
var url = string.Format("{0}/api?t=caps", indexerSettings.Url.TrimEnd('/'));
|
|
|
|
|
|
|
|
if (indexerSettings.ApiKey.IsNotNullOrWhiteSpace())
|
|
|
|
{
|
|
|
|
url += "&apikey=" + indexerSettings.ApiKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
var request = new HttpRequest(url, HttpAccept.Rss);
|
|
|
|
|
|
|
|
HttpResponse response;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
response = _httpClient.Get(request);
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.Debug(ex, "Failed to get newznab api capabilities from {0}", indexerSettings.Url);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
capabilities = ParseCapabilities(response);
|
|
|
|
}
|
|
|
|
catch (XmlException ex)
|
|
|
|
{
|
|
|
|
_logger.Debug(ex, "Failed to parse newznab api capabilities for {0}.", indexerSettings.Url);
|
|
|
|
ex.WithData(response);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
_logger.Error(ex, "Failed to determine newznab api capabilities for {0}, using the defaults instead till Lidarr restarts.", indexerSettings.Url);
|
|
|
|
}
|
|
|
|
|
|
|
|
return capabilities;
|
|
|
|
}
|
|
|
|
|
|
|
|
private NewznabCapabilities ParseCapabilities(HttpResponse response)
|
|
|
|
{
|
|
|
|
var capabilities = new NewznabCapabilities();
|
|
|
|
|
|
|
|
var xmlRoot = XDocument.Parse(response.Content).Element("caps");
|
|
|
|
|
|
|
|
var xmlLimits = xmlRoot.Element("limits");
|
|
|
|
if (xmlLimits != null)
|
|
|
|
{
|
|
|
|
capabilities.DefaultPageSize = int.Parse(xmlLimits.Attribute("default").Value);
|
|
|
|
capabilities.MaxPageSize = int.Parse(xmlLimits.Attribute("max").Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
var xmlSearching = xmlRoot.Element("searching");
|
|
|
|
if (xmlSearching != null)
|
|
|
|
{
|
|
|
|
var xmlBasicSearch = xmlSearching.Element("search");
|
|
|
|
if (xmlBasicSearch == null || xmlBasicSearch.Attribute("available").Value != "yes")
|
|
|
|
{
|
|
|
|
capabilities.SupportedSearchParameters = null;
|
|
|
|
}
|
|
|
|
else if (xmlBasicSearch.Attribute("supportedParams") != null)
|
|
|
|
{
|
|
|
|
capabilities.SupportedSearchParameters = xmlBasicSearch.Attribute("supportedParams").Value.Split(',');
|
|
|
|
}
|
|
|
|
|
|
|
|
var xmlTvSearch = xmlSearching.Element("tv-search");
|
|
|
|
if (xmlTvSearch == null || xmlTvSearch.Attribute("available").Value != "yes")
|
|
|
|
{
|
|
|
|
capabilities.SupportedTvSearchParameters = null;
|
|
|
|
}
|
|
|
|
else if (xmlTvSearch.Attribute("supportedParams") != null)
|
|
|
|
{
|
|
|
|
capabilities.SupportedTvSearchParameters = xmlTvSearch.Attribute("supportedParams").Value.Split(',');
|
|
|
|
capabilities.SupportsAggregateIdSearch = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
var xmlAudioSearch = xmlSearching.Element("audio-search");
|
|
|
|
if (xmlAudioSearch == null || xmlAudioSearch.Attribute("available").Value != "yes")
|
|
|
|
{
|
|
|
|
capabilities.SupportedAudioSearchParameters = null;
|
|
|
|
}
|
|
|
|
else if (xmlAudioSearch.Attribute("supportedParams") != null)
|
|
|
|
{
|
|
|
|
capabilities.SupportedAudioSearchParameters = xmlAudioSearch.Attribute("supportedParams").Value.Split(',');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var xmlCategories = xmlRoot.Element("categories");
|
|
|
|
if (xmlCategories != null)
|
|
|
|
{
|
|
|
|
foreach (var xmlCategory in xmlCategories.Elements("category"))
|
|
|
|
{
|
|
|
|
var cat = new NewznabCategory
|
|
|
|
{
|
|
|
|
Id = int.Parse(xmlCategory.Attribute("id").Value),
|
|
|
|
Name = xmlCategory.Attribute("name").Value,
|
|
|
|
Description = xmlCategory.Attribute("description") != null ? xmlCategory.Attribute("description").Value : string.Empty,
|
|
|
|
Subcategories = new List<NewznabCategory>()
|
|
|
|
};
|
|
|
|
|
|
|
|
foreach (var xmlSubcat in xmlCategory.Elements("subcat"))
|
|
|
|
{
|
|
|
|
cat.Subcategories.Add(new NewznabCategory
|
|
|
|
{
|
|
|
|
Id = int.Parse(xmlSubcat.Attribute("id").Value),
|
|
|
|
Name = xmlSubcat.Attribute("name").Value,
|
|
|
|
Description = xmlSubcat.Attribute("description") != null ? xmlSubcat.Attribute("description").Value : string.Empty
|
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
capabilities.Categories.Add(cat);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return capabilities;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|