using System; using System.Collections.Generic; using System.Collections.Specialized; using System.Globalization; using System.Linq; using System.Text; using System.Text.RegularExpressions; using System.Threading.Tasks; using CsQuery; 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.Linq; using NLog; namespace Jackett.Common.Indexers { public class TorrentSyndikat : BaseWebIndexer { private string SearchUrl { get { return SiteLink + "browse.php"; } } private string LoginUrl { get { return SiteLink + "eing2.php"; } } private string CaptchaUrl { get { return SiteLink + "simpleCaptcha.php?numImages=1"; } } private TimeZoneInfo germanyTz; private new ConfigurationDataBasicLoginWithRSSAndDisplay configData { get { return (ConfigurationDataBasicLoginWithRSSAndDisplay)base.configData; } set { base.configData = value; } } public TorrentSyndikat(IIndexerConfigurationService configService, WebClient w, Logger l, IProtectionService ps) : base(name: "Torrent-Syndikat", description: "A German general tracker", link: "https://torrent-syndikat.org/", caps: new TorznabCapabilities(), configService: configService, client: w, logger: l, p: ps, configData: new ConfigurationDataBasicLoginWithRSSAndDisplay()) { Encoding = Encoding.UTF8; Language = "de-de"; Type = "private"; TorznabCaps.SupportsImdbMovieSearch = true; this.configData.DisplayText.Value = "Only the results from the first search result page are shown, adjust your profile settings to show the maximum."; this.configData.DisplayText.Name = "Notice"; AddCategoryMapping(2, TorznabCatType.PC, "Apps / Windows"); AddCategoryMapping(13, TorznabCatType.PC, "Apps / Linux"); AddCategoryMapping(4, TorznabCatType.PCMac, "Apps / MacOS"); AddCategoryMapping(6, TorznabCatType.PC, "Apps / Misc"); AddCategoryMapping(50, TorznabCatType.PCGames, "Spiele / Windows"); AddCategoryMapping(51, TorznabCatType.PCGames, "Spiele / MacOS"); AddCategoryMapping(52, TorznabCatType.PCGames, "Spiele / Linux"); AddCategoryMapping(8, TorznabCatType.ConsoleOther, "Spiele / Playstation"); AddCategoryMapping(7, TorznabCatType.ConsoleOther, "Spiele / Nintendo"); AddCategoryMapping(32, TorznabCatType.ConsoleOther, "Spiele / XBOX"); AddCategoryMapping(42, TorznabCatType.MoviesUHD, "Filme / 2160p"); AddCategoryMapping(9, TorznabCatType.MoviesHD, "Filme / 1080p"); AddCategoryMapping(20, TorznabCatType.MoviesHD, "Filme / 720p"); AddCategoryMapping(10, TorznabCatType.MoviesSD, "Filme / SD"); AddCategoryMapping(43, TorznabCatType.TVUHD, "Serien / 2160p"); AddCategoryMapping(53, TorznabCatType.TVHD, "Serien / 1080p"); AddCategoryMapping(54, TorznabCatType.TVHD, "Serien / 720p"); AddCategoryMapping(15, TorznabCatType.TVSD, "Serien / SD"); AddCategoryMapping(30, TorznabCatType.TVSport, "Serien / Sport"); AddCategoryMapping(44, TorznabCatType.TVUHD, "Serienpacks / 2160p"); AddCategoryMapping(55, TorznabCatType.TVHD, "Serienpacks / 1080p"); AddCategoryMapping(56, TorznabCatType.TVHD, "Serienpacks / 720p"); AddCategoryMapping(27, TorznabCatType.TVSD, "Serienpacks / SD"); AddCategoryMapping(24, TorznabCatType.AudioLossless, "Audio / Musik / FLAC"); AddCategoryMapping(25, TorznabCatType.AudioMP3, "Audio / Musik / MP3"); AddCategoryMapping(35, TorznabCatType.AudioOther, "Audio / Other"); AddCategoryMapping(18, TorznabCatType.AudioAudiobook, "Audio / aBooks"); AddCategoryMapping(33, TorznabCatType.AudioVideo, "Audio / Videos"); AddCategoryMapping(17, TorznabCatType.Books, "Misc / eBooks"); AddCategoryMapping(5, TorznabCatType.PCPhoneOther, "Misc / Mobile"); AddCategoryMapping(39, TorznabCatType.Other, "Misc / Bildung"); AddCategoryMapping(36, TorznabCatType.TVFOREIGN, "Englisch / Serien"); AddCategoryMapping(57, TorznabCatType.TVFOREIGN, "Englisch / Serienpacks"); AddCategoryMapping(37, TorznabCatType.MoviesForeign, "Englisch / Filme"); AddCategoryMapping(47, TorznabCatType.Books, "Englisch / eBooks"); AddCategoryMapping(48, TorznabCatType.Other, "Englisch / Bildung"); AddCategoryMapping(49, TorznabCatType.TVSport, "Englisch / Sport"); TimeZoneInfo.TransitionTime startTransition = TimeZoneInfo.TransitionTime.CreateFloatingDateRule(new DateTime(1, 1, 1, 3, 0, 0), 3, 5, DayOfWeek.Sunday); TimeZoneInfo.TransitionTime endTransition = TimeZoneInfo.TransitionTime.CreateFloatingDateRule(new DateTime(1, 1, 1, 4, 0, 0), 10, 5, DayOfWeek.Sunday); TimeSpan delta = new TimeSpan(1, 0, 0); TimeZoneInfo.AdjustmentRule adjustment = TimeZoneInfo.AdjustmentRule.CreateAdjustmentRule(new DateTime(1999, 10, 1), DateTime.MaxValue.Date, delta, startTransition, endTransition); TimeZoneInfo.AdjustmentRule[] adjustments = { adjustment }; germanyTz = TimeZoneInfo.CreateCustomTimeZone("W. Europe Standard Time", new TimeSpan(1, 0, 0), "(GMT+01:00) W. Europe Standard Time", "W. Europe Standard Time", "W. Europe DST Time", adjustments); } public override async Task ApplyConfiguration(JToken configJson) { LoadValuesFromJson(configJson); CookieHeader = ""; var result1 = await RequestStringWithCookies(CaptchaUrl); var json1 = JObject.Parse(result1.Content); var captchaSelection = json1["images"][0]["hash"]; var pairs = new Dictionary { { "username", configData.Username.Value }, { "password", configData.Password.Value }, { "captchaSelection", (string)captchaSelection }, { "submitme", "X" } }; var result2 = await RequestLoginAndFollowRedirect(LoginUrl, pairs, result1.Cookies, true, null, null, true); await ConfigureIfOK(result2.Cookies, result2.Content.Contains("/logout.php"), () => { var errorMessage = result2.Content; throw new ExceptionWithConfigData(errorMessage, configData); }); return IndexerConfigurationStatus.RequiresTesting; } protected override async Task> PerformQuery(TorznabQuery query) { List releases = new List(); var searchString = query.GetQueryString(); var searchUrl = SearchUrl; var queryCollection = new NameValueCollection(); queryCollection.Add("incldead", "1"); queryCollection.Add("rel_type", "0"); // Alle if(query.ImdbID != null) { queryCollection.Add("searchin", "imdb"); queryCollection.Add("search", query.ImdbID); } else { queryCollection.Add("searchin", "title"); if (!string.IsNullOrWhiteSpace(searchString)) { // use AND+wildcard operator to avoid getting to many useless results var searchStringArray = Regex.Split(searchString.Trim(), "[ _.-]+", RegexOptions.Compiled).ToList(); searchStringArray = searchStringArray.Select(x => "+" + x).ToList(); // add AND operators var searchStringFinal = String.Join(" ", searchStringArray); queryCollection.Add("search", searchStringFinal); } } foreach (var cat in MapTorznabCapsToTrackers(query)) { queryCollection.Add("c" + cat, "1"); } searchUrl += "?" + queryCollection.GetQueryString(); var results = await RequestStringWithCookiesAndRetry(searchUrl); if (results.IsRedirect) { await ApplyConfiguration(null); results = await RequestStringWithCookiesAndRetry(searchUrl); } try { CQ dom = results.Content; var rows = dom["table.torrent_table > tbody > tr"]; var globalFreeleech = dom.Find("legend:contains(\"Freeleech\")+ul > li > b:contains(\"Freeleech\")").Any(); foreach (var row in rows.Skip(1)) { var release = new ReleaseInfo(); release.MinimumRatio = 1; release.MinimumSeedTime = 96 * 60 * 60; var qRow = row.Cq(); var catStr = row.ChildElements.ElementAt(0).FirstElementChild.GetAttribute("href").Split('=')[1].Split('&')[0]; release.Category = MapTrackerCatToNewznab(catStr); var qLink = row.ChildElements.ElementAt(2).FirstElementChild.Cq(); release.Link = new Uri(SiteLink + qLink.Attr("href")); var torrentId = qLink.Attr("href").Split('=').Last(); var descCol = row.ChildElements.ElementAt(1); var qCommentLink = descCol.FirstElementChild.Cq(); var torrentTag = descCol.Cq().Find("span.torrent-tag"); var torrentTags = torrentTag.Elements.Select(x => x.InnerHTML).ToList(); release.Title = qCommentLink.Attr("title"); release.Description = String.Join(", ", torrentTags); release.Comments = new Uri(SiteLink + qCommentLink.Attr("href").Replace("&hit=1", "")); release.Guid = release.Comments; var torrent_details = descCol.Cq().Find(".torrent_details").Get(0); var rawDateStr = torrent_details.ChildNodes.ElementAt(torrent_details.ChildNodes.Length - 3).Cq().Text(); var dateStr = rawDateStr.Trim().Replace("von", "").Trim(); DateTime dateGerman; if (dateStr.StartsWith("Heute ")) dateGerman = DateTime.SpecifyKind(DateTime.UtcNow.Date, DateTimeKind.Unspecified) + TimeSpan.Parse(dateStr.Split(' ')[1]); else if (dateStr.StartsWith("Gestern ")) dateGerman = DateTime.SpecifyKind(DateTime.UtcNow.Date, DateTimeKind.Unspecified) + TimeSpan.Parse(dateStr.Split(' ')[1]) - TimeSpan.FromDays(1); else dateGerman = DateTime.SpecifyKind(DateTime.ParseExact(dateStr, "dd.MM.yyyy HH:mm", CultureInfo.InvariantCulture), DateTimeKind.Unspecified); DateTime pubDateUtc = TimeZoneInfo.ConvertTimeToUtc(dateGerman, germanyTz); release.PublishDate = pubDateUtc.ToLocalTime(); var imdbLink = descCol.Cq().Find("a[href*=\"&searchin=imdb\"]"); if (imdbLink.Any()) release.Imdb = ParseUtil.GetLongFromString(imdbLink.Attr("href")); var sizeFileCountRowChilds = row.ChildElements.ElementAt(5).ChildElements; release.Size = ReleaseInfo.GetBytes(sizeFileCountRowChilds.ElementAt(0).Cq().Text()); release.Files = ParseUtil.CoerceInt(sizeFileCountRowChilds.ElementAt(2).Cq().Text()); release.Seeders = ParseUtil.CoerceInt(row.ChildElements.ElementAt(7).Cq().Text()); release.Peers = ParseUtil.CoerceInt(row.ChildElements.ElementAt(8).Cq().Text()) + release.Seeders; var grabs = qRow.Find("td:nth-child(7)").Text(); release.Grabs = ParseUtil.CoerceInt(grabs); if (globalFreeleech) release.DownloadVolumeFactor = 0; else if (qRow.Find("span.torrent-tag-free").Length >= 1) release.DownloadVolumeFactor = 0; else release.DownloadVolumeFactor = 1; release.UploadVolumeFactor = 1; releases.Add(release); } } catch (Exception ex) { OnParseError(results.Content, ex); } return releases; } } }