2017-10-29 06:50:47 +00:00
|
|
|
|
using System;
|
2017-04-15 08:45:10 +00:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Text.RegularExpressions;
|
2017-10-29 06:50:47 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2017-04-15 08:45:10 +00:00
|
|
|
|
using System.Xml.Linq;
|
2017-10-29 06:50:47 +00:00
|
|
|
|
using CsQuery;
|
2018-03-10 08:05:56 +00:00
|
|
|
|
using Jackett.Common.Models;
|
|
|
|
|
using Jackett.Common.Models.IndexerConfig;
|
|
|
|
|
using Jackett.Common.Services.Interfaces;
|
|
|
|
|
using Jackett.Common.Utils;
|
|
|
|
|
using Jackett.Common.Utils.Clients;
|
2017-10-29 06:50:47 +00:00
|
|
|
|
using Newtonsoft.Json.Linq;
|
|
|
|
|
using NLog;
|
2018-03-10 08:05:56 +00:00
|
|
|
|
using static Jackett.Common.Models.IndexerConfig.ConfigurationData;
|
|
|
|
|
using static Jackett.Common.Utils.ParseUtil;
|
2017-04-15 08:45:10 +00:00
|
|
|
|
|
2018-03-10 08:05:56 +00:00
|
|
|
|
namespace Jackett.Common.Indexers
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
2017-07-10 20:58:44 +00:00
|
|
|
|
public class XSpeeds : BaseWebIndexer
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
2017-10-29 06:50:47 +00:00
|
|
|
|
private string LandingUrl => SiteLink + "login.php";
|
|
|
|
|
private string LoginUrl => SiteLink + "takelogin.php";
|
|
|
|
|
private string GetRSSKeyUrl => SiteLink + "getrss.php";
|
|
|
|
|
private string SearchUrl => SiteLink + "browse.php";
|
|
|
|
|
private string RSSUrl => SiteLink + "rss.php?secret_key={0}&feedtype=download&timezone=0&showrows=50&categories=all";
|
|
|
|
|
private string CommentUrl => SiteLink + "details.php?id={0}";
|
|
|
|
|
private string DownloadUrl => SiteLink + "download.php?id={0}";
|
2017-04-15 08:45:10 +00:00
|
|
|
|
|
2017-10-29 06:50:47 +00:00
|
|
|
|
private new ConfigurationDataBasicLoginWithRSSAndDisplay configData
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
2017-10-29 06:50:47 +00:00
|
|
|
|
get { return (ConfigurationDataBasicLoginWithRSSAndDisplay)base.configData; }
|
2017-04-15 08:45:10 +00:00
|
|
|
|
set { base.configData = value; }
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-05 09:42:03 +00:00
|
|
|
|
public XSpeeds(IIndexerConfigurationService configService, WebClient wc, Logger l, IProtectionService ps)
|
2017-04-15 08:45:10 +00:00
|
|
|
|
: base(name: "XSpeeds",
|
2017-10-01 09:26:07 +00:00
|
|
|
|
description: "XSpeeds (XS) is a Private Torrent Tracker for MOVIES / TV / GENERAL",
|
2017-04-15 08:45:10 +00:00
|
|
|
|
link: "https://www.xspeeds.eu/",
|
|
|
|
|
caps: TorznabUtil.CreateDefaultTorznabTVCaps(),
|
2017-07-10 20:58:44 +00:00
|
|
|
|
configService: configService,
|
2017-04-15 08:45:10 +00:00
|
|
|
|
client: wc,
|
|
|
|
|
logger: l,
|
|
|
|
|
p: ps,
|
|
|
|
|
configData: new ConfigurationDataBasicLoginWithRSSAndDisplay())
|
|
|
|
|
{
|
2017-11-05 09:42:03 +00:00
|
|
|
|
Encoding = Encoding.UTF8;
|
2017-04-15 08:45:10 +00:00
|
|
|
|
Language = "en-us";
|
|
|
|
|
Type = "private";
|
|
|
|
|
|
|
|
|
|
configData.DisplayText.Value = "Expect an initial delay (often around 10 seconds) due to XSpeeds CloudFlare DDoS protection";
|
|
|
|
|
configData.DisplayText.Name = "Notice";
|
|
|
|
|
|
2016-11-04 17:59:58 +00:00
|
|
|
|
AddCategoryMapping(70, TorznabCatType.TVAnime); // Anime
|
|
|
|
|
AddCategoryMapping(4, TorznabCatType.PC); // Apps
|
|
|
|
|
AddCategoryMapping(82, TorznabCatType.PCMac); // Mac
|
|
|
|
|
AddCategoryMapping(80, TorznabCatType.AudioAudiobook); // Audiobooks
|
|
|
|
|
AddCategoryMapping(66, TorznabCatType.MoviesBluRay); // Blu-Ray
|
|
|
|
|
AddCategoryMapping(48, TorznabCatType.Books); // Books Magazines
|
|
|
|
|
AddCategoryMapping(68, TorznabCatType.MoviesOther); // Cams/TS
|
|
|
|
|
AddCategoryMapping(65, TorznabCatType.TVDocumentary); // Documentaries
|
|
|
|
|
AddCategoryMapping(10, TorznabCatType.MoviesDVD); // DVDR
|
|
|
|
|
AddCategoryMapping(72, TorznabCatType.MoviesForeign); // Foreign
|
|
|
|
|
AddCategoryMapping(74, TorznabCatType.TVOTHER); // Kids
|
|
|
|
|
AddCategoryMapping(44, TorznabCatType.TVSport); // MMA
|
|
|
|
|
AddCategoryMapping(11, TorznabCatType.Movies); // Movie Boxsets
|
|
|
|
|
AddCategoryMapping(12, TorznabCatType.Movies); // Movies
|
|
|
|
|
AddCategoryMapping(13, TorznabCatType.Audio); // Music
|
|
|
|
|
AddCategoryMapping(15, TorznabCatType.AudioVideo); // Music Videos
|
|
|
|
|
AddCategoryMapping(32, TorznabCatType.ConsoleNDS); // NDS Games
|
|
|
|
|
AddCategoryMapping(9, TorznabCatType.Other); // Other
|
|
|
|
|
AddCategoryMapping(6, TorznabCatType.PCGames); // PC Games
|
|
|
|
|
AddCategoryMapping(45, TorznabCatType.Other); // Pictures
|
|
|
|
|
AddCategoryMapping(31, TorznabCatType.ConsolePS4); // Playstation
|
|
|
|
|
AddCategoryMapping(71, TorznabCatType.TV); // PPV
|
|
|
|
|
AddCategoryMapping(54, TorznabCatType.TV); // Soaps
|
|
|
|
|
AddCategoryMapping(20, TorznabCatType.TVSport); // Sports
|
2017-04-15 08:45:10 +00:00
|
|
|
|
AddCategoryMapping(86, TorznabCatType.TVSport); // MotorSports
|
|
|
|
|
AddCategoryMapping(89, TorznabCatType.TVSport); // Olympics 2016
|
2016-11-04 17:59:58 +00:00
|
|
|
|
AddCategoryMapping(88, TorznabCatType.TVSport); // World Cup
|
|
|
|
|
AddCategoryMapping(83, TorznabCatType.Movies); // TOTM
|
|
|
|
|
AddCategoryMapping(21, TorznabCatType.TVSD); // TV Boxsets
|
|
|
|
|
AddCategoryMapping(76, TorznabCatType.TVHD); // HD Boxsets
|
|
|
|
|
AddCategoryMapping(47, TorznabCatType.TVHD); // TV-HD
|
|
|
|
|
AddCategoryMapping(16, TorznabCatType.TVSD); // TV-SD
|
|
|
|
|
AddCategoryMapping(7, TorznabCatType.ConsoleWii); // Wii Games
|
|
|
|
|
AddCategoryMapping(43, TorznabCatType.TVSport); // Wrestling
|
2017-04-15 08:45:10 +00:00
|
|
|
|
AddCategoryMapping(8, TorznabCatType.ConsoleXbox); // Xbox Games
|
|
|
|
|
|
|
|
|
|
// RSS Textual categories
|
|
|
|
|
AddCategoryMapping("Anime", TorznabCatType.TVAnime);
|
|
|
|
|
AddCategoryMapping("Apps", TorznabCatType.PC);
|
|
|
|
|
AddCategoryMapping("Mac", TorznabCatType.PCMac);
|
|
|
|
|
AddCategoryMapping("Audiobooks", TorznabCatType.AudioAudiobook);
|
|
|
|
|
AddCategoryMapping("Blu-Ray", TorznabCatType.MoviesBluRay);
|
|
|
|
|
AddCategoryMapping("Books Magazines", TorznabCatType.Books);
|
|
|
|
|
AddCategoryMapping("Cams/TS", TorznabCatType.MoviesOther);
|
|
|
|
|
AddCategoryMapping("Documentaries", TorznabCatType.TVDocumentary);
|
|
|
|
|
AddCategoryMapping("DVDR", TorznabCatType.MoviesDVD);
|
|
|
|
|
AddCategoryMapping("Foreign", TorznabCatType.MoviesForeign);
|
|
|
|
|
AddCategoryMapping("Kids", TorznabCatType.TVOTHER);
|
|
|
|
|
AddCategoryMapping("MMA", TorznabCatType.TVSport);
|
|
|
|
|
AddCategoryMapping("Movie Boxsets", TorznabCatType.Movies);
|
|
|
|
|
AddCategoryMapping("Movies", TorznabCatType.Movies);
|
|
|
|
|
AddCategoryMapping("Music", TorznabCatType.Audio);
|
|
|
|
|
AddCategoryMapping("Music Videos", TorznabCatType.AudioVideo);
|
|
|
|
|
AddCategoryMapping("NDS Games", TorznabCatType.ConsoleNDS);
|
|
|
|
|
AddCategoryMapping("Other", TorznabCatType.Other);
|
|
|
|
|
AddCategoryMapping("PC Games", TorznabCatType.PCGames);
|
|
|
|
|
AddCategoryMapping("Pictures", TorznabCatType.Other);
|
|
|
|
|
AddCategoryMapping("Playstation", TorznabCatType.ConsolePS4);
|
|
|
|
|
AddCategoryMapping("PPV", TorznabCatType.TV);
|
|
|
|
|
AddCategoryMapping("Soaps", TorznabCatType.TV);
|
|
|
|
|
AddCategoryMapping("Sports", TorznabCatType.TVSport);
|
|
|
|
|
AddCategoryMapping("MotorSports", TorznabCatType.TVSport);
|
|
|
|
|
AddCategoryMapping("Olympics 2016", TorznabCatType.TVSport);
|
|
|
|
|
AddCategoryMapping("World Cup", TorznabCatType.TVSport);
|
|
|
|
|
AddCategoryMapping("TOTM", TorznabCatType.Movies);
|
|
|
|
|
AddCategoryMapping("TV Boxsets", TorznabCatType.TVSD);
|
|
|
|
|
AddCategoryMapping("HD Boxsets", TorznabCatType.TVHD);
|
|
|
|
|
AddCategoryMapping("TV-HD", TorznabCatType.TVHD);
|
|
|
|
|
AddCategoryMapping("TV-SD", TorznabCatType.TVSD);
|
|
|
|
|
AddCategoryMapping("Wii Games", TorznabCatType.ConsoleWii);
|
|
|
|
|
AddCategoryMapping("Wrestling", TorznabCatType.TVSport);
|
|
|
|
|
AddCategoryMapping("Xbox Games", TorznabCatType.ConsoleXbox);
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-21 13:06:58 +00:00
|
|
|
|
public override async Task<ConfigurationData> GetConfigurationForSetup()
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
|
|
|
|
var loginPage = await RequestStringWithCookies(LandingUrl);
|
|
|
|
|
CQ dom = loginPage.Content;
|
|
|
|
|
CQ qCaptchaImg = dom.Find("img#regimage").First();
|
|
|
|
|
if (qCaptchaImg.Length > 0)
|
2017-10-29 06:50:47 +00:00
|
|
|
|
{
|
2017-02-21 13:06:58 +00:00
|
|
|
|
var CaptchaUrl = qCaptchaImg.Attr("src");
|
|
|
|
|
var captchaImage = await RequestBytesWithCookies(CaptchaUrl, loginPage.Cookies, RequestType.GET, LandingUrl);
|
2017-04-15 08:45:10 +00:00
|
|
|
|
|
|
|
|
|
var CaptchaImage = new ImageItem { Name = "Captcha Image" };
|
|
|
|
|
var CaptchaText = new StringItem { Name = "Captcha Text" };
|
|
|
|
|
|
|
|
|
|
CaptchaImage.Value = captchaImage.Content;
|
|
|
|
|
|
|
|
|
|
configData.AddDynamic("CaptchaImage", CaptchaImage);
|
|
|
|
|
configData.AddDynamic("CaptchaText", CaptchaText);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
logger.Debug(string.Format("{0}: No captcha image found", ID));
|
2017-02-21 13:06:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return configData;
|
2017-04-15 08:45:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-06-28 05:31:38 +00:00
|
|
|
|
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
|
|
|
|
LoadValuesFromJson(configJson);
|
|
|
|
|
|
|
|
|
|
var pairs = new Dictionary<string, string>
|
|
|
|
|
{
|
|
|
|
|
{"username", configData.Username.Value},
|
|
|
|
|
{"password", configData.Password.Value}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var CaptchaText = (StringItem)configData.GetDynamic("CaptchaText");
|
|
|
|
|
if (CaptchaText != null)
|
|
|
|
|
{
|
|
|
|
|
pairs.Add("imagestring", CaptchaText.Value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//var result = await RequestLoginAndFollowRedirect(LoginUrl, pairs, null, true, null, SiteLink, true);
|
|
|
|
|
var result = await RequestLoginAndFollowRedirect(LoginUrl, pairs, null, true, SearchUrl, LandingUrl, true);
|
|
|
|
|
await ConfigureIfOK(result.Cookies, result.Content != null && result.Content.Contains("logout.php"),
|
|
|
|
|
() =>
|
|
|
|
|
{
|
|
|
|
|
CQ dom = result.Content;
|
|
|
|
|
var errorMessage = dom[".left_side table:eq(0) tr:eq(1)"].Text().Trim().Replace("\n\t", " ");
|
|
|
|
|
throw new ExceptionWithConfigData(errorMessage, configData);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
// Get RSS key
|
|
|
|
|
var rssParams = new Dictionary<string, string>
|
|
|
|
|
{
|
|
|
|
|
{"feedtype", "download"},
|
|
|
|
|
{"timezone", "0"},
|
|
|
|
|
{"showrows", "50"}
|
|
|
|
|
};
|
|
|
|
|
var rssPage = await PostDataWithCookies(GetRSSKeyUrl, rssParams, result.Cookies);
|
|
|
|
|
var match = Regex.Match(rssPage.Content, "(?<=secret_key\\=)([a-zA-z0-9]*)");
|
|
|
|
|
configData.RSSKey.Value = match.Success ? match.Value : string.Empty;
|
|
|
|
|
if (string.IsNullOrWhiteSpace(configData.RSSKey.Value))
|
|
|
|
|
throw new Exception("Failed to get RSS Key");
|
|
|
|
|
SaveConfig();
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
IsConfigured = false;
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
return IndexerConfigurationStatus.RequiresTesting;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-03 05:15:47 +00:00
|
|
|
|
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
|
|
|
|
var releases = new List<ReleaseInfo>();
|
|
|
|
|
var searchString = query.GetQueryString();
|
|
|
|
|
var prevCook = CookieHeader + "";
|
|
|
|
|
|
|
|
|
|
// If we have no query use the RSS Page as their server is slow enough at times!
|
|
|
|
|
if (query.IsTest || string.IsNullOrWhiteSpace(searchString))
|
|
|
|
|
{
|
|
|
|
|
var rssPage = await RequestStringWithCookiesAndRetry(string.Format(RSSUrl, configData.RSSKey.Value));
|
|
|
|
|
try
|
|
|
|
|
{
|
2017-10-29 06:50:47 +00:00
|
|
|
|
if (rssPage.Content.EndsWith("\0"))
|
|
|
|
|
{
|
2017-04-15 08:45:10 +00:00
|
|
|
|
rssPage.Content = rssPage.Content.Substring(0, rssPage.Content.Length - 1);
|
|
|
|
|
}
|
|
|
|
|
rssPage.Content = RemoveInvalidXmlChars(rssPage.Content);
|
|
|
|
|
var rssDoc = XDocument.Parse(rssPage.Content);
|
|
|
|
|
|
|
|
|
|
foreach (var item in rssDoc.Descendants("item"))
|
|
|
|
|
{
|
|
|
|
|
var title = item.Descendants("title").First().Value;
|
|
|
|
|
var description = item.Descendants("description").First().Value;
|
|
|
|
|
var link = item.Descendants("link").First().Value;
|
|
|
|
|
var category = item.Descendants("category").First().Value;
|
|
|
|
|
var date = item.Descendants("pubDate").First().Value;
|
|
|
|
|
|
|
|
|
|
var torrentIdMatch = Regex.Match(link, "(?<=id=)(\\d)*");
|
|
|
|
|
var torrentId = torrentIdMatch.Success ? torrentIdMatch.Value : string.Empty;
|
|
|
|
|
if (string.IsNullOrWhiteSpace(torrentId))
|
|
|
|
|
throw new Exception("Missing torrent id");
|
|
|
|
|
|
|
|
|
|
var infoMatch = Regex.Match(description, @"Category:\W(?<cat>.*)\W\/\WSeeders:\W(?<seeders>[\d\,]*)\W\/\WLeechers:\W(?<leechers>[\d\,]*)\W\/\WSize:\W(?<size>[\d\.]*\W\S*)");
|
|
|
|
|
if (!infoMatch.Success)
|
|
|
|
|
throw new Exception("Unable to find info");
|
|
|
|
|
|
|
|
|
|
var release = new ReleaseInfo
|
|
|
|
|
{
|
|
|
|
|
Title = title,
|
|
|
|
|
Description = title,
|
|
|
|
|
Guid = new Uri(string.Format(DownloadUrl, torrentId)),
|
|
|
|
|
Comments = new Uri(string.Format(CommentUrl, torrentId)),
|
2017-10-29 06:50:47 +00:00
|
|
|
|
PublishDate = DateTime.ParseExact(date, "yyyy-MM-dd H:mm:ss", CultureInfo.InvariantCulture), //2015-08-08 21:20:31
|
2017-04-15 08:45:10 +00:00
|
|
|
|
Link = new Uri(string.Format(DownloadUrl, torrentId)),
|
|
|
|
|
Seeders = ParseUtil.CoerceInt(infoMatch.Groups["seeders"].Value),
|
|
|
|
|
Peers = ParseUtil.CoerceInt(infoMatch.Groups["leechers"].Value),
|
|
|
|
|
Size = ReleaseInfo.GetBytes(infoMatch.Groups["size"].Value),
|
|
|
|
|
Category = MapTrackerCatToNewznab(category)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
release.Peers += release.Seeders;
|
|
|
|
|
releases.Add(release);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
logger.Error("XSpeeds: Error while parsing the RSS feed:");
|
|
|
|
|
logger.Error(rssPage.Content);
|
|
|
|
|
throw ex;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (query.IsTest || !string.IsNullOrWhiteSpace(searchString))
|
|
|
|
|
{
|
|
|
|
|
if (searchString.Length < 3 && !query.IsTest)
|
|
|
|
|
{
|
|
|
|
|
OnParseError("", new Exception("Minimum search length is 3"));
|
|
|
|
|
return releases;
|
|
|
|
|
}
|
|
|
|
|
var searchParams = new Dictionary<string, string> {
|
|
|
|
|
{ "do", "search" },
|
|
|
|
|
{ "keywords", searchString },
|
|
|
|
|
{ "search_type", "t_name" },
|
|
|
|
|
{ "category", "0" },
|
|
|
|
|
{ "include_dead_torrents", "no" }
|
|
|
|
|
};
|
|
|
|
|
var pairs = new Dictionary<string, string> {
|
|
|
|
|
{ "username", configData.Username.Value },
|
|
|
|
|
{ "password", configData.Password.Value }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var searchPage = await PostDataWithCookiesAndRetry(SearchUrl, searchParams, CookieHeader);
|
|
|
|
|
// Occasionally the cookies become invalid, login again if that happens
|
|
|
|
|
if (searchPage.IsRedirect)
|
|
|
|
|
{
|
|
|
|
|
await ApplyConfiguration(null);
|
|
|
|
|
searchPage = await PostDataWithCookiesAndRetry(SearchUrl, searchParams, CookieHeader);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
CQ dom = searchPage.Content;
|
|
|
|
|
var rows = dom["table#sortabletable > tbody > tr:has(div > a[href*=\"details.php?id=\"])"];
|
|
|
|
|
foreach (var row in rows)
|
|
|
|
|
{
|
|
|
|
|
var release = new ReleaseInfo();
|
|
|
|
|
var qRow = row.Cq();
|
|
|
|
|
|
|
|
|
|
var qDetails = qRow.Find("div > a[href*=\"details.php?id=\"]"); // details link, release name get's shortened if it's to long
|
|
|
|
|
var qTitle = qRow.Find("td:eq(1) .tooltip-content div:eq(0)"); // use Title from tooltip
|
2017-10-29 06:50:47 +00:00
|
|
|
|
if (!qTitle.Any()) // fallback to Details link if there's no tooltip
|
2017-04-15 08:45:10 +00:00
|
|
|
|
{
|
|
|
|
|
qTitle = qDetails;
|
|
|
|
|
}
|
|
|
|
|
release.Title = qTitle.Text();
|
|
|
|
|
|
|
|
|
|
release.Guid = new Uri(qRow.Find("td:eq(2) a").Attr("href"));
|
|
|
|
|
release.Link = release.Guid;
|
|
|
|
|
release.Comments = new Uri(qDetails.Attr("href"));
|
2017-10-29 06:50:47 +00:00
|
|
|
|
release.PublishDate = DateTime.ParseExact(qRow.Find("td:eq(1) div").Last().Text().Trim(), "dd-MM-yyyy H:mm", CultureInfo.InvariantCulture, DateTimeStyles.AssumeUniversal); //08-08-2015 12:51
|
2017-04-15 08:45:10 +00:00
|
|
|
|
release.Seeders = ParseUtil.CoerceInt(qRow.Find("td:eq(6)").Text());
|
|
|
|
|
release.Peers = release.Seeders + ParseUtil.CoerceInt(qRow.Find("td:eq(7)").Text().Trim());
|
|
|
|
|
release.Size = ReleaseInfo.GetBytes(qRow.Find("td:eq(4)").Text().Trim());
|
|
|
|
|
|
|
|
|
|
var qBanner = qRow.Find("td:eq(1) .tooltip-content img").First();
|
|
|
|
|
if (qBanner.Length > 0)
|
|
|
|
|
release.BannerUrl = new Uri(qBanner.Attr("src"));
|
|
|
|
|
|
|
|
|
|
var cat = row.Cq().Find("td:eq(0) a").First().Attr("href");
|
|
|
|
|
var catSplit = cat.LastIndexOf('=');
|
|
|
|
|
if (catSplit > -1)
|
|
|
|
|
cat = cat.Substring(catSplit + 1);
|
|
|
|
|
release.Category = MapTrackerCatToNewznab(cat);
|
|
|
|
|
|
|
|
|
|
var grabs = qRow.Find("td:nth-child(6)").Text();
|
|
|
|
|
release.Grabs = ParseUtil.CoerceInt(grabs);
|
|
|
|
|
|
|
|
|
|
if (qRow.Find("img[alt^=\"Free Torrent\"]").Length >= 1)
|
|
|
|
|
release.DownloadVolumeFactor = 0;
|
|
|
|
|
else if (qRow.Find("img[alt^=\"Silver Torrent\"]").Length >= 1)
|
|
|
|
|
release.DownloadVolumeFactor = 0.5;
|
|
|
|
|
else
|
|
|
|
|
release.DownloadVolumeFactor = 1;
|
|
|
|
|
|
|
|
|
|
if (qRow.Find("img[alt^=\"x2 Torrent\"]").Length >= 1)
|
|
|
|
|
release.UploadVolumeFactor = 2;
|
|
|
|
|
else
|
|
|
|
|
release.UploadVolumeFactor = 1;
|
|
|
|
|
|
|
|
|
|
releases.Add(release);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
OnParseError(searchPage.Content, ex);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!CookieHeader.Trim().Equals(prevCook.Trim()))
|
|
|
|
|
{
|
|
|
|
|
SaveConfig();
|
|
|
|
|
}
|
|
|
|
|
return releases;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|