subsplease: refactor search

This commit is contained in:
Bogdan
2024-07-06 21:52:14 +03:00
parent 31091870eb
commit 565f8c482a

View File

@@ -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; }
} }
}
} }