mirror of
https://git.luna-app.eu/50n50/sources
synced 2025-12-21 21:26:19 +01:00
116 lines
3.5 KiB
JavaScript
116 lines
3.5 KiB
JavaScript
async function searchResults(keyword) {
|
|
const results = [];
|
|
const response = await fetchv2(`https://crimsonfansubs.com/?s=${keyword}`);
|
|
const html = await response.text();
|
|
|
|
const regex = /<article class="bs"[^>]*>.*?<a href="([^"]+)"[^>]*>.*?<img src="([^"]+)"[^>]*>.*?<h2[^>]*>(.*?)<\/h2>/gs;
|
|
|
|
let match;
|
|
while ((match = regex.exec(html)) !== null) {
|
|
results.push({
|
|
title: match[3].trim(),
|
|
image: match[2].trim(),
|
|
href: match[1].trim()
|
|
});
|
|
}
|
|
|
|
return JSON.stringify(results);
|
|
}
|
|
|
|
async function extractDetails(url) {
|
|
const results = [];
|
|
const response = await fetchv2(url);
|
|
const html = await response.text();
|
|
|
|
const match = html.match(/<div class="entry-content"[^>]*>([\s\S]*?)<\/div>/);
|
|
|
|
let description = "N/A";
|
|
if (match) {
|
|
description = match[1]
|
|
.replace(/<[^>]+>/g, '')
|
|
.replace(/&#(\d+);/g, (_, code) => String.fromCharCode(code))
|
|
.replace(/"/g, '"')
|
|
.replace(/'/g, "'")
|
|
.replace(/&/g, "&")
|
|
.trim();
|
|
}
|
|
|
|
results.push({
|
|
description: description,
|
|
aliases: 'N/A',
|
|
airdate: 'N/A'
|
|
});
|
|
|
|
return JSON.stringify(results);
|
|
}
|
|
|
|
async function extractEpisodes(url) {
|
|
const results = [];
|
|
const response = await fetchv2(url);
|
|
const html = await response.text();
|
|
|
|
const regex = /<a href="([^"]+)">\s*<div class="epl-num">([\d.]+)<\/div>/g;
|
|
|
|
let match;
|
|
while ((match = regex.exec(html)) !== null) {
|
|
results.push({
|
|
href: match[1].trim(),
|
|
number: parseInt(match[2], 10)
|
|
});
|
|
}
|
|
results.reverse();
|
|
return JSON.stringify(results);
|
|
}
|
|
|
|
async function extractStreamUrl(url) {
|
|
try {
|
|
const response = await fetchv2(url);
|
|
const html = await response.text();
|
|
|
|
let videoId;
|
|
|
|
const geoMatch = html.match(/https:\/\/geo\.dailymotion\.com\/player\/([a-zA-Z0-9]+)\.js"\s*data-video="([a-zA-Z0-9]+)"/);
|
|
if (geoMatch) {
|
|
videoId = geoMatch[2];
|
|
} else {
|
|
const embedMatch = html.match(/https:\/\/www\.dailymotion\.com\/embed\/video\/([a-zA-Z0-9]+)/);
|
|
if (embedMatch) {
|
|
videoId = embedMatch[1];
|
|
} else {
|
|
return "no dailymotion video found";
|
|
}
|
|
}
|
|
|
|
const metaRes = await fetchv2(`https://www.dailymotion.com/player/metadata/video/${videoId}`);
|
|
const metaJson = await metaRes.json();
|
|
const hlsLink = metaJson.qualities?.auto?.[0]?.url;
|
|
if (!hlsLink) return "no hls";
|
|
|
|
async function getBestHls(hlsUrl) {
|
|
try {
|
|
const res = await fetchv2(hlsUrl);
|
|
const text = await res.text();
|
|
const regex = /#EXT-X-STREAM-INF:.*RESOLUTION=(\d+)x(\d+).*?\n(https?:\/\/[^\n]+)/g;
|
|
const streams = [];
|
|
let match;
|
|
while ((match = regex.exec(text)) !== null) {
|
|
streams.push({ width: parseInt(match[1]), height: parseInt(match[2]), url: match[3] });
|
|
}
|
|
if (streams.length === 0) return hlsUrl;
|
|
streams.sort((a, b) => b.height - a.height);
|
|
return streams[0].url;
|
|
} catch {
|
|
return hlsUrl;
|
|
}
|
|
}
|
|
|
|
const bestHls = await getBestHls(hlsLink);
|
|
return bestHls;
|
|
|
|
} catch {
|
|
const empty = "{ streams: [";
|
|
console.log("Extracted stream result:" + JSON.stringify(empty));
|
|
return JSON.stringify(empty);
|
|
}
|
|
}
|
|
|