Skip to content

Commit faa02b2

Browse files
committed
HDVB
1 parent c84d39f commit faa02b2

File tree

6 files changed

+299
-1
lines changed

6 files changed

+299
-1
lines changed

AppInit.cs

+3
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
using Lampac.Models.LITE.Collaps;
88
using Lampac.Models.LITE.Filmix;
99
using Lampac.Models.LITE;
10+
using Lampac.Models.LITE.HDVB;
1011

1112
namespace Lampac
1213
{
@@ -87,6 +88,8 @@ public class AppInit
8788

8889
public KodikSettings Kodik = new KodikSettings("https://kodikapi.com", "http://kodik.biz", "", "", true);
8990

91+
public HDVBSettings HDVB = new HDVBSettings("https://apivb.info", "");
92+
9093

9194
public ProxySettings proxy = new ProxySettings();
9295
}

Controllers/ApiController.cs

+3
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,9 @@ public ActionResult Lite()
4040
if (!string.IsNullOrWhiteSpace(AppInit.conf.Kodik.token))
4141
addonline += "{name:'Kodik',url:'{localhost}/kodik'},";
4242

43+
if (!string.IsNullOrWhiteSpace(AppInit.conf.HDVB.token))
44+
addonline += "{name:'HDVB',url:'{localhost}/hdvb'},";
45+
4346
file = file.Replace("{addonline}", addonline);
4447
file = file.Replace("{localhost}", $"{AppInit.Host(HttpContext)}/lite");
4548

Controllers/LITE/HDVB.cs

