mirror of
https://github.com/Jackett/Jackett.git
synced 2025-09-17 17:34:09 +02:00
subsplease: refactor search
This commit is contained in:
@@ -3,7 +3,6 @@ using System.Collections.Generic;
|
|||||||
using System.Collections.Specialized;
|
using System.Collections.Specialized;
|
||||||
using System.Diagnostics.CodeAnalysis;
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Jackett.Common.Extensions;
|
using Jackett.Common.Extensions;
|
||||||
@@ -12,6 +11,7 @@ using Jackett.Common.Models;
|
|||||||
using Jackett.Common.Models.IndexerConfig;
|
using Jackett.Common.Models.IndexerConfig;
|
||||||
using Jackett.Common.Services.Interfaces;
|
using Jackett.Common.Services.Interfaces;
|
||||||
using Jackett.Common.Utils;
|
using Jackett.Common.Utils;
|
||||||
|
using Jackett.Common.Utils.Clients;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using Newtonsoft.Json.Linq;
|
using Newtonsoft.Json.Linq;
|
||||||
using NLog;
|
using NLog;
|
||||||
@@ -35,11 +35,7 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
|
|
||||||
public override TorznabCapabilities TorznabCaps => SetCapabilities();
|
public override TorznabCapabilities TorznabCaps => SetCapabilities();
|
||||||
|
|
||||||
private string ApiEndpoint => SiteLink + "api/?";
|
public SubsPlease(IIndexerConfigurationService configService, WebClient wc, Logger l, IProtectionService ps, ICacheService cs)
|
||||||
|
|
||||||
private static readonly Regex _RegexSize = new Regex(@"\&xl=(?<size>\d+)", RegexOptions.Compiled | RegexOptions.IgnoreCase);
|
|
||||||
|
|
||||||
public SubsPlease(IIndexerConfigurationService configService, Utils.Clients.WebClient wc, Logger l, IProtectionService ps, ICacheService cs)
|
|
||||||
: base(configService: configService,
|
: base(configService: configService,
|
||||||
client: wc,
|
client: wc,
|
||||||
logger: l,
|
logger: l,
|
||||||
@@ -69,6 +65,16 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
return caps;
|
return caps;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public override IIndexerRequestGenerator GetRequestGenerator()
|
||||||
|
{
|
||||||
|
return new SubsPleaseRequestGenerator(SiteLink);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IParseIndexerResponse GetParser()
|
||||||
|
{
|
||||||
|
return new SubsPleaseParser(SiteLink);
|
||||||
|
}
|
||||||
|
|
||||||
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
|
public override async Task<IndexerConfigurationStatus> ApplyConfiguration(JToken configJson)
|
||||||
{
|
{
|
||||||
LoadValuesFromJson(configJson);
|
LoadValuesFromJson(configJson);
|
||||||
@@ -80,91 +86,128 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
return IndexerConfigurationStatus.Completed;
|
return IndexerConfigurationStatus.Completed;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the search string is empty use the latest releases
|
|
||||||
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
|
protected override async Task<IEnumerable<ReleaseInfo>> PerformQuery(TorznabQuery query)
|
||||||
=> query.IsTest || string.IsNullOrWhiteSpace(query.SearchTerm)
|
{
|
||||||
? await FetchNewReleases()
|
var releases = await base.PerformQuery(query);
|
||||||
: await PerformSearch(query);
|
|
||||||
|
|
||||||
private async Task<IEnumerable<ReleaseInfo>> PerformSearch(TorznabQuery query)
|
if (query.SearchTerm.IsNotNullOrWhiteSpace())
|
||||||
|
{
|
||||||
|
releases = releases.Where(release => query.MatchQueryStringAND(release.Title));
|
||||||
|
|
||||||
|
// If we detected a resolution in the search terms earlier, filter by it
|
||||||
|
var resolutionMatch = Regex.Match(query.SearchTerm, @"\d{3,4}p", RegexOptions.IgnoreCase);
|
||||||
|
|
||||||
|
if (resolutionMatch.Success)
|
||||||
|
{
|
||||||
|
releases = releases.Where(release => release.Title.IndexOf(resolutionMatch.Value, StringComparison.OrdinalIgnoreCase) >= 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return releases;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class SubsPleaseRequestGenerator : IIndexerRequestGenerator
|
||||||
|
{
|
||||||
|
private readonly string _siteLink;
|
||||||
|
|
||||||
|
private static readonly Regex _ResolutionRegex = new Regex(@"\d{3,4}p", RegexOptions.Compiled | RegexOptions.IgnoreCase);
|
||||||
|
|
||||||
|
public SubsPleaseRequestGenerator(string siteLink)
|
||||||
|
{
|
||||||
|
_siteLink = siteLink;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IndexerPageableRequestChain GetSearchRequests(TorznabQuery query)
|
||||||
|
{
|
||||||
|
var pageableRequests = new IndexerPageableRequestChain();
|
||||||
|
|
||||||
|
var queryParameters = new NameValueCollection
|
||||||
|
{
|
||||||
|
{ "tz", "UTC" }
|
||||||
|
};
|
||||||
|
|
||||||
|
if (query.SearchTerm.IsNullOrWhiteSpace())
|
||||||
|
{
|
||||||
|
queryParameters.Set("f", "latest");
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
// If the search terms contain [SubsPlease] or SubsPlease, remove them from the query sent to the API
|
// If the search terms contain [SubsPlease] or SubsPlease, remove them from the query sent to the API
|
||||||
var searchTerm = Regex.Replace(query.SearchTerm, "\\[?SubsPlease\\]?\\s*", string.Empty, RegexOptions.IgnoreCase).Trim();
|
var searchTerm = Regex.Replace(query.SearchTerm, "\\[?SubsPlease\\]?\\s*", string.Empty, RegexOptions.IgnoreCase).Trim();
|
||||||
|
|
||||||
// If the search terms contain a resolution, remove it from the query sent to the API
|
// If the search terms contain a resolution, remove it from the query sent to the API
|
||||||
var resMatch = Regex.Match(searchTerm, "\\d{3,4}[p|P]");
|
var resolutionMatch = _ResolutionRegex.Match(searchTerm);
|
||||||
if (resMatch.Success)
|
|
||||||
|
if (resolutionMatch.Success)
|
||||||
{
|
{
|
||||||
searchTerm = searchTerm.Replace(resMatch.Value, string.Empty);
|
searchTerm = searchTerm.Replace(resolutionMatch.Value, string.Empty);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only include season > 1 in searchTerm, format as S2 rather than S02
|
// Only include season > 1 in searchTerm, format as S2 rather than S02
|
||||||
if (query.Season != 0)
|
if (query.Season.HasValue && query.Season.Value > 1)
|
||||||
{
|
{
|
||||||
searchTerm = query.Season == 1 ? searchTerm : searchTerm + $" S{query.Season}";
|
searchTerm += $" S{query.Season}";
|
||||||
query.Season = 0;
|
query.Season = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
var queryParameters = new NameValueCollection
|
queryParameters.Set("f", "search");
|
||||||
|
queryParameters.Set("s", searchTerm);
|
||||||
|
}
|
||||||
|
|
||||||
|
pageableRequests.Add(GetRequest(queryParameters));
|
||||||
|
|
||||||
|
return pageableRequests;
|
||||||
|
}
|
||||||
|
|
||||||
|
private IEnumerable<IndexerRequest> GetRequest(NameValueCollection queryParameters)
|
||||||
{
|
{
|
||||||
{ "f", "search" },
|
var searchUrl = $"{_siteLink}api/?{queryParameters.GetQueryString()}";
|
||||||
{ "tz", "America/New_York" },
|
|
||||||
{ "s", searchTerm }
|
var webRequest = new WebRequest
|
||||||
|
{
|
||||||
|
Url = searchUrl,
|
||||||
|
Headers = new Dictionary<string, string>
|
||||||
|
{
|
||||||
|
{ "Accept", "application/json" },
|
||||||
|
}
|
||||||
};
|
};
|
||||||
var response = await RequestWithCookiesAndRetryAsync(ApiEndpoint + queryParameters.GetQueryString());
|
|
||||||
if (response.Status != HttpStatusCode.OK)
|
yield return new IndexerRequest(webRequest);
|
||||||
{
|
}
|
||||||
throw new WebException($"SubsPlease search returned unexpected result. Expected 200 OK but got {response.Status}.", WebExceptionStatus.ProtocolError);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var results = ParseApiResults(response.ContentString);
|
public class SubsPleaseParser : IParseIndexerResponse
|
||||||
var filteredResults = results.Where(release => query.MatchQueryStringAND(release.Title));
|
|
||||||
|
|
||||||
// If we detected a resolution in the search terms earlier, filter by it
|
|
||||||
if (resMatch.Success)
|
|
||||||
{
|
{
|
||||||
filteredResults = filteredResults.Where(release => release.Title.IndexOf(resMatch.Value, StringComparison.OrdinalIgnoreCase) >= 0);
|
private readonly string _siteLink;
|
||||||
|
|
||||||
|
private static readonly Regex _RegexSize = new Regex(@"\&xl=(?<size>\d+)", RegexOptions.Compiled | RegexOptions.IgnoreCase);
|
||||||
|
|
||||||
|
public SubsPleaseParser(string siteLink)
|
||||||
|
{
|
||||||
|
_siteLink = siteLink;
|
||||||
}
|
}
|
||||||
|
|
||||||
return filteredResults;
|
public IList<ReleaseInfo> ParseResponse(IndexerResponse indexerResponse)
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<IEnumerable<ReleaseInfo>> FetchNewReleases()
|
|
||||||
{
|
{
|
||||||
var queryParameters = new NameValueCollection
|
var releases = new List<ReleaseInfo>();
|
||||||
{
|
|
||||||
{ "f", "latest" },
|
|
||||||
{ "tz", "America/New_York" }
|
|
||||||
};
|
|
||||||
var response = await RequestWithCookiesAndRetryAsync(ApiEndpoint + queryParameters.GetQueryString());
|
|
||||||
if (response.Status != HttpStatusCode.OK)
|
|
||||||
{
|
|
||||||
throw new WebException($"SubsPlease search returned unexpected result. Expected 200 OK but got {response.Status}.", WebExceptionStatus.ProtocolError);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ParseApiResults(response.ContentString);
|
|
||||||
}
|
|
||||||
|
|
||||||
private List<ReleaseInfo> ParseApiResults(string json)
|
|
||||||
{
|
|
||||||
var releaseInfo = new List<ReleaseInfo>();
|
|
||||||
|
|
||||||
// When there are no results, the API returns an empty array or empty response instead of an object
|
// When there are no results, the API returns an empty array or empty response instead of an object
|
||||||
if (string.IsNullOrWhiteSpace(json) || json == "[]")
|
if (indexerResponse.Content.IsNullOrWhiteSpace() || indexerResponse.Content == "[]")
|
||||||
{
|
{
|
||||||
return releaseInfo;
|
return releases;
|
||||||
}
|
}
|
||||||
|
|
||||||
var releases = JsonConvert.DeserializeObject<Dictionary<string, SubsPleaseRelease>>(json);
|
var jsonResponse = JsonConvert.DeserializeObject<Dictionary<string, SubsPleaseRelease>>(indexerResponse.Content);
|
||||||
|
|
||||||
foreach (var keyValue in releases)
|
foreach (var r in jsonResponse.Values)
|
||||||
{
|
{
|
||||||
var r = keyValue.Value;
|
foreach (var d in r.Downloads)
|
||||||
|
|
||||||
var baseRelease = new ReleaseInfo
|
|
||||||
{
|
{
|
||||||
Details = new Uri(SiteLink + $"shows/{r.Page}/"),
|
var release = new ReleaseInfo
|
||||||
PublishDate = r.ReleaseDate.DateTime,
|
{
|
||||||
|
Details = new Uri($"{_siteLink}shows/{r.Page}/"),
|
||||||
|
PublishDate = r.ReleaseDate.LocalDateTime,
|
||||||
Files = 1,
|
Files = 1,
|
||||||
Category = new List<int> { TorznabCatType.TVAnime.ID },
|
Category = new List<int> { TorznabCatType.TVAnime.ID },
|
||||||
Seeders = 1,
|
Seeders = 1,
|
||||||
@@ -175,26 +218,28 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
UploadVolumeFactor = 1
|
UploadVolumeFactor = 1
|
||||||
};
|
};
|
||||||
|
|
||||||
if (r.Episode.ToLowerInvariant() == "movie")
|
if (r.ImageUrl.IsNotNullOrWhiteSpace())
|
||||||
{
|
{
|
||||||
baseRelease.Category.Add(TorznabCatType.MoviesOther.ID);
|
release.Poster = new Uri(_siteLink + r.ImageUrl.TrimStart('/'));
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (var d in r.Downloads)
|
if (r.Episode.ToLowerInvariant() == "movie")
|
||||||
{
|
{
|
||||||
var release = (ReleaseInfo)baseRelease.Clone();
|
release.Category.Add(TorznabCatType.MoviesOther.ID);
|
||||||
|
}
|
||||||
|
|
||||||
// Ex: [SubsPlease] Shingeki no Kyojin (The Final Season) - 64 (1080p)
|
// Ex: [SubsPlease] Shingeki no Kyojin (The Final Season) - 64 (1080p)
|
||||||
release.Title += $"[SubsPlease] {r.Show} - {r.Episode} ({d.Resolution}p)";
|
release.Title = $"[SubsPlease] {r.Show} - {r.Episode} ({d.Resolution}p)";
|
||||||
release.MagnetUri = new Uri(d.Magnet);
|
release.MagnetUri = new Uri(d.Magnet);
|
||||||
release.Link = null;
|
release.Link = null;
|
||||||
release.Guid = new Uri(d.Magnet);
|
release.Guid = new Uri(d.Magnet);
|
||||||
release.Size = GetReleaseSize(d);
|
release.Size = GetReleaseSize(d);
|
||||||
|
|
||||||
releaseInfo.Add(release);
|
releases.Add(release);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return releaseInfo;
|
return releases;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static long GetReleaseSize(SubsPleaseDownloadInfo info)
|
private static long GetReleaseSize(SubsPleaseDownloadInfo info)
|
||||||
@@ -220,6 +265,7 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
_ => 1.Gigabytes()
|
_ => 1.Gigabytes()
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public class SubsPleaseRelease
|
public class SubsPleaseRelease
|
||||||
{
|
{
|
||||||
@@ -231,6 +277,8 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
public string Episode { get; set; }
|
public string Episode { get; set; }
|
||||||
public SubsPleaseDownloadInfo[] Downloads { get; set; }
|
public SubsPleaseDownloadInfo[] Downloads { get; set; }
|
||||||
public string Xdcc { get; set; }
|
public string Xdcc { get; set; }
|
||||||
|
|
||||||
|
[JsonProperty("image_url")]
|
||||||
public string ImageUrl { get; set; }
|
public string ImageUrl { get; set; }
|
||||||
public string Page { get; set; }
|
public string Page { get; set; }
|
||||||
}
|
}
|
||||||
@@ -241,5 +289,4 @@ namespace Jackett.Common.Indexers.Definitions
|
|||||||
public string Resolution { get; set; }
|
public string Resolution { get; set; }
|
||||||
public string Magnet { get; set; }
|
public string Magnet { get; set; }
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user