mirror of
https://github.com/sussy-code/smov.git
synced 2025-01-23 02:41:38 +01:00
157 lines
No EOL
4.6 KiB
JavaScript
157 lines
No EOL
4.6 KiB
JavaScript
import Fuse from 'fuse.js'
|
|
import JSON5 from 'json5'
|
|
|
|
function getCorsUrl(url) {
|
|
return `https://hidden-inlet-27205.herokuapp.com/${url}`;
|
|
}
|
|
|
|
async function getVideoUrl(config) {
|
|
const accessToken = await getAccessToken(config);
|
|
const now = Math.floor(Date.now() / 1e3);
|
|
|
|
let url = '';
|
|
|
|
if (config.type === 'movie') {
|
|
url = getCorsUrl(`https://lookmovie.io/manifests/movies/json/${config.id}/${now}/${accessToken}/master.m3u8`);
|
|
} else if (config.type === 'show') {
|
|
url = getCorsUrl(`https://lookmovie.io/manifests/shows/json/${accessToken}/${now}/${config.id}/master.m3u8`);
|
|
}
|
|
|
|
if (url) {
|
|
const videoOpts = await fetch(url).then((d) => d.json());
|
|
|
|
// Find video URL and return it (with a check for a full url if needed)
|
|
const opts = ["1080p", "1080", "720p", "720", "480p", "480", "auto"]
|
|
|
|
let videoUrl = "";
|
|
for (let res of opts) {
|
|
if (videoOpts[res] && !videoOpts[res].includes('dummy') && !videoOpts[res].includes('earth-1984') && !videoUrl) {
|
|
videoUrl = videoOpts[res]
|
|
}
|
|
}
|
|
|
|
return videoUrl.startsWith("/") ? getCorsUrl(`https://lookmovie.io/${videoUrl}`) : getCorsUrl(videoUrl);
|
|
}
|
|
|
|
return "Invalid type.";
|
|
}
|
|
|
|
async function getAccessToken(config) {
|
|
let url = '';
|
|
|
|
if (config.type === 'movie') {
|
|
url = getCorsUrl(`https://lookmovie.io/api/v1/security/movie-access?id_movie=${config.id}&token=1&sk=&step=1`);
|
|
} else if (config.type === 'show') {
|
|
url = getCorsUrl(`https://lookmovie.io/api/v1/security/show-access?slug=${config.slug}&token=&step=2`);
|
|
}
|
|
|
|
const data = await fetch(url).then((d) => d.json());
|
|
|
|
const token = data?.data?.accessToken;
|
|
if (token) return token;
|
|
|
|
return "Invalid type provided in config";
|
|
}
|
|
|
|
async function getEpisodes(slug) {
|
|
const url = getCorsUrl(`https://lookmovie.io/shows/view/${slug}`);
|
|
const pageReq = await fetch(url).then((d) => d.text());
|
|
|
|
const data = JSON5.parse("{" +
|
|
pageReq
|
|
.slice(pageReq.indexOf(`show_storage`))
|
|
.split("};")[0]
|
|
.split("= {")[1]
|
|
.trim() +
|
|
"}"
|
|
);
|
|
|
|
return data.seasons
|
|
}
|
|
|
|
async function getStreamUrl(slug, type, season, episode) {
|
|
const url = getCorsUrl(`https://lookmovie.io/${type}s/view/${slug}`);
|
|
const pageReq = await fetch(url).then((d) => d.text());
|
|
|
|
const data = JSON5.parse("{" +
|
|
pageReq
|
|
.slice(pageReq.indexOf(`${type}_storage`))
|
|
.split("};")[0]
|
|
.split("= {")[1]
|
|
.trim() +
|
|
"}"
|
|
);
|
|
|
|
let id = '';
|
|
|
|
if (type === "movie") {
|
|
id = data.id_movie;
|
|
} else if (type === "show") {
|
|
const episodeObj = data.seasons.find((v) => { return v.season === season && v.episode === episode; });
|
|
|
|
if (episodeObj) {
|
|
id = episodeObj.id_episode;
|
|
}
|
|
}
|
|
|
|
if (id === '') {
|
|
return { url: '' }
|
|
}
|
|
|
|
const videoUrl = await getVideoUrl({
|
|
slug: slug,
|
|
id: id,
|
|
type: type,
|
|
});
|
|
|
|
return { url: videoUrl }
|
|
}
|
|
|
|
async function findContent(searchTerm, type) {
|
|
// const searchUrl = getCorsUrl(`https://lookmovie.io/api/v1/${type}s/search/?q=${encodeURIComponent(searchTerm)}`);
|
|
const searchUrl = getCorsUrl(`https://lookmovie.io/${type}s/search/?q=${encodeURIComponent(searchTerm)}`);
|
|
const searchRes = await fetch(searchUrl).then((d) => d.text());
|
|
|
|
// Parse DOM to find search results on full search page
|
|
const parser = new DOMParser();
|
|
const doc = parser.parseFromString(searchRes, "text/html");
|
|
const nodes = Array.from(doc.querySelectorAll('.movie-item-style-1'));
|
|
const results = nodes.map(node => {
|
|
return {
|
|
type,
|
|
title: node.querySelector('h6 a').innerText,
|
|
year: node.querySelector('.year').innerText,
|
|
slug: node.querySelector('a').href.split('/').pop(),
|
|
}
|
|
});
|
|
|
|
const fuse = new Fuse(results, { threshold: 0.3, distance: 200, keys: ["title"] });
|
|
const matchedResults = fuse
|
|
.search(searchTerm.toString())
|
|
.map((result) => result.item);
|
|
|
|
if (matchedResults.length === 0) {
|
|
return { options: [] }
|
|
}
|
|
|
|
if (matchedResults.length > 1) {
|
|
const res = { options: [] };
|
|
|
|
matchedResults.forEach((r) => res.options.push({
|
|
title: r.title,
|
|
slug: r.slug,
|
|
type: r.type,
|
|
year: r.year
|
|
}));
|
|
|
|
return res;
|
|
} else {
|
|
const { title, slug, type, year } = matchedResults[0];
|
|
|
|
return {
|
|
options: [{ title, slug, type, year }]
|
|
}
|
|
}
|
|
}
|
|
|
|
export { findContent, getStreamUrl, getEpisodes }; |