using System; using System.Collections.Generic; using System.Collections.Specialized; using System.Globalization; using System.Linq; using System.Text; 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 { // ReSharper disable once InconsistentNaming public class GimmePeers : BaseWebIndexer { private string BrowseUrl => SiteLink + "browse.php"; private string LoginUrl => SiteLink + "takelogin.php"; private new ConfigurationDataBasicLogin configData { get { return (ConfigurationDataBasicLogin)base.configData; } set { base.configData = value; } } public GimmePeers(IIndexerConfigurationService configService, WebClient wc, Logger l, IProtectionService ps) : base(name: "GimmePeers", description: "Formerly ILT", link: "https://www.gimmepeers.com/", caps: new TorznabCapabilities(), configService: configService, client: wc, logger: l, p: ps, configData: new ConfigurationDataBasicLogin()) { Encoding = Encoding.GetEncoding("iso-8859-1"); Language = "en-us"; Type = "private"; AddCategoryMapping(1, TorznabCatType.TVAnime); AddCategoryMapping(5, TorznabCatType.BooksEbook); AddCategoryMapping(10, TorznabCatType.PCGames); AddCategoryMapping(11, TorznabCatType.ConsolePS3); AddCategoryMapping(11, TorznabCatType.ConsolePS4); AddCategoryMapping(11, TorznabCatType.ConsolePSP); AddCategoryMapping(12, TorznabCatType.ConsoleXBOX360DLC); AddCategoryMapping(12, TorznabCatType.ConsoleXbox); AddCategoryMapping(12, TorznabCatType.ConsoleXbox360); AddCategoryMapping(12, TorznabCatType.ConsoleXboxOne); AddCategoryMapping(6, TorznabCatType.Audio); AddCategoryMapping(21, TorznabCatType.TV); AddCategoryMapping(20, TorznabCatType.TVSD); AddCategoryMapping(21, TorznabCatType.TVHD); AddCategoryMapping(22, TorznabCatType.TV); AddCategoryMapping(24, TorznabCatType.TVSD); AddCategoryMapping(25, TorznabCatType.TVHD); AddCategoryMapping(50, TorznabCatType.XXX); AddCategoryMapping(49, TorznabCatType.XXXDVD); AddCategoryMapping(50, TorznabCatType.XXXx264); AddCategoryMapping(13, TorznabCatType.Movies3D); AddCategoryMapping(14, TorznabCatType.MoviesBluRay); AddCategoryMapping(15, TorznabCatType.MoviesDVD); AddCategoryMapping(16, TorznabCatType.MoviesHD); AddCategoryMapping(17, TorznabCatType.Movies); AddCategoryMapping(19, TorznabCatType.Movies); AddCategoryMapping(26, TorznabCatType.MoviesUHD); } public override async Task ApplyConfiguration(JToken configJson) { LoadValuesFromJson(configJson); var pairs = new Dictionary { { "username", configData.Username.Value }, { "password", configData.Password.Value }, { "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"), () => { CQ dom = result.Content; var messageEl = dom["body > div"].First(); var errorMessage = messageEl.Text().Trim(); throw new ExceptionWithConfigData(errorMessage, configData); }); return IndexerConfigurationStatus.RequiresTesting; } protected override async Task> PerformQuery(TorznabQuery query) { var releases = new List(); var searchString = query.GetQueryString(); 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"); } else { foreach (var cat in MapTorznabCapsToTrackers(query)) { queryCollection.Add("c" + cat, "1"); } queryCollection.Add("incldead", "0"); } searchUrl += "?" + queryCollection.GetQueryString(); await ProcessPage(releases, searchUrl); return releases; } private async Task ProcessPage(List releases, string searchUrl) { var response = await RequestStringWithCookiesAndRetry(searchUrl, null, BrowseUrl); if (response.IsRedirect) { // re login await ApplyConfiguration(null); response = await RequestStringWithCookiesAndRetry(searchUrl, null, BrowseUrl); } var results = response.Content; try { CQ dom = results; var rows = dom[".browsetable:last tr"]; //the class for the table changed foreach (var row in rows.Skip(1)) { var release = new ReleaseInfo(); var link = row.Cq().Find("td:eq(1) a:eq(0)").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; } // 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); } var qLink = row.Cq().Find("td:eq(2) a").First(); release.Link = new Uri(SiteLink + qLink.Attr("href")); var added = row.Cq().Find("td:eq(6)").First().Text().Trim(); //column changed from 7 to 6 var date = added.Substring(0, 10); var time = added.Substring(11, 8); //date layout wasn't quite right var dateTime = date + time; release.PublishDate = DateTime.ParseExact(dateTime, "yyyy-MM-ddHH:mm:ss", CultureInfo.InvariantCulture, DateTimeStyles.AssumeUniversal).ToLocalTime(); var sizeStr = row.Cq().Find("td:eq(5)").First().Text().Trim(); //size column moved from 8 to 5 release.Size = ReleaseInfo.GetBytes(sizeStr); release.Seeders = ParseUtil.CoerceInt(row.Cq().Find("td:eq(10)").First().Text().Trim()); release.Peers = ParseUtil.CoerceInt(row.Cq().Find("td:eq(11)").First().Text().Trim()) + release.Seeders; releases.Add(release); } } catch (Exception ex) { OnParseError(results, ex); } } } }