mirror of
https://github.com/lampame/lampac-ukraine.git
synced 2026-04-16 17:32:20 +00:00
Cloacking
This commit is contained in:
parent
80bef4adbc
commit
7a4f3f18e9
@ -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,33 +31,14 @@ namespace Uaflix.Controllers
|
||||
async public Task<ActionResult> 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)));
|
||||
|
||||
@ -65,244 +47,45 @@ namespace Uaflix.Controllers
|
||||
|
||||
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);
|
||||
else // Выбор эпизода
|
||||
{
|
||||
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?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");
|
||||
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<List<Uaflix.Models.EpisodeLinkInfo>> 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<Uaflix.Models.EpisodeLinkInfo> 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<HeadersModel>() { 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<Uaflix.Models.EpisodeLinkInfo>() { 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<Uaflix.Models.EpisodeLinkInfo>();
|
||||
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<Uaflix.Models.PlayResult> 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<HeadersModel>() { 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<List<(string link, string quality)>> ParseAllZetvideoSources(string iframeUrl)
|
||||
{
|
||||
var result = new List<(string link, string quality)>();
|
||||
var html = await Http.Get(iframeUrl, headers: new List<HeadersModel>() { 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<List<(string link, string quality)>> ParseAllAshdiSources(string iframeUrl)
|
||||
{
|
||||
var result = new List<(string link, string quality)>();
|
||||
var html = await Http.Get(iframeUrl, headers: new List<HeadersModel>() { 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<SubtitleTpl?> GetAshdiSubtitles(string id)
|
||||
{
|
||||
var html = await Http.Get($"https://ashdi.vip/vod/{id}", headers: new List<HeadersModel>() { 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
|
||||
}
|
||||
}
|
||||
|
||||
@ -13,6 +13,8 @@ namespace Uaflix
|
||||
/// </summary>
|
||||
public static void loaded(InitspaceModel initspace)
|
||||
{
|
||||
// streamproxy: false - замовчуванням вимкнено, але модуль сумісний з streamproxy=true
|
||||
// Клоакінг посилань для серіалів дозволяє працювати незалежно від налаштування streamproxy
|
||||
UaFlix = new OnlinesSettings("Uaflix", "https://uafix.net", streamproxy: false, useproxy: false)
|
||||
{
|
||||
displayname = "🇺🇦 UaFlix",
|
||||
|
||||
@ -11,6 +11,7 @@ using Shared.Engine;
|
||||
using Uaflix.Models;
|
||||
using System.Linq;
|
||||
using Shared.Models.Templates;
|
||||
using System.Net;
|
||||
|
||||
namespace Uaflix
|
||||
{
|
||||
@ -62,27 +63,195 @@ 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<Uaflix.Models.EpisodeLinkInfo>() { new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl } };
|
||||
res = new List<Uaflix.Models.EpisodeLinkInfo>() { new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl, title = filmTitle } };
|
||||
_hybridCache.Set(memKey, res, cacheTime(20));
|
||||
return res;
|
||||
}
|
||||
|
||||
// Для серіалів використовується GetPaginationInfo
|
||||
return null;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_onLog($"UaFlix search error: {ex.Message}");
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
public async Task<FilmInfo> GetFilmInfo(string filmUrl)
|
||||
{
|
||||
string memKey = $"UaFlix:filminfo:{filmUrl}";
|
||||
if (_hybridCache.TryGetValue(memKey, out FilmInfo res))
|
||||
return res;
|
||||
|
||||
try
|
||||
{
|
||||
var headers = new List<HeadersModel>() { 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);
|
||||
|
||||
res = new List<Uaflix.Models.EpisodeLinkInfo>();
|
||||
var episodeNodes = doc.DocumentNode.SelectNodes("//div[contains(@class, 'frels2')]//a[contains(@class, 'vi-img')]");
|
||||
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<PaginationInfo> 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<HeadersModel>() { 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<EpisodeLinkInfo>();
|
||||
var seasonUrls = new HashSet<string>();
|
||||
|
||||
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.Reverse().ToList())
|
||||
foreach (var episodeNode in episodeNodes)
|
||||
{
|
||||
string episodeUrl = episodeNode.GetAttributeValue("href", "");
|
||||
if (!episodeUrl.StartsWith("http"))
|
||||
@ -91,7 +260,7 @@ namespace Uaflix
|
||||
var match = Regex.Match(episodeUrl, @"season-(\d+).*?episode-(\d+)");
|
||||
if (match.Success)
|
||||
{
|
||||
res.Add(new Uaflix.Models.EpisodeLinkInfo
|
||||
allEpisodes.Add(new EpisodeLinkInfo
|
||||
{
|
||||
url = episodeUrl,
|
||||
title = episodeNode.SelectSingleNode(".//div[@class='vi-rate']")?.InnerText.Trim() ?? $"Епізод {match.Groups[2].Value}",
|
||||
@ -101,24 +270,28 @@ namespace Uaflix
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (res.Count == 0)
|
||||
paginationInfo.Episodes = allEpisodes.OrderBy(e => e.season).ThenBy(e => e.episode).ToList();
|
||||
|
||||
if (paginationInfo.Episodes.Any())
|
||||
{
|
||||
var iframe = doc.DocumentNode.SelectSingleNode("//div[contains(@class, 'video-box')]//iframe[contains(@src, 'ashdi.vip/serial/')]");
|
||||
if (iframe != null)
|
||||
var uniqueSeasons = paginationInfo.Episodes.Select(e => e.season).Distinct().OrderBy(se => se);
|
||||
foreach (var season in uniqueSeasons)
|
||||
{
|
||||
res.Add(new Uaflix.Models.EpisodeLinkInfo() { url = filmUrl, season = 1, episode = 1 });
|
||||
paginationInfo.Seasons[season] = 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (res.Count > 0)
|
||||
_hybridCache.Set(memKey, res, cacheTime(20));
|
||||
|
||||
return res;
|
||||
if (paginationInfo.Episodes.Count > 0)
|
||||
{
|
||||
_hybridCache.Set(memKey, paginationInfo, cacheTime(20));
|
||||
return paginationInfo;
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_onLog($"UaFlix search error: {ex.Message}");
|
||||
_onLog($"UaFlix GetPaginationInfo error: {ex.Message}");
|
||||
}
|
||||
return null;
|
||||
}
|
||||
@ -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;
|
||||
}
|
||||
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user