]+id="wrapper"[^>]+data-id="(\d+)"[^>]*>/);
if (!idMatch) throw new Error("movie_id not found");
const movieId = idMatch[1];
const epListResp = await fetchv2(`https://hianime.to/ajax/v2/episode/list/${movieId}`);
const epListJson = await epListResp.json();
const epHtml = epListJson.html;
const epRegex = /
]+class="ssl-item\s+ep-item"[^>]+data-number="(\d+)"[^>]+data-id="(\d+)"[^>]*>/g;
let match;
while ((match = epRegex.exec(epHtml)) !== null) {
results.push({
href: match[2],
number: parseInt(match[1], 10)
});
}
return JSON.stringify(results);
} catch (err) {
console.error(err);
return JSON.stringify([{ id: "Error", href: "Error", number: "Error", title: "Error" }]);
}
}
async function extractStreamUrl(ID) {
try {
const serversResp = await fetchv2(`https://hianime.to/ajax/v2/episode/servers?episodeId=${ID}`);
const serversJson = await serversResp.json();
const serversHtml = serversJson.html;
const subServerMatch = serversHtml.match(/ {
try {
const sourcesResp = await fetchv2(`https://hianime.to/ajax/v2/episode/sources?id=${serverId}`);
const sourcesJson = await sourcesResp.json();
const iframeUrl = sourcesJson.link;
if (!iframeUrl) return null;
const iframeResp = await fetchv2(iframeUrl, headers);
const iframeHtml = await iframeResp.text();
const videoTagMatch = iframeHtml.match(/data-id="([^"]+)"/);
if (!videoTagMatch) return null;
const fileId = videoTagMatch[1];
const nonceMatch = iframeHtml.match(/\b[a-zA-Z0-9]{48}\b/) ||
iframeHtml.match(/\b([a-zA-Z0-9]{16})\b.*?\b([a-zA-Z0-9]{16})\b.*?\b([a-zA-Z0-9]{16})\b/);
if (!nonceMatch) return null;
const nonce = nonceMatch.length === 4 ?
nonceMatch[1] + nonceMatch[2] + nonceMatch[3] :
nonceMatch[0];
const urlParts = iframeUrl.split('/');
const protocol = iframeUrl.startsWith('https') ? 'https:' : 'http:';
const hostname = urlParts[2];
const defaultDomain = `${protocol}//${hostname}/`;
const getSourcesUrl = `${defaultDomain}embed-2/v3/e-1/getSources?id=${fileId}&_k=${nonce}`;
const getSourcesResp = await fetchv2(getSourcesUrl, headers);
const getSourcesJson = await getSourcesResp.json();
console.log(JSON.stringify(getSourcesJson));
const videoUrl = getSourcesJson.sources?.[0]?.file || "";
if (!videoUrl) return null;
const streamHeaders = {
"Referer": defaultDomain,
"User-Agent": "Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/137.0.0.0 Mobile Safari/537.36"
};
return {
title: title,
streamUrl: videoUrl,
headers: streamHeaders,
sourcesData: getSourcesJson
};
} catch (e) {
console.log(`${title} failed:`, e);
return null;
}
};
const serverPromises = [];
if (subServerId) serverPromises.push(processServer(subServerId, "SUB"));
if (dubServerId) serverPromises.push(processServer(dubServerId, "DUB"));
const results = await Promise.all(serverPromises);
const streams = results.filter(r => r !== null);
if (streams.length === 0) {
return "https://error.org/";
}
const englishTrack = streams[0].sourcesData.tracks?.find(t => t.kind === "captions" && t.label === "English");
const subtitle = englishTrack ? englishTrack.file : "";
const finalStreams = streams.map(s => ({
title: s.title,
streamUrl: s.streamUrl,
headers: s.headers
}));
console.log(JSON.stringify({
streams: finalStreams,
subtitle: subtitle
}));
return JSON.stringify({
streams: finalStreams,
subtitles: subtitle
});
} catch (err) {
console.error(err);
return "https://error.org/";
}
}
function decodeHtmlEntities(text) {
if (!text) {
return "";
}
return text
.replace(/(\d+);/g, (match, dec) => String.fromCharCode(dec))
.replace(/([0-9a-fA-F]+);/g, (match, hex) => String.fromCharCode(parseInt(hex, 16)))
.replace(/"/g, "\"")
.replace(/&/g, "&")
.replace(/</g, "<")
.replace(/>/g, ">")
.replace(/ /g, " ");
}