Jackett/src/Jackett.Common/Indexers/Abstract/AvistazTracker.cs

206 lines
8.7 KiB
C#
Raw Normal View History

2020-02-09 02:35:16 +00:00
using System;
2015-08-23 20:28:21 +00:00
using System.Collections.Generic;
using System.Collections.Specialized;
2015-08-23 20:28:21 +00:00
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
2015-08-23 20:28:21 +00:00
using System.Threading.Tasks;
using AngleSharp.Dom;
using AngleSharp.Html.Parser;
using Jackett.Common.Models;
using Jackett.Common.Models.IndexerConfig;
using Jackett.Common.Services.Interfaces;
using Jackett.Common.Utils;
using Jackett.Common.Utils.Clients;
using Newtonsoft.Json;
2015-08-23 20:28:21 +00:00
using Newtonsoft.Json.Linq;
using NLog;
namespace Jackett.Common.Indexers.Abstract
2015-08-23 20:28:21 +00:00
{
public abstract class AvistazTracker : BaseWebIndexer
2015-08-23 20:28:21 +00:00
{
private string LoginUrl => SiteLink + "auth/login";
private string SearchUrl => SiteLink + "torrents?";
private string IMDBSearch => SiteLink + "ajax/movies/3?term=";
2015-08-23 20:28:21 +00:00
private new ConfigurationDataBasicLogin configData
2015-08-23 20:28:21 +00:00
{
get => (ConfigurationDataBasicLogin)base.configData;
set => base.configData = value;
2015-08-23 20:28:21 +00:00
}
2018-04-06 12:32:46 +00:00
// hook to adjust the search term
protected string GetSearchTerm(TorznabQuery query) => $"{query.SearchTerm} {query.GetEpisodeSearchString()}";
2018-04-06 12:32:46 +00:00
protected AvistazTracker(string name, string link, string description, IIndexerConfigurationService configService,
WebClient client, Logger logger, IProtectionService p, TorznabCapabilities caps)
: base(name,
description: description,
link: link,
caps: caps,
configService: configService,
client: client,
logger: logger,
p: p,
configData: new ConfigurationDataBasicLogin())
2015-08-23 20:28:21 +00:00
{
Encoding = Encoding.UTF8;
2016-12-09 17:20:58 +00:00
Language = "en-us";
2015-08-23 20:28:21 +00:00
AddCategoryMapping(1, TorznabCatType.Movies);
AddCategoryMapping(1, TorznabCatType.MoviesForeign);
AddCategoryMapping(1, TorznabCatType.MoviesHD);
AddCategoryMapping(1, TorznabCatType.MoviesSD);
AddCategoryMapping(2, TorznabCatType.TV);
AddCategoryMapping(3, TorznabCatType.Audio);
}
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
2015-08-23 20:28:21 +00:00
{
LoadValuesFromJson(configJson);
2015-08-23 20:28:21 +00:00
var loginPage = await RequestStringWithCookies(LoginUrl, string.Empty);
2016-09-07 17:46:03 +00:00
var token = new Regex("<meta name=\"_token\" content=\"(.*?)\">").Match(loginPage.Content).Groups[1].ToString();
2015-08-23 20:28:21 +00:00
var pairs = new Dictionary<string, string> {
{ "_token", token },
2016-09-07 17:46:03 +00:00
{ "email_username", configData.Username.Value },
2015-08-23 20:28:21 +00:00
{ "password", configData.Password.Value },
2016-09-07 17:46:03 +00:00
{ "remember", "1" }
};
2015-08-23 20:28:21 +00:00
var result = await RequestLoginAndFollowRedirect(LoginUrl, pairs, loginPage.Cookies, true, null, LoginUrl);
await ConfigureIfOK(result.Cookies, result.Content != null && result.Content.Contains("auth/logout"), () =>
{
var parser = new HtmlParser();
var dom = parser.ParseDocument(result.Content);
var messageEl = dom.QuerySelector(".form-error");
2015-08-23 20:28:21 +00:00
var errorMessage = messageEl.Text().Trim();
throw new ExceptionWithConfigData(errorMessage, configData);
});
return IndexerConfigurationStatus.RequiresTesting;
}
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
2015-08-23 20:28:21 +00:00
{
var releases = new List<ReleaseInfo>();
var categoryMapping = MapTorznabCapsToTrackers(query).Distinct().ToList();
var qc = new NameValueCollection
2015-08-23 20:28:21 +00:00
{
{"in", "1"},
{"type", categoryMapping.Any() ? categoryMapping.First() : "0"} // type=0 => all categories
};
2015-08-23 20:28:21 +00:00
// imdb search
if (!string.IsNullOrWhiteSpace(query.ImdbID))
{
var movieId = await GetMovieId(query.ImdbID);
if (movieId == null)
return releases; // movie not found or service broken => return 0 results
qc.Add("movie_id", movieId);
}
else
qc.Add("search", GetSearchTerm(query));
2015-08-23 20:28:21 +00:00
var episodeSearchUrl = SearchUrl + qc.GetQueryString();
2015-08-23 20:28:21 +00:00
var response = await RequestStringWithCookiesAndRetry(episodeSearchUrl);
if (response.IsRedirect)
{
// re-login
await ApplyConfiguration(null);
response = await RequestStringWithCookiesAndRetry(episodeSearchUrl);
2017-03-07 12:46:43 +00:00
}
2015-08-23 20:28:21 +00:00
try
{
var parser = new HtmlParser();
var dom = parser.ParseDocument(response.Content);
var rows = dom.QuerySelectorAll("table:has(thead) > tbody > tr");
2015-08-23 20:28:21 +00:00
foreach (var row in rows)
{
var release = new ReleaseInfo
{
MinimumRatio = 1,
MinimumSeedTime = 172800 // 48 hours
};
2015-08-23 20:28:21 +00:00
var qLink = row.QuerySelector("a.torrent-filename");
2015-08-23 20:28:21 +00:00
release.Title = qLink.Text().Trim();
release.Comments = new Uri(qLink.GetAttribute("href"));
2015-08-23 20:28:21 +00:00
release.Guid = release.Comments;
var qDownload = row.QuerySelector("a.torrent-download-icon");
release.Link = new Uri(qDownload.GetAttribute("href"));
2015-08-23 20:28:21 +00:00
var qBanner = row.QuerySelector("img.img-tor-poster")?.GetAttribute("data-poster-mid");
if (qBanner != null)
release.BannerUrl = new Uri(qBanner);
var dateStr = row.QuerySelector("td:nth-of-type(4) > span").Text().Trim();
2015-08-23 20:28:21 +00:00
release.PublishDate = DateTimeUtil.FromTimeAgo(dateStr);
var sizeStr = row.QuerySelector("td:nth-of-type(6) > span").Text().Trim();
2015-08-23 20:28:21 +00:00
release.Size = ReleaseInfo.GetBytes(sizeStr);
release.Seeders = ParseUtil.CoerceInt(row.QuerySelector("td:nth-of-type(7)").Text().Trim());
release.Peers = ParseUtil.CoerceInt(row.QuerySelector("td:nth-of-type(8)").Text().Trim()) + release.Seeders;
2015-08-23 20:28:21 +00:00
var cat = row.QuerySelectorAll("td:nth-of-type(1) i").First().GetAttribute("class")
2016-09-07 17:46:03 +00:00
.Replace("torrent-icon", string.Empty)
.Replace("fa fa-", string.Empty)
.Replace("film", "1")
.Replace("tv", "2")
.Replace("music", "3")
2015-08-23 20:28:21 +00:00
.Replace("text-pink", string.Empty);
release.Category = MapTrackerCatToNewznab(cat.Trim());
var grabs = row.QuerySelector("td:nth-child(9)").Text();
release.Grabs = ParseUtil.CoerceInt(grabs);
if (row.QuerySelectorAll("i.fa-star").Any())
release.DownloadVolumeFactor = 0;
else if (row.QuerySelectorAll("i.fa-star-half-o").Any())
release.DownloadVolumeFactor = 0.5;
else
release.DownloadVolumeFactor = 1;
if (row.QuerySelectorAll("i.fa-diamond").Any())
release.UploadVolumeFactor = 2;
else
release.UploadVolumeFactor = 1;
2015-08-23 20:28:21 +00:00
releases.Add(release);
}
}
catch (Exception ex)
{
OnParseError(response.Content, ex);
}
return releases;
}
private async Task<string> GetMovieId(string imdbId)
{
try
{
var imdbUrl = IMDBSearch + imdbId;
var imdbHeaders = new Dictionary<string, string> { { "X-Requested-With", "XMLHttpRequest" } };
var imdbResponse = await RequestStringWithCookiesAndRetry(imdbUrl, null, null, imdbHeaders);
if (imdbResponse.IsRedirect)
{
// re-login
await ApplyConfiguration(null);
imdbResponse = await RequestStringWithCookiesAndRetry(imdbUrl, null, null, imdbHeaders);
}
var json = JsonConvert.DeserializeObject<dynamic>(imdbResponse.Content);
return (string)((JArray)json["data"])[0]["id"];
}
catch (Exception)
{
return null;
}
}
2015-08-23 20:28:21 +00:00
}
2020-02-09 02:35:16 +00:00
}