mirror of
https://github.com/vega-org/vega-providers.git
synced 2026-04-17 23:51:44 +00:00
init
This commit is contained in:
85
providers/mod/catalog.ts
Normal file
85
providers/mod/catalog.ts
Normal file
@@ -0,0 +1,85 @@
|
||||
export const catalogList = [
|
||||
{
|
||||
title: 'Latest',
|
||||
filter: '',
|
||||
},
|
||||
{
|
||||
title: 'Netflix',
|
||||
filter: '/ott/netflix',
|
||||
},
|
||||
{
|
||||
title: 'HBO Max',
|
||||
filter: '/ott/hbo-max',
|
||||
},
|
||||
{
|
||||
title: 'Amazon Prime',
|
||||
filter: '/ott/amazon-prime-video',
|
||||
},
|
||||
];
|
||||
|
||||
export const modGenresList = [
|
||||
{
|
||||
title: 'Apple TV+',
|
||||
filter: '/ott/apple-tv',
|
||||
},
|
||||
{
|
||||
title: 'Disney+',
|
||||
filter: '/ott/disney-plus',
|
||||
},
|
||||
{
|
||||
title: 'Hulu',
|
||||
filter: '/ott/hulu',
|
||||
},
|
||||
{
|
||||
title: 'Crunchyroll',
|
||||
filter: '/ott/crunchyroll',
|
||||
},
|
||||
{
|
||||
title: 'Action',
|
||||
filter: '/movies-by-genre/action/',
|
||||
},
|
||||
{
|
||||
title: 'Adventure',
|
||||
filter: '/movies-by-genre/adventure/',
|
||||
},
|
||||
{
|
||||
title: 'Animation',
|
||||
filter: '/movies-by-genre/animated/',
|
||||
},
|
||||
{
|
||||
title: 'Comedy',
|
||||
filter: '/movies-by-genre/comedy/',
|
||||
},
|
||||
{
|
||||
title: 'Crime',
|
||||
filter: '/movies-by-genre/crime/',
|
||||
},
|
||||
{
|
||||
title: 'Documentary',
|
||||
filter: '/movies-by-genre/documentary/',
|
||||
},
|
||||
{
|
||||
title: 'Fantasy',
|
||||
filter: '/movies-by-genre/fantasy/',
|
||||
},
|
||||
{
|
||||
title: 'Horror',
|
||||
filter: '/movies-by-genre/horror/',
|
||||
},
|
||||
{
|
||||
title: 'Mystery',
|
||||
filter: '/movies-by-genre/mystery/',
|
||||
},
|
||||
{
|
||||
title: 'Romance',
|
||||
filter: '/movies-by-genre/romance/',
|
||||
},
|
||||
{
|
||||
title: 'Thriller',
|
||||
filter: '/movies-by-genre/thriller/',
|
||||
},
|
||||
{
|
||||
title: 'Sci-Fi',
|
||||
filter: '/movies-by-genre/sci-fi/',
|
||||
},
|
||||
];
|
||||
17
providers/mod/index.ts
Normal file
17
providers/mod/index.ts
Normal file
@@ -0,0 +1,17 @@
|
||||
import {modGenresList, catalogList} from './catalog';
|
||||
import {modGetInfo} from './modGetInfo';
|
||||
import {modGetEpisodeLinks} from './modGetEpisodesList';
|
||||
import {modGetPosts, modGetPostsSearch} from './modGetPosts';
|
||||
import {modGetStream} from './modGetStream';
|
||||
import {ProviderType} from '../types';
|
||||
|
||||
export const modMovies: ProviderType = {
|
||||
catalog: catalogList,
|
||||
genres: modGenresList,
|
||||
GetMetaData: modGetInfo,
|
||||
GetHomePosts: modGetPosts,
|
||||
GetStream: modGetStream,
|
||||
GetEpisodeLinks: modGetEpisodeLinks,
|
||||
// nonStreamableServer: ['Gdrive-Instant'],
|
||||
GetSearchPosts: modGetPostsSearch,
|
||||
};
|
||||
52
providers/mod/modGetEpisodesList.ts
Normal file
52
providers/mod/modGetEpisodesList.ts
Normal file
@@ -0,0 +1,52 @@
|
||||
import {EpisodeLink, ProviderContext} from '../types';
|
||||
|
||||
export 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 [];
|
||||
}
|
||||
};
|
||||
71
providers/mod/modGetInfo.ts
Normal file
71
providers/mod/modGetInfo.ts
Normal file
@@ -0,0 +1,71 @@
|
||||
import {Info, Link, ProviderContext} from '../types';
|
||||
|
||||
export const modGetInfo = async function ({
|
||||
link,
|
||||
providerContext,
|
||||
}: {
|
||||
link: string;
|
||||
providerContext: ProviderContext;
|
||||
}): Promise<Info> {
|
||||
try {
|
||||
const {axios, cheerio} = providerContext;
|
||||
const url = link;
|
||||
const res = await axios.get(url);
|
||||
const data = res.data;
|
||||
const $ = cheerio.load(data);
|
||||
const meta = {
|
||||
title: $('.imdbwp__title').text(),
|
||||
synopsis: $('.imdbwp__teaser').text(),
|
||||
image: $('.imdbwp__thumb').find('img').attr('src') || '',
|
||||
imdbId: $('.imdbwp__link').attr('href')?.split('/')[4] || '',
|
||||
type: $('.thecontent').text().toLocaleLowerCase().includes('season')
|
||||
? 'series'
|
||||
: 'movie',
|
||||
};
|
||||
const links: Link[] = [];
|
||||
|
||||
$('h3,h4').map((i, element) => {
|
||||
const seriesTitle = $(element).text();
|
||||
// const batchZipLink = $(element)
|
||||
// .next("p")
|
||||
// .find(".maxbutton-batch-zip,.maxbutton-zip-download")
|
||||
// .attr("href");
|
||||
const episodesLink = $(element)
|
||||
.next('p')
|
||||
.find(
|
||||
'.maxbutton-episode-links,.maxbutton-g-drive,.maxbutton-af-download',
|
||||
)
|
||||
.attr('href');
|
||||
const movieLink = $(element)
|
||||
.next('p')
|
||||
.find('.maxbutton-download-links')
|
||||
.attr('href');
|
||||
|
||||
if (
|
||||
movieLink ||
|
||||
(episodesLink && episodesLink !== 'javascript:void(0);')
|
||||
) {
|
||||
links.push({
|
||||
title: seriesTitle.replace('Download ', '').trim() || 'Download',
|
||||
episodesLink: episodesLink || '',
|
||||
directLinks: movieLink
|
||||
? [{link: movieLink, title: 'Movie', type: 'movie'}]
|
||||
: [],
|
||||
quality: seriesTitle?.match(/\d+p\b/)?.[0] || '',
|
||||
});
|
||||
}
|
||||
});
|
||||
// console.log('mod meta', links);
|
||||
return {...meta, linkList: links};
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
return {
|
||||
title: '',
|
||||
synopsis: '',
|
||||
image: '',
|
||||
imdbId: '',
|
||||
type: 'movie',
|
||||
linkList: [],
|
||||
};
|
||||
}
|
||||
};
|
||||
74
providers/mod/modGetPosts.ts
Normal file
74
providers/mod/modGetPosts.ts
Normal file
@@ -0,0 +1,74 @@
|
||||
import {Post, ProviderContext} from '../types';
|
||||
|
||||
export const modGetPosts = async function ({
|
||||
filter,
|
||||
page,
|
||||
signal,
|
||||
providerContext,
|
||||
}: {
|
||||
filter: string;
|
||||
page: number;
|
||||
providerValue: string;
|
||||
signal: AbortSignal;
|
||||
providerContext: ProviderContext;
|
||||
}): Promise<Post[]> {
|
||||
const {getBaseUrl, axios, cheerio} = providerContext;
|
||||
const baseUrl = await getBaseUrl('Moviesmod');
|
||||
const url = `${baseUrl + filter}/page/${page}/`;
|
||||
return posts({url, signal, axios, cheerio});
|
||||
};
|
||||
|
||||
export const modGetPostsSearch = async function ({
|
||||
searchQuery,
|
||||
page,
|
||||
signal,
|
||||
providerContext,
|
||||
}: {
|
||||
searchQuery: string;
|
||||
page: number;
|
||||
providerValue: string;
|
||||
signal: AbortSignal;
|
||||
providerContext: ProviderContext;
|
||||
}): Promise<Post[]> {
|
||||
const {getBaseUrl, axios, cheerio} = providerContext;
|
||||
const baseUrl = await getBaseUrl('Moviesmod');
|
||||
const url = `${baseUrl}/search/${searchQuery}/page/${page}/`;
|
||||
return posts({url, signal, axios, cheerio});
|
||||
};
|
||||
|
||||
async function posts({
|
||||
url,
|
||||
signal,
|
||||
axios,
|
||||
cheerio,
|
||||
}: {
|
||||
url: string;
|
||||
signal: AbortSignal;
|
||||
axios: ProviderContext['axios'];
|
||||
cheerio: ProviderContext['cheerio'];
|
||||
}): Promise<Post[]> {
|
||||
try {
|
||||
const res = await axios.get(url, {signal});
|
||||
const data = res.data;
|
||||
const $ = cheerio.load(data);
|
||||
const catalog: Post[] = [];
|
||||
$('.post-cards')
|
||||
.find('article')
|
||||
.map((i, element) => {
|
||||
const title = $(element).find('a').attr('title');
|
||||
const link = $(element).find('a').attr('href');
|
||||
const image = $(element).find('img').attr('src');
|
||||
if (title && link && image) {
|
||||
catalog.push({
|
||||
title: title,
|
||||
link: link,
|
||||
image: image,
|
||||
});
|
||||
}
|
||||
});
|
||||
return catalog;
|
||||
} catch (err) {
|
||||
console.error('modGetPosts error ', err);
|
||||
return [];
|
||||
}
|
||||
}
|
||||
288
providers/mod/modGetStream.ts
Normal file
288
providers/mod/modGetStream.ts
Normal file
@@ -0,0 +1,288 @@
|
||||
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 modGetStream = 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);
|
||||
}
|
||||
// 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.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.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
|
||||
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);
|
||||
}
|
||||
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);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user