Jackett/src/Jackett.Common/Indexers/Sharewood.cs

332 lines
15 KiB
C#

using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Globalization;
using System.Linq;
using System.Net;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using Jackett.Common.Models;
using Jackett.Common.Models.IndexerConfig;
using Jackett.Common.Services.Interfaces;
using Jackett.Common.Utils;
using Newtonsoft.Json.Linq;
using NLog;
using static Jackett.Common.Models.IndexerConfig.ConfigurationData;
using WebClient = Jackett.Common.Utils.Clients.WebClient;
namespace Jackett.Common.Indexers
{
[ExcludeFromCodeCoverage]
public class ShareWood : BaseWebIndexer
{
private readonly Dictionary<string, string> _apiHeaders = new Dictionary<string, string>
{
{"Accept", "application/json"},
{"Content-Type", "application/json"}
};
// API DOC: https://github.com/Jackett/Jackett/issues/10269
private string SearchUrl => SiteLink + "api/" + configData.Passkey.Value;
private new ConfigurationDataPasskey configData => (ConfigurationDataPasskey)base.configData;
public ShareWood(IIndexerConfigurationService configService, WebClient wc, Logger l, IProtectionService ps,
ICacheService cs)
: base(
id: "sharewoodapi",
name: "Sharewood API",
description: "Sharewood is a Semi-Private FRENCH Torrent Tracker for GENERAL",
link: "https://www.sharewood.tv/",
caps: new TorznabCapabilities
{
TvSearchParams = new List<TvSearchParam>
{
TvSearchParam.Q, TvSearchParam.Season, TvSearchParam.Ep
},
MovieSearchParams = new List<MovieSearchParam>
{
MovieSearchParam.Q
},
MusicSearchParams = new List<MusicSearchParam>
{
MusicSearchParam.Q
},
BookSearchParams = new List<BookSearchParam>
{
BookSearchParam.Q
}
},
configService: configService,
client: wc,
logger: l,
p: ps,
cacheService: cs,
configData: new ConfigurationDataPasskey()
)
{
Encoding = Encoding.UTF8;
Language = "fr-fr";
Type = "semi-private";
// requestDelay for API Limit (1 request per 2 seconds)
webclient.requestDelay = 2.1;
//CATEGORIES
//AddCategoryMapping(1, TorznabCatType.Movies, "Vidéos");
//AddCategoryMapping(1, TorznabCatType.TV, "Vidéos");
//AddCategoryMapping(2, TorznabCatType.Audio, "Audio");
//AddCategoryMapping(3, TorznabCatType.PC, "Application");
//AddCategoryMapping(4, TorznabCatType.Books, "Ebooks");
//AddCategoryMapping(5, TorznabCatType.PCGames, "Jeu-Vidéo");
//AddCategoryMapping(6, TorznabCatType.OtherMisc, "Formation");
//AddCategoryMapping(7, TorznabCatType.XXX, "XXX");
//SUBCATEGORIES
AddCategoryMapping(9, TorznabCatType.Movies, "Films");
AddCategoryMapping(10, TorznabCatType.TV, "Série");
AddCategoryMapping(11, TorznabCatType.MoviesOther, "Film Animation");
AddCategoryMapping(12, TorznabCatType.TVAnime, "Série Animation");
AddCategoryMapping(13, TorznabCatType.TVDocumentary, "Documentaire");
AddCategoryMapping(14, TorznabCatType.TVOther, "Emission TV");
AddCategoryMapping(15, TorznabCatType.TVOther, "Spectacle/Concert");
AddCategoryMapping(16, TorznabCatType.TVSport, "Sport");
AddCategoryMapping(17, TorznabCatType.AudioOther, "Karaoké Vidéo");
AddCategoryMapping(18, TorznabCatType.AudioOther, "Karaoké");
AddCategoryMapping(20, TorznabCatType.Audio, "Musique");
AddCategoryMapping(21, TorznabCatType.AudioOther, "Podcast");
AddCategoryMapping(22, TorznabCatType.Audio, "Sample");
AddCategoryMapping(23, TorznabCatType.AudioAudiobook, "Ebook Audio");
AddCategoryMapping(24, TorznabCatType.Books, "BD");
AddCategoryMapping(25, TorznabCatType.BooksComics, "Comic");
AddCategoryMapping(26, TorznabCatType.BooksOther, "Manga");
AddCategoryMapping(27, TorznabCatType.Books, "Livre");
AddCategoryMapping(28, TorznabCatType.BooksMags, "Presse");
AddCategoryMapping(29, TorznabCatType.Audio, "Application Linux");
AddCategoryMapping(30, TorznabCatType.PC, "Application Window");
AddCategoryMapping(31, TorznabCatType.PCMac, "Application Mac");
AddCategoryMapping(34, TorznabCatType.PCMobileiOS, "Application Smartphone/Tablette");
AddCategoryMapping(34, TorznabCatType.PCMobileAndroid, "Application Smartphone/Tablette");
AddCategoryMapping(35, TorznabCatType.Other, "GPS");
AddCategoryMapping(36, TorznabCatType.Audio, "Jeux Linux");
AddCategoryMapping(37, TorznabCatType.PCGames, "Jeux Windows");
AddCategoryMapping(39, TorznabCatType.ConsoleNDS, "Jeux Nintendo");
AddCategoryMapping(39, TorznabCatType.ConsoleWii, "Jeux Nintendo");
AddCategoryMapping(39, TorznabCatType.ConsoleWiiware, "Jeux Nintendo");
AddCategoryMapping(39, TorznabCatType.Console3DS, "Jeux Nintendo");
AddCategoryMapping(39, TorznabCatType.ConsoleWiiU, "Jeux Nintendo");
AddCategoryMapping(41, TorznabCatType.PCMobileAndroid, "PC/Mobile-Android");
AddCategoryMapping(42, TorznabCatType.PCGames, "Jeux Microsoft");
AddCategoryMapping(44, TorznabCatType.XXX, "XXX Films");
AddCategoryMapping(45, TorznabCatType.XXXOther, "XXX Hentai");
AddCategoryMapping(47, TorznabCatType.XXXImageSet, "XXX Images");
AddCategoryMapping(48, TorznabCatType.XXXOther, "XXX Jeu-Vidéo");
AddCategoryMapping(50, TorznabCatType.OtherMisc, "Formation Logiciels");
AddCategoryMapping(49, TorznabCatType.OtherMisc, "Formations Vidéos");
AddCategoryMapping(51, TorznabCatType.XXXOther, "XXX Ebooks");
AddCategoryMapping(52, TorznabCatType.AudioVideo, "Vidéos-Clips");
AddCategoryMapping(51, TorznabCatType.XXXOther, "XXX Ebooks");
AddCategoryMapping(51, TorznabCatType.XXXOther, "XXX Ebooks");
var FreeLeechOnly = new BoolConfigurationItem("Search freeleech only");
configData.AddDynamic("freeleechonly", FreeLeechOnly);
var ReplaceMulti = new BoolConfigurationItem("Replace MULTI by another language in release name");
configData.AddDynamic("replacemulti", ReplaceMulti);
// Configure the language select option for MULTI
var languageSelect = new SingleSelectConfigurationItem("Replace MULTI by this language", new Dictionary<string, string>
{
{"FRENCH", "FRENCH"},
{"MULTI.FRENCH", "MULTI.FRENCH"},
{"ENGLISH", "ENGLISH"},
{"MULTI.ENGLISH", "MULTI.ENGLISH" },
{"VOSTFR", "VOSTFR"},
{"MULTI.VOSTFR", "MULTI.VOSTFR"}
})
{ Value = "FRENCH" };
;
configData.AddDynamic("languageid", languageSelect);
var ReplaceVostfr = new BoolConfigurationItem("Replace VOSTFR with ENGLISH");
configData.AddDynamic("replacevostfr", ReplaceVostfr);
EnableConfigurableRetryAttempts();
}
private string MultiRename(string term, string replacement)
{
replacement = " " + replacement + " ";
term = Regex.Replace(term, @"(?i)(\smulti\s)", replacement);
term = Regex.Replace(term, @"(?i)(\smulti$)", replacement);
return term;
}
private string VostfrRename(string term, string replacement)
{
term = Regex.Replace(term, @"(?i)(vostfr)", replacement);
term = Regex.Replace(term, @"(?i)(subfrench)", replacement);
return term;
}
private bool GetFreeLeech => ((BoolConfigurationItem)configData.GetDynamic("freeleechonly")).Value;
private bool GetReplaceMulti => ((BoolConfigurationItem)configData.GetDynamic("replacemulti")).Value;
private string GetLang => ((SingleSelectConfigurationItem)configData.GetDynamic("languageid")).Value;
private bool GetReplaceVostfr => ((BoolConfigurationItem)configData.GetDynamic("replacevostfr")).Value;
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
{
LoadValuesFromJson(configJson);
if (configData.Passkey.Value.Length != 32)
throw new Exception("Invalid Passkey configured. Expected length: 32");
var releases = await PerformQuery(new TorznabQuery());
await ConfigureIfOK(string.Empty, releases.Any(),
() => throw new Exception("Could not find releases."));
return IndexerConfigurationStatus.Completed;
}
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
{
var releases = new List<ReleaseInfo>();
var Mapcats = MapTorznabCapsToTrackers(query);
if (Mapcats.Count == 0)
{ // NO CATEGORIES ==> RSS SEARCH
Mapcats.Add("1000");
}
foreach (var cats in Mapcats)
{
var searchUrl = SearchUrl;
var qc = new List<KeyValuePair<string, string>> // NameValueCollection don't support cat[]=19&cat[]=6
{
{"limit", "25"}
};
if (Convert.ToInt32(cats) != 1000)
{
qc.Add("subcategory", cats);
}
if (query.GetQueryString() != "")
{
qc.Add("name", query.GetQueryString());
searchUrl = searchUrl + "/search";
}
else
{
searchUrl = searchUrl + "/last-torrents";
}
searchUrl = searchUrl + "?" + qc.GetQueryString();
var response = await RequestWithCookiesAsync(searchUrl);
if (response.Status == HttpStatusCode.Unauthorized)
{
response = await RequestWithCookiesAsync(searchUrl);
}
else if (response.Status != HttpStatusCode.OK)
throw new Exception($"Unknown error in search: {response.ContentString}");
try
{
var rows = JArray.Parse(response.ContentString);
foreach (var row in rows)
{
var id = row.Value<string>("id");
var link = new Uri($"{SearchUrl}/{id}/download");
var urlStr = row.Value<string>("slug");
var details = new Uri($"{SiteLink}torrents/{urlStr}.{id}");
DateTime publishDate = DateTime.Parse(row.Value<string>("created_at"), CultureInfo.InvariantCulture);
var cat = row.Value<string>("subcategory_id");
if (Convert.ToInt32(cats) != 1000)
{
if (Convert.ToInt32(cats) < 8) //USE CATEGORIES
{
cat = row.Value<string>("category_id");
}
else //USE SUBCATEGORIES
{
cat = row.Value<string>("subcategory_id");
}
}
long dlVolumeFactor = 1;
long ulVolumeFactor = 1;
if (row.Value<bool>("free") == true)
{
dlVolumeFactor = 0;
}
if (row.Value<bool>("doubleup") == true)
{
ulVolumeFactor = 2;
}
var sizeString = row.Value<string>("size");
var title = row.Value<string>("name");
//SPECIAL CASES
if (GetFreeLeech == true && dlVolumeFactor == 1)
continue;
if (GetReplaceMulti == true)
{
title = MultiRename(title, GetLang);
}
if (GetReplaceVostfr == true)
{
title = VostfrRename(title, "ENGLISH");
}
var release = new ReleaseInfo
{
Title = title,
Link = link,
Details = details,
Guid = details,
Category = MapTrackerCatToNewznab(cat),
PublishDate = publishDate,
Size = ReleaseInfo.GetBytes(sizeString),
Grabs = row.Value<long>("times_completed"),
Seeders = row.Value<long>("seeders"),
Peers = row.Value<long>("leechers") + row.Value<long>("seeders"),
DownloadVolumeFactor = dlVolumeFactor,
UploadVolumeFactor = ulVolumeFactor,
MinimumRatio = 1,
MinimumSeedTime = 259200 // 72 hours
};
releases.Add(release);
}
}
catch (Exception ex)
{
OnParseError(response.ContentString, ex);
}
}
return releases;
}
public override async Task<byte[]> Download(Uri link)
{
var response = await RequestWithCookiesAsync(link.ToString());
if (response.Status == HttpStatusCode.Unauthorized)
{
response = await RequestWithCookiesAsync(link.ToString());
}
else if (response.Status != HttpStatusCode.OK)
throw new Exception($"Unknown error in download: {response.ContentBytes}");
return response.ContentBytes;
}
}
}