From 7a4f3f18e9035e9213ab260c458b92477f09e981 Mon Sep 17 00:00:00 2001 From: Felix Date: Wed, 24 Sep 2025 20:01:07 +0300 Subject: [PATCH] Cloacking --- Uaflix/Controller.cs | 277 +++++------------------------------------ Uaflix/ModInit.cs | 2 + Uaflix/UaflixInvoke.cs | 258 +++++++++++++++++++++++++++++++------- 3 files changed, 248 insertions(+), 289 deletions(-) diff --git a/Uaflix/Controller.cs b/Uaflix/Controller.cs index 8c10443..7d33808 100644 --- a/Uaflix/Controller.cs +++ b/Uaflix/Controller.cs @@ -1,4 +1,5 @@ -using Shared.Engine; +using Shared.Models.Templates; +using Shared.Engine; using System; using System.Threading.Tasks; using Microsoft.AspNetCore.Mvc; @@ -30,279 +31,61 @@ namespace Uaflix.Controllers async public Task Index(long id, string imdb_id, long kinopoisk_id, string title, string original_title, string original_language, int year, string source, int serial, string account_email, string t, int s = -1, int e = -1, bool play = false, bool rjson = false) { var init = await loadKit(ModInit.UaFlix); - if (!init.enable) + if (await IsBadInitialization(init)) return Forbid(); var invoke = new UaflixInvoke(init, hybridCache, OnLog, proxyManager); - var episodesInfo = await invoke.Search(imdb_id, kinopoisk_id, title, original_title, year, serial == 0); - if (episodesInfo == null) - return Content("Uaflix", "text/html; charset=utf-8"); - if (play) { - var episode = episodesInfo.FirstOrDefault(ep => ep.season == s && ep.episode == e); - if (serial == 0) // для фильма берем первый - episode = episodesInfo.FirstOrDefault(); - - if (episode == null) - return Content("Uaflix", "text/html; charset=utf-8"); - - var playResult = await invoke.ParseEpisode(episode.url); - - if (!string.IsNullOrEmpty(playResult.ashdi_url)) - { - string ashdi_kp = Regex.Match(playResult.ashdi_url, "/serial/([0-9]+)").Groups[1].Value; - if (!string.IsNullOrEmpty(ashdi_kp)) - return Redirect($"/ashdi?kinopoisk_id={ashdi_kp}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&s={s}&e={e}"); - } - + var playResult = await invoke.ParseEpisode(t); if (playResult.streams != null && playResult.streams.Count > 0) return Redirect(HostStreamProxy(init, accsArgs(playResult.streams.First().link))); - + return Content("Uaflix", "text/html; charset=utf-8"); } if (serial == 1) { + var paginationInfo = await invoke.GetPaginationInfo(imdb_id, kinopoisk_id, title, original_title, year); + if (paginationInfo == null || paginationInfo.Episodes == null) + return Content("Uaflix", "text/html; charset=utf-8"); + if (s == -1) // Выбор сезона { - var seasons = episodesInfo.GroupBy(ep => ep.season).ToDictionary(k => k.Key, v => v.ToList()); - var season_tpl = new SeasonTpl(seasons.Count); - foreach (var season in seasons.OrderBy(i => i.Key)) + var seasons = paginationInfo.Episodes.Select(se => se.season).Distinct().OrderBy(se => se); + var season_tpl = new SeasonTpl(seasons.Count()); + + foreach (var season in seasons) { - string link = $"{host}/uaflix?imdb_id={imdb_id}&kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&year={year}&serial=1&s={season.Key}"; - season_tpl.Append($"Сезон {season.Key}", link, $"{season.Key}"); + string link = $"{host}/uaflix?imdb_id={imdb_id}&kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&year={year}&serial=1&s={season}"; + season_tpl.Append($"Сезон {season}", link, $"{season}"); } return rjson ? Content(season_tpl.ToJson(), "application/json; charset=utf-8") : Content(season_tpl.ToHtml(), "text/html; charset=utf-8"); } - - // Выбор эпизода - var episodes = episodesInfo.Where(ep => ep.season == s).OrderBy(ep => ep.episode).ToList(); - var movie_tpl = new MovieTpl(title, original_title, episodes.Count); - foreach(var ep in episodes) + else // Выбор эпизода { - string link = $"{host}/uaflix?imdb_id={imdb_id}&kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&year={year}&serial=1&s={s}&e={ep.episode}&play=true"; - movie_tpl.Append(ep.title, accsArgs(link), method: "play"); + var episodes = paginationInfo.Episodes.Where(ep => ep.season == s).OrderBy(ep => ep.episode).ToList(); + var episode_tpl = new EpisodeTpl(); + foreach(var ep in episodes) + { + string link = $"{host}/uaflix?t={HttpUtility.UrlEncode(ep.url)}&play=true"; + episode_tpl.Append(ep.title, title, ep.season.ToString(), ep.episode.ToString(), accsArgs(link)); + } + return rjson ? Content(episode_tpl.ToJson(), "application/json; charset=utf-8") : Content(episode_tpl.ToHtml(), "text/html; charset=utf-8"); } - return rjson ? Content(movie_tpl.ToJson(), "application/json; charset=utf-8") : Content(movie_tpl.ToHtml(), "text/html; charset=utf-8"); } else // Фильм { - string link = $"{host}/uaflix?imdb_id={imdb_id}&kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&year={year}&play=true"; + var episodesInfo = await invoke.Search(imdb_id, kinopoisk_id, title, original_title, year, true); + if (episodesInfo == null || episodesInfo.Count == 0) + return Content("Uaflix", "text/html; charset=utf-8"); + + string link = $"{host}/uaflix?t={HttpUtility.UrlEncode(episodesInfo[0].url)}&play=true"; var tpl = new MovieTpl(title, original_title, 1); - tpl.Append(title, accsArgs(link), method: "play"); + tpl.Append(episodesInfo[0].title, accsArgs(link), method: "play"); return rjson ? Content(tpl.ToJson(), "application/json; charset=utf-8") : Content(tpl.ToHtml(), "text/html; charset=utf-8"); } } - - async ValueTask> search(OnlinesSettings init, string imdb_id, long kinopoisk_id, string title, string original_title, int year, bool isfilm = false) - { - string memKey = $"UaFlix:search:{kinopoisk_id}:{imdb_id}"; - if (hybridCache.TryGetValue(memKey, out List res)) - return res; - - try - { - string filmTitle = !string.IsNullOrEmpty(title) ? title : original_title; - string searchUrl = $"{init.host}/index.php?do=search&subaction=search&story={HttpUtility.UrlEncode(filmTitle)}"; - var headers = new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", init.host) }; - - var searchHtml = await Http.Get(searchUrl, headers: headers); - var doc = new HtmlDocument(); - doc.LoadHtml(searchHtml); - - var filmNodes = doc.DocumentNode.SelectNodes("//a[contains(@class, 'sres-wrap')]"); - if (filmNodes == null) return null; - - string filmUrl = null; - foreach (var filmNode in filmNodes) - { - var h2Node = filmNode.SelectSingleNode(".//h2"); - if (h2Node == null || !h2Node.InnerText.Trim().ToLower().Contains(filmTitle.ToLower())) continue; - - var descNode = filmNode.SelectSingleNode(".//div[contains(@class, 'sres-desc')]"); - if (year > 0 && (descNode?.InnerText ?? "").Contains(year.ToString())) - { - filmUrl = filmNode.GetAttributeValue("href", ""); - break; - } - } - - if (filmUrl == null) - filmUrl = filmNodes.FirstOrDefault()?.GetAttributeValue("href", ""); - - if (!filmUrl.StartsWith("http")) - filmUrl = init.host + filmUrl; - - if (isfilm) - { - res = new List() { new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl } }; - hybridCache.Set(memKey, res, cacheTime(20)); - return res; - } - - var filmHtml = await Http.Get(filmUrl, headers: headers); - doc.LoadHtml(filmHtml); - - res = new List(); - var episodeNodes = doc.DocumentNode.SelectNodes("//div[contains(@class, 'frels2')]//a[contains(@class, 'vi-img')]"); - if (episodeNodes != null) - { - foreach (var episodeNode in episodeNodes.Reverse().ToList()) - { - string episodeUrl = episodeNode.GetAttributeValue("href", ""); - if (!episodeUrl.StartsWith("http")) - episodeUrl = init.host + episodeUrl; - - var match = Regex.Match(episodeUrl, @"season-(\d+).*?episode-(\d+)"); - if (match.Success) - { - res.Add(new Uaflix.Models.EpisodeLinkInfo - { - url = episodeUrl, - title = episodeNode.SelectSingleNode(".//div[@class='vi-rate']")?.InnerText.Trim() ?? $"Епізод {match.Groups[2].Value}", - season = int.Parse(match.Groups[1].Value), - episode = int.Parse(match.Groups[2].Value) - }); - } - } - } - - if (res.Count == 0) - { - var iframe = doc.DocumentNode.SelectSingleNode("//div[contains(@class, 'video-box')]//iframe[contains(@src, 'ashdi.vip/serial/')]"); - if (iframe != null) - { - res.Add(new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl, season = 1, episode = 1 }); - } - } - - if (res.Count > 0) - hybridCache.Set(memKey, res, cacheTime(20)); - - return res; - } - catch (Exception ex) - { - OnLog($"UaFlix search error: {ex.Message}"); - } - return null; - } - - async Task ParseEpisode(OnlinesSettings init, string url) - { - var result = new Uaflix.Models.PlayResult() { streams = new List<(string, string)>() }; - try - { - string html = await Http.Get(url, headers: new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", init.host) }); - var doc = new HtmlDocument(); - doc.LoadHtml(html); - - var iframe = doc.DocumentNode.SelectSingleNode("//div[contains(@class, 'video-box')]//iframe"); - if (iframe != null) - { - string iframeUrl = iframe.GetAttributeValue("src", "").Replace("&", "&"); - if (iframeUrl.StartsWith("//")) - iframeUrl = "https:" + iframeUrl; - - if (iframeUrl.Contains("ashdi.vip/serial/")) - { - result.ashdi_url = iframeUrl; - return result; - } - - if (iframeUrl.Contains("zetvideo.net")) - result.streams = await ParseAllZetvideoSources(iframeUrl); - else if (iframeUrl.Contains("ashdi.vip")) - { - result.streams = await ParseAllAshdiSources(iframeUrl); - var idMatch = Regex.Match(iframeUrl, @"_(\d+)|vod/(\d+)"); - if (idMatch.Success) - { - string ashdiId = idMatch.Groups[1].Success ? idMatch.Groups[1].Value : idMatch.Groups[2].Value; - result.subtitles = await GetAshdiSubtitles(ashdiId); - } - } - } - } - catch (Exception ex) - { - OnLog($"ParseEpisode error: {ex.Message}"); - } - return result; - } - - #region Parsers - async Task> ParseAllZetvideoSources(string iframeUrl) - { - var result = new List<(string link, string quality)>(); - var html = await Http.Get(iframeUrl, headers: new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", "https://zetvideo.net/") }); - if (string.IsNullOrEmpty(html)) return result; - - var doc = new HtmlDocument(); - doc.LoadHtml(html); - - var script = doc.DocumentNode.SelectSingleNode("//script[contains(text(), 'file:')]"); - if (script != null) - { - var match = Regex.Match(script.InnerText, @"file:\s*""([^""]+\.m3u8)"); - if (match.Success) - { - result.Add((match.Groups[1].Value, "1080p")); - return result; - } - } - - var sourceNodes = doc.DocumentNode.SelectNodes("//source[contains(@src, '.m3u8')]"); - if (sourceNodes != null) - { - foreach (var node in sourceNodes) - { - result.Add((node.GetAttributeValue("src", null), node.GetAttributeValue("label", null) ?? node.GetAttributeValue("res", null) ?? "1080p")); - } - } - return result; - } - - async Task> ParseAllAshdiSources(string iframeUrl) - { - var result = new List<(string link, string quality)>(); - var html = await Http.Get(iframeUrl, headers: new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", "https://ashdi.vip/") }); - if (string.IsNullOrEmpty(html)) return result; - - var doc = new HtmlDocument(); - doc.LoadHtml(html); - - var sourceNodes = doc.DocumentNode.SelectNodes("//source[contains(@src, '.m3u8')]"); - if (sourceNodes != null) - { - foreach (var node in sourceNodes) - { - result.Add((node.GetAttributeValue("src", null), node.GetAttributeValue("label", null) ?? node.GetAttributeValue("res", null) ?? "1080p")); - } - } - return result; - } - - async Task GetAshdiSubtitles(string id) - { - var html = await Http.Get($"https://ashdi.vip/vod/{id}", headers: new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", "https://ashdi.vip/") }); - string subtitle = new Regex("subtitle(\")?:\"([^\"]+)\"").Match(html).Groups[2].Value; - if (!string.IsNullOrEmpty(subtitle)) - { - var match = new Regex("\\[([^\\]]+)\\](https?://[^\\,]+)").Match(subtitle); - var st = new Shared.Models.Templates.SubtitleTpl(); - while (match.Success) - { - st.Append(match.Groups[1].Value, match.Groups[2].Value); - match = match.NextMatch(); - } - if (!st.IsEmpty()) - return st; - } - return null; - } - #endregion } } diff --git a/Uaflix/ModInit.cs b/Uaflix/ModInit.cs index e0afb3d..1ae5a25 100644 --- a/Uaflix/ModInit.cs +++ b/Uaflix/ModInit.cs @@ -13,6 +13,8 @@ namespace Uaflix /// public static void loaded(InitspaceModel initspace) { + // streamproxy: false - замовчуванням вимкнено, але модуль сумісний з streamproxy=true + // Клоакінг посилань для серіалів дозволяє працювати незалежно від налаштування streamproxy UaFlix = new OnlinesSettings("Uaflix", "https://uafix.net", streamproxy: false, useproxy: false) { displayname = "🇺🇦 UaFlix", diff --git a/Uaflix/UaflixInvoke.cs b/Uaflix/UaflixInvoke.cs index 0e27eaa..014e99d 100644 --- a/Uaflix/UaflixInvoke.cs +++ b/Uaflix/UaflixInvoke.cs @@ -11,6 +11,7 @@ using Shared.Engine; using Uaflix.Models; using System.Linq; using Shared.Models.Templates; +using System.Net; namespace Uaflix { @@ -62,59 +63,24 @@ namespace Uaflix } } - if (filmUrl == null) + if (string.IsNullOrEmpty(filmUrl)) filmUrl = filmNodes.FirstOrDefault()?.GetAttributeValue("href", ""); + if (string.IsNullOrEmpty(filmUrl)) + return null; + if (!filmUrl.StartsWith("http")) filmUrl = _init.host + filmUrl; if (isfilm) { - res = new List() { new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl } }; + res = new List() { new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl, title = filmTitle } }; _hybridCache.Set(memKey, res, cacheTime(20)); return res; } - var filmHtml = await Http.Get(filmUrl, headers: headers, proxy: _proxyManager.Get()); - doc.LoadHtml(filmHtml); - - res = new List(); - var episodeNodes = doc.DocumentNode.SelectNodes("//div[contains(@class, 'frels2')]//a[contains(@class, 'vi-img')]"); - if (episodeNodes != null) - { - foreach (var episodeNode in episodeNodes.Reverse().ToList()) - { - string episodeUrl = episodeNode.GetAttributeValue("href", ""); - if (!episodeUrl.StartsWith("http")) - episodeUrl = _init.host + episodeUrl; - - var match = Regex.Match(episodeUrl, @"season-(\d+).*?episode-(\d+)"); - if (match.Success) - { - res.Add(new Uaflix.Models.EpisodeLinkInfo - { - url = episodeUrl, - title = episodeNode.SelectSingleNode(".//div[@class='vi-rate']")?.InnerText.Trim() ?? $"Епізод {match.Groups[2].Value}", - season = int.Parse(match.Groups[1].Value), - episode = int.Parse(match.Groups[2].Value) - }); - } - } - } - - if (res.Count == 0) - { - var iframe = doc.DocumentNode.SelectSingleNode("//div[contains(@class, 'video-box')]//iframe[contains(@src, 'ashdi.vip/serial/')]"); - if (iframe != null) - { - res.Add(new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl, season = 1, episode = 1 }); - } - } - - if (res.Count > 0) - _hybridCache.Set(memKey, res, cacheTime(20)); - - return res; + // Для серіалів використовується GetPaginationInfo + return null; } catch (Exception ex) { @@ -122,7 +88,214 @@ namespace Uaflix } return null; } + + public async Task GetFilmInfo(string filmUrl) + { + string memKey = $"UaFlix:filminfo:{filmUrl}"; + if (_hybridCache.TryGetValue(memKey, out FilmInfo res)) + return res; + try + { + var headers = new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", _init.host) }; + var filmHtml = await Http.Get(filmUrl, headers: headers, proxy: _proxyManager.Get()); + var doc = new HtmlDocument(); + doc.LoadHtml(filmHtml); + + var result = new FilmInfo + { + Url = filmUrl + }; + + var titleNode = doc.DocumentNode.SelectSingleNode("//h1[@class='h1-title']"); + if (titleNode != null) + { + result.Title = titleNode.InnerText.Trim(); + } + + var metaDuration = doc.DocumentNode.SelectSingleNode("//meta[@property='og:video:duration']"); + if (metaDuration != null) + { + string durationStr = metaDuration.GetAttributeValue("content", ""); + if (int.TryParse(durationStr, out int duration)) + { + result.Duration = duration; + } + } + + var metaActors = doc.DocumentNode.SelectSingleNode("//meta[@property='og:video:actor']"); + if (metaActors != null) + { + string actorsStr = metaActors.GetAttributeValue("content", ""); + result.Actors = actorsStr.Split(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries) + .Select(a => a.Trim()) + .ToList(); + } + + var metaDirector = doc.DocumentNode.SelectSingleNode("//meta[@property='og:video:director']"); + if (metaDirector != null) + { + result.Director = metaDirector.GetAttributeValue("content", ""); + } + + var descNode = doc.DocumentNode.SelectSingleNode("//div[@id='main-descr']//div[@itemprop='description']"); + if (descNode != null) + { + result.Description = descNode.InnerText.Trim(); + } + + var posterNode = doc.DocumentNode.SelectSingleNode("//img[@itemprop='image']"); + if (posterNode != null) + { + result.PosterUrl = posterNode.GetAttributeValue("src", ""); + if (!result.PosterUrl.StartsWith("http") && !string.IsNullOrEmpty(result.PosterUrl)) + { + result.PosterUrl = _init.host + result.PosterUrl; + } + } + + _hybridCache.Set(memKey, result, cacheTime(60)); + return result; + } + catch (Exception ex) + { + _onLog($"UaFlix GetFilmInfo error: {ex.Message}"); + } + return null; + } + + public async Task GetPaginationInfo(string imdb_id, long kinopoisk_id, string title, string original_title, int year) + { + string memKey = $"UaFlix:pagination:{kinopoisk_id}:{imdb_id}"; + if (_hybridCache.TryGetValue(memKey, out PaginationInfo res)) + return res; + + try + { + string filmTitle = !string.IsNullOrEmpty(title) ? title : original_title; + string searchUrl = $"{_init.host}/index.php?do=search&subaction=search&story={System.Web.HttpUtility.UrlEncode(filmTitle)}"; + var headers = new List() { new HeadersModel("User-Agent", "Mozilla/5.0"), new HeadersModel("Referer", _init.host) }; + + var searchHtml = await Http.Get(searchUrl, headers: headers, proxy: _proxyManager.Get()); + var searchDoc = new HtmlDocument(); + searchDoc.LoadHtml(searchHtml); + + var filmNodes = searchDoc.DocumentNode.SelectNodes("//a[contains(@class, 'sres-wrap')]"); + if (filmNodes == null) return null; + + string filmUrl = null; + foreach (var filmNode in filmNodes) + { + var h2Node = filmNode.SelectSingleNode(".//h2"); + if (h2Node == null || !h2Node.InnerText.Trim().ToLower().Contains(filmTitle.ToLower())) continue; + + var descNode = filmNode.SelectSingleNode(".//div[contains(@class, 'sres-desc')]"); + if (year > 0 && (descNode?.InnerText ?? "").Contains(year.ToString())) + { + filmUrl = filmNode.GetAttributeValue("href", ""); + break; + } + } + + if (string.IsNullOrEmpty(filmUrl)) + filmUrl = filmNodes.FirstOrDefault()?.GetAttributeValue("href", ""); + + if (string.IsNullOrEmpty(filmUrl)) + return null; + + if (!filmUrl.StartsWith("http")) + filmUrl = _init.host + filmUrl; + + var filmHtml = await Http.Get(filmUrl, headers: headers, proxy: _proxyManager.Get()); + var filmDoc = new HtmlDocument(); + filmDoc.LoadHtml(filmHtml); + + var paginationInfo = new PaginationInfo + { + SerialUrl = filmUrl + }; + + var allEpisodes = new List(); + var seasonUrls = new HashSet(); + + var seasonNodes = filmDoc.DocumentNode.SelectNodes("//div[contains(@class, 'sez-wr')]//a"); + if (seasonNodes == null) + seasonNodes = filmDoc.DocumentNode.SelectNodes("//div[contains(@class, 'fss-box')]//a"); + if (seasonNodes != null && seasonNodes.Count > 0) + { + foreach (var node in seasonNodes) + { + string pageUrl = node.GetAttributeValue("href", null); + if (!string.IsNullOrEmpty(pageUrl)) + { + if (!pageUrl.StartsWith("http")) + pageUrl = _init.host + pageUrl; + + seasonUrls.Add(pageUrl); + } + } + } + else + { + seasonUrls.Add(filmUrl); + } + + var seasonTasks = seasonUrls.Select(url => Http.Get(url, headers: headers, proxy: _proxyManager.Get()).AsTask()); + var seasonPagesHtml = await Task.WhenAll(seasonTasks); + + foreach (var html in seasonPagesHtml) + { + var pageDoc = new HtmlDocument(); + pageDoc.LoadHtml(html); + + var episodeNodes = pageDoc.DocumentNode.SelectNodes("//div[contains(@class, 'frels')]//a[contains(@class, 'vi-img')]"); + if (episodeNodes != null) + { + foreach (var episodeNode in episodeNodes) + { + string episodeUrl = episodeNode.GetAttributeValue("href", ""); + if (!episodeUrl.StartsWith("http")) + episodeUrl = _init.host + episodeUrl; + + var match = Regex.Match(episodeUrl, @"season-(\d+).*?episode-(\d+)"); + if (match.Success) + { + allEpisodes.Add(new EpisodeLinkInfo + { + url = episodeUrl, + title = episodeNode.SelectSingleNode(".//div[@class='vi-rate']")?.InnerText.Trim() ?? $"Епізод {match.Groups[2].Value}", + season = int.Parse(match.Groups[1].Value), + episode = int.Parse(match.Groups[2].Value) + }); + } + } + } + } + + paginationInfo.Episodes = allEpisodes.OrderBy(e => e.season).ThenBy(e => e.episode).ToList(); + + if (paginationInfo.Episodes.Any()) + { + var uniqueSeasons = paginationInfo.Episodes.Select(e => e.season).Distinct().OrderBy(se => se); + foreach (var season in uniqueSeasons) + { + paginationInfo.Seasons[season] = 1; + } + } + + if (paginationInfo.Episodes.Count > 0) + { + _hybridCache.Set(memKey, paginationInfo, cacheTime(20)); + return paginationInfo; + } + } + catch (Exception ex) + { + _onLog($"UaFlix GetPaginationInfo error: {ex.Message}"); + } + return null; + } + public async Task ParseEpisode(string url) { var result = new Uaflix.Models.PlayResult() { streams = new List<(string, string)>() }; @@ -163,6 +336,7 @@ namespace Uaflix { _onLog($"ParseEpisode error: {ex.Message}"); } + _onLog($"ParseEpisode result: streams.count={result.streams.Count}, ashdi_url={result.ashdi_url}"); return result; }