Jackett/src/Jackett/Indexers/TorrentBytes.cs

179 lines
7.0 KiB
C#
Raw Normal View History

2015-07-30 20:09:38 +00:00
using CsQuery;
using Jackett.Models;
using Jackett.Services;
using Jackett.Utils;
using Jackett.Utils.Clients;
using Newtonsoft.Json.Linq;
using NLog;
using System;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Globalization;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using System.Web;
using Jackett.Models.IndexerConfig;
2015-07-30 20:09:38 +00:00
namespace Jackett.Indexers
{
public class TorrentBytes : BaseIndexer, IIndexer
{
private string BrowseUrl { get { return SiteLink + "browse.php"; } }
private string LoginUrl { get { return SiteLink + "takelogin.php"; } }
new ConfigurationDataBasicLogin configData
{
get { return (ConfigurationDataBasicLogin)base.configData; }
set { base.configData = value; }
}
2015-07-30 20:09:38 +00:00
public TorrentBytes(IIndexerManagerService i, IWebClient wc, Logger l)
: base(name: "TorrentBytes",
description: "A decade of torrentbytes",
link: "https://www.torrentbytes.net/",
caps: TorznabCapsUtil.CreateDefaultTorznabTVCaps(),
manager: i,
client: wc,
logger: l,
configData: new ConfigurationDataBasicLogin())
2015-07-30 20:09:38 +00:00
{
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);
AddCategoryMapping(6, TorznabCatType.AudioLossy);
AddCategoryMapping(46, TorznabCatType.Movies);
AddCategoryMapping(1, TorznabCatType.Apps);
AddCategoryMapping(2, TorznabCatType.Apps);
AddCategoryMapping(23, TorznabCatType.Anime);
AddCategoryMapping(21, TorznabCatType.XXX);
AddCategoryMapping(9, TorznabCatType.XXXSD);
AddCategoryMapping(39, TorznabCatType.XXXHD);
AddCategoryMapping(29, TorznabCatType.XXXSD);
AddCategoryMapping(24, TorznabCatType.XXXImg);
}
public async Task ApplyConfiguration(JToken configJson)
{
configData.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;
var messageEl = dom["body > div"].First();
var errorMessage = messageEl.Text().Trim();
throw new ExceptionWithConfigData(errorMessage, configData);
2015-07-30 20:09:38 +00:00
});
}
public async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
{
var releases = new List<ReleaseInfo>();
var searchString = query.SanitizedSearchTerm + " " + query.GetEpisodeSearchString();
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();
// 15 results per page - really don't want to call the server twice but only 15 results per page is a bit crap!
await ProcessPage(releases, searchUrl);
2015-07-30 20:09:38 +00:00
await ProcessPage(releases, searchUrl + "&page=1");
return releases;
}
private async Task ProcessPage(List<ReleaseInfo> releases, string searchUrl)
{
var response = await RequestStringWithCookiesAndRetry(searchUrl, null, BrowseUrl);
var results = response.Content;
try
{
CQ dom = results;
var rows = dom["#content table:eq(4) tr"];
foreach (var row in rows.Skip(1))
{
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.Text().Trim();
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;
}
var cat = row.Cq().Find("td:eq(0) a").First().Attr("href").Substring(15);
release.Category = MapTrackerCatToNewznab(cat);
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 = DateTimeUtil.FromTimeAgo(added);
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;
releases.Add(release);
}
}
catch (Exception ex)
{
OnParseError(results, ex);
}
}
}
}