Jackett/src/Jackett/Indexers/MoreThanTV.cs

267 lines
11 KiB
C#
Raw Normal View History

using Jackett.Models;
2015-07-19 17:18:54 +00:00
using Jackett.Services;
using Jackett.Utils.Clients;
2015-04-19 02:05:38 +00:00
using Newtonsoft.Json.Linq;
2015-07-19 00:27:41 +00:00
using NLog;
2015-04-19 02:05:38 +00:00
using System;
using System.Collections.Generic;
using System.Globalization;
2015-04-19 02:05:38 +00:00
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
2015-04-19 02:05:38 +00:00
using System.Threading.Tasks;
using System.Web;
using AngleSharp.Dom;
using AngleSharp.Parser.Html;
using CsQuery;
using Jackett.Models.IndexerConfig;
using Jackett.Utils;
2015-04-19 02:05:38 +00:00
namespace Jackett.Indexers
{
public class MoreThanTV : BaseIndexer
2015-04-19 02:05:38 +00:00
{
private string LoginUrl => SiteLink + "login.php";
private string SearchUrl => SiteLink + "ajax.php?action=browse&searchstr=";
private string DownloadUrl => SiteLink + "torrents.php?action=download&id=";
private string GuidUrl => SiteLink + "torrents.php?torrentid=";
2015-04-19 02:05:38 +00:00
private ConfigurationDataBasicLogin ConfigData => (ConfigurationDataBasicLogin) configData;
2015-08-07 19:09:13 +00:00
public MoreThanTV(IIndexerManagerService i, IWebClient c, Logger l, IProtectionService ps)
: base(name: "MoreThanTV",
description: "ROMANIAN Private Torrent Tracker for TV / MOVIES, and the internal tracker for the release group DRACULA.",
link: "https://www.morethan.tv/",
2015-08-13 22:41:21 +00:00
caps: new TorznabCapabilities(TorznabCatType.TV,
TorznabCatType.Movies),
manager: i,
client: c,
logger: l,
2015-08-07 19:09:13 +00:00
p: ps,
configData: new ConfigurationDataBasicLogin())
2015-04-19 02:05:38 +00:00
{
Encoding = Encoding.GetEncoding("UTF-8");
2016-12-09 17:20:58 +00:00
Language = "en-us";
Type = "private";
2015-04-19 02:05:38 +00:00
}
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
2015-04-19 02:05:38 +00:00
{
LoadValuesFromJson(configJson);
var pairs = new Dictionary<string, string> {
{ "username", ConfigData.Username.Value },
{ "password", ConfigData.Password.Value },
{ "login", "Log in" },
{ "keeplogged", "1" }
};
2015-04-19 02:05:38 +00:00
2015-10-12 18:58:40 +00:00
var preRequest = await RequestStringWithCookiesAndRetry(LoginUrl, string.Empty);
var result = await RequestLoginAndFollowRedirect(LoginUrl, pairs, preRequest.Cookies, true, SearchUrl, SiteLink);
2015-10-12 18:58:40 +00:00
await ConfigureIfOK(result.Cookies, result.Content != null && result.Content.Contains("status\":\"success\""), () =>
2015-04-19 02:05:38 +00:00
{
CQ dom = result.Content;
2015-04-19 02:05:38 +00:00
dom["#loginform > table"].Remove();
var errorMessage = dom["#loginform"].Text().Trim().Replace("\n\t", " ");
throw new ExceptionWithConfigData(errorMessage, ConfigData);
});
return IndexerConfigurationStatus.RequiresTesting;
2015-04-19 02:05:38 +00:00
}
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
2015-04-19 02:05:38 +00:00
{
var isTv = TorznabCatType.QueryContainsParentCategory(query.Categories, new List<int> { TorznabCatType.TV.ID });
var releases = new List<ReleaseInfo>();
var searchQuery = query.GetQueryString();
var searchQuerySingleEpisodes = Regex.Replace(searchQuery, @"(S\d{2})$", "$1*"); // If we're just seaching for a season (no episode) append an * to include all episodes of that season.
await GetReleases(releases, query, searchQuerySingleEpisodes);
// Search for torrent groups (complete seasons)
var seasonMatch = new Regex(@".*\s[Ss]{1}\d{2}$").Match(query.GetQueryString());
if (seasonMatch.Success)
2015-08-13 22:41:21 +00:00
{
var newSearchQuery = Regex.Replace(searchQuery, @"[Ss]{1}\d{2}", $"Season {query.Season}");
await GetReleases(releases, query, newSearchQuery);
2015-08-13 22:41:21 +00:00
}
return releases;
2015-04-19 02:05:38 +00:00
}
private string GetTorrentSearchUrl(int[] categories, string searchQuery)
2015-04-19 02:05:38 +00:00
{
var extra = "";
if (Array.IndexOf(categories, TorznabCatType.Movies.ID) > -1)
extra += "&filter_cat%5B1%5D=1";
if (Array.IndexOf(categories, TorznabCatType.TV.ID) > -1)
extra += "&filter_cat%5B2%5D=1";
return SiteLink + $"torrents.php?searchstr={HttpUtility.UrlEncode(searchQuery)}&tags_type=1&order_by=time&order_way=desc&group_results=1{extra}&action=basic&searchsubmit=1";
}
private async Task GetReleases(ICollection<ReleaseInfo> releases, TorznabQuery query, string searchQuery)
{
var searchUrl = GetTorrentSearchUrl(query.Categories, searchQuery);
var response = await RequestStringWithCookiesAndRetry(searchUrl);
if (response.IsRedirect)
{
// re login
await ApplyConfiguration(null);
response = await RequestStringWithCookiesAndRetry(searchUrl);
2017-03-03 17:52:19 +00:00
}
2015-04-19 02:05:38 +00:00
try
{
var parser = new HtmlParser();
var document = parser.Parse(response.Content);
var groups = document.QuerySelectorAll(".torrent_table > tbody > tr.group");
var torrents = document.QuerySelectorAll(".torrent_table > tbody > tr.torrent");
// Loop through all torrent (season) groups
foreach (var group in groups)
{
var showName = group.QuerySelector(".tp-showname a").InnerHtml.Replace("(", "").Replace(")", "").Replace(' ', '.');
var season = group.QuerySelector(".big_info a").InnerHtml;
// Loop through all group items
var previousElement = group;
var qualityEdition = string.Empty;
while (true)
2015-04-19 02:05:38 +00:00
{
var groupItem = previousElement.NextElementSibling;
2015-05-04 04:12:14 +00:00
if (groupItem == null) break;
2015-05-04 04:12:14 +00:00
if (!groupItem.ClassList[0].Equals("group_torrent") ||
!groupItem.ClassList[1].StartsWith("groupid_")) break;
// Found a new edition
if (groupItem.ClassList[2].Equals("edition"))
2015-04-19 02:05:38 +00:00
{
qualityEdition = groupItem.QuerySelector(".edition_info strong").TextContent.Split('/')[1].Trim();
2015-04-19 02:05:38 +00:00
}
else if (groupItem.ClassList[2].StartsWith("edition_"))
{
if (qualityEdition.Equals(string.Empty)) break;
// Parse required data
var downloadAnchor = groupItem.QuerySelectorAll("td a").Last();
var qualityData = downloadAnchor.InnerHtml.Split('/');
if (qualityData.Length < 2)
throw new Exception($"We expected 2 or more quality datas, instead we have {qualityData.Length}.");
// Build title
var title = string.Join(".", new List<string>
{
showName,
SeasonToShortSeason(season),
qualityData[1].Trim(),
qualityEdition, // Audio quality should be after this one. Unobtainable at the moment.
$"{qualityData[0].Trim()}-MTV"
});
releases.Add(GetReleaseInfo(groupItem, downloadAnchor, title, TorznabCatType.TV.ID));
}
else
{
break;
}
previousElement = groupItem;
}
}
// Loop through all torrents
foreach (var torrent in torrents)
{
// Parse required data
var downloadAnchor = torrent.QuerySelector(".big_info > .group_info > a");
var title = downloadAnchor.TextContent;
int category;
var categories = torrent.QuerySelector(".cats_col div").ClassList;
if (categories.Contains("cats_tv"))
{
category = TorznabCatType.TV.ID;
}
else if (categories.Contains("cats_movies"))
{
category = TorznabCatType.Movies.ID;
2015-05-04 04:12:14 +00:00
}
2017-04-17 04:22:00 +00:00
else if (categories.Contains("cats_other"))
{
category = TorznabCatType.Other.ID;
}
else
{
throw new Exception("Couldn't find category.");
}
releases.Add(GetReleaseInfo(torrent, downloadAnchor, title, category));
2015-05-04 04:12:14 +00:00
}
}
catch (Exception ex)
{
OnParseError(response.Content, ex);
2015-04-19 02:05:38 +00:00
}
}
2015-04-19 02:05:38 +00:00
private ReleaseInfo GetReleaseInfo(IElement row, IElement downloadAnchor, string title, int category)
{
// Parse required data
var downloadAnchorHref = downloadAnchor.Attributes["href"].Value;
var torrentId = downloadAnchorHref.Substring(downloadAnchorHref.LastIndexOf('=') + 1);
var qFiles = row.QuerySelector("td:nth-last-child(6)");
var files = ParseUtil.CoerceLong(qFiles.TextContent);
var publishDate = DateTime.ParseExact(row.QuerySelector(".time.tooltip").Attributes["title"].Value, "MMM dd yyyy, HH:mm", CultureInfo.InvariantCulture, DateTimeStyles.AssumeUniversal).ToLocalTime();
var torrentData = row.QuerySelectorAll(".number_column"); // Size (xx.xx GB[ (Max)]) Snatches (xx) Seeders (xx) Leechers (xx)
if (torrentData.Length != 4)
throw new Exception($"We expected 4 torrent datas, instead we have {torrentData.Length}.");
if (torrentId.Contains('#'))
torrentId = torrentId.Split('#')[0];
2016-12-08 06:31:31 +00:00
var size = ReleaseInfo.GetBytes(torrentData[0].TextContent);
var grabs = int.Parse(torrentData[1].TextContent);
var seeders = int.Parse(torrentData[2].TextContent);
var guid = new Uri(GuidUrl + torrentId);
// Build releaseinfo
return new ReleaseInfo
{
Title = title,
Category = new List<int> { category }, // Who seasons movies right
Link = new Uri(DownloadUrl + torrentId),
PublishDate = publishDate,
Seeders = seeders,
Peers = seeders,
Files = files,
Size = size,
Grabs = grabs,
Guid = guid,
Comments = guid,
DownloadVolumeFactor = 0, // ratioless tracker
UploadVolumeFactor = 1
};
}
// Changes "Season 1" to "S01"
private static string SeasonToShortSeason(string season)
{
var seasonMatch = new Regex(@"Season (?<seasonNumber>\d{1,2})").Match(season);
if (seasonMatch.Success)
{
season = $"S{int.Parse(seasonMatch.Groups["seasonNumber"].Value):00}";
}
return season;
}
2015-04-19 02:05:38 +00:00
}
}