mirror of
https://github.com/Jackett/Jackett.git
synced 2025-09-17 17:34:09 +02:00
184 lines
6.7 KiB
C#
184 lines
6.7 KiB
C#
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.Globalization;
|
|
using System.Linq;
|
|
using System.Net;
|
|
using System.Net.Http;
|
|
using System.Net.Http.Headers;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
using System.Web;
|
|
using Jackett.Models.IndexerConfig;
|
|
|
|
namespace Jackett.Indexers
|
|
{
|
|
public class T411 : BaseIndexer, IIndexer
|
|
{
|
|
private readonly string CommentsUrl = "";
|
|
const string ApiUrl = "http://api.t411.io";
|
|
const string AuthUrl = ApiUrl + "/auth";
|
|
const string SearchUrl = ApiUrl + "/torrents/search/{0}";
|
|
const string DownloadUrl = ApiUrl + "/torrents/download/{0}";
|
|
|
|
HttpClientHandler handler;
|
|
HttpClient client;
|
|
|
|
new ConfigurationDataLoginTokin configData
|
|
{
|
|
get { return (ConfigurationDataLoginTokin)base.configData; }
|
|
set { base.configData = value; }
|
|
}
|
|
|
|
public T411(IIndexerManagerService i, Logger l, IWebClient wc, IProtectionService ps)
|
|
: base(name: "T411",
|
|
description: "French Torrent Tracker",
|
|
link: "http://www.t411.io/",
|
|
caps: TorznabUtil.CreateDefaultTorznabTVCaps(),
|
|
manager: i,
|
|
client: wc,
|
|
logger: l,
|
|
p: ps,
|
|
configData: new ConfigurationDataLoginTokin())
|
|
{
|
|
CommentsUrl = SiteLink + "/torrents/{0}";
|
|
IsConfigured = false;
|
|
handler = new HttpClientHandler
|
|
{
|
|
AllowAutoRedirect = true
|
|
};
|
|
client = new HttpClient(handler);
|
|
}
|
|
|
|
async Task<string> GetAuthToken(bool forceFetch = false)
|
|
{
|
|
if (!forceFetch && configData.LastTokenFetchDateTime > DateTime.Now - TimeSpan.FromHours(48))
|
|
{
|
|
return configData.ApiToken.Value;
|
|
}
|
|
|
|
var pairs = new Dictionary<string, string> {
|
|
{ "username", configData.Username.Value },
|
|
{ "password", configData.Password.Value }
|
|
};
|
|
|
|
var content = new FormUrlEncodedContent(pairs);
|
|
|
|
var response = await client.PostAsync(AuthUrl, content);
|
|
var responseContent = await response.Content.ReadAsStringAsync();
|
|
var jsonResponse = JObject.Parse(responseContent);
|
|
if (jsonResponse["error"] != null)
|
|
{
|
|
throw new ApplicationException((string)jsonResponse["error"]);
|
|
}
|
|
configData.ApiToken.Value = (string)jsonResponse["token"];
|
|
configData.LastTokenFetchDateTime = DateTime.Now;
|
|
return configData.ApiToken.Value;
|
|
}
|
|
|
|
public async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
|
|
{
|
|
configData.LoadValuesFromJson(configJson);
|
|
|
|
Exception tokenFetchEx = null;
|
|
try
|
|
{
|
|
await GetAuthToken(true);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
tokenFetchEx = new ExceptionWithConfigData(ex.Message, configData);
|
|
}
|
|
|
|
await ConfigureIfOK(string.Empty, tokenFetchEx == null, () =>
|
|
{
|
|
throw tokenFetchEx;
|
|
});
|
|
|
|
return IndexerConfigurationStatus.RequiresTesting;
|
|
}
|
|
|
|
// Override to load legacy config format
|
|
public override void LoadFromSavedConfiguration(JToken jsonConfig)
|
|
{
|
|
if (jsonConfig is JObject)
|
|
{
|
|
configData.ApiToken.Value = jsonConfig.Value<string>("token"); ;
|
|
configData.Username.Value = jsonConfig.Value<string>("username");
|
|
configData.Password.Value = jsonConfig.Value<string>("password");
|
|
SaveConfig();
|
|
IsConfigured = true;
|
|
return;
|
|
}
|
|
|
|
base.LoadFromSavedConfiguration(jsonConfig);
|
|
}
|
|
|
|
public async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
|
|
{
|
|
var releases = new List<ReleaseInfo>();
|
|
var searchTerm = string.IsNullOrEmpty(query.SanitizedSearchTerm) ? "%20" : query.SanitizedSearchTerm;
|
|
var searchString = searchTerm + " " + query.GetEpisodeSearchString();
|
|
var episodeSearchUrl = string.Format(SearchUrl, HttpUtility.UrlEncode(searchString));
|
|
|
|
var message = new HttpRequestMessage();
|
|
message.Method = HttpMethod.Get;
|
|
message.RequestUri = new Uri(episodeSearchUrl);
|
|
message.Headers.TryAddWithoutValidation("Authorization", await GetAuthToken());
|
|
|
|
var response = await client.SendAsync(message);
|
|
var results = await response.Content.ReadAsStringAsync();
|
|
|
|
var jsonResult = JObject.Parse(results);
|
|
try
|
|
{
|
|
var items = (JArray)jsonResult["torrents"];
|
|
foreach (var item in items)
|
|
{
|
|
var release = new ReleaseInfo();
|
|
|
|
release.MinimumRatio = 1;
|
|
release.MinimumSeedTime = 172800;
|
|
var torrentId = (string)item["id"];
|
|
release.Link = new Uri(string.Format(DownloadUrl, torrentId));
|
|
release.Title = (string)item["name"];
|
|
release.Description = release.Title;
|
|
release.Comments = new Uri(string.Format(CommentsUrl, (string)item["rewritename"]));
|
|
release.Guid = release.Comments;
|
|
|
|
var dateUtc = DateTime.ParseExact((string)item["added"], "yyyy-MM-dd HH:mm:ss", CultureInfo.InvariantCulture);
|
|
release.PublishDate = DateTime.SpecifyKind(dateUtc, DateTimeKind.Utc).ToLocalTime();
|
|
|
|
release.Seeders = ParseUtil.CoerceInt((string)item["seeders"]);
|
|
release.Peers = ParseUtil.CoerceInt((string)item["leechers"]) + release.Seeders;
|
|
|
|
release.Size = ParseUtil.CoerceLong((string)item["size"]);
|
|
|
|
releases.Add(release);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
OnParseError(results, ex);
|
|
}
|
|
return releases;
|
|
}
|
|
|
|
public override async Task<byte[]> Download(Uri link)
|
|
{
|
|
var message = new HttpRequestMessage();
|
|
message.Method = HttpMethod.Get;
|
|
message.RequestUri = link;
|
|
message.Headers.TryAddWithoutValidation("Authorization", await GetAuthToken());
|
|
|
|
var response = await client.SendAsync(message);
|
|
return await response.Content.ReadAsByteArrayAsync();
|
|
}
|
|
}
|
|
}
|