Jackett/src/Jackett.Common/Indexers/TorrentBytes.cs

204 lines
8.4 KiB
C#
Raw Normal View History

2017-10-29 06:50:47 +00:00
using System;
2015-07-30 20:09:38 +00:00
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Globalization;
using System.Text;
using System.Threading.Tasks;
2017-10-29 06:50:47 +00:00
using CsQuery;
using Jackett.Models;
using Jackett.Models.IndexerConfig;
Feature/netcore preparation (#2035) * Move to use package reference for restoring nuget packages. * Return a task result for this async method. * Update to a supported version of the .NET Framework. This also has the side effect of allowing us to automatically generate our binding redirects on build. * Set the solution to target VS2017 * Update test solution csproj file to support being built through MSBuild 15 * Move to use package reference for restoring nuget packages. * Return a task result for this async method. * Update to a supported version of the .NET Framework. This also has the side effect of allowing us to automatically generate our binding redirects on build. * Set the solution to target VS2017 * Update test solution csproj file to support being built through MSBuild 15 * DateTimeRoutines does not have Nuget packages that support .NET Standard (and therefore .NET Core). We will have to include them for now until we can get rid of this dependency. * Move the interfaces into their own files. This will be useful when we share them between the .NET Core and .NET Framework WebAPI * Stage services that need to point to the new interface namespace. * Update CurlSharp to fix memory leak issue and support better runtime compatibility with OSX and Linux * Start spliting some interfaces into their own files - this will help by allowing us to split them out in the future into a seperate project so the actual implementations can stay within their respective architectures when required
2017-10-29 10:19:09 +00:00
using Jackett.Services.Interfaces;
2017-10-29 06:50:47 +00:00
using Jackett.Utils;
using Jackett.Utils.Clients;
using Newtonsoft.Json.Linq;
using NLog;
2015-07-30 20:09:38 +00:00
namespace Jackett.Indexers
{
public class TorrentBytes : BaseWebIndexer
2015-07-30 20:09:38 +00:00
{
private string BrowseUrl { get { return SiteLink + "browse.php"; } }
private string LoginUrl { get { return SiteLink + "takelogin.php"; } }
2017-10-29 06:50:47 +00:00
private new ConfigurationDataBasicLogin configData
{
get { return (ConfigurationDataBasicLogin)base.configData; }
set { base.configData = value; }
}
public TorrentBytes(IIndexerConfigurationService configService, WebClient wc, Logger l, IProtectionService ps)
2015-07-30 20:09:38 +00:00
: base(name: "TorrentBytes",
description: "A decade of torrentbytes",
link: "https://www.torrentbytes.net/",
2015-08-11 18:48:25 +00:00
caps: TorznabUtil.CreateDefaultTorznabTVCaps(),
configService: configService,
2015-07-30 20:09:38 +00:00
client: wc,
logger: l,
2015-08-07 19:09:13 +00:00
p: ps,
configData: new ConfigurationDataBasicLogin("For best results, change the 'Torrents per page' setting to 30 or greater (100 recommended) in your profile on the TorrentBytes webpage."))
2015-07-30 20:09:38 +00:00
{
Encoding = Encoding.GetEncoding("iso-8859-1");
2016-12-09 17:20:58 +00:00
Language = "en-us";
2017-01-28 06:07:44 +00:00
Type = "semi-private";
2015-07-30 20:09:38 +00:00
AddCategoryMapping(41, TorznabCatType.TV);
AddCategoryMapping(33, TorznabCatType.TVSD);
AddCategoryMapping(38, TorznabCatType.TVHD);
AddCategoryMapping(32, TorznabCatType.TVSD);
AddCategoryMapping(37, TorznabCatType.TVSD);
AddCategoryMapping(44, TorznabCatType.TVSD);
AddCategoryMapping(40, TorznabCatType.Movies);
AddCategoryMapping(19, TorznabCatType.MoviesSD);
AddCategoryMapping(5, TorznabCatType.MoviesHD);
AddCategoryMapping(20, TorznabCatType.MoviesSD);
AddCategoryMapping(28, TorznabCatType.MoviesForeign);
AddCategoryMapping(45, TorznabCatType.MoviesSD);
AddCategoryMapping(43, TorznabCatType.Audio);
AddCategoryMapping(48, TorznabCatType.AudioLossless);
2015-08-12 17:31:59 +00:00
AddCategoryMapping(6, TorznabCatType.AudioMP3);
2015-07-30 20:09:38 +00:00
AddCategoryMapping(46, TorznabCatType.Movies);
2015-08-12 17:31:59 +00:00
AddCategoryMapping(1, TorznabCatType.PC);
AddCategoryMapping(2, TorznabCatType.PC);
AddCategoryMapping(23, TorznabCatType.TVAnime);
2015-07-30 20:09:38 +00:00
AddCategoryMapping(21, TorznabCatType.XXX);
2015-08-12 17:31:59 +00:00
AddCategoryMapping(9, TorznabCatType.XXXXviD);
AddCategoryMapping(39, TorznabCatType.XXXx264);
AddCategoryMapping(29, TorznabCatType.XXXXviD);
AddCategoryMapping(24, TorznabCatType.XXXImageset);
2015-07-30 20:09:38 +00:00
}
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
2015-07-30 20:09:38 +00:00
{
LoadValuesFromJson(configJson);
2015-07-30 20:09:38 +00:00
var pairs = new Dictionary<string, string> {
{ "username", configData.Username.Value },
{ "password", configData.Password.Value },
2015-07-30 20:09:38 +00:00
{ "returnto", "/" },
{ "login", "Log in!" }
};
var loginPage = await RequestStringWithCookies(SiteLink, string.Empty);
var result = await RequestLoginAndFollowRedirect(LoginUrl, pairs, loginPage.Cookies, true, SiteLink, SiteLink);
await ConfigureIfOK(result.Cookies, result.Content != null && result.Content.Contains("logout.php"), () =>
2015-07-30 20:09:38 +00:00
{
CQ dom = result.Content;
2017-02-16 16:36:40 +00:00
var messageEl = dom["td.embedded"].First();
var errorMessage = messageEl.Text();
if (string.IsNullOrWhiteSpace(errorMessage))
errorMessage = result.Content;
throw new ExceptionWithConfigData(errorMessage, configData);
2015-07-30 20:09:38 +00:00
});
return IndexerConfigurationStatus.RequiresTesting;
2015-07-30 20:09:38 +00:00
}
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
2015-07-30 20:09:38 +00:00
{
var releases = new List<ReleaseInfo>();
var searchString = query.GetQueryString();
2015-07-30 20:09:38 +00:00
var searchUrl = BrowseUrl;
var trackerCats = MapTorznabCapsToTrackers(query);
var queryCollection = new NameValueCollection();
// Tracker can only search OR return things in categories
if (!string.IsNullOrWhiteSpace(searchString))
{
queryCollection.Add("search", searchString);
queryCollection.Add("cat", "0");
queryCollection.Add("sc", "1");
}
else
{
foreach (var cat in MapTorznabCapsToTrackers(query))
{
queryCollection.Add("c" + cat, "1");
}
queryCollection.Add("incldead", "0");
}
searchUrl += "?" + queryCollection.GetQueryString();
await ProcessPage(releases, searchUrl);
2015-07-30 20:09:38 +00:00
return releases;
}
private async Task ProcessPage(List<ReleaseInfo> releases, string searchUrl)
{
var response = await RequestStringWithCookiesAndRetry(searchUrl, null, BrowseUrl);
// On IP change the cookies become invalid, login again and retry
if (response.IsRedirect)
{
await ApplyConfiguration(null);
response = await RequestStringWithCookiesAndRetry(searchUrl, null, BrowseUrl);
}
2015-07-30 20:09:38 +00:00
var results = response.Content;
try
{
CQ dom = results;
var rows = dom["table > tbody:has(tr > td.colhead) > tr:not(:has(td.colhead))"];
foreach (var row in rows)
2015-07-30 20:09:38 +00:00
{
var release = new ReleaseInfo();
var link = row.Cq().Find("td:eq(1) a:eq(1)").First();
release.Guid = new Uri(SiteLink + link.Attr("href"));
release.Comments = release.Guid;
release.Title = link.Get(0).FirstChild.ToString();
2015-07-30 20:09:38 +00:00
release.Description = release.Title;
// If we search an get no results, we still get a table just with no info.
if (string.IsNullOrWhiteSpace(release.Title))
{
break;
}
// Check if the release has been assigned a category
if (row.Cq().Find("td:eq(0) a").Length > 0)
{
var cat = row.Cq().Find("td:eq(0) a").First().Attr("href").Substring(15);
release.Category = MapTrackerCatToNewznab(cat);
}
2015-07-30 20:09:38 +00:00
var qLink = row.Cq().Find("td:eq(1) a").First();
release.Link = new Uri(SiteLink + qLink.Attr("href"));
var added = row.Cq().Find("td:eq(4)").First().Text().Trim();
release.PublishDate = DateTime.ParseExact(added, "yyyy-MM-ddHH:mm:ss", CultureInfo.InvariantCulture);
2015-07-30 20:09:38 +00:00
var sizeStr = row.Cq().Find("td:eq(6)").First().Text().Trim();
release.Size = ReleaseInfo.GetBytes(sizeStr);
release.Seeders = ParseUtil.CoerceInt(row.Cq().Find("td:eq(8)").First().Text().Trim());
release.Peers = ParseUtil.CoerceInt(row.Cq().Find("td:eq(9)").First().Text().Trim()) + release.Seeders;
var files = row.Cq().Find("td:nth-child(3)").Text();
release.Files = ParseUtil.CoerceInt(files);
var grabs = row.Cq().Find("td:nth-child(8)").Text();
if (grabs != "----")
release.Grabs = ParseUtil.CoerceInt(grabs);
if (row.Cq().Find("font[color=\"green\"]:contains(F):contains(L)").Length >= 1)
release.DownloadVolumeFactor = 0;
else
release.DownloadVolumeFactor = 1;
release.UploadVolumeFactor = 1;
2015-07-30 20:09:38 +00:00
releases.Add(release);
}
}
catch (Exception ex)
{
OnParseError(results, ex);
}
}
}
}