+259
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,259 @@
1+
using System;
2+
using System.Text.RegularExpressions;
3+
using System.Threading.Tasks;
4+
using Microsoft.AspNetCore.Mvc;
5+
using Microsoft.Extensions.Caching.Memory;
6+
using System.Collections.Generic;
7+
using System.Web;
8+
using Newtonsoft.Json.Linq;
9+
using System.Linq;
10+
using Lampac.Engine;
11+
using Lampac.Engine.CORE;
12+
using Lampac.Models.LITE.HDVB;
13+
14+
namespace Lampac.Controllers.LITE
15+
{
16+
public class HDVB : BaseController
17+
{
18+
[HttpGet]
19+
[Route("lite/hdvb")]
20+
async public Task<ActionResult> Index(long kinopoisk_id, string title, string original_title, int serial, int t, int s = -1)
21+
{
22+
if (kinopoisk_id == 0)
23+
return Content(string.Empty);
24+
25+
JArray data = await search(memoryCache, kinopoisk_id);
26+
if (data == null)
27+
return Content(string.Empty);
28+
29+
bool firstjson = true;
30+
string html = "<div class=\"videos__line\">";
31+
32+
if (serial == 0)
33+
{
34+
#region Фильм
35+
foreach (var m in data)
36+
{
37+
string link = $"{AppInit.Host(HttpContext)}/lite/hdvb/video?kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&iframe={HttpUtility.UrlEncode(m.Value<string>("iframe_url"))}";
38+
html += "<div class=\"videos__item videos__movie selector " + (firstjson ? "focused" : "") + "\" media=\"\" data-json='{\"method\":\"call\",\"url\":\"" + link + "\"}'><div class=\"videos__item-imgbox videos__movie-imgbox\"></div><div class=\"videos__item-title\">" + m.Value<string>("translator") + "</div></div>";
39+
firstjson = false;
40+
}
41+
#endregion
42+
}
43+
else
44+
{
45+
#region Перевод
46+
for (int i = 0; i < data.Count; i++)
47+
{
48+
string link = $"{AppInit.Host(HttpContext)}/lite/hdvb?serial=1&kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&t={i}";
49+
50+
html += "<div class=\"videos__button selector " + (t == i ? "active" : "") + "\" data-json='{\"method\":\"link\",\"url\":\"" + link + "\"}'>" + data[i].Value<string>("translator") + "</div>";
51+
}
52+
53+
html += "</div>";
54+
#endregion
55+
56+
#region Сериал
57+
firstjson = true;
58+
html += "<div class=\"videos__line\">";
59+
60+
if (s == -1)
61+
{
62+
var serial_episodes = data[t].Value<JArray>("serial_episodes");
63+
64+
for (int i = 0; i < serial_episodes.Count; i++)
65+
{
66+
string link = $"{AppInit.Host(HttpContext)}/lite/hdvb?serial=1&kinopoisk_id={kinopoisk_id}&title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&t={t}&s={i}";
67+
68+
html += "<div class=\"videos__item videos__season selector " + (firstjson ? "focused" : "") + "\" data-json='{\"method\":\"link\",\"url\":\"" + link + "\"}'><div class=\"videos__season-layers\"></div><div class=\"videos__item-imgbox videos__season-imgbox\"><div class=\"videos__item-title videos__season-title\">" + $"{serial_episodes[i].Value<int>("season_number")} сезон" + "</div></div></div>";
69+
firstjson = false;
70+
}
71+
}
72+
else
73+
{
74+
foreach (int episode in data[t].Value<JArray>("serial_episodes")[s].Value<JArray>("episodes").ToObject<List<int>>())
75+
{
76+
string iframe = HttpUtility.UrlEncode(data[t].Value<string>("iframe_url"));
77+
string translator = HttpUtility.UrlEncode(data[t].Value<string>("translator"));
78+
int season = data[t].Value<JArray>("serial_episodes")[s].Value<int>("season_number");
79+
80+
string link = $"{AppInit.Host(HttpContext)}/lite/hdvb/serial?title={HttpUtility.UrlEncode(title)}&original_title={HttpUtility.UrlEncode(original_title)}&iframe={iframe}&t={translator}&s={season}&e={episode}";
81+
82+
html += "<div class=\"videos__item videos__movie selector " + (firstjson ? "focused" : "") + "\" media=\"\" s=\"" + s + "\" e=\"" + episode + "\" data-json='{\"method\":\"call\",\"url\":\"" + link + "\",\"title\":\"" + $"{title ?? original_title} ({episode} серия)" + "\"}'><div class=\"videos__item-imgbox videos__movie-imgbox\"></div><div class=\"videos__item-title\">" + $"{episode} серия" + "</div></div>";
83+
firstjson = false;
84+
}
85+
}
86+
#endregion
87+
}
88+
89+
return Content(html + "</div>", "text/html; charset=utf-8");
90+
}
91+
92+
#region Video
93+
[HttpGet]
94+
[Route("lite/hdvb/video")]
95+
async public Task<ActionResult> Video(string iframe, string title, string original_title)
96+
{
97+
string memKey = $"video:view:video:{iframe}";
98+
if (!memoryCache.TryGetValue(memKey, out List<(string q, string url)> streams))
99+
{
100+
string html = await HttpClient.Get(iframe, referer: $"{AppInit.conf.HDVB.apihost}/", timeoutSeconds: 8);
101+
if (html == null)
102+
return Content(string.Empty);
103+
104+
#region urim3u8
105+
string vid = "vid1666694269";
106+
string href = Regex.Match(html, "\"href\":\"([^\"]+)\"").Groups[1].Value;
107+
string csrftoken = Regex.Match(html, "\"key\":\"([^\"]+)\"").Groups[1].Value.Replace("\\", "");
108+
string file = Regex.Match(html, "\"file\":\"([^\"]+)\"").Groups[1].Value.Replace("\\", "");
109+
file = Regex.Replace(file, "^/playlist/", "/");
110+
file = Regex.Replace(file, "\\.txt$", "");
111+
112+
if (string.IsNullOrWhiteSpace(href) || string.IsNullOrWhiteSpace(file) || string.IsNullOrWhiteSpace(csrftoken))
113+
return Content(string.Empty);
114+
115+
string urim3u8 = await HttpClient.Post($"https://{vid}.{href}/playlist/{file}.txt", "", timeoutSeconds: 8, addHeaders: new List<(string name, string val)>()
116+
{
117+
("cache-control", "no-cache"),
118+
("dnt", "1"),
119+
("origin", $"https://{vid}.{href}"),
120+
("referer", iframe),
121+
("sec-ch-ua", "\"Chromium\";v=\"106\", \"Google Chrome\";v=\"106\", \"Not;A=Brand\";v=\"99\""),
122+
("sec-ch-ua-mobile", "?0"),
123+
("sec-ch-ua-platform", "\"Windows\""),
124+
("sec-fetch-dest", "empty"),
125+
("sec-fetch-mode", "cors"),
126+
("sec-fetch-site", "same-origin"),
127+
("x-csrf-token", csrftoken)
128+
});
129+
130+
if (urim3u8 == null || !urim3u8.Contains("/index.m3u8"))
131+
return Content(string.Empty);
132+
#endregion
133+
134+
string m3u8 = await HttpClient.Get(urim3u8, timeoutSeconds: 8);
135+
if (m3u8 == null)
136+
return Content(string.Empty);
137+
138+
streams = new List<(string q, string url)>();
139+
var match = new Regex("/([0-9]+)/index\\.m3u8", RegexOptions.IgnoreCase).Match(m3u8);
140+
while (match.Success)
141+
{
142+
if (!string.IsNullOrWhiteSpace(match.Groups[1].Value))
143+
streams.Insert(0, ($"{match.Groups[1].Value}p", urim3u8.Replace("/index.m3u8", $"/{match.Groups[1].Value}/index.m3u8")));
144+
145+
match = match.NextMatch();
146+
}
147+
148+
memoryCache.Set(memKey, streams, TimeSpan.FromMinutes(10));
149+
}
150+
151+
string streansquality = string.Empty;
152+
foreach (var l in streams)
153+
streansquality += $"\"{l.q}\":\"" + l.url + "\",";
154+
155+
return Content("{\"method\":\"play\",\"url\":\"" + streams[0].url + "\",\"title\":\"" + (title ?? original_title) + "\", \"quality\": {" + Regex.Replace(streansquality, ",$", "") + "}}", "application/json; charset=utf-8");
156+
}
157+
#endregion
158+
159+
#region Serial
160+
[HttpGet]
161+
[Route("lite/hdvb/serial")]
162+
async public Task<ActionResult> Serial(string iframe, string t, string s, string e, string title, string original_title)
163+
{
164+
string memKey = $"video:view:serial:{iframe}:{t}:{s}:{e}";
165+
if (!memoryCache.TryGetValue(memKey, out List<(string q, string url)> streams))
166+
{
167+
string html = await HttpClient.Get(iframe, referer: $"{AppInit.conf.HDVB.apihost}/", timeoutSeconds: 8);
168+
if (html == null)
169+
return Content(string.Empty);
170+
171+
#region playlist
172+
string vid = "vid1666694269";
173+
string href = Regex.Match(html, "\"href\":\"([^\"]+)\"").Groups[1].Value;
174+
string csrftoken = Regex.Match(html, "\"key\":\"([^\"]+)\"").Groups[1].Value.Replace("\\", "");
175+
string file = Regex.Match(html, "\"file\":\"([^\"]+)\"").Groups[1].Value.Replace("\\", "");
176+
file = Regex.Replace(file, "^/playlist/", "/");
177+
file = Regex.Replace(file, "\\.txt$", "");
178+
179+
if (string.IsNullOrWhiteSpace(href) || string.IsNullOrWhiteSpace(file) || string.IsNullOrWhiteSpace(csrftoken))
180+
return Content(string.Empty);
181+
182+
var headers = new List<(string name, string val)>()
183+
{
184+
("cache-control", "no-cache"),
185+
("dnt", "1"),
186+
("origin", $"https://{vid}.{href}"),
187+
("referer", iframe),
188+
("sec-ch-ua", "\"Chromium\";v=\"106\", \"Google Chrome\";v=\"106\", \"Not;A=Brand\";v=\"99\""),
189+
("sec-ch-ua-mobile", "?0"),
190+
("sec-ch-ua-platform", "\"Windows\""),
191+
("sec-fetch-dest", "empty"),
192+
("sec-fetch-mode", "cors"),
193+
("sec-fetch-site", "same-origin"),
194+
("x-csrf-token", csrftoken)
195+
};
196+
197+
var playlist = await HttpClient.Post<List<Folder>>($"https://{vid}.{href}/playlist/{file}.txt", "", timeoutSeconds: 8, addHeaders: headers);
198+
if (playlist == null || playlist.Count == 0)
199+
return Content(string.Empty);
200+
#endregion
201+
202+
#region urim3u8
203+
file = playlist.First(i => i.id == s).folder.First(i => i.episode == e).folder.First(i => i.title == t).file;
204+
if (string.IsNullOrWhiteSpace(file))
205+
return Content(string.Empty);
206+
207+
file = Regex.Replace(file, "^/playlist/", "/");
208+
file = Regex.Replace(file, "\\.txt$", "");
209+
210+
string urim3u8 = await HttpClient.Post($"https://{vid}.{href}/playlist/{file}.txt", "", timeoutSeconds: 8, addHeaders: headers);
211+
if (urim3u8 == null || !urim3u8.Contains("/index.m3u8"))
212+
return Content(string.Empty);
213+
#endregion
214+
215+
string m3u8 = await HttpClient.Get(urim3u8, timeoutSeconds: 8);
216+
if (m3u8 == null)
217+
return Content(string.Empty);
218+
219+
streams = new List<(string q, string url)>();
220+
var match = new Regex("/([0-9]+)/index\\.m3u8", RegexOptions.IgnoreCase).Match(m3u8);
221+
while (match.Success)
222+
{
223+
if (!string.IsNullOrWhiteSpace(match.Groups[1].Value))
224+
streams.Insert(0, ($"{match.Groups[1].Value}p", urim3u8.Replace("/index.m3u8", $"/{match.Groups[1].Value}/index.m3u8")));
225+
226+
match = match.NextMatch();
227+
}
228+
229+
memoryCache.Set(memKey, streams, TimeSpan.FromMinutes(10));
230+
}
231+
232+
string streansquality = string.Empty;
233+
foreach (var l in streams)
234+
streansquality += $"\"{l.q}\":\"" + l.url + "\",";
235+
236+
return Content("{\"method\":\"play\",\"url\":\"" + streams[0].url + "\",\"title\":\"" + (title ?? original_title) + "\", \"quality\": {" + Regex.Replace(streansquality, ",$", "") + "}}", "application/json; charset=utf-8");
237+
}
238+
#endregion
239+
240+
241+
#region search
242+
async ValueTask<JArray> search(IMemoryCache memoryCache, long kinopoisk_id)
243+
{
244+
string memKey = $"hdvb:view:{kinopoisk_id}";
245+
246+
if (!memoryCache.TryGetValue(memKey, out JArray root))
247+
{
248+
root = await HttpClient.Get<JArray>($"{AppInit.conf.HDVB.apihost}/api/videos.json?token={AppInit.conf.HDVB.token}&id_kp={kinopoisk_id}", timeoutSeconds: 8);
249+
if (root == null || root.Count == 0)
250+
return null;
251+
252+
memoryCache.Set(memKey, root, TimeSpan.FromMinutes(10));
253+
}
254+
255+
return root;
256+
}
257+
#endregion
258+
}
259+
}

Models/LITE/HDVB/Folder.cs

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
using System.Collections.Generic;
2+
3+
namespace Lampac.Models.LITE.HDVB
4+
{
5+
public class Folder
6+
{
7+
public string id { get; set; }
8+
9+
public string episode { get; set; }
10+
11+
public List<Folder> folder { get; set; }
12+
13+
public string title { get; set; }
14+
15+
public string file { get; set; }
16+
}
17+
}

Models/LITE/HDVB/HDVBSettings.cs

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
namespace Lampac.Models.LITE.HDVB
2+
{
3+
public class HDVBSettings
4+
{
5+
public HDVBSettings(string apihost, string token)
6+
{
7+
this.apihost = apihost;
8+
this.token = token;
9+
}
10+
11+
12+
public string apihost { get; set; }
13+
14+
public string token { get; set; }
15+
}
16+
}

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616
***
1717

1818
* Public online - Videocdn, Rezka, Collaps, Filmix
19-
* Private online - Bazon, Alloha, Kodik
19+
* Private online - Bazon, Alloha, HDVB, Kodik
2020
* Public Trackers - kinozal.tv, nnmclub.to, rutor.info, megapeer.vip, torrent.by, bitru.org, anilibria.tv
2121
* Private Trackers - toloka.to, rutracker.net, underver.se, selezen.net, animelayer.ru
2222
* Клубничка bongacams.com, chaturbate.com, ebalovo.pro, eporner.com, hqporner.com, porntrex.com, spankbang.com, xhamster.com, xnxx.com, xvideos.com

0 commit comments

Comments
 (0)