mirror of
https://github.com/movie-web/movie-web.git
synced 2025-01-07 23:26:00 +00:00
add series support & improvements
This commit is contained in:
parent
e52b29a1a1
commit
7b75c36d21
|
@ -1,4 +1,5 @@
|
||||||
import { proxiedFetch } from "../helpers/fetch";
|
import { proxiedFetch } from "../helpers/fetch";
|
||||||
|
import { MWProviderContext } from "../helpers/provider";
|
||||||
import { registerProvider } from "../helpers/register";
|
import { registerProvider } from "../helpers/register";
|
||||||
import { MWStreamQuality, MWStreamType } from "../helpers/streams";
|
import { MWStreamQuality, MWStreamType } from "../helpers/streams";
|
||||||
import { MWMediaType } from "../metadata/types";
|
import { MWMediaType } from "../metadata/types";
|
||||||
|
@ -13,39 +14,134 @@ const qualityMap: Record<number, MWStreamQuality> = {
|
||||||
1080: MWStreamQuality.Q1080P,
|
1080: MWStreamQuality.Q1080P,
|
||||||
};
|
};
|
||||||
|
|
||||||
interface MovieSearchList {
|
interface SearchRes {
|
||||||
title: string;
|
title: string;
|
||||||
|
year?: number;
|
||||||
|
href: string;
|
||||||
id: string;
|
id: string;
|
||||||
year: number;
|
}
|
||||||
|
|
||||||
|
function getStreamFromEmbed(stream: string) {
|
||||||
|
const embedPage = new DOMParser().parseFromString(stream, "text/html");
|
||||||
|
const source = embedPage.querySelector("#vjsplayer > source");
|
||||||
|
if (!source) {
|
||||||
|
throw new Error("Unable to fetch stream");
|
||||||
|
}
|
||||||
|
|
||||||
|
const streamSrc = source.getAttribute("src");
|
||||||
|
const streamRes = source.getAttribute("res");
|
||||||
|
|
||||||
|
if (!streamSrc || !streamRes) throw new Error("Unable to find stream");
|
||||||
|
|
||||||
|
return {
|
||||||
|
streamUrl: streamSrc,
|
||||||
|
quality:
|
||||||
|
streamRes && typeof +streamRes === "number"
|
||||||
|
? qualityMap[+streamRes]
|
||||||
|
: MWStreamQuality.QUNKNOWN,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchMovie(targetSource: SearchRes) {
|
||||||
|
const stream = await proxiedFetch<any>(`/embed/${targetSource.id}`, {
|
||||||
|
baseURL: hdwatchedBase,
|
||||||
|
});
|
||||||
|
|
||||||
|
const embedPage = new DOMParser().parseFromString(stream, "text/html");
|
||||||
|
const source = embedPage.querySelector("#vjsplayer > source");
|
||||||
|
if (!source) {
|
||||||
|
throw new Error("Unable to fetch movie stream");
|
||||||
|
}
|
||||||
|
|
||||||
|
return getStreamFromEmbed(stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchSeries(
|
||||||
|
targetSource: SearchRes,
|
||||||
|
{ media, episode, progress }: MWProviderContext
|
||||||
|
) {
|
||||||
|
if (media.meta.type !== MWMediaType.SERIES)
|
||||||
|
throw new Error("Media type mismatch");
|
||||||
|
|
||||||
|
const seasonNumber = media.meta.seasonData.number;
|
||||||
|
const episodeNumber = media.meta.seasonData.episodes.find(
|
||||||
|
(e) => e.id === episode
|
||||||
|
)?.number;
|
||||||
|
|
||||||
|
if (!seasonNumber || !episodeNumber)
|
||||||
|
throw new Error("Unable to get season or episode number");
|
||||||
|
|
||||||
|
const seriesPage = await proxiedFetch<any>(
|
||||||
|
`${targetSource.href}?season=${media.meta.seasonData.number}`,
|
||||||
|
{
|
||||||
|
baseURL: hdwatchedBase,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const seasonPage = new DOMParser().parseFromString(seriesPage, "text/html");
|
||||||
|
const pageElements = seasonPage.querySelectorAll("div.i-container");
|
||||||
|
|
||||||
|
const seriesList: SearchRes[] = [];
|
||||||
|
pageElements.forEach((pageElement) => {
|
||||||
|
const href = pageElement.querySelector("a")?.getAttribute("href") || "";
|
||||||
|
const title =
|
||||||
|
pageElement?.querySelector("span.content-title")?.textContent || "";
|
||||||
|
|
||||||
|
seriesList.push({
|
||||||
|
title,
|
||||||
|
href,
|
||||||
|
id: href.split("/")[2], // Format: /free/{id}/{series-slug}-season-{season-number}-episode-{episode-number}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const targetEpisode = seriesList.find(
|
||||||
|
(episodeEl) =>
|
||||||
|
episodeEl.title.trim().toLowerCase() === `episode ${episodeNumber}`
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!targetEpisode) throw new Error("Unable to find episode");
|
||||||
|
|
||||||
|
progress(70);
|
||||||
|
|
||||||
|
const stream = await proxiedFetch<any>(`/embed/${targetEpisode.id}`, {
|
||||||
|
baseURL: hdwatchedBase,
|
||||||
|
});
|
||||||
|
|
||||||
|
const embedPage = new DOMParser().parseFromString(stream, "text/html");
|
||||||
|
const source = embedPage.querySelector("#vjsplayer > source");
|
||||||
|
if (!source) {
|
||||||
|
throw new Error("Unable to fetch movie stream");
|
||||||
|
}
|
||||||
|
|
||||||
|
return getStreamFromEmbed(stream);
|
||||||
}
|
}
|
||||||
|
|
||||||
registerProvider({
|
registerProvider({
|
||||||
id: "hdwatched",
|
id: "hdwatched",
|
||||||
displayName: "HDwatched",
|
displayName: "HDwatched",
|
||||||
rank: 50,
|
rank: 50,
|
||||||
type: [MWMediaType.MOVIE],
|
type: [MWMediaType.MOVIE, MWMediaType.SERIES],
|
||||||
async scrape({ media, progress }) {
|
async scrape(options) {
|
||||||
|
const { media, progress } = options;
|
||||||
if (!this.type.includes(media.meta.type)) {
|
if (!this.type.includes(media.meta.type)) {
|
||||||
throw new Error("Unsupported type");
|
throw new Error("Unsupported type");
|
||||||
}
|
}
|
||||||
|
|
||||||
progress(20);
|
|
||||||
|
|
||||||
const search = await proxiedFetch<any>(`/search/${media.imdbId}`, {
|
const search = await proxiedFetch<any>(`/search/${media.imdbId}`, {
|
||||||
baseURL: hdwatchedBase,
|
baseURL: hdwatchedBase,
|
||||||
});
|
});
|
||||||
|
|
||||||
const searchPage = new DOMParser().parseFromString(search, "text/html");
|
const searchPage = new DOMParser().parseFromString(search, "text/html");
|
||||||
const movieElements = searchPage.querySelectorAll("div.i-container");
|
const pageElements = searchPage.querySelectorAll("div.i-container");
|
||||||
|
|
||||||
const movieSearchList: MovieSearchList[] = [];
|
const searchList: SearchRes[] = [];
|
||||||
movieElements.forEach((movieElement) => {
|
pageElements.forEach((pageElement) => {
|
||||||
const href = movieElement.querySelector("a")?.getAttribute("href") || "";
|
const href = pageElement.querySelector("a")?.getAttribute("href") || "";
|
||||||
const title =
|
const title =
|
||||||
movieElement?.querySelector("span.content-title")?.textContent || "";
|
pageElement?.querySelector("span.content-title")?.textContent || "";
|
||||||
const year =
|
const year =
|
||||||
parseInt(
|
parseInt(
|
||||||
movieElement
|
pageElement
|
||||||
?.querySelector("div.duration")
|
?.querySelector("div.duration")
|
||||||
?.textContent?.trim()
|
?.textContent?.trim()
|
||||||
?.split(" ")
|
?.split(" ")
|
||||||
|
@ -53,50 +149,45 @@ registerProvider({
|
||||||
10
|
10
|
||||||
) || 0;
|
) || 0;
|
||||||
|
|
||||||
movieSearchList.push({
|
searchList.push({
|
||||||
title,
|
title,
|
||||||
year,
|
year,
|
||||||
id: href.split("/")[2], // Format: /free/{id}}/{movie-slug} | Example: /free/18804/iron-man-231
|
href,
|
||||||
|
id: href.split("/")[2], // Format: /free/{id}/{movie-slug} or /series/{id}/{series-slug}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
progress(50);
|
progress(20);
|
||||||
|
|
||||||
const targetMovie = movieSearchList.find(
|
const targetSource = searchList.find(
|
||||||
(movie) => movie.year === (media.meta.year ? +media.meta.year : 0) // Compare year to make the search more robust
|
(source) => source.year === (media.meta.year ? +media.meta.year : 0) // Compare year to make the search more robust
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!targetMovie) {
|
if (!targetSource) {
|
||||||
throw new Error("Could not find stream");
|
throw new Error("Could not find stream");
|
||||||
}
|
}
|
||||||
|
|
||||||
const stream = await proxiedFetch<any>(`/embed/${targetMovie.id}`, {
|
progress(40);
|
||||||
baseURL: hdwatchedBase,
|
|
||||||
});
|
|
||||||
|
|
||||||
progress(80);
|
|
||||||
|
|
||||||
const embedPage = new DOMParser().parseFromString(stream, "text/html");
|
|
||||||
const source = embedPage.querySelector("#vjsplayer > source");
|
|
||||||
if (!source) {
|
|
||||||
throw new Error("Could not find stream");
|
|
||||||
}
|
|
||||||
|
|
||||||
const streamSrc = source.getAttribute("src");
|
|
||||||
const streamRes = source.getAttribute("res");
|
|
||||||
|
|
||||||
if (!streamSrc) {
|
|
||||||
throw new Error("Could not find stream");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if (media.meta.type === MWMediaType.SERIES) {
|
||||||
|
const series = await fetchSeries(targetSource, options);
|
||||||
return {
|
return {
|
||||||
embeds: [],
|
embeds: [],
|
||||||
stream: {
|
stream: {
|
||||||
streamUrl: streamSrc,
|
streamUrl: series.streamUrl,
|
||||||
quality:
|
quality: series.quality,
|
||||||
streamRes && typeof +streamRes === "number"
|
type: MWStreamType.MP4,
|
||||||
? qualityMap[+streamRes]
|
captions: [],
|
||||||
: MWStreamQuality.QUNKNOWN,
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const movie = await fetchMovie(targetSource);
|
||||||
|
return {
|
||||||
|
embeds: [],
|
||||||
|
stream: {
|
||||||
|
streamUrl: movie.streamUrl,
|
||||||
|
quality: movie.quality,
|
||||||
type: MWStreamType.MP4,
|
type: MWStreamType.MP4,
|
||||||
captions: [],
|
captions: [],
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in a new issue