parent
80beea9bdb
commit
a922586aba
@ -0,0 +1,550 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Collections.Specialized;
|
||||||
|
using System.Globalization;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Text;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using AngleSharp.Html.Parser;
|
||||||
|
using FluentValidation;
|
||||||
|
using NLog;
|
||||||
|
using NzbDrone.Common.Http;
|
||||||
|
using NzbDrone.Core.Annotations;
|
||||||
|
using NzbDrone.Core.Configuration;
|
||||||
|
using NzbDrone.Core.Indexers.Exceptions;
|
||||||
|
using NzbDrone.Core.IndexerSearch.Definitions;
|
||||||
|
using NzbDrone.Core.Messaging.Events;
|
||||||
|
using NzbDrone.Core.Parser;
|
||||||
|
using NzbDrone.Core.Parser.Model;
|
||||||
|
using NzbDrone.Core.Validation;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Indexers.Definitions
|
||||||
|
{
|
||||||
|
public class Anidub : TorrentIndexerBase<AnidubSettings>
|
||||||
|
{
|
||||||
|
public override string Name => "Anidub";
|
||||||
|
public override string[] IndexerUrls => new string[] { "https://tr.anidub.com/" };
|
||||||
|
public override string Description => "Anidub is russian anime voiceover group and eponymous anime tracker.";
|
||||||
|
public override string Language => "ru-ru";
|
||||||
|
public override Encoding Encoding => Encoding.UTF8;
|
||||||
|
public override DownloadProtocol Protocol => DownloadProtocol.Torrent;
|
||||||
|
public override IndexerPrivacy Privacy => IndexerPrivacy.SemiPublic;
|
||||||
|
public override IndexerCapabilities Capabilities => SetCapabilities();
|
||||||
|
|
||||||
|
public Anidub(IHttpClient httpClient, IEventAggregator eventAggregator, IIndexerStatusService indexerStatusService, IConfigService configService, Logger logger)
|
||||||
|
: base(httpClient, eventAggregator, indexerStatusService, configService, logger)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IIndexerRequestGenerator GetRequestGenerator()
|
||||||
|
{
|
||||||
|
return new AnidubRequestGenerator() { Settings = Settings, Capabilities = Capabilities };
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IParseIndexerResponse GetParser()
|
||||||
|
{
|
||||||
|
return new AnidubParser(Settings, Capabilities.Categories) { HttpClient = _httpClient, Logger = _logger };
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async Task DoLogin()
|
||||||
|
{
|
||||||
|
UpdateCookies(null, null);
|
||||||
|
|
||||||
|
var requestBuilder = new HttpRequestBuilder(Settings.BaseUrl + "index.php")
|
||||||
|
{
|
||||||
|
LogResponseContent = true,
|
||||||
|
AllowAutoRedirect = true
|
||||||
|
};
|
||||||
|
|
||||||
|
var mainPage = await _httpClient.ExecuteAsync(new HttpRequest(Settings.BaseUrl));
|
||||||
|
|
||||||
|
requestBuilder.Method = Common.Http.HttpMethod.POST;
|
||||||
|
requestBuilder.PostProcess += r => r.RequestTimeout = TimeSpan.FromSeconds(15);
|
||||||
|
requestBuilder.SetCookies(mainPage.GetCookies());
|
||||||
|
|
||||||
|
var authLoginRequest = requestBuilder
|
||||||
|
.AddFormParameter("login_name", Settings.Username)
|
||||||
|
.AddFormParameter("login_password", Settings.Password)
|
||||||
|
.AddFormParameter("login", "submit")
|
||||||
|
.SetHeader("Content-Type", "application/x-www-form-urlencoded")
|
||||||
|
.Build();
|
||||||
|
|
||||||
|
var response = await ExecuteAuth(authLoginRequest);
|
||||||
|
|
||||||
|
if (response.Content != null && !CheckIfLoginNeeded(response))
|
||||||
|
{
|
||||||
|
UpdateCookies(response.GetCookies(), DateTime.Now + TimeSpan.FromDays(30));
|
||||||
|
_logger.Debug("Anidub authentication succeeded");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
const string ErrorSelector = "#content .berror .berror_c";
|
||||||
|
var parser = new HtmlParser();
|
||||||
|
var document = await parser.ParseDocumentAsync(response.Content);
|
||||||
|
var errorMessage = document.QuerySelector(ErrorSelector).TextContent.Trim();
|
||||||
|
throw new IndexerAuthException("Anidub authentication failed. Error: " + errorMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override bool CheckIfLoginNeeded(HttpResponse httpResponse)
|
||||||
|
{
|
||||||
|
if (httpResponse.Content.Contains("index.php?action=logout"))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private IndexerCapabilities SetCapabilities()
|
||||||
|
{
|
||||||
|
var caps = new IndexerCapabilities
|
||||||
|
{
|
||||||
|
TvSearchParams = new List<TvSearchParam>
|
||||||
|
{
|
||||||
|
TvSearchParam.Q, TvSearchParam.Season, TvSearchParam.Ep
|
||||||
|
},
|
||||||
|
MovieSearchParams = new List<MovieSearchParam>
|
||||||
|
{
|
||||||
|
MovieSearchParam.Q
|
||||||
|
},
|
||||||
|
BookSearchParams = new List<BookSearchParam>
|
||||||
|
{
|
||||||
|
BookSearchParam.Q
|
||||||
|
},
|
||||||
|
MusicSearchParams = new List<MusicSearchParam>
|
||||||
|
{
|
||||||
|
MusicSearchParam.Q
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
caps.Categories.AddCategoryMapping(2, NewznabStandardCategory.TVAnime, "Аниме TV");
|
||||||
|
caps.Categories.AddCategoryMapping(3, NewznabStandardCategory.Movies, "Аниме Фильмы");
|
||||||
|
caps.Categories.AddCategoryMapping(4, NewznabStandardCategory.TVAnime, "Аниме OVA");
|
||||||
|
caps.Categories.AddCategoryMapping(5, NewznabStandardCategory.TVAnime, "Аниме OVA |- Аниме ONA");
|
||||||
|
caps.Categories.AddCategoryMapping(6, NewznabStandardCategory.TV, "Дорамы / Японские Сериалы и Фильмы");
|
||||||
|
caps.Categories.AddCategoryMapping(7, NewznabStandardCategory.TV, "Дорамы / Корейские Сериалы и Фильмы");
|
||||||
|
caps.Categories.AddCategoryMapping(8, NewznabStandardCategory.TV, "Дорамы / Китайские Сериалы и Фильмы");
|
||||||
|
caps.Categories.AddCategoryMapping(9, NewznabStandardCategory.TV, "Дорамы");
|
||||||
|
caps.Categories.AddCategoryMapping(10, NewznabStandardCategory.TVAnime, "Аниме TV / Аниме Ongoing");
|
||||||
|
caps.Categories.AddCategoryMapping(11, NewznabStandardCategory.TVAnime, "Аниме TV / Многосерийный сёнэн");
|
||||||
|
caps.Categories.AddCategoryMapping(12, NewznabStandardCategory.Other, "Аниме Ongoing Анонсы");
|
||||||
|
caps.Categories.AddCategoryMapping(13, NewznabStandardCategory.XXX, "18+");
|
||||||
|
caps.Categories.AddCategoryMapping(14, NewznabStandardCategory.TVAnime, "Аниме TV / Законченные сериалы");
|
||||||
|
caps.Categories.AddCategoryMapping(15, NewznabStandardCategory.BooksComics, "Манга");
|
||||||
|
caps.Categories.AddCategoryMapping(16, NewznabStandardCategory.Audio, "OST");
|
||||||
|
caps.Categories.AddCategoryMapping(17, NewznabStandardCategory.Audio, "Подкасты");
|
||||||
|
return caps;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class AnidubRequestGenerator : IIndexerRequestGenerator
|
||||||
|
{
|
||||||
|
public AnidubSettings Settings { get; set; }
|
||||||
|
public IndexerCapabilities Capabilities { get; set; }
|
||||||
|
|
||||||
|
public AnidubRequestGenerator()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
private IEnumerable<IndexerRequest> GetPagedRequests(string term, int[] categories)
|
||||||
|
{
|
||||||
|
var requestUrl = string.Empty;
|
||||||
|
var isSearch = !string.IsNullOrWhiteSpace(term);
|
||||||
|
|
||||||
|
if (isSearch)
|
||||||
|
{
|
||||||
|
requestUrl = string.Format("{0}/index.php?do=search", Settings.BaseUrl.TrimEnd('/'));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
requestUrl = Settings.BaseUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
var request = new IndexerRequest(requestUrl, HttpAccept.Html);
|
||||||
|
|
||||||
|
if (isSearch)
|
||||||
|
{
|
||||||
|
request.HttpRequest.Method = NzbDrone.Common.Http.HttpMethod.POST;
|
||||||
|
var postData = new NameValueCollection
|
||||||
|
{
|
||||||
|
{ "do", "search" },
|
||||||
|
{ "subaction", "search" },
|
||||||
|
{ "search_start", "1" },
|
||||||
|
{ "full_search", "1" },
|
||||||
|
{ "result_from", "1" },
|
||||||
|
|
||||||
|
// Remove season and episode info from search term cause it breaks search
|
||||||
|
{ "story", Regex.Replace(term, @"(?:[SsEe]?\d{1,4}){1,2}$", "").TrimEnd() },
|
||||||
|
{ "titleonly", "3" },
|
||||||
|
{ "searchuser", "" },
|
||||||
|
{ "replyless", "0" },
|
||||||
|
{ "replylimit", "0" },
|
||||||
|
{ "searchdate", "0" },
|
||||||
|
{ "beforeafter", "after" },
|
||||||
|
{ "sortby", "" },
|
||||||
|
{ "resorder", "desc" },
|
||||||
|
{ "showposts", "1" },
|
||||||
|
{ "catlist[]", "0" }
|
||||||
|
};
|
||||||
|
var headers = new NameValueCollection
|
||||||
|
{
|
||||||
|
{ "Content-Type", "application/x-www-form-urlencoded" }
|
||||||
|
};
|
||||||
|
|
||||||
|
request.HttpRequest.SetContent(postData.GetQueryString());
|
||||||
|
request.HttpRequest.Headers.Add(headers);
|
||||||
|
}
|
||||||
|
|
||||||
|
yield return request;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IndexerPageableRequestChain GetSearchRequests(MovieSearchCriteria searchCriteria)
|
||||||
|
{
|
||||||
|
var pageableRequests = new IndexerPageableRequestChain();
|
||||||
|
|
||||||
|
pageableRequests.Add(GetPagedRequests(string.Format("{0}", searchCriteria.SanitizedSearchTerm), searchCriteria.Categories));
|
||||||
|
|
||||||
|
return pageableRequests;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IndexerPageableRequestChain GetSearchRequests(TvSearchCriteria searchCriteria)
|
||||||
|
{
|
||||||
|
var pageableRequests = new IndexerPageableRequestChain();
|
||||||
|
|
||||||
|
pageableRequests.Add(GetPagedRequests(string.Format("{0}", searchCriteria.SanitizedTvSearchString), searchCriteria.Categories));
|
||||||
|
|
||||||
|
return pageableRequests;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IndexerPageableRequestChain GetSearchRequests(BasicSearchCriteria searchCriteria)
|
||||||
|
{
|
||||||
|
var pageableRequests = new IndexerPageableRequestChain();
|
||||||
|
|
||||||
|
pageableRequests.Add(GetPagedRequests(string.Format("{0}", searchCriteria.SanitizedSearchTerm), searchCriteria.Categories));
|
||||||
|
|
||||||
|
return pageableRequests;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IndexerPageableRequestChain GetSearchRequests(MusicSearchCriteria searchCriteria)
|
||||||
|
{
|
||||||
|
var pageableRequests = new IndexerPageableRequestChain();
|
||||||
|
|
||||||
|
pageableRequests.Add(GetPagedRequests(string.Format("{0}", searchCriteria.SanitizedSearchTerm), searchCriteria.Categories));
|
||||||
|
|
||||||
|
return pageableRequests;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IndexerPageableRequestChain GetSearchRequests(BookSearchCriteria searchCriteria)
|
||||||
|
{
|
||||||
|
var pageableRequests = new IndexerPageableRequestChain();
|
||||||
|
|
||||||
|
pageableRequests.Add(GetPagedRequests(string.Format("{0}", searchCriteria.SanitizedSearchTerm), searchCriteria.Categories));
|
||||||
|
|
||||||
|
return pageableRequests;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Func<IDictionary<string, string>> GetCookies { get; set; }
|
||||||
|
public Action<IDictionary<string, string>, DateTime?> CookiesUpdater { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class AnidubParser : IParseIndexerResponse
|
||||||
|
{
|
||||||
|
private readonly AnidubSettings _settings;
|
||||||
|
private readonly IndexerCapabilitiesCategories _categories;
|
||||||
|
public IHttpClient HttpClient { get; set; }
|
||||||
|
public Logger Logger { get; set; }
|
||||||
|
|
||||||
|
private static Dictionary<string, string> CategoriesMap => new Dictionary<string, string>
|
||||||
|
{
|
||||||
|
{ "/anime_tv/full", "14" },
|
||||||
|
{ "/anime_tv/anime_ongoing", "10" },
|
||||||
|
{ "/anime_tv/shonen", "11" },
|
||||||
|
{ "/anime_tv", "2" },
|
||||||
|
{ "/xxx", "13" },
|
||||||
|
{ "/manga", "15" },
|
||||||
|
{ "/ost", "16" },
|
||||||
|
{ "/podcast", "17" },
|
||||||
|
{ "/anime_movie", "3" },
|
||||||
|
{ "/anime_ova/anime_ona", "5" },
|
||||||
|
{ "/anime_ova", "4" },
|
||||||
|
{ "/dorama/japan_dorama", "6" },
|
||||||
|
{ "/dorama/korea_dorama", "7" },
|
||||||
|
{ "/dorama/china_dorama", "8" },
|
||||||
|
{ "/dorama", "9" },
|
||||||
|
{ "/anons_ongoing", "12" }
|
||||||
|
};
|
||||||
|
|
||||||
|
public AnidubParser(AnidubSettings settings, IndexerCapabilitiesCategories categories)
|
||||||
|
{
|
||||||
|
_settings = settings;
|
||||||
|
_categories = categories;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetTitle(AngleSharp.Html.Dom.IHtmlDocument content, AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
var domTitle = content.QuerySelector("#news-title");
|
||||||
|
var baseTitle = domTitle.TextContent.Trim();
|
||||||
|
var quality = GetQuality(tabNode.ParentElement);
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(quality))
|
||||||
|
{
|
||||||
|
return $"{baseTitle} [{quality}]";
|
||||||
|
}
|
||||||
|
|
||||||
|
return baseTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetQuality(AngleSharp.Dom.IElement releaseNode)
|
||||||
|
{
|
||||||
|
// For some releases there's no block with quality
|
||||||
|
if (string.IsNullOrWhiteSpace(releaseNode.Id))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var quality = releaseNode.Id.Trim();
|
||||||
|
switch (quality.ToLowerInvariant())
|
||||||
|
{
|
||||||
|
case "tv720":
|
||||||
|
return "HDTV 720p";
|
||||||
|
case "tv1080":
|
||||||
|
return "HDTV 1080p";
|
||||||
|
case "bd720":
|
||||||
|
return "BDRip 720p";
|
||||||
|
case "bd1080":
|
||||||
|
return "BDRip 1080p";
|
||||||
|
case "hwp":
|
||||||
|
return "SDTV";
|
||||||
|
default:
|
||||||
|
return quality.ToUpperInvariant();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int GetReleaseLeechers(AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
const string LeechersSelector = ".list.down > .li_swing_m";
|
||||||
|
|
||||||
|
var leechersStr = tabNode.QuerySelector(LeechersSelector).TextContent;
|
||||||
|
int.TryParse(leechersStr, out var leechers);
|
||||||
|
return leechers;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int GetReleaseSeeders(AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
const string SeedersSelector = ".list.down > .li_distribute_m";
|
||||||
|
|
||||||
|
var seedersStr = tabNode.QuerySelector(SeedersSelector).TextContent;
|
||||||
|
int.TryParse(seedersStr, out var seeders);
|
||||||
|
return seeders;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int GetReleaseGrabs(AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
const string GrabsSelector = ".list.down > .li_download_m";
|
||||||
|
|
||||||
|
var grabsStr = tabNode.QuerySelector(GrabsSelector).TextContent;
|
||||||
|
int.TryParse(grabsStr, out var grabs);
|
||||||
|
return grabs;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetDateFromDocument(AngleSharp.Html.Dom.IHtmlDocument content)
|
||||||
|
{
|
||||||
|
const string DateSelector = ".story_inf > li:nth-child(2)";
|
||||||
|
|
||||||
|
var domDate = content.QuerySelector(DateSelector).LastChild;
|
||||||
|
|
||||||
|
if (domDate?.NodeName != "#text")
|
||||||
|
{
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
return domDate.NodeValue.Trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
private DateTime GetDateFromShowPage(AngleSharp.Html.Dom.IHtmlDocument content)
|
||||||
|
{
|
||||||
|
const string dateFormat = "d-MM-yyyy";
|
||||||
|
const string dateTimeFormat = dateFormat + ", HH:mm";
|
||||||
|
|
||||||
|
// Would be better to use AssumeLocal and provide "ru-RU" culture,
|
||||||
|
// but doesn't work cross-platform
|
||||||
|
const DateTimeStyles style = DateTimeStyles.AssumeUniversal;
|
||||||
|
|
||||||
|
var culture = CultureInfo.InvariantCulture;
|
||||||
|
|
||||||
|
var dateText = GetDateFromDocument(content);
|
||||||
|
|
||||||
|
//Correct way but will not always work on cross-platform
|
||||||
|
//var localTimeZone = TimeZoneInfo.FindSystemTimeZoneById("Russian Standard Time");
|
||||||
|
//var nowLocal = TimeZoneInfo.ConvertTime(DateTime.UtcNow, localTimeZone);
|
||||||
|
|
||||||
|
// Russian Standard Time is +03:00, no DST
|
||||||
|
const int russianStandardTimeDiff = 3;
|
||||||
|
var nowLocal = DateTime.UtcNow.AddHours(russianStandardTimeDiff);
|
||||||
|
|
||||||
|
dateText = dateText
|
||||||
|
.Replace("Вчера", nowLocal.AddDays(-1).ToString(dateFormat))
|
||||||
|
.Replace("Сегодня", nowLocal.ToString(dateFormat));
|
||||||
|
|
||||||
|
if (DateTime.TryParseExact(dateText, dateTimeFormat, culture, style, out var date))
|
||||||
|
{
|
||||||
|
var utcDate = date.ToUniversalTime();
|
||||||
|
return utcDate.AddHours(-russianStandardTimeDiff);
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger.Warn($"[AniDub] Date time couldn't be parsed on. Date text: {dateText}");
|
||||||
|
|
||||||
|
return DateTime.UtcNow;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static long GetReleaseSize(AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
const string SizeSelector = ".list.down > .red";
|
||||||
|
|
||||||
|
var sizeStr = tabNode.QuerySelector(SizeSelector).TextContent;
|
||||||
|
return ReleaseInfo.GetBytes(sizeStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
private string GetReleaseLink(AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
return $"{_settings.BaseUrl}engine/download.php?id={GetTorrentId(tabNode)}";
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string GetTorrentId(AngleSharp.Dom.IElement tabNode)
|
||||||
|
{
|
||||||
|
var nodeId = tabNode.Id;
|
||||||
|
|
||||||
|
// Format is "torrent_{id}_info"
|
||||||
|
return nodeId
|
||||||
|
.Replace("torrent_", string.Empty)
|
||||||
|
.Replace("_info", string.Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
private ICollection<IndexerCategory> ParseCategories(string uriPath)
|
||||||
|
{
|
||||||
|
var categoriesMap = CategoriesMap;
|
||||||
|
|
||||||
|
return categoriesMap
|
||||||
|
.Where(categoryMap => uriPath.StartsWith(categoryMap.Key))
|
||||||
|
.Select(categoryMap => _categories.MapTrackerCatToNewznab(categoryMap.Value))
|
||||||
|
.FirstOrDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
private IList<TorrentInfo> ParseRelease(IndexerResponse indexerResponse)
|
||||||
|
{
|
||||||
|
var torrentInfos = new List<TorrentInfo>();
|
||||||
|
var parser = new HtmlParser();
|
||||||
|
var dom = parser.ParseDocument(indexerResponse.Content);
|
||||||
|
|
||||||
|
foreach (var t in dom.QuerySelectorAll("#tabs .torrent_c > div"))
|
||||||
|
{
|
||||||
|
var release = new TorrentInfo
|
||||||
|
{
|
||||||
|
Title = GetTitle(dom, t),
|
||||||
|
InfoUrl = indexerResponse.Request.Url.ToString(),
|
||||||
|
DownloadVolumeFactor = 0,
|
||||||
|
UploadVolumeFactor = 1,
|
||||||
|
|
||||||
|
Guid = indexerResponse.Request.Url.ToString() + t.Id,
|
||||||
|
Seeders = GetReleaseSeeders(t),
|
||||||
|
Peers = GetReleaseSeeders(t) + GetReleaseLeechers(t),
|
||||||
|
Grabs = GetReleaseGrabs(t),
|
||||||
|
Categories = ParseCategories(indexerResponse.Request.Url.Path),
|
||||||
|
PublishDate = GetDateFromShowPage(dom),
|
||||||
|
DownloadUrl = GetReleaseLink(t),
|
||||||
|
Size = GetReleaseSize(t),
|
||||||
|
Resolution = GetQuality(t)
|
||||||
|
};
|
||||||
|
torrentInfos.Add(release);
|
||||||
|
}
|
||||||
|
|
||||||
|
return torrentInfos;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IList<ReleaseInfo> ParseResponse(IndexerResponse indexerResponse)
|
||||||
|
{
|
||||||
|
var torrentInfos = new List<ReleaseInfo>();
|
||||||
|
|
||||||
|
var parser = new HtmlParser();
|
||||||
|
var dom = parser.ParseDocument(indexerResponse.Content);
|
||||||
|
var domQuery = string.Empty;
|
||||||
|
|
||||||
|
if (indexerResponse.Request.Url.Query.Contains("do=search"))
|
||||||
|
{
|
||||||
|
domQuery = ".searchitem > h3 > a";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
domQuery = "#dle-content > .story > .story_h > .lcol > h2 > a";
|
||||||
|
}
|
||||||
|
|
||||||
|
var links = dom.QuerySelectorAll(domQuery);
|
||||||
|
foreach (var link in links)
|
||||||
|
{
|
||||||
|
var url = link.GetAttribute("href");
|
||||||
|
|
||||||
|
var releaseRequest = new IndexerRequest(url, HttpAccept.Html);
|
||||||
|
var releaseResponse = new IndexerResponse(releaseRequest, HttpClient.Execute(releaseRequest.HttpRequest));
|
||||||
|
|
||||||
|
// Throw common http errors here before we try to parse
|
||||||
|
if (releaseResponse.HttpResponse.HasHttpError)
|
||||||
|
{
|
||||||
|
if ((int)releaseResponse.HttpResponse.StatusCode == 429)
|
||||||
|
{
|
||||||
|
throw new TooManyRequestsException(releaseRequest.HttpRequest, releaseResponse.HttpResponse);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
throw new IndexerException(releaseResponse, "Http error code: " + releaseResponse.HttpResponse.StatusCode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
torrentInfos.AddRange(ParseRelease(releaseResponse));
|
||||||
|
}
|
||||||
|
|
||||||
|
return torrentInfos.ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Action<IDictionary<string, string>, DateTime?> CookiesUpdater { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class AnidubSettingsValidator : AbstractValidator<AnidubSettings>
|
||||||
|
{
|
||||||
|
public AnidubSettingsValidator()
|
||||||
|
{
|
||||||
|
RuleFor(c => c.Username).NotEmpty();
|
||||||
|
RuleFor(c => c.Password).NotEmpty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class AnidubSettings : IIndexerSettings
|
||||||
|
{
|
||||||
|
private static readonly AnidubSettingsValidator Validator = new AnidubSettingsValidator();
|
||||||
|
|
||||||
|
public AnidubSettings()
|
||||||
|
{
|
||||||
|
Username = "";
|
||||||
|
Password = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
[FieldDefinition(1, Label = "Base Url", Type = FieldType.Select, SelectOptionsProviderAction = "getUrls", HelpText = "Select which baseurl Prowlarr will use for requests to the site")]
|
||||||
|
public string BaseUrl { get; set; }
|
||||||
|
|
||||||
|
[FieldDefinition(2, Label = "Username", HelpText = "Site Username", Privacy = PrivacyLevel.UserName)]
|
||||||
|
public string Username { get; set; }
|
||||||
|
|
||||||
|
[FieldDefinition(3, Label = "Password", HelpText = "Site Password", Privacy = PrivacyLevel.Password, Type = FieldType.Password)]
|
||||||
|
public string Password { get; set; }
|
||||||
|
|
||||||
|
[FieldDefinition(4)]
|
||||||
|
public IndexerBaseSettings BaseSettings { get; set; } = new IndexerBaseSettings();
|
||||||
|
|
||||||
|
public NzbDroneValidationResult Validate()
|
||||||
|
{
|
||||||
|
return new NzbDroneValidationResult(Validator.Validate(this));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue