mirror of
https://github.com/vega-org/vega-providers.git
synced 2026-04-17 23:51:44 +00:00
fix: remove episodesLink from meta data in getMeta function fix: add unlocked cookie header in extractKmhdLink function fix: remove episodesLink from linkList in kmMovies meta fix: add base page worker links in topmovies stream fix: refactor instant link handling in topmovies stream
337 lines
11 KiB
TypeScript
337 lines
11 KiB
TypeScript
import { Stream, ProviderContext, EpisodeLink } from "../types";
|
|
|
|
const headers = {
|
|
Accept:
|
|
"text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7",
|
|
"Cache-Control": "no-store",
|
|
"Accept-Language": "en-US,en;q=0.9",
|
|
DNT: "1",
|
|
"sec-ch-ua":
|
|
'"Not_A Brand";v="8", "Chromium";v="120", "Microsoft Edge";v="120"',
|
|
"sec-ch-ua-mobile": "?0",
|
|
"sec-ch-ua-platform": '"Windows"',
|
|
"Sec-Fetch-Dest": "document",
|
|
"Sec-Fetch-Mode": "navigate",
|
|
Cookie: "popads_user_id=6ba8fe60a481387a3249f05aa058822d",
|
|
"Sec-Fetch-Site": "none",
|
|
"Sec-Fetch-User": "?1",
|
|
"Upgrade-Insecure-Requests": "1",
|
|
"User-Agent":
|
|
"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36 Edg/120.0.0.0",
|
|
};
|
|
|
|
export const getStream = async function ({
|
|
link: url,
|
|
type,
|
|
providerContext,
|
|
}: {
|
|
link: string;
|
|
type: string;
|
|
providerContext: ProviderContext;
|
|
}): Promise<Stream[]> {
|
|
const { axios, cheerio } = providerContext;
|
|
try {
|
|
const modGetEpisodeLinks = async function ({
|
|
url,
|
|
providerContext,
|
|
}: {
|
|
url: string;
|
|
providerContext: ProviderContext;
|
|
}): Promise<EpisodeLink[]> {
|
|
const { axios, cheerio } = providerContext;
|
|
try {
|
|
if (url.includes("url=")) {
|
|
url = atob(url.split("url=")[1]);
|
|
}
|
|
const res = await axios.get(url);
|
|
const html = res.data;
|
|
let $ = cheerio.load(html);
|
|
if (url.includes("url=")) {
|
|
const newUrl = $("meta[http-equiv='refresh']")
|
|
.attr("content")
|
|
?.split("url=")[1];
|
|
const res2 = await axios.get(newUrl || url);
|
|
const html2 = res2.data;
|
|
$ = cheerio.load(html2);
|
|
}
|
|
const episodeLinks: EpisodeLink[] = [];
|
|
$("h3,h4").map((i, element) => {
|
|
const seriesTitle = $(element).text();
|
|
const episodesLink = $(element).find("a").attr("href");
|
|
if (episodesLink && episodesLink !== "#") {
|
|
episodeLinks.push({
|
|
title: seriesTitle.trim() || "No title found",
|
|
link: episodesLink || "",
|
|
});
|
|
}
|
|
});
|
|
$("a.maxbutton").map((i, element) => {
|
|
const seriesTitle = $(element).children("span").text();
|
|
const episodesLink = $(element).attr("href");
|
|
if (episodesLink && episodesLink !== "#") {
|
|
episodeLinks.push({
|
|
title: seriesTitle.trim() || "No title found",
|
|
link: episodesLink || "",
|
|
});
|
|
}
|
|
});
|
|
return episodeLinks;
|
|
} catch (err) {
|
|
console.error(err);
|
|
return [];
|
|
}
|
|
};
|
|
console.log("modGetStream", type, url);
|
|
if (type === "movie") {
|
|
const servers = await modGetEpisodeLinks({ url, providerContext });
|
|
url = servers[0].link || url;
|
|
}
|
|
|
|
let downloadLink = await modExtractor(url, providerContext);
|
|
|
|
// console.log(downloadLink.data);
|
|
|
|
const ddl = downloadLink?.data?.match(/content="0;url=(.*?)"/)?.[1] || url;
|
|
// console.log('ddl', url);
|
|
|
|
// console.log(ddl);
|
|
// console.log(ddl);
|
|
const servers: Stream[] = [];
|
|
const driveLink = await isDriveLink(ddl);
|
|
const driveRes = await axios.get(driveLink, { headers });
|
|
const driveHtml = driveRes.data;
|
|
const $drive = cheerio.load(driveHtml);
|
|
|
|
try {
|
|
const resumeBot = $drive(".btn.btn-light").attr("href") || "";
|
|
const resumeBotRes = await axios.get(resumeBot, { headers });
|
|
const resumeBotToken = resumeBotRes.data.match(
|
|
/formData\.append\('token', '([a-f0-9]+)'\)/
|
|
)[1];
|
|
const resumeBotBody = new FormData();
|
|
resumeBotBody.append("token", resumeBotToken);
|
|
const resumeBotPath = resumeBotRes.data.match(
|
|
/fetch\('\/download\?id=([a-zA-Z0-9\/+]+)'/
|
|
)[1];
|
|
const resumeBotBaseUrl = resumeBot.split("/download")[0];
|
|
// console.log(
|
|
// 'resumeBotPath',
|
|
// resumeBotBaseUrl + '/download?id=' + resumeBotPath,
|
|
// );
|
|
// console.log('resumeBotBody', resumeBotToken);
|
|
|
|
const resumeBotDownload = await fetch(
|
|
resumeBotBaseUrl + "/download?id=" + resumeBotPath,
|
|
{
|
|
method: "POST",
|
|
body: resumeBotBody,
|
|
headers: {
|
|
Referer: resumeBot,
|
|
Cookie: "PHPSESSID=7e9658ce7c805dab5bbcea9046f7f308",
|
|
},
|
|
}
|
|
);
|
|
const resumeBotDownloadData = await resumeBotDownload.json();
|
|
console.log("resumeBotDownloadData", resumeBotDownloadData.url);
|
|
servers.push({
|
|
server: "ResumeBot",
|
|
link: resumeBotDownloadData.url,
|
|
type: "mkv",
|
|
});
|
|
} catch (err) {
|
|
console.log("ResumeBot link not found", err);
|
|
}
|
|
// Base page worker
|
|
try {
|
|
const baseWorkerStream = $drive(".btn-success");
|
|
baseWorkerStream.each((i, el) => {
|
|
const link = (el as any).attribs?.href;
|
|
if (link) {
|
|
servers.push({
|
|
server: "Resume Worker " + (i + 1),
|
|
link: link,
|
|
type: "mkv",
|
|
});
|
|
}
|
|
});
|
|
} catch (err) {
|
|
console.log("Base page worker link not found", err);
|
|
}
|
|
// CF workers type 1
|
|
try {
|
|
const cfWorkersLink = driveLink.replace("/file", "/wfile") + "?type=1";
|
|
const cfWorkersRes = await axios.get(cfWorkersLink, { headers });
|
|
const cfWorkersHtml = cfWorkersRes.data;
|
|
const $cfWorkers = cheerio.load(cfWorkersHtml);
|
|
const cfWorkersStream = $cfWorkers(".btn-success");
|
|
cfWorkersStream.each((i, el) => {
|
|
const link = (el as any).attribs?.href;
|
|
if (link) {
|
|
servers.push({
|
|
server: "Cf Worker 1." + i,
|
|
link: link,
|
|
type: "mkv",
|
|
});
|
|
}
|
|
});
|
|
} catch (err) {
|
|
console.log("CF workers link not found", err);
|
|
}
|
|
|
|
// CF workers type 2
|
|
try {
|
|
const cfWorkersLink = driveLink.replace("/file", "/wfile") + "?type=2";
|
|
const cfWorkersRes = await axios.get(cfWorkersLink, { headers });
|
|
const cfWorkersHtml = cfWorkersRes.data;
|
|
const $cfWorkers = cheerio.load(cfWorkersHtml);
|
|
const cfWorkersStream = $cfWorkers(".btn-success");
|
|
cfWorkersStream.each((i, el) => {
|
|
const link = (el as any).attribs?.href;
|
|
if (link) {
|
|
servers.push({
|
|
server: "Cf Worker 2." + i,
|
|
link: link,
|
|
type: "mkv",
|
|
});
|
|
}
|
|
});
|
|
} catch (err) {
|
|
console.log("CF workers link not found", err);
|
|
}
|
|
|
|
// gdrive
|
|
|
|
//*******
|
|
//instant link 1
|
|
//*******
|
|
// try {
|
|
// const seed = $drive(".btn-danger").attr("href") || "";
|
|
// const instantToken = seed.split("=")[1];
|
|
// console.log("InstantToken", instantToken);
|
|
// const InstantFromData = new FormData();
|
|
// InstantFromData.append("keys", instantToken);
|
|
// const videoSeedUrl = seed.split("/").slice(0, 3).join("/") + "/api";
|
|
// console.log("videoSeedUrl", videoSeedUrl);
|
|
// const instantLinkRes = await fetch(videoSeedUrl, {
|
|
// method: "POST",
|
|
// body: InstantFromData,
|
|
// headers: {
|
|
// "x-token": videoSeedUrl,
|
|
// },
|
|
// });
|
|
// const instantLinkData = await instantLinkRes.json();
|
|
// // console.log('instantLinkData', instantLinkData);
|
|
// if (instantLinkData.error === false) {
|
|
// const instantLink = instantLinkData.url;
|
|
// servers.push({
|
|
// server: "Gdrive-Instant",
|
|
// link: instantLink,
|
|
// type: "mkv",
|
|
// });
|
|
// } else {
|
|
// console.log("Instant link not found", instantLinkData);
|
|
// }
|
|
// } catch (err) {
|
|
// console.log("Instant link not found", err);
|
|
// }
|
|
|
|
//*******
|
|
//instant link 2
|
|
//*******
|
|
try {
|
|
const seed = $drive(".btn-danger").attr("href") || "";
|
|
const newLinkRes = await fetch(seed, {
|
|
method: "HEAD",
|
|
headers,
|
|
redirect: "manual",
|
|
});
|
|
let newLink = seed;
|
|
if (newLinkRes.status >= 300 && newLinkRes.status < 400) {
|
|
newLink = newLinkRes.headers.get("location") || seed;
|
|
} else if (newLinkRes.url && newLinkRes.url !== seed) {
|
|
// Fallback: check if URL changed (redirect was followed)
|
|
newLink = newLinkRes.url || newLinkRes.url;
|
|
} else {
|
|
newLink = newLinkRes.headers.get("location") || seed;
|
|
}
|
|
console.log("Gdrive-Instant-2 link", newLink?.split("?url=")[1]);
|
|
servers.push({
|
|
server: "Gdrive-Instant-2",
|
|
link: newLink?.split("?url=")[1] || newLink,
|
|
type: "mkv",
|
|
});
|
|
} catch (err) {
|
|
console.log("Instant link not found", err);
|
|
}
|
|
|
|
return servers;
|
|
} catch (err) {
|
|
console.log("getStream error", err);
|
|
return [];
|
|
}
|
|
};
|
|
|
|
const isDriveLink = async (ddl: string) => {
|
|
if (ddl.includes("drive")) {
|
|
const driveLeach = await fetch(ddl);
|
|
const driveLeachData = await driveLeach.text();
|
|
const pathMatch = driveLeachData.match(
|
|
/window\.location\.replace\("([^"]+)"\)/
|
|
);
|
|
const path = pathMatch?.[1];
|
|
const mainUrl = ddl.split("/")[2];
|
|
console.log(`driveUrl = https://${mainUrl}${path}`);
|
|
return `https://${mainUrl}${path}`;
|
|
} else {
|
|
return ddl;
|
|
}
|
|
};
|
|
|
|
async function modExtractor(url: string, providerContext: ProviderContext) {
|
|
const { axios, cheerio } = providerContext;
|
|
try {
|
|
const wpHttp = url.split("sid=")[1];
|
|
var bodyFormData0 = new FormData();
|
|
bodyFormData0.append("_wp_http", wpHttp);
|
|
const res = await fetch(url.split("?")[0], {
|
|
method: "POST",
|
|
body: bodyFormData0,
|
|
});
|
|
const data = await res.text();
|
|
// console.log('', data);
|
|
const html = data;
|
|
const $ = cheerio.load(html);
|
|
|
|
// find input with name="_wp_http2"
|
|
const wpHttp2 = $("input").attr("name", "_wp_http2").val();
|
|
|
|
console.log("wpHttp2", wpHttp2);
|
|
|
|
// form data
|
|
var bodyFormData = new FormData();
|
|
bodyFormData.append("_wp_http2", wpHttp2);
|
|
const formUrl1 = $("form").attr("action");
|
|
const formUrl = formUrl1 || url.split("?")[0];
|
|
|
|
const res2 = await fetch(formUrl, {
|
|
method: "POST",
|
|
body: bodyFormData,
|
|
});
|
|
const html2: any = await res2.text();
|
|
const link = html2.match(/setAttribute\("href",\s*"(.*?)"/)[1];
|
|
console.log(link);
|
|
const cookie = link.split("=")[1];
|
|
console.log("cookie", cookie);
|
|
|
|
const downloadLink = await axios.get(link, {
|
|
headers: {
|
|
Referer: formUrl,
|
|
Cookie: `${cookie}=${wpHttp2}`,
|
|
},
|
|
});
|
|
return downloadLink;
|
|
} catch (err) {
|
|
console.log("modGetStream error", err);
|
|
}
|
|
}
|