From e3d6ec93c77071e44c8025412d23936c15cf5c36 Mon Sep 17 00:00:00 2001 From: thehairy Date: Mon, 22 May 2023 20:07:19 +0200 Subject: [PATCH 01/96] chore: some corrections in the german translation --- src/setup/locales/de/translation.json | 44 +++++++++++++-------------- 1 file changed, 22 insertions(+), 22 deletions(-) diff --git a/src/setup/locales/de/translation.json b/src/setup/locales/de/translation.json index 5edad4d0..d58923c0 100644 --- a/src/setup/locales/de/translation.json +++ b/src/setup/locales/de/translation.json @@ -3,8 +3,8 @@ "name": "movie-web" }, "search": { - "loading_series": "Auf der Suche nach Ihrer Lieblingsserie...", - "loading_movie": "Auf der Suche nach Ihren Lieblingsfilmen...", + "loading_series": "Auf der Suche nach deiner Lieblingsserie...", + "loading_movie": "Auf der Suche nach deinen Lieblingsfilmen...", "loading": "Wird geladen...", "allResults": "Das ist alles, was wir haben!", "noResults": "Wir haben nichts gefunden!", @@ -12,15 +12,15 @@ "headingTitle": "Suchergebnisse", "bookmarks": "Favoriten", "continueWatching": "Weiter ansehen", - "title": "Was willst du sehen?", - "placeholder": "Was willst du sehen?" + "title": "Was willst du gucken?", + "placeholder": "Was willst du gucken?" }, "media": { "movie": "Filme", "series": "Serie", - "stopEditing": "Beenden Sie die Bearbeitung", + "stopEditing": "Beenden die Bearbeitung", "errors": { - "genericTitle": "Hoppla, etwas ist falsch gegangen!", + "genericTitle": "Hoppla, etwas ist schiefgegangen!", "failedMeta": "Metadaten konnten nicht geladen werden", "mediaFailed": "Wir konnten die angeforderten Medien nicht abrufen.", "videoFailed": "Beim Abspielen des angeforderten Videos ist ein Fehler aufgetreten. <0>Discord Oder weiter <1>GitHub." @@ -48,17 +48,17 @@ "searchBar": { "movie": "Film", "series": "Serie", - "Search": "Forschen" + "Search": "Suchen" }, "videoPlayer": { "findingBestVideo": "Auf der Suche nach dem besten Video für Sie", - "noVideos": "Entschuldigung, wir konnten keine Videos für Sie finden", + "noVideos": "Entschuldigung, wir konnten keine Videos finden", "loading": "Wird geladen...", "backToHome": "Zurück zur Startseite", "backToHomeShort": "Rückmeldung", "seasonAndEpisode": "S{{season}} E{{episode}}", - "timeLeft": "{{timeLeft}} bleibt", - "finishAt": "Ende um {{timeFinished, datetime}}", + "timeLeft": "{{timeLeft}} verbleibend", + "finishAt": "Endet um {{timeFinished, datetime}}", "buttons": { "episodes": "Folgen", "source": "Quelle", @@ -71,13 +71,13 @@ "popouts": { "back": "Zurück", "sources": "Quellen", - "seasons": "Saison", + "seasons": "Staffel", "captions": "Untertitel", "playbackSpeed": "Lesegeschwindigkeit", "customPlaybackSpeed": "Benutzerdefinierte Wiedergabegeschwindigkeit", "captionPreferences": { - "title": "Personifizieren", - "delay": "Zeitlimit", + "title": "Bearbeiten", + "delay": "Verzögerung", "fontSize": "Größe", "opacity": "Opazität", "color": "Farbe" @@ -93,17 +93,17 @@ "embedsError": "Beim Laden der eingebetteter Medien ist ein Problem aufgetreten" }, "descriptions": { - "sources": "Welchen Anbieter möchten Sie nutzen?", - "embeds": "Wählen Sie das Video aus, das Sie ansehen möchten", - "seasons": "Wählen Sie die Staffel aus, die Sie sehen möchten", - "episode": "Wählen Sie eine Folge aus", - "captions": "Wählen Sie eine Untertitelsprache", - "captionPreferences": "Passen Sie das Erscheinungsbild von Untertiteln an", + "sources": "Welchen Anbieter möchtest du nutzen?", + "embeds": "Wähle das Video aus, das du ansehen möchten", + "seasons": "Wähle die Staffel aus, die du sehen möchten", + "episode": "Wähle eine Folge aus", + "captions": "Wähle eine Untertitelsprache", + "captionPreferences": "Passe das Erscheinungsbild von Untertiteln an", "playbackSpeed": "Wiedergabegeschwindigkeit ändern" } }, "errors": { - "fatalError": "Der Videoplayer hat einen Fehler festgestellt, bitte melden Sie ihn dem Server <0>Discord Oder weiter <1>GitHub." + "fatalError": "Der Videoplayer hat einen Fehler festgestellt, bitte melde ihn dem Server <0>Discord Oder weiter <1>GitHub." } }, "settings": { @@ -115,13 +115,13 @@ "newSiteTitle": "Neue Version verfügbar!", "newDomain": "https://movie-web.app", "newDomainText": "movie-web zieht in Kürze auf eine neue Domain um: <0>https://movie-web.app. <1>Die alte Website funktioniert nicht mehr {{date}}.", - "tireless": "Wir haben unermüdlich an diesem neuen Update gearbeitet und hoffen, dass Ihnen das gefällt, was wir in den letzten Monaten vorbereitet haben.", + "tireless": "Wir haben unermüdlich an diesem neuen Update gearbeitet und hoffen, dass dir gefällt, was wir in den letzten Monaten vorbereitet haben.", "leaveAnnouncement": "Bring mich dahin!" }, "casting": { "casting": "An Gerät übertragen..." }, "errors": { - "offline": "Ihre Internetverbindung ist instabil" + "offline": "Internetverbindung ist instabil" } } From 77a0c36a5880ef23bc0e42caa82b3194cb9a5504 Mon Sep 17 00:00:00 2001 From: JORDAAR <69628820+Jordaar@users.noreply.github.com> Date: Thu, 25 May 2023 00:15:22 +0530 Subject: [PATCH 02/96] add sflix provider --- src/backend/providers/sflix.ts | 99 ++++++++++++++++++++++++++++++++++ 1 file changed, 99 insertions(+) create mode 100644 src/backend/providers/sflix.ts diff --git a/src/backend/providers/sflix.ts b/src/backend/providers/sflix.ts new file mode 100644 index 00000000..f33f6951 --- /dev/null +++ b/src/backend/providers/sflix.ts @@ -0,0 +1,99 @@ +import { proxiedFetch } from "../helpers/fetch"; +import { registerProvider } from "../helpers/register"; +import { MWStreamQuality, MWStreamType } from "../helpers/streams"; +import { MWMediaType } from "../metadata/types"; + +const sflixBase = "https://sflix.video"; + +registerProvider({ + id: "sflix", + displayName: "Sflix", + rank: 135, + type: [MWMediaType.MOVIE, MWMediaType.SERIES], + async scrape({ media, episode, progress }) { + let searchQuery = `${media.meta.title} `; + + if (media.meta.type === MWMediaType.MOVIE) + searchQuery += media.meta.year ?? ""; + + if (media.meta.type === MWMediaType.SERIES) + searchQuery += `S${String(media.meta.seasonData.number).padStart( + 2, + "0" + )}`; + + const search = await proxiedFetch( + `/?s=${encodeURIComponent(searchQuery)}`, + { + baseURL: sflixBase, + } + ); + const searchPage = new DOMParser().parseFromString(search, "text/html"); + + const moviePageUrl = searchPage + .querySelector(".movies-list .ml-item:first-child a") + ?.getAttribute("href"); + if (!moviePageUrl) throw new Error("Movie does not exist"); + + progress(25); + + const movie = await proxiedFetch(moviePageUrl); + const moviePage = new DOMParser().parseFromString(movie, "text/html"); + + progress(45); + + let outerEmbedSrc = null; + if (media.meta.type === MWMediaType.MOVIE) { + outerEmbedSrc = moviePage + .querySelector("iframe") + ?.getAttribute("data-lazy-src"); + } else if (media.meta.type === MWMediaType.SERIES) { + const series = Array.from(moviePage.querySelectorAll(".desc p a")).map( + (a) => ({ + title: a.getAttribute("title"), + link: a.getAttribute("href"), + }) + ); + + const episodeNumber = media.meta.seasonData.episodes.find( + (e) => e.id === episode + )?.number; + + const targetSeries = series.find((s) => + s.title?.endsWith(String(episodeNumber).padStart(2, "0")) + ); + if (!targetSeries) throw new Error("Episode does not exist"); + + outerEmbedSrc = targetSeries.link; + } + if (!outerEmbedSrc) throw new Error("Outer embed source not found"); + + progress(65); + + const outerEmbed = await proxiedFetch(outerEmbedSrc); + const outerEmbedPage = new DOMParser().parseFromString( + outerEmbed, + "text/html" + ); + + const embedSrc = outerEmbedPage + .querySelector("iframe") + ?.getAttribute("src"); + if (!embedSrc) throw new Error("Embed source not found"); + + const embed = await proxiedFetch(embedSrc); + + const streamUrl = embed.match(/file\s*:\s*"([^"]+\.mp4)"/)?.[1]; + if (!streamUrl) throw new Error("Unable to get stream"); + + return { + embeds: [], + stream: { + streamUrl, + quality: MWStreamQuality.Q1080P, + type: MWStreamType.MP4, + captions: [], + }, + }; + }, +}); From ffc772727acfc2cf65d93c363a8d797e6394fc1e Mon Sep 17 00:00:00 2001 From: JORDAAR <69628820+Jordaar@users.noreply.github.com> Date: Thu, 25 May 2023 00:16:00 +0530 Subject: [PATCH 03/96] register sflix provider --- src/backend/index.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/backend/index.ts b/src/backend/index.ts index 812a0558..eb0ad897 100644 --- a/src/backend/index.ts +++ b/src/backend/index.ts @@ -8,6 +8,7 @@ import "./providers/netfilm"; import "./providers/m4ufree"; import "./providers/hdwatched"; import "./providers/2embed"; +import "./providers/sflix"; // embeds import "./embeds/streamm4u"; From a648f456946b34c5120dc87d6439fa0478a7aa97 Mon Sep 17 00:00:00 2001 From: Jip Fr Date: Thu, 25 May 2023 22:54:35 +0200 Subject: [PATCH 04/96] feat(player): add T query param for starting time --- src/hooks/useQueryParams.ts | 22 ++++++++++++++++++ .../ProgressListenerController.tsx | 23 ++++++++++++++++++- 2 files changed, 44 insertions(+), 1 deletion(-) create mode 100644 src/hooks/useQueryParams.ts diff --git a/src/hooks/useQueryParams.ts b/src/hooks/useQueryParams.ts new file mode 100644 index 00000000..c10289a9 --- /dev/null +++ b/src/hooks/useQueryParams.ts @@ -0,0 +1,22 @@ +import { useMemo } from "react"; +import { useLocation } from "react-router-dom"; + +export function useQueryParams() { + const loc = useLocation(); + + const queryParams = useMemo(() => { + // Basic absolutely-not-fool-proof URL query param parser + const obj: Record = {}; + for (const [key, value] of loc.search + .slice(1) + .split("&") + .map((e) => e.split("="))) { + const valueAsNum = Number(value); + obj[key] = Number.isNaN(valueAsNum) ? value : valueAsNum; + } + + return obj; + }, [loc]); + + return queryParams; +} diff --git a/src/video/components/controllers/ProgressListenerController.tsx b/src/video/components/controllers/ProgressListenerController.tsx index 78e27bc8..e3f7c3ac 100644 --- a/src/video/components/controllers/ProgressListenerController.tsx +++ b/src/video/components/controllers/ProgressListenerController.tsx @@ -1,6 +1,7 @@ import throttle from "lodash.throttle"; import { useEffect, useMemo, useRef } from "react"; +import { useQueryParams } from "@/hooks/useQueryParams"; import { useVideoPlayerDescriptor } from "@/video/state/hooks"; import { useControls } from "@/video/state/logic/controls"; import { useMediaPlaying } from "@/video/state/logic/mediaplaying"; @@ -20,6 +21,7 @@ export function ProgressListenerController(props: Props) { const misc = useMisc(descriptor); const didInitialize = useRef(null); const lastTime = useRef(props.startAt ?? 0); + const queryParams = useQueryParams(); // time updates (throttled) const updateTime = useMemo( @@ -56,9 +58,28 @@ export function ProgressListenerController(props: Props) { useEffect(() => { if (lastStateProviderId.current === stateProviderId) return; if (mediaPlaying.isFirstLoading) return; + lastStateProviderId.current = stateProviderId; + + if ((queryParams.t ?? null) !== null) { + // Convert `t` param to time. Supports only seconds, but also `3:30` or `1:30:02` + const timeArr = queryParams.t.toString().split(":").map(Number); + + const hours = timeArr[timeArr.length - 3] ?? 0; + const minutes = Math.min(timeArr[timeArr.length - 2] ?? 0, 59); + const seconds = Math.min( + timeArr[timeArr.length - 1] ?? 0, + minutes > 0 ? 59 : Infinity + ); + console.log(hours, minutes, seconds, 123); + const timeInSeconds = hours * 60 * 60 + minutes * 60 + seconds; + + controls.setTime(timeInSeconds); + return; + } + controls.setTime(lastTime.current); - }, [controls, mediaPlaying, stateProviderId]); + }, [controls, mediaPlaying, stateProviderId, queryParams]); useEffect(() => { // if it initialized, but media starts loading for the first time again. From 5e0e223851fdfae5932026f5d9b1d25c7ac5c9fd Mon Sep 17 00:00:00 2001 From: Jip Fr Date: Thu, 25 May 2023 22:57:00 +0200 Subject: [PATCH 05/96] style: make scrollbar style global --- src/setup/index.css | 16 ++++++++++++++++ .../components/popouts/PopoutProviderAction.tsx | 2 -- src/video/components/popouts/Popouts.css | 15 --------------- 3 files changed, 16 insertions(+), 17 deletions(-) delete mode 100644 src/video/components/popouts/Popouts.css diff --git a/src/setup/index.css b/src/setup/index.css index af81851d..c17b8258 100644 --- a/src/setup/index.css +++ b/src/setup/index.css @@ -178,4 +178,20 @@ input[type=range].styled-slider.slider-progress::-ms-fill-lower { background: var(--slider-progress-background); border: none; border-right-width: 0; +} + +::-webkit-scrollbar-track { + background-color: transparent; +} + +::-webkit-scrollbar-thumb { + background-color: theme("colors.denim-500"); + border: 5px solid transparent; + border-left: 0; + background-clip: content-box; +} + +::-webkit-scrollbar { + /* For some reason the styles don't get applied without the width */ + width: 13px; } \ No newline at end of file diff --git a/src/video/components/popouts/PopoutProviderAction.tsx b/src/video/components/popouts/PopoutProviderAction.tsx index 5882dc7f..a29a4e09 100644 --- a/src/video/components/popouts/PopoutProviderAction.tsx +++ b/src/video/components/popouts/PopoutProviderAction.tsx @@ -9,8 +9,6 @@ import { useVideoPlayerDescriptor } from "@/video/state/hooks"; import { useControls } from "@/video/state/logic/controls"; import { useInterface } from "@/video/state/logic/interface"; -import "./Popouts.css"; - function ShowPopout(props: { popoutId: string | null; onClose: () => void }) { const popoutMap = { settings: , diff --git a/src/video/components/popouts/Popouts.css b/src/video/components/popouts/Popouts.css deleted file mode 100644 index 5f8cfd89..00000000 --- a/src/video/components/popouts/Popouts.css +++ /dev/null @@ -1,15 +0,0 @@ -.popout-wrapper ::-webkit-scrollbar-track { - background-color: transparent; -} - -.popout-wrapper ::-webkit-scrollbar-thumb { - background-color: theme("colors.denim-500"); - border: 5px solid transparent; - border-left: 0; - background-clip: content-box; -} - -.popout-wrapper ::-webkit-scrollbar { - /* For some reason the styles don't get applied without the width */ - width: 13px; -} \ No newline at end of file From 01b019365d8b9eba1a29c9dfd64707439d8b2acb Mon Sep 17 00:00:00 2001 From: Jip Fr Date: Thu, 25 May 2023 23:01:42 +0200 Subject: [PATCH 06/96] Yeet log --- src/video/components/controllers/ProgressListenerController.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/video/components/controllers/ProgressListenerController.tsx b/src/video/components/controllers/ProgressListenerController.tsx index e3f7c3ac..257b44ac 100644 --- a/src/video/components/controllers/ProgressListenerController.tsx +++ b/src/video/components/controllers/ProgressListenerController.tsx @@ -71,7 +71,7 @@ export function ProgressListenerController(props: Props) { timeArr[timeArr.length - 1] ?? 0, minutes > 0 ? 59 : Infinity ); - console.log(hours, minutes, seconds, 123); + const timeInSeconds = hours * 60 * 60 + minutes * 60 + seconds; controls.setTime(timeInSeconds); From 525f9d0b74464e8e993571b1fcd4bf6c87a03d33 Mon Sep 17 00:00:00 2001 From: Jip Fr Date: Fri, 26 May 2023 00:38:51 +0200 Subject: [PATCH 07/96] chore(player): revert timeArr order for improved readability --- .../controllers/ProgressListenerController.tsx | 14 ++++++-------- 1 file changed, 6 insertions(+), 8 deletions(-) diff --git a/src/video/components/controllers/ProgressListenerController.tsx b/src/video/components/controllers/ProgressListenerController.tsx index 257b44ac..68cf8fc3 100644 --- a/src/video/components/controllers/ProgressListenerController.tsx +++ b/src/video/components/controllers/ProgressListenerController.tsx @@ -62,15 +62,13 @@ export function ProgressListenerController(props: Props) { lastStateProviderId.current = stateProviderId; if ((queryParams.t ?? null) !== null) { - // Convert `t` param to time. Supports only seconds, but also `3:30` or `1:30:02` - const timeArr = queryParams.t.toString().split(":").map(Number); + // Convert `t` param to time. Supports having only seconds (like `?t=192`), but also `3:30` or `1:30:02` - const hours = timeArr[timeArr.length - 3] ?? 0; - const minutes = Math.min(timeArr[timeArr.length - 2] ?? 0, 59); - const seconds = Math.min( - timeArr[timeArr.length - 1] ?? 0, - minutes > 0 ? 59 : Infinity - ); + const timeArr = queryParams.t.toString().split(":").map(Number).reverse(); // This is an array of [seconds, ?minutes, ?hours] as ints. + + const hours = timeArr[2] ?? 0; + const minutes = Math.min(timeArr[1] ?? 0, 59); + const seconds = Math.min(timeArr[0] ?? 0, minutes > 0 ? 59 : Infinity); const timeInSeconds = hours * 60 * 60 + minutes * 60 + seconds; From d586899dbf9f941c351d32d39c3d6b16f9aeca70 Mon Sep 17 00:00:00 2001 From: Isra Date: Thu, 25 May 2023 22:38:58 -0500 Subject: [PATCH 08/96] Pirate speak! --- src/setup/i18n.ts | 4 + src/setup/iso6391.ts | 7 ++ src/setup/locales/pirate/translation.json | 124 ++++++++++++++++++++++ 3 files changed, 135 insertions(+) create mode 100644 src/setup/locales/pirate/translation.json diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index c5094776..6116434a 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -8,6 +8,7 @@ import de from "./locales/de/translation.json"; import en from "./locales/en/translation.json"; import fr from "./locales/fr/translation.json"; import nl from "./locales/nl/translation.json"; +import pirate from "./locales/pirate/translation.json"; import tr from "./locales/tr/translation.json"; import zh from "./locales/zh/translation.json"; @@ -33,6 +34,9 @@ const locales = { cs: { translation: cs, }, + pirate: { + translation: pirate, + }, }; i18n // pass the i18n instance to react-i18next. diff --git a/src/setup/iso6391.ts b/src/setup/iso6391.ts index 28d42806..3499155b 100644 --- a/src/setup/iso6391.ts +++ b/src/setup/iso6391.ts @@ -1,5 +1,6 @@ export type LangCode = | "none" + | "pirate" | "aa" | "ab" | "ae" @@ -219,6 +220,12 @@ export const captionLanguages: CaptionLanguageOption[] = [ name: "None", nativeName: "Lorem ipsum", }, + { + id: "pirate", + englishName: "Pirate", + name: "Pirate English", + nativeName: "Pirate English", + }, { id: "aa", englishName: "Afar", diff --git a/src/setup/locales/pirate/translation.json b/src/setup/locales/pirate/translation.json new file mode 100644 index 00000000..e87a7d2d --- /dev/null +++ b/src/setup/locales/pirate/translation.json @@ -0,0 +1,124 @@ +{ + "global": { + "name": "movie-web" + }, + "search": { + "loading_series": "Fetchin' yer favorite series...", + "loading_movie": "Fetchin' yer favorite movies...", + "loadin'": "Loadin'...", + "allResults": "That be all we 'ave, me hearty!", + "noResults": "We couldn't find anythin' that matches yer search!", + "allFailed": "Failed t' find media, walk the plank and try again!", + "headingTitle": "Search results", + "bookmarks": "Treasure Maps", + "continueWatchin'": "Continue Watchin'", + "title": "Wha' be ye wantin' to watch, me matey?", + "placeholder": "Wha' be ye searchin' for?" + }, + "media": { + "movie": "Movie", + "series": "Series", + "stopEditin'": "Stop editin'", + "errors": { + "genericTitle": "Shiver me timbers! It broke!", + "failedMeta": "Ye can't trust the compass, failed to load meta", + "mediaFailed": "We failed t' request the media ye asked fer, check yer internet connection, or Davy Jones's locker awaits ye!", + "videoFailed": "Blimey! We encountered an error while playin' the video ye requested. If this keeps happening please report the issue to the <0>Discord server or on <1>GitHub." + } + }, + "seasons": { + "seasonAndEpisode": "S{{season}} E{{episode}}" + }, + "notFound": { + "genericTitle": "Ahoy! I see nothin' on the horizon.", + "backArrow": "Back to the port", + "media": { + "title": "Avast ye! Couldn't find that media", + "description": "We couldn't find the media ye requested. Either it's been scuttled or ye tampered with the URL, ye scallywag!" + }, + "provider": { + "title": "Walk the plank! This provider has been disabled", + "description": "We had issues wit' the provider or 'twas too unstable t' use, so we had t' disable it. Try another one, arrr!" + }, + "page": { + "title": "Avast ye! Couldn't find that page.", + "description": "Arrr! We searched every inch o' the vessel: from the bilge to the crow's nest, from the keel to the topmast, but avast! We couldn't find the page ye be lookin' fer, me heartie." + } + }, + "searchBar": { + "movie": "Movie", + "series": "Series", + "Search": "Search" + }, + "videoPlayer": { + "findingBestVideo": "Finding the best video fer ye, hoist the colors!", + "noVideos": "Blistering barnacles, couldn't find any videos fer ye. Ye need a better map!", + "loading": "Loading...", + "backToHome": "Back to the port, mates!", + "backToHomeShort": "Back", + "seasonAndEpisode": "S{{season}} E{{episode}}", + "timeLeft": "{{timeLeft}} left", + "finishAt": "Finish at {{timeFinished}}", + "buttons": { + "episodes": "Episodes", + "source": "Source", + "captions": "Captions", + "download": "Download", + "settings": "Settings", + "pictureInPicture": "Spyglass view", + "playbackSpeed": "Set sail faster!" + }, + "popouts": { + "back": "Avast ye, go back!", + "sources": "Wha' provider do ye want to use?", + "seasons": "Choose which season you wants to watch!", + "captions": "Select a subtitle language, me hearty!", + "playbackSpeed": "Make the video faster than Blackbeard's ship!", + "customPlaybackSpeed": "Set a custom playback speed", + "captionPreferences": { + "title": "Customize yer captions", + "delay": "Delay", + "fontSize": "Size", + "opacity": "Opacity", + "color": "Color" + }, + "episode": "E{{index}} - {{title}}", + "noCaptions": "No captions, hoist the Jolly Roger!", + "linkedCaptions": "Linked captions, drop anchor!", + "customCaption": "Custom caption, arrr!", + "uploadCustomCaption": "Upload yer own caption!", + "noEmbeds": "No embeds we be found fer this source", + + "errors": { + "loadingWentWong": "Shiver me timbers! Somethin' went wrong loadin' the episodes fer {{seasonTitle}}", + "embedsError": "Blimey! Somethin' went wrong loadin' the embeds fer this thin' that ye like" + }, + "descriptions": { + "sources": "Wha' provider do ye wants to use?", + "embeds": "Choose which video to view", + "seasons": "Choose which season ye wants to watch", + "episode": "Pick an episode", + "captions": "Choose a subtitle language", + "captionPreferences": "Make subtitles look how ye wants it", + "playbackSpeed": "Change the playback speed" + } + }, + "errors": { + "fatalError": "Blow me down! The video player encounted a fatal error, please report it to the <0>Discord server or on <1>GitHub." + } + }, + "settings": { + "title": "Settings", + "language": "Language", + "captionLanguage": "Caption Language" + }, + "v3": { + "newSiteTitle": "New version now released!", + "newDomain": "https://movie-web.app", + "newDomainText": "movie-web will soon be movin' to a new domain: <0>https://movie-web.app. Make sure to update all yer bookmarks as <1>the ole website will stop workin' on {{date}}.", + "tireless": "We've worked tirelessly on this new update, we hope ye will enjoy wha' we've been cookin' up fer the past months.", + "leaveAnnouncement": "Take me thar!" + }, + "casting": { "casting": "Casting to device..." }, + "errors": { "offline": "Avast! Check yer internet connection" } +} From be03a8eb42072e7fc907743287788c9813c9e80b Mon Sep 17 00:00:00 2001 From: zisra <100528712+zisra@users.noreply.github.com> Date: Fri, 26 May 2023 08:01:55 -0500 Subject: [PATCH 09/96] Update src/setup/locales/pirate/translation.json Co-authored-by: Jip Frijlink --- src/setup/locales/pirate/translation.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/setup/locales/pirate/translation.json b/src/setup/locales/pirate/translation.json index e87a7d2d..c084ae34 100644 --- a/src/setup/locales/pirate/translation.json +++ b/src/setup/locales/pirate/translation.json @@ -66,7 +66,7 @@ "download": "Download", "settings": "Settings", "pictureInPicture": "Spyglass view", - "playbackSpeed": "Set sail faster!" + "playbackSpeed": "Set sail!" }, "popouts": { "back": "Avast ye, go back!", From 519e74480ecdd38ed787547edcea8b7a8b3e8d7a Mon Sep 17 00:00:00 2001 From: zisra <100528712+zisra@users.noreply.github.com> Date: Fri, 26 May 2023 10:45:45 -0500 Subject: [PATCH 10/96] Update translation.json --- src/setup/locales/pirate/translation.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/setup/locales/pirate/translation.json b/src/setup/locales/pirate/translation.json index c084ae34..a4a92117 100644 --- a/src/setup/locales/pirate/translation.json +++ b/src/setup/locales/pirate/translation.json @@ -73,7 +73,7 @@ "sources": "Wha' provider do ye want to use?", "seasons": "Choose which season you wants to watch!", "captions": "Select a subtitle language, me hearty!", - "playbackSpeed": "Make the video faster than Blackbeard's ship!", + "playbackSpeed": "Change the speed of Blackbeard's ship!", "customPlaybackSpeed": "Set a custom playback speed", "captionPreferences": { "title": "Customize yer captions", From 3c096c069cb11e27083eeebf37cac764e3bf5600 Mon Sep 17 00:00:00 2001 From: JORDAAR <69628820+Jordaar@users.noreply.github.com> Date: Sat, 27 May 2023 02:27:04 +0530 Subject: [PATCH 11/96] lower rank --- src/backend/providers/sflix.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/backend/providers/sflix.ts b/src/backend/providers/sflix.ts index f33f6951..4121046b 100644 --- a/src/backend/providers/sflix.ts +++ b/src/backend/providers/sflix.ts @@ -8,7 +8,7 @@ const sflixBase = "https://sflix.video"; registerProvider({ id: "sflix", displayName: "Sflix", - rank: 135, + rank: 50, type: [MWMediaType.MOVIE, MWMediaType.SERIES], async scrape({ media, episode, progress }) { let searchQuery = `${media.meta.title} `; From 02135527c1e58dad76c4fcb75f3c50b8fe910a60 Mon Sep 17 00:00:00 2001 From: Jip Fr Date: Fri, 26 May 2023 23:04:11 +0200 Subject: [PATCH 12/96] Use URLSearchParams --- src/hooks/useQueryParams.ts | 11 +++-------- 1 file changed, 3 insertions(+), 8 deletions(-) diff --git a/src/hooks/useQueryParams.ts b/src/hooks/useQueryParams.ts index c10289a9..be8c3c86 100644 --- a/src/hooks/useQueryParams.ts +++ b/src/hooks/useQueryParams.ts @@ -6,14 +6,9 @@ export function useQueryParams() { const queryParams = useMemo(() => { // Basic absolutely-not-fool-proof URL query param parser - const obj: Record = {}; - for (const [key, value] of loc.search - .slice(1) - .split("&") - .map((e) => e.split("="))) { - const valueAsNum = Number(value); - obj[key] = Number.isNaN(valueAsNum) ? value : valueAsNum; - } + const obj: Record = Object.fromEntries( + new URLSearchParams(loc.search).entries() + ); return obj; }, [loc]); From 30e5ae7121525521a99d52881d51570213eda688 Mon Sep 17 00:00:00 2001 From: frost768 Date: Mon, 29 May 2023 22:10:07 +0300 Subject: [PATCH 13/96] add missing translation keys and polish translation --- src/components/popout/FloatingCard.tsx | 2 +- src/setup/i18n.ts | 4 + src/setup/locales/en/translation.json | 11 +- src/setup/locales/pl/translation.json | 137 ++++++++++++++++++ src/setup/locales/tr/translation.json | 11 +- .../popouts/EpisodeSelectionPopout.tsx | 15 +- 6 files changed, 171 insertions(+), 9 deletions(-) create mode 100644 src/setup/locales/pl/translation.json diff --git a/src/components/popout/FloatingCard.tsx b/src/components/popout/FloatingCard.tsx index b4fd250c..47bfcb59 100644 --- a/src/components/popout/FloatingCard.tsx +++ b/src/components/popout/FloatingCard.tsx @@ -154,7 +154,7 @@ export const FloatingCardView = { className="flex cursor-pointer items-center space-x-2 transition-colors duration-200 hover:text-white" > - Close + {t("videoPlayer.popouts.close")} ); diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index 6116434a..d3aa261f 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -9,6 +9,7 @@ import en from "./locales/en/translation.json"; import fr from "./locales/fr/translation.json"; import nl from "./locales/nl/translation.json"; import pirate from "./locales/pirate/translation.json"; +import pl from "./locales/pl/translation.json"; import tr from "./locales/tr/translation.json"; import zh from "./locales/zh/translation.json"; @@ -37,6 +38,9 @@ const locales = { pirate: { translation: pirate, }, + pl: { + translation: pl, + }, }; i18n // pass the i18n instance to react-i18next. diff --git a/src/setup/locales/en/translation.json b/src/setup/locales/en/translation.json index d8b81da4..d90b568b 100644 --- a/src/setup/locales/en/translation.json +++ b/src/setup/locales/en/translation.json @@ -71,7 +71,16 @@ "popouts": { "back": "Go back", "sources": "Sources", - "seasons": "Seasons", + "close": "Close", + "seasons": { + "title":"Seasons", + "other": "Other seasons", + "noSeason": "No season" + }, + "episodes": { + "unknown": "Unknown episode", + "noEpisode": "No episode" + }, "captions": "Captions", "playbackSpeed": "Playback speed", "customPlaybackSpeed": "Custom playback speed", diff --git a/src/setup/locales/pl/translation.json b/src/setup/locales/pl/translation.json new file mode 100644 index 00000000..a85d9c70 --- /dev/null +++ b/src/setup/locales/pl/translation.json @@ -0,0 +1,137 @@ +{ + "global": { + "name": "movie-web" + }, + "search": { + "loading_series": "Szukamy twoich ulubionych seriali...", + "loading_movie": "Szukamy twoich ulubionych filmów...", + "loading": "Wczytywanie...", + "allResults": "To wszystko co mamy!", + "noResults": "Nie mogliśmy niczego znaleźć!", + "allFailed": "Nie udało się znaleźć mediów, Spróbuj ponownie!", + "headingTitle": "Wyniki wyszukiwania", + "bookmarks": "Zakładki", + "continueWatching": "Kontynuuj oglądanie", + "title": "Co chciałbyś obejrzeć?", + "placeholder": "Co chciałbyś obejrzeć?" + }, + "media": { + "movie": "Film", + "series": "Serial", + "stopEditing": "Zatrzymaj edycje", + "errors": { + "genericTitle": "Ups, popsuło się!", + "failedMeta": "Nie udało się wczytać metadanych", + "mediaFailed": "Nie udało nam się zarządać mediów, sprawdź połączenie sieciowe i spróbuj ponownie.", + "videoFailed": "Napotkaliśmy błąd podczas odtwarzania rządanego video. Jeśli problem będzie się powtarzać prosimy o zgłoszenie problemu na <0>Serwer Discord lub na <1>GitHub." + } + }, + "seasons": { + "seasonAndEpisode": "S{{season}} E{{episode}}" + }, + "notFound": { + "genericTitle": "Nie znaleziono", + "backArrow": "Wróć na stronę główną", + "media": { + "title": "Nie można znaleźć multimediów", + "description": "Nie mogliśmy znaleźć rządanych multimediów. Albo zostały usunięte, albo grzebałeś przy adresie URL." + }, + "provider": { + "title": "Ten dostawca został wyłączony", + "description": "Mieliśmy problemy z tym dostawcą, albo był zbyt niestabilny, więc musieliśmy go wyłączyć." + }, + "page": { + "title": "Nie można znaleźć tej strony", + "description": "Szukaliśmy wszędzie: w koszu, w szafie a nawet w piwnicy, ale nie byliśmy w stanie znaleźć strony której szukasz." + } + }, + "searchBar": { + "movie": "Filmy", + "series": "Seriale", + "Search": "Szukaj" + }, + "videoPlayer": { + "findingBestVideo": "Szukamy najlepszego video dla ciebie", + "noVideos": "Oj, Nie mogliśmy znaleźć żadnego video", + "loading": "Wczytywanie...", + "backToHome": "Wróć na stronę główną", + "backToHomeShort": "Wróć", + "seasonAndEpisode": "S{{season}} E{{episode}}", + "timeLeft": "Pozostało {{timeLeft}}", + "finishAt": "Zakończ na {{timeFinished, datetime}}", + "buttons": { + "episodes": "Odcinki", + "source": "Źródło", + "captions": "Napisy", + "download": "Pobierz", + "settings": "Ustawienia", + "pictureInPicture": "Obraz w obrazie (PIP)", + "playbackSpeed": "Prędkość odtwarzania" + }, + "popouts": { + "close": "Zamknąć", + "seasons": { + "title":"Sezony", + "other": "Inne sezony", + "noSeason": "Brak sezonu" + }, + "episodes": { + "unknown": "Nieznany odcinki", + "noEpisode": "Brak odcinki" + }, + "back": "Wróć", + "sources": "Źródła", + "captions": "Napisy", + "playbackSpeed": "Prędkość odtwarzania", + "customPlaybackSpeed": "Niestandardowa prędkość odtwarzania", + "captionPreferences": { + "title": "Personalizuj", + "delay": "Opóźnienie", + "fontSize": "Rozmiar", + "opacity": "Przeźroczystość", + "color": "Kolor" + }, + "episode": "E{{index}} - {{title}}", + "noCaptions": "Brak napisów", + "linkedCaptions": "Załączone napisy", + "customCaption": "Napisy niestandardowe", + "uploadCustomCaption": "Załącz", + "noEmbeds": "Nie znaleziono osadzonych mediów dla tego źródła", + + "errors": { + "loadingWentWong": "Coś poszło nie tak {{seasonTitle}}", + "embedsError": "Coś poszło nie tak przy wczytywaniu osadzonych mediów" + }, + "descriptions": { + "sources": "Którego dostawcy chciałbyś używać?", + "embeds": "Wybierz, które video chcesz zobaczyć", + "seasons": "Wybierz, który sezon chcesz obejrzeć", + "episode": "Wybierz odcinek", + "captions": "Zmień język napisów", + "captionPreferences": "Ustaw napisy, tak jak ci to odpowiada", + "playbackSpeed": "Zmień prędkość odtwarzania" + } + }, + "errors": { + "fatalError": "Odtwarzacz napotkał poważny błąd, Prosimy o złoszenie tego na <0>Serwer Discord lub na <1>GitHub." + } + }, + "settings": { + "title": "Ustawienia", + "language": "Język", + "captionLanguage": "Język napisów" + }, + "v3": { + "newSiteTitle": "Nowa wersja została wydana!", + "newDomain": "https://movie-web.app", + "newDomainText": "movie-web przeniesie się wkrótce na nowy adres: <0>https://movie-web.app. Prosimy zaaktualizować swoje zakładki ponieważ <1>stara strona przestanie działać {{date}}.", + "tireless": "Pracowaliśmy niestrudzenie nad tą aktualizacją, Mamy nadzieję że będziecie zadowoleni z tego nad czym pracowaliśmy przez ostatnie parę miesięcy.", + "leaveAnnouncement": "Zabierz mnie tam!" + }, + "casting": { + "casting": "Przesyłanie do urządzenia..." + }, + "errors": { + "offline": "Sprawdź swoje połączenie sieciowe" + } +} diff --git a/src/setup/locales/tr/translation.json b/src/setup/locales/tr/translation.json index 326cc35d..bab6cd1d 100644 --- a/src/setup/locales/tr/translation.json +++ b/src/setup/locales/tr/translation.json @@ -71,7 +71,16 @@ "popouts": { "back": "Geri git", "sources": "Kaynaklar", - "seasons": "Sezonlar", + "close":"Kapat", + "seasons": { + "title":"Sezonlar", + "other": "Diğer sezonlar", + "noSeason": "Sezon yok" + }, + "episodes": { + "unknown": "Bilinmeyen bölüm", + "noEpisode": "Bölüm yok" + }, "captions": "Altyazılar", "playbackSpeed": "Oynatma hızı", "customPlaybackSpeed": "Özel oynatma hızı", diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index bd152378..c80045bd 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -99,10 +99,10 @@ export function EpisodeSelectionPopout() { <> navigate("/episodes")} - backText={`To ${currentSeasonInfo?.title.toLowerCase()}`} + backText={currentSeasonInfo?.title} /> {currentSeasonInfo @@ -115,12 +115,15 @@ export function EpisodeSelectionPopout() { {season.title} )) - : "No season"} + : t("videoPlayer.popouts.seasons.noSeason")} navigate("/episodes/seasons")} className="flex cursor-pointer items-center space-x-2 transition-colors duration-200 hover:text-white" > - Other seasons + {t("videoPlayer.popouts.seasons.other")} } @@ -181,7 +184,7 @@ export function EpisodeSelectionPopout() { })} )) - : "No episodes"} + : t("videoPlayer.popouts.episodes.noEpisode")} )} From cc4f64032a92f82170de9e687c70e66f1bd17d96 Mon Sep 17 00:00:00 2001 From: Federico Benedetti Date: Sat, 3 Jun 2023 11:55:57 +0200 Subject: [PATCH 14/96] Add Italian language support --- src/setup/i18n.ts | 4 + src/setup/locales/it/translation.json | 128 ++++++++++++++++++++++++++ 2 files changed, 132 insertions(+) create mode 100644 src/setup/locales/it/translation.json diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index 6116434a..aa69517a 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -11,11 +11,15 @@ import nl from "./locales/nl/translation.json"; import pirate from "./locales/pirate/translation.json"; import tr from "./locales/tr/translation.json"; import zh from "./locales/zh/translation.json"; +import it from "./locales/it/translation.json"; const locales = { en: { translation: en, }, + it: { + translation: it, + }, nl: { translation: nl, }, diff --git a/src/setup/locales/it/translation.json b/src/setup/locales/it/translation.json new file mode 100644 index 00000000..7c28992c --- /dev/null +++ b/src/setup/locales/it/translation.json @@ -0,0 +1,128 @@ +{ + "global": { + "name": "movie-web" + }, + "search": { + "loading_series": "Recupero delle tue serie preferite...", + "loading_movie": "Recupero dei tuoi film preferiti...", + "loading": "Caricamento...", + "allResults": "Ecco tutto ciò che abbiamo!", + "noResults": "Non abbiamo trovato nulla!", + "allFailed": "Impossibile trovare i media, riprova!", + "headingTitle": "Risultati della ricerca", + "bookmarks": "Segnalibri", + "continueWatching": "Continua a guardare", + "title": "Cosa vuoi guardare?", + "placeholder": "Cosa vuoi guardare?" + }, + "media": { + "movie": "Film", + "series": "Serie", + "stopEditing": "Interrompi modifica", + "errors": { + "genericTitle": "Ops, qualcosa si è rotto!", + "failedMeta": "Caricamento dei metadati non riuscito", + "mediaFailed": "Impossibile richiedere il media che hai richiesto, controlla la tua connessione internet e riprova.", + "videoFailed": "Si è verificato un errore durante la riproduzione del video che hai richiesto. Se ciò continua a accadere, segnala il problema sul <0>server Discord o su <1>GitHub." + } + }, + "seasons": { + "seasonAndEpisode": "S{{season}} E{{episode}}" + }, + "notFound": { + "genericTitle": "Non trovato", + "backArrow": "Torna alla home", + "media": { + "title": "Impossibile trovare quel media", + "description": "Non siamo riusciti a trovare il media richiesto. È stato rimosso o hai manomesso l'URL." + }, + "provider": { + "title": "Questo provider è stato disabilitato", + "description": "Abbiamo riscontrato problemi con il provider o era troppo instabile da utilizzare, quindi abbiamo dovuto disabilitarlo." + }, + "page": { + "title": "Impossibile trovare quella pagina", + "description": "Abbiamo cercato ovunque: sotto i bidoni, nell'armadio, dietro il proxy, ma alla fine non siamo riusciti a trovare la pagina che stai cercando." + } + }, + "searchBar": { + "movie": "Film", + "series": "Serie", + "Search": "Cerca" + }, + "videoPlayer": { + "findingBestVideo": "Ricerca del miglior video per te", + "noVideos": "Ops, non è stato possibile trovare alcun video per te", + "loading": "Caricamento...", + "backToHome": "Torna alla home", + "backToHomeShort": "Indietro", + "seasonAndEpisode": "S{{season}} E{{episode}}", + "timeLeft": "{{timeLeft}} rimanente", + "finishAt": "Fine alle {{timeFinished, datetime}}", + "buttons": { + "episodes": "Episodi", + "source": "Fonte", + "captions": "Sottotitoli", + "download": "Download", + "settings": "Impostazioni", + "pictureInPicture": "Picture in Picture", + "playbackSpeed": "Velocità di riproduzione" + }, + "popouts": { + "back": "Torna indietro", + "sources": "Fonti", + "seasons": "Stagioni", + "captions": "Sottotitoli", + "playbackSpeed": "Velocità di riproduzione", + "customPlaybackSpeed": "Velocità di riproduzione personalizzata", + "captionPreferences": { + "title": "Personalizza", + "delay": "Ritardo", + "fontSize": "Dimensione carattere", + "opacity": "Opacità", + "color": "Colore" + }, + "episode": "E{{index}} - {{title}}", + "noCaptions": "Nessun sottotitolo", + "linkedCaptions": "Sottotitoli collegati", + "customCaption": "Sottotitolo personalizzato", + "uploadCustomCaption": "Carica sottotitolo", + "noEmbeds": "Nessun embed è stato trovato per questa fonte", + + "errors": { + "loadingWentWong": "Si è verificato un problema durante il caricamento degli episodi per {{seasonTitle}}", + "embedsError": "Si è verificato un problema durante il caricamento degli embed per questa cosa che ti piace" + }, + "descriptions": { + "sources": "Quale provider desideri utilizzare?", + "embeds": "Scegli quale video visualizzare", + "seasons": "Scegli quale stagione vuoi guardare", + "episode": "Scegli un episodio", + "captions": "Scegli una lingua per i sottotitoli", + "captionPreferences": "Personalizza l'aspetto dei sottotitoli", + "playbackSpeed": "Cambia la velocità di riproduzione" + } + }, + "errors": { + "fatalError": "Il lettore video ha riscontrato un errore fatale, segnalalo sul <0>server Discord o su <1>GitHub." + } + }, + "settings": { + "title": "Impostazioni", + "language": "Lingua", + "captionLanguage": "Lingua dei sottotitoli" + }, + "v3": { + "newSiteTitle": "Nuova versione ora disponibile!", + "newDomain": "https://movie-web.app", + "newDomainText": "movie-web si sposterà presto su un nuovo dominio: <0>https://movie-web.app. Assicurati di aggiornare tutti i tuoi segnalibri poiché <1>il vecchio sito smetterà di funzionare il {{date}}.", + "tireless": "Abbiamo lavorato instancabilmente su questo nuovo aggiornamento, speriamo che ti piaccia quello su cui abbiamo lavorato negli ultimi mesi.", + "leaveAnnouncement": "Portami lì!" + }, + "casting": { + "casting": "Trasmissione su dispositivo in corso..." + }, + "errors": { + "offline": "Controlla la tua connessione internet" + } +} From b7033a31c453f6cfe70117db5042c1a0c645f9a0 Mon Sep 17 00:00:00 2001 From: Federico Benedetti Date: Sat, 3 Jun 2023 12:15:19 +0200 Subject: [PATCH 15/96] Fix locale import position --- src/setup/i18n.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index aa69517a..e5f9358f 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -7,11 +7,11 @@ import cs from "./locales/cs/translation.json"; import de from "./locales/de/translation.json"; import en from "./locales/en/translation.json"; import fr from "./locales/fr/translation.json"; +import it from "./locales/it/translation.json"; import nl from "./locales/nl/translation.json"; import pirate from "./locales/pirate/translation.json"; import tr from "./locales/tr/translation.json"; import zh from "./locales/zh/translation.json"; -import it from "./locales/it/translation.json"; const locales = { en: { From 18bde24b3a21bd2c3df793a76fd072e476febf0f Mon Sep 17 00:00:00 2001 From: cloud <62519659+lem6ns@users.noreply.github.com> Date: Sun, 11 Jun 2023 11:29:55 -0600 Subject: [PATCH 16/96] feat(provider): Remote Stream --- src/backend/index.ts | 1 + src/backend/providers/remotestream.ts | 49 +++++++++++++++++++++++++++ 2 files changed, 50 insertions(+) create mode 100644 src/backend/providers/remotestream.ts diff --git a/src/backend/index.ts b/src/backend/index.ts index eb0ad897..d62f557b 100644 --- a/src/backend/index.ts +++ b/src/backend/index.ts @@ -9,6 +9,7 @@ import "./providers/m4ufree"; import "./providers/hdwatched"; import "./providers/2embed"; import "./providers/sflix"; +import "./providers/remotestream"; // embeds import "./embeds/streamm4u"; diff --git a/src/backend/providers/remotestream.ts b/src/backend/providers/remotestream.ts new file mode 100644 index 00000000..cf19f826 --- /dev/null +++ b/src/backend/providers/remotestream.ts @@ -0,0 +1,49 @@ +import { mwFetch } from "@/backend/helpers/fetch"; +import { registerProvider } from "@/backend/helpers/register"; +import { MWStreamQuality, MWStreamType } from "@/backend/helpers/streams"; +import { MWMediaType } from "@/backend/metadata/types"; + +const remotestreamBase = `https://fsa.remotestre.am`; + +registerProvider({ + id: "remotestream", + displayName: "Remote Stream", + disabled: false, + rank: 50, + type: [MWMediaType.MOVIE, MWMediaType.SERIES], + + async scrape({ media, episode, progress }) { + if (!this.type.includes(media.meta.type)) { + throw new Error("Unsupported type"); + } + + progress(30); + const type = media.meta.type === MWMediaType.MOVIE ? "Movies" : "Shows"; + let playlistLink = `${remotestreamBase}/${type}/${media.tmdbId}`; + + if (media.meta.type === MWMediaType.SERIES) { + const seasonNumber = media.meta.seasonData.number; + const episodeNumber = media.meta.seasonData.episodes.find( + (e) => e.id === episode + )?.number; + + playlistLink += `/${seasonNumber}/${episodeNumber}.m3u8`; + } else { + playlistLink += `/${media.tmdbId}.m3u8`; + } + + const streamRes = await mwFetch(playlistLink); + if (streamRes.type !== "application/x-mpegurl") + throw new Error("No watchable item found"); + progress(90); + return { + embeds: [], + stream: { + streamUrl: playlistLink, + quality: MWStreamQuality.QUNKNOWN, + type: MWStreamType.HLS, + captions: [], + }, + }; + }, +}); From 893a385f0043a989cbb39ab1772737294c49ca9b Mon Sep 17 00:00:00 2001 From: cloud <62519659+lem6ns@users.noreply.github.com> Date: Sun, 11 Jun 2023 11:34:57 -0600 Subject: [PATCH 17/96] fix(remotestream): additional path for tv --- src/backend/providers/remotestream.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/backend/providers/remotestream.ts b/src/backend/providers/remotestream.ts index cf19f826..8439d2c1 100644 --- a/src/backend/providers/remotestream.ts +++ b/src/backend/providers/remotestream.ts @@ -27,7 +27,7 @@ registerProvider({ (e) => e.id === episode )?.number; - playlistLink += `/${seasonNumber}/${episodeNumber}.m3u8`; + playlistLink += `/${seasonNumber}/${episodeNumber}/${episodeNumber}.m3u8`; } else { playlistLink += `/${media.tmdbId}.m3u8`; } From ef782974fe20115e2993e482f1a7a48f8087b287 Mon Sep 17 00:00:00 2001 From: cloud <62519659+lem6ns@users.noreply.github.com> Date: Sun, 11 Jun 2023 11:36:05 -0600 Subject: [PATCH 18/96] fix(remotestream): Duplicate rank number --- src/backend/providers/remotestream.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/backend/providers/remotestream.ts b/src/backend/providers/remotestream.ts index 8439d2c1..02c0f199 100644 --- a/src/backend/providers/remotestream.ts +++ b/src/backend/providers/remotestream.ts @@ -9,7 +9,7 @@ registerProvider({ id: "remotestream", displayName: "Remote Stream", disabled: false, - rank: 50, + rank: 55, type: [MWMediaType.MOVIE, MWMediaType.SERIES], async scrape({ media, episode, progress }) { From 1a613287f8453672c83d9adb0fdbbb77d0f7772b Mon Sep 17 00:00:00 2001 From: cloud <62519659+lem6ns@users.noreply.github.com> Date: Sun, 11 Jun 2023 10:44:51 -0600 Subject: [PATCH 19/96] feat(provider): streamflix --- src/backend/index.ts | 1 + src/backend/providers/streamflix.ts | 70 +++++++++++++++++++++++++++++ 2 files changed, 71 insertions(+) create mode 100644 src/backend/providers/streamflix.ts diff --git a/src/backend/index.ts b/src/backend/index.ts index eb0ad897..447c2165 100644 --- a/src/backend/index.ts +++ b/src/backend/index.ts @@ -9,6 +9,7 @@ import "./providers/m4ufree"; import "./providers/hdwatched"; import "./providers/2embed"; import "./providers/sflix"; +import "./providers/streamflix"; // embeds import "./embeds/streamm4u"; diff --git a/src/backend/providers/streamflix.ts b/src/backend/providers/streamflix.ts new file mode 100644 index 00000000..90dd4975 --- /dev/null +++ b/src/backend/providers/streamflix.ts @@ -0,0 +1,70 @@ +import { proxiedFetch } from "@/backend/helpers/fetch"; +import { registerProvider } from "@/backend/helpers/register"; +import { + MWCaptionType, + MWStreamQuality, + MWStreamType, +} from "@/backend/helpers/streams"; +import { MWMediaType } from "@/backend/metadata/types"; + +const streamflixBase = "https://us-west2-compute-proxied.streamflix.one"; + +const qualityMap: Record = { + 360: MWStreamQuality.Q360P, + 540: MWStreamQuality.Q540P, + 480: MWStreamQuality.Q480P, + 720: MWStreamQuality.Q720P, + 1080: MWStreamQuality.Q1080P, +}; + +registerProvider({ + id: "streamflix", + displayName: "StreamFlix", + disabled: false, + rank: 69, + type: [MWMediaType.MOVIE, MWMediaType.SERIES], + + async scrape({ media, episode, progress }) { + if (!this.type.includes(media.meta.type)) { + throw new Error("Unsupported type"); + } + + progress(30); + const type = media.meta.type === MWMediaType.MOVIE ? "movies" : "tv"; + let seasonNumber: number | undefined; + let episodeNumber: number | undefined; + + if (media.meta.type === MWMediaType.SERIES) { + // can't do type === "tv" here :( + seasonNumber = media.meta.seasonData.number; + episodeNumber = media.meta.seasonData.episodes.find( + (e: any) => e.id === episode + )?.number; + } + + const streamRes = await proxiedFetch(`/api/player/${type}`, { + baseURL: streamflixBase, + params: { + id: media.tmdbId, + s: seasonNumber, + e: episodeNumber, + }, + }); + if (!streamRes.headers.Referer) throw new Error("No watchable item found"); + progress(90); + return { + embeds: [], + stream: { + streamUrl: streamRes.sources[0].url, + quality: qualityMap[streamRes.sources[0].quality], + type: MWStreamType.HLS, + captions: streamRes.subtitles.map((s: Record) => ({ + needsProxy: true, + url: s.url, + type: MWCaptionType.VTT, + langIso: s.lang, + })), + }, + }; + }, +}); From 40e45ae103f29796e7725e368f3605f8d292ce56 Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 12 Jun 2023 20:06:46 +0200 Subject: [PATCH 20/96] partial refactor --- src/backend/metadata/search_new.ts | 21 +++ src/backend/metadata/tmdb.ts | 78 +++++++++ src/backend/metadata/trakttv.ts | 166 ++++++++++++++++++ src/backend/metadata/types_new.ts | 264 +++++++++++++++++++++++++++++ 4 files changed, 529 insertions(+) create mode 100644 src/backend/metadata/search_new.ts create mode 100644 src/backend/metadata/tmdb.ts create mode 100644 src/backend/metadata/trakttv.ts create mode 100644 src/backend/metadata/types_new.ts diff --git a/src/backend/metadata/search_new.ts b/src/backend/metadata/search_new.ts new file mode 100644 index 00000000..4506514a --- /dev/null +++ b/src/backend/metadata/search_new.ts @@ -0,0 +1,21 @@ +import { SimpleCache } from "@/utils/cache"; + +import { Trakt, mediaTypeToTTV } from "./trakttv"; +import { MWMediaMeta, MWQuery } from "./types"; + +const cache = new SimpleCache(); +cache.setCompare((a, b) => { + return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); +}); +cache.initialize(); + +export async function searchForMedia(query: MWQuery): Promise { + if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; + const { searchQuery, type } = query; + + const contentType = mediaTypeToTTV(type); + + const results = await Trakt.search(searchQuery, contentType); + cache.set(query, results, 3600); + return results; +} diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts new file mode 100644 index 00000000..0700945b --- /dev/null +++ b/src/backend/metadata/tmdb.ts @@ -0,0 +1,78 @@ +import { conf } from "@/setup/config"; + +import { + DetailedMeta, + MWMediaType, + TMDBMediaStatic, + TMDBMovieData, + TMDBShowData, +} from "./types"; +import { mwFetch } from "../helpers/fetch"; + +export abstract class Tmdb { + private static baseURL = "https://api.themoviedb.org/3"; + + private static headers = { + accept: "application/json", + Authorization: `Bearer ${conf().TMDB_API_KEY}`, + }; + + private static async get(url: string): Promise { + const res = await mwFetch(url, { + headers: Tmdb.headers, + baseURL: Tmdb.baseURL, + }); + return res; + } + + public static getMediaDetails: TMDBMediaStatic["getMediaDetails"] = async ( + id: string, + type: MWMediaType + ) => { + let data; + + switch (type) { + case "movie": + data = await Tmdb.get(`/movie/${id}`); + break; + case "series": + data = await Tmdb.get(`/tv/${id}`); + break; + default: + throw new Error("Invalid media type"); + } + + return data; + }; + + public static getMediaPoster(posterPath: string | null): string | undefined { + if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; + } + + /* public static async getMetaFromId( + type: MWMediaType, + id: string, + seasonId?: string + ): Promise { + console.log("getMetaFromId", type, id, seasonId); + + const details = await Tmdb.getMediaDetails(id, type); + + if (!details) return null; + + let imdbId; + if (type === MWMediaType.MOVIE) { + imdbId = (details as TMDBMovieData).imdb_id ?? undefined; + } + + if (!meta.length) return null; + + console.log(meta); + + return { + meta, + imdbId, + tmdbId: id, + }; + } */ +} diff --git a/src/backend/metadata/trakttv.ts b/src/backend/metadata/trakttv.ts new file mode 100644 index 00000000..5fb67a17 --- /dev/null +++ b/src/backend/metadata/trakttv.ts @@ -0,0 +1,166 @@ +import { conf } from "@/setup/config"; + +import { Tmdb } from "./tmdb"; +import { + DetailedMeta, + MWMediaMeta, + MWMediaType, + MWSeasonMeta, + TMDBShowData, + TTVContentTypes, + TTVMediaResult, + TTVSearchResult, + TTVSeasonMetaResult, +} from "./types"; +import { mwFetch } from "../helpers/fetch"; + +export function mediaTypeToTTV(type: MWMediaType): TTVContentTypes { + if (type === MWMediaType.MOVIE) return "movie"; + if (type === MWMediaType.SERIES) return "show"; + throw new Error("unsupported type"); +} + +export function TTVMediaToMediaType(type: string): MWMediaType { + if (type === "movie") return MWMediaType.MOVIE; + if (type === "show") return MWMediaType.SERIES; + throw new Error("unsupported type"); +} + +export function formatTTVMeta( + media: TTVMediaResult, + season?: TTVSeasonMetaResult +): MWMediaMeta { + const type = TTVMediaToMediaType(media.object_type); + let seasons: undefined | MWSeasonMeta[]; + if (type === MWMediaType.SERIES) { + seasons = media.seasons + ?.sort((a, b) => a.season_number - b.season_number) + .map( + (v): MWSeasonMeta => ({ + title: v.title, + id: v.id.toString(), + number: v.season_number, + }) + ); + } + + return { + title: media.title, + id: media.id.toString(), + year: media.original_release_year?.toString(), + poster: media.poster, + type, + seasons: seasons as any, + seasonData: season + ? ({ + id: season.id.toString(), + number: season.season_number, + title: season.title, + episodes: season.episodes + .sort((a, b) => a.episode_number - b.episode_number) + .map((v) => ({ + id: v.id.toString(), + number: v.episode_number, + title: v.title, + })), + } as any) + : (undefined as any), + }; +} + +export function TTVMediaToId(media: MWMediaMeta): string { + return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); +} + +export function decodeTTVId( + paramId: string +): { id: string; type: MWMediaType } | null { + const [prefix, type, id] = paramId.split("-", 3); + if (prefix !== "TTV") return null; + let mediaType; + try { + mediaType = TTVMediaToMediaType(type); + } catch { + return null; + } + return { + type: mediaType, + id, + }; +} + +export async function formatTTVSearchResult( + result: TTVSearchResult +): Promise { + const type = TTVMediaToMediaType(result.type); + const media = result[result.type]; + + if (!media) throw new Error("invalid result"); + + const details = await Tmdb.getMediaDetails( + media.ids.tmdb.toString(), + TTVMediaToMediaType(result.type) + ); + console.log(details); + + const seasons = + type === MWMediaType.SERIES + ? (details as TMDBShowData).seasons?.map((v) => ({ + id: v.id, + title: v.name, + season_number: v.season_number, + })) + : undefined; + + return { + title: media.title, + poster: Tmdb.getMediaPoster(details.poster_path), + id: media.ids.trakt, + original_release_year: media.year, + ttv_entity_id: media.ids.slug, + object_type: mediaTypeToTTV(type), + seasons, + }; +} + +export abstract class Trakt { + private static baseURL = "https://api.trakt.tv"; + + private static headers = { + "Content-Type": "application/json", + "trakt-api-version": "2", + "trakt-api-key": conf().TRAKT_CLIENT_ID, + }; + + private static async get(url: string): Promise { + const res = await mwFetch(url, { + headers: Trakt.headers, + baseURL: Trakt.baseURL, + }); + return res; + } + + public static async search( + query: string, + type: "movie" | "show" + ): Promise { + const data = await Trakt.get( + `/search/${type}?query=${encodeURIComponent(query)}` + ); + + const formatted = await Promise.all( + // eslint-disable-next-line no-return-await + data.map(async (v) => await formatTTVSearchResult(v)) + ); + return formatted.map((v) => formatTTVMeta(v)); + } + + public static async getMetaFromId( + type: MWMediaType, + id: string, + seasonId?: string + ): Promise { + console.log("getMetaFromId", type, id, seasonId); + return null; + } +} diff --git a/src/backend/metadata/types_new.ts b/src/backend/metadata/types_new.ts new file mode 100644 index 00000000..f954ff6a --- /dev/null +++ b/src/backend/metadata/types_new.ts @@ -0,0 +1,264 @@ +export enum MWMediaType { + MOVIE = "movie", + SERIES = "series", + ANIME = "anime", +} + +export type MWSeasonMeta = { + id: string; + number: number; + title: string; +}; + +export type MWSeasonWithEpisodeMeta = { + id: string; + number: number; + title: string; + episodes: { + id: string; + number: number; + title: string; + }[]; +}; + +type MWMediaMetaBase = { + title: string; + id: string; + year?: string; + poster?: string; +}; + +type MWMediaMetaSpecific = + | { + type: MWMediaType.MOVIE | MWMediaType.ANIME; + seasons: undefined; + } + | { + type: MWMediaType.SERIES; + seasons: MWSeasonMeta[]; + seasonData: MWSeasonWithEpisodeMeta; + }; + +export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; + +export interface MWQuery { + searchQuery: string; + type: MWMediaType; +} + +export type TTVContentTypes = "movie" | "show"; + +export type TTVSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type TTVEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type TTVMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + ttv_entity_id: string; + object_type: TTVContentTypes; + seasons?: TTVSeasonShort[]; +}; + +export type TTVSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: TTVEpisodeShort[]; +}; + +export interface TTVSearchResult { + type: "movie" | "show"; + score: number; + movie?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + imdb: string; + tmdb: number; + }; + }; + show?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + tvdb: number; + imdb: string; + tmdb: number; + }; + }; +} + +export interface DetailedMeta { + meta: MWMediaMeta; + imdbId?: string; + tmdbId?: string; +} + +export interface TMDBShowData { + adult: boolean; + backdrop_path: string | null; + created_by: { + id: number; + credit_id: string; + name: string; + gender: number; + profile_path: string | null; + }[]; + episode_run_time: number[]; + first_air_date: string; + genres: { + id: number; + name: string; + }[]; + homepage: string; + id: number; + in_production: boolean; + languages: string[]; + last_air_date: string; + last_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + name: string; + next_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + networks: { + id: number; + logo_path: string; + name: string; + origin_country: string; + }[]; + number_of_episodes: number; + number_of_seasons: number; + origin_country: string[]; + original_language: string; + original_name: string; + overview: string; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + seasons: { + air_date: string; + episode_count: number; + id: number; + name: string; + overview: string; + poster_path: string | null; + season_number: number; + }[]; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string; + type: string; + vote_average: number; + vote_count: number; +} + +export interface TMDBMovieData { + adult: boolean; + backdrop_path: string | null; + belongs_to_collection: { + id: number; + name: string; + poster_path: string | null; + backdrop_path: string | null; + } | null; + budget: number; + genres: { + id: number; + name: string; + }[]; + homepage: string | null; + id: number; + imdb_id: string | null; + original_language: string; + original_title: string; + overview: string | null; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + release_date: string; + revenue: number; + runtime: number | null; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string | null; + title: string; + video: boolean; + vote_average: number; + vote_count: number; +} + +export type TMDBMediaDetailsPromise = Promise; + +export interface TMDBMediaStatic { + getMediaDetails( + id: string, + type: MWMediaType.SERIES + ): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; +} From dfe67157d42308c5ce421ebb3a852a306d6a9716 Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 12 Jun 2023 20:17:42 +0200 Subject: [PATCH 21/96] preliminary refactor --- src/backend/metadata/search.ts | 48 +----- src/backend/metadata/search_new.ts | 21 --- src/backend/metadata/search_old.ts | 59 +++++++ src/backend/metadata/tmdb.ts | 1 - src/backend/metadata/types.ts | 217 ++++++++++++++++++++++++ src/backend/metadata/types_new.ts | 264 ----------------------------- src/backend/metadata/types_old.ts | 47 +++++ src/setup/config.ts | 4 + 8 files changed, 332 insertions(+), 329 deletions(-) delete mode 100644 src/backend/metadata/search_new.ts create mode 100644 src/backend/metadata/search_old.ts delete mode 100644 src/backend/metadata/types_new.ts create mode 100644 src/backend/metadata/types_old.ts diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 10cbb285..4506514a 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -1,14 +1,7 @@ import { SimpleCache } from "@/utils/cache"; -import { - JWContentTypes, - JWMediaResult, - JW_API_BASE, - formatJWMeta, - mediaTypeToJW, -} from "./justwatch"; +import { Trakt, mediaTypeToTTV } from "./trakttv"; import { MWMediaMeta, MWQuery } from "./types"; -import { proxiedFetch } from "../helpers/fetch"; const cache = new SimpleCache(); cache.setCompare((a, b) => { @@ -16,44 +9,13 @@ cache.setCompare((a, b) => { }); cache.initialize(); -type JWSearchQuery = { - content_types: JWContentTypes[]; - page: number; - page_size: number; - query: string; -}; - -type JWPage = { - items: T[]; - page: number; - page_size: number; - total_pages: number; - total_results: number; -}; - export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const contentType = mediaTypeToJW(type); - const body: JWSearchQuery = { - content_types: [contentType], - page: 1, - query: searchQuery, - page_size: 40, - }; + const contentType = mediaTypeToTTV(type); - const data = await proxiedFetch>( - "/content/titles/en_US/popular", - { - baseURL: JW_API_BASE, - params: { - body: JSON.stringify(body), - }, - } - ); - - const returnData = data.items.map((v) => formatJWMeta(v)); - cache.set(query, returnData, 3600); // cache for an hour - return returnData; + const results = await Trakt.search(searchQuery, contentType); + cache.set(query, results, 3600); + return results; } diff --git a/src/backend/metadata/search_new.ts b/src/backend/metadata/search_new.ts deleted file mode 100644 index 4506514a..00000000 --- a/src/backend/metadata/search_new.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { SimpleCache } from "@/utils/cache"; - -import { Trakt, mediaTypeToTTV } from "./trakttv"; -import { MWMediaMeta, MWQuery } from "./types"; - -const cache = new SimpleCache(); -cache.setCompare((a, b) => { - return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); -}); -cache.initialize(); - -export async function searchForMedia(query: MWQuery): Promise { - if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; - const { searchQuery, type } = query; - - const contentType = mediaTypeToTTV(type); - - const results = await Trakt.search(searchQuery, contentType); - cache.set(query, results, 3600); - return results; -} diff --git a/src/backend/metadata/search_old.ts b/src/backend/metadata/search_old.ts new file mode 100644 index 00000000..05be994d --- /dev/null +++ b/src/backend/metadata/search_old.ts @@ -0,0 +1,59 @@ +import { SimpleCache } from "@/utils/cache"; + +import { + JWContentTypes, + JWMediaResult, + JW_API_BASE, + formatJWMeta, + mediaTypeToJW, +} from "./justwatch"; +import { MWMediaMeta, MWQuery } from "./types_old"; +import { proxiedFetch } from "../helpers/fetch"; + +const cache = new SimpleCache(); +cache.setCompare((a, b) => { + return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); +}); +cache.initialize(); + +type JWSearchQuery = { + content_types: JWContentTypes[]; + page: number; + page_size: number; + query: string; +}; + +type JWPage = { + items: T[]; + page: number; + page_size: number; + total_pages: number; + total_results: number; +}; + +export async function searchForMedia(query: MWQuery): Promise { + if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; + const { searchQuery, type } = query; + + const contentType = mediaTypeToJW(type); + const body: JWSearchQuery = { + content_types: [contentType], + page: 1, + query: searchQuery, + page_size: 40, + }; + + const data = await proxiedFetch>( + "/content/titles/en_US/popular", + { + baseURL: JW_API_BASE, + params: { + body: JSON.stringify(body), + }, + } + ); + + const returnData = data.items.map((v) => formatJWMeta(v)); + cache.set(query, returnData, 3600); // cache for an hour + return returnData; +} diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 0700945b..460e13b4 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -1,7 +1,6 @@ import { conf } from "@/setup/config"; import { - DetailedMeta, MWMediaType, TMDBMediaStatic, TMDBMovieData, diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index 2723fbe7..f954ff6a 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -45,3 +45,220 @@ export interface MWQuery { searchQuery: string; type: MWMediaType; } + +export type TTVContentTypes = "movie" | "show"; + +export type TTVSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type TTVEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type TTVMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + ttv_entity_id: string; + object_type: TTVContentTypes; + seasons?: TTVSeasonShort[]; +}; + +export type TTVSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: TTVEpisodeShort[]; +}; + +export interface TTVSearchResult { + type: "movie" | "show"; + score: number; + movie?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + imdb: string; + tmdb: number; + }; + }; + show?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + tvdb: number; + imdb: string; + tmdb: number; + }; + }; +} + +export interface DetailedMeta { + meta: MWMediaMeta; + imdbId?: string; + tmdbId?: string; +} + +export interface TMDBShowData { + adult: boolean; + backdrop_path: string | null; + created_by: { + id: number; + credit_id: string; + name: string; + gender: number; + profile_path: string | null; + }[]; + episode_run_time: number[]; + first_air_date: string; + genres: { + id: number; + name: string; + }[]; + homepage: string; + id: number; + in_production: boolean; + languages: string[]; + last_air_date: string; + last_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + name: string; + next_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + networks: { + id: number; + logo_path: string; + name: string; + origin_country: string; + }[]; + number_of_episodes: number; + number_of_seasons: number; + origin_country: string[]; + original_language: string; + original_name: string; + overview: string; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + seasons: { + air_date: string; + episode_count: number; + id: number; + name: string; + overview: string; + poster_path: string | null; + season_number: number; + }[]; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string; + type: string; + vote_average: number; + vote_count: number; +} + +export interface TMDBMovieData { + adult: boolean; + backdrop_path: string | null; + belongs_to_collection: { + id: number; + name: string; + poster_path: string | null; + backdrop_path: string | null; + } | null; + budget: number; + genres: { + id: number; + name: string; + }[]; + homepage: string | null; + id: number; + imdb_id: string | null; + original_language: string; + original_title: string; + overview: string | null; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + release_date: string; + revenue: number; + runtime: number | null; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string | null; + title: string; + video: boolean; + vote_average: number; + vote_count: number; +} + +export type TMDBMediaDetailsPromise = Promise; + +export interface TMDBMediaStatic { + getMediaDetails( + id: string, + type: MWMediaType.SERIES + ): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; +} diff --git a/src/backend/metadata/types_new.ts b/src/backend/metadata/types_new.ts deleted file mode 100644 index f954ff6a..00000000 --- a/src/backend/metadata/types_new.ts +++ /dev/null @@ -1,264 +0,0 @@ -export enum MWMediaType { - MOVIE = "movie", - SERIES = "series", - ANIME = "anime", -} - -export type MWSeasonMeta = { - id: string; - number: number; - title: string; -}; - -export type MWSeasonWithEpisodeMeta = { - id: string; - number: number; - title: string; - episodes: { - id: string; - number: number; - title: string; - }[]; -}; - -type MWMediaMetaBase = { - title: string; - id: string; - year?: string; - poster?: string; -}; - -type MWMediaMetaSpecific = - | { - type: MWMediaType.MOVIE | MWMediaType.ANIME; - seasons: undefined; - } - | { - type: MWMediaType.SERIES; - seasons: MWSeasonMeta[]; - seasonData: MWSeasonWithEpisodeMeta; - }; - -export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; - -export interface MWQuery { - searchQuery: string; - type: MWMediaType; -} - -export type TTVContentTypes = "movie" | "show"; - -export type TTVSeasonShort = { - title: string; - id: number; - season_number: number; -}; - -export type TTVEpisodeShort = { - title: string; - id: number; - episode_number: number; -}; - -export type TTVMediaResult = { - title: string; - poster?: string; - id: number; - original_release_year?: number; - ttv_entity_id: string; - object_type: TTVContentTypes; - seasons?: TTVSeasonShort[]; -}; - -export type TTVSeasonMetaResult = { - title: string; - id: string; - season_number: number; - episodes: TTVEpisodeShort[]; -}; - -export interface TTVSearchResult { - type: "movie" | "show"; - score: number; - movie?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - imdb: string; - tmdb: number; - }; - }; - show?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - tvdb: number; - imdb: string; - tmdb: number; - }; - }; -} - -export interface DetailedMeta { - meta: MWMediaMeta; - imdbId?: string; - tmdbId?: string; -} - -export interface TMDBShowData { - adult: boolean; - backdrop_path: string | null; - created_by: { - id: number; - credit_id: string; - name: string; - gender: number; - profile_path: string | null; - }[]; - episode_run_time: number[]; - first_air_date: string; - genres: { - id: number; - name: string; - }[]; - homepage: string; - id: number; - in_production: boolean; - languages: string[]; - last_air_date: string; - last_episode_to_air: { - id: number; - name: string; - overview: string; - vote_average: number; - vote_count: number; - air_date: string; - episode_number: number; - production_code: string; - runtime: number | null; - season_number: number; - show_id: number; - still_path: string | null; - } | null; - name: string; - next_episode_to_air: { - id: number; - name: string; - overview: string; - vote_average: number; - vote_count: number; - air_date: string; - episode_number: number; - production_code: string; - runtime: number | null; - season_number: number; - show_id: number; - still_path: string | null; - } | null; - networks: { - id: number; - logo_path: string; - name: string; - origin_country: string; - }[]; - number_of_episodes: number; - number_of_seasons: number; - origin_country: string[]; - original_language: string; - original_name: string; - overview: string; - popularity: number; - poster_path: string | null; - production_companies: { - id: number; - logo_path: string | null; - name: string; - origin_country: string; - }[]; - production_countries: { - iso_3166_1: string; - name: string; - }[]; - seasons: { - air_date: string; - episode_count: number; - id: number; - name: string; - overview: string; - poster_path: string | null; - season_number: number; - }[]; - spoken_languages: { - english_name: string; - iso_639_1: string; - name: string; - }[]; - status: string; - tagline: string; - type: string; - vote_average: number; - vote_count: number; -} - -export interface TMDBMovieData { - adult: boolean; - backdrop_path: string | null; - belongs_to_collection: { - id: number; - name: string; - poster_path: string | null; - backdrop_path: string | null; - } | null; - budget: number; - genres: { - id: number; - name: string; - }[]; - homepage: string | null; - id: number; - imdb_id: string | null; - original_language: string; - original_title: string; - overview: string | null; - popularity: number; - poster_path: string | null; - production_companies: { - id: number; - logo_path: string | null; - name: string; - origin_country: string; - }[]; - production_countries: { - iso_3166_1: string; - name: string; - }[]; - release_date: string; - revenue: number; - runtime: number | null; - spoken_languages: { - english_name: string; - iso_639_1: string; - name: string; - }[]; - status: string; - tagline: string | null; - title: string; - video: boolean; - vote_average: number; - vote_count: number; -} - -export type TMDBMediaDetailsPromise = Promise; - -export interface TMDBMediaStatic { - getMediaDetails( - id: string, - type: MWMediaType.SERIES - ): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; -} diff --git a/src/backend/metadata/types_old.ts b/src/backend/metadata/types_old.ts new file mode 100644 index 00000000..2723fbe7 --- /dev/null +++ b/src/backend/metadata/types_old.ts @@ -0,0 +1,47 @@ +export enum MWMediaType { + MOVIE = "movie", + SERIES = "series", + ANIME = "anime", +} + +export type MWSeasonMeta = { + id: string; + number: number; + title: string; +}; + +export type MWSeasonWithEpisodeMeta = { + id: string; + number: number; + title: string; + episodes: { + id: string; + number: number; + title: string; + }[]; +}; + +type MWMediaMetaBase = { + title: string; + id: string; + year?: string; + poster?: string; +}; + +type MWMediaMetaSpecific = + | { + type: MWMediaType.MOVIE | MWMediaType.ANIME; + seasons: undefined; + } + | { + type: MWMediaType.SERIES; + seasons: MWSeasonMeta[]; + seasonData: MWSeasonWithEpisodeMeta; + }; + +export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; + +export interface MWQuery { + searchQuery: string; + type: MWMediaType; +} diff --git a/src/setup/config.ts b/src/setup/config.ts index f1db01da..c24117bb 100644 --- a/src/setup/config.ts +++ b/src/setup/config.ts @@ -8,6 +8,7 @@ interface Config { TMDB_API_KEY: string; CORS_PROXY_URL: string; NORMAL_ROUTER: boolean; + TRAKT_CLIENT_ID: string; } export interface RuntimeConfig { @@ -18,6 +19,7 @@ export interface RuntimeConfig { TMDB_API_KEY: string; NORMAL_ROUTER: boolean; PROXY_URLS: string[]; + TRAKT_CLIENT_ID: string; } const env: Record = { @@ -28,6 +30,7 @@ const env: Record = { DISCORD_LINK: undefined, CORS_PROXY_URL: import.meta.env.VITE_CORS_PROXY_URL, NORMAL_ROUTER: import.meta.env.VITE_NORMAL_ROUTER, + TRAKT_CLIENT_ID: import.meta.env.VITE_TRAKT_CLIENT_ID, }; const alerts = [] as string[]; @@ -62,5 +65,6 @@ export function conf(): RuntimeConfig { .split(",") .map((v) => v.trim()), NORMAL_ROUTER: getKey("NORMAL_ROUTER", "false") === "true", + TRAKT_CLIENT_ID: getKey("TRAKT_CLIENT_ID"), }; } From 1eac9f886eac6b3d1ecc59ae89396f9caa9c4938 Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 12 Jun 2023 21:25:24 +0200 Subject: [PATCH 22/96] more refactorings --- src/backend/metadata/getmeta.ts | 8 ++--- src/backend/metadata/justwatch.ts | 42 +++++-------------------- src/backend/metadata/search_old.ts | 9 ++---- src/backend/metadata/types.ts | 49 ++++++++++++++++++++++++++++++ 4 files changed, 62 insertions(+), 46 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 6b3b9a30..c4771451 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -1,13 +1,13 @@ import { FetchError } from "ofetch"; +import { formatJWMeta, mediaTypeToJW } from "./justwatch"; import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, - formatJWMeta, - mediaTypeToJW, -} from "./justwatch"; -import { MWMediaMeta, MWMediaType } from "./types"; + MWMediaMeta, + MWMediaType, +} from "./types"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; type JWExternalIdType = diff --git a/src/backend/metadata/justwatch.ts b/src/backend/metadata/justwatch.ts index 5c79c1e3..27c5aa4c 100644 --- a/src/backend/metadata/justwatch.ts +++ b/src/backend/metadata/justwatch.ts @@ -1,38 +1,10 @@ -import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types"; - -export const JW_API_BASE = "https://apis.justwatch.com"; -export const JW_IMAGE_BASE = "https://images.justwatch.com"; - -export type JWContentTypes = "movie" | "show"; - -export type JWSeasonShort = { - title: string; - id: number; - season_number: number; -}; - -export type JWEpisodeShort = { - title: string; - id: number; - episode_number: number; -}; - -export type JWMediaResult = { - title: string; - poster?: string; - id: number; - original_release_year?: number; - jw_entity_id: string; - object_type: JWContentTypes; - seasons?: JWSeasonShort[]; -}; - -export type JWSeasonMetaResult = { - title: string; - id: string; - season_number: number; - episodes: JWEpisodeShort[]; -}; +import { + JWContentTypes, + JWMediaResult, + JWSeasonMetaResult, + JW_IMAGE_BASE, +} from "./types"; +import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types_old"; export function mediaTypeToJW(type: MWMediaType): JWContentTypes { if (type === MWMediaType.MOVIE) return "movie"; diff --git a/src/backend/metadata/search_old.ts b/src/backend/metadata/search_old.ts index 05be994d..f12f62d2 100644 --- a/src/backend/metadata/search_old.ts +++ b/src/backend/metadata/search_old.ts @@ -1,12 +1,7 @@ import { SimpleCache } from "@/utils/cache"; -import { - JWContentTypes, - JWMediaResult, - JW_API_BASE, - formatJWMeta, - mediaTypeToJW, -} from "./justwatch"; +import { formatJWMeta, mediaTypeToJW } from "./justwatch"; +import { JWContentTypes, JWMediaResult, JW_API_BASE } from "./types"; import { MWMediaMeta, MWQuery } from "./types_old"; import { proxiedFetch } from "../helpers/fetch"; diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index f954ff6a..9e87d49d 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -262,3 +262,52 @@ export interface TMDBMediaStatic { getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; } + +export type JWContentTypes = "movie" | "show"; + +export type JWSearchQuery = { + content_types: JWContentTypes[]; + page: number; + page_size: number; + query: string; +}; + +export type JWPage = { + items: T[]; + page: number; + page_size: number; + total_pages: number; + total_results: number; +}; + +export const JW_API_BASE = "https://apis.justwatch.com"; +export const JW_IMAGE_BASE = "https://images.justwatch.com"; + +export type JWSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type JWEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type JWMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + jw_entity_id: string; + object_type: JWContentTypes; + seasons?: JWSeasonShort[]; +}; + +export type JWSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: JWEpisodeShort[]; +}; From e5ddb9816275251d242f79a52827d7b4bfdfad70 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 10:41:54 +0200 Subject: [PATCH 23/96] finish initial refactor --- src/backend/metadata/getmeta.ts | 55 +++++++++++++++++++++++++++++++++ src/backend/metadata/search.ts | 1 + src/backend/metadata/tmdb.ts | 27 ---------------- src/backend/metadata/trakttv.ts | 45 ++++++++++++++++++++------- src/backend/metadata/types.ts | 12 +++++++ 5 files changed, 101 insertions(+), 39 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index c4771451..e428199e 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -1,12 +1,17 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; +import { Tmdb } from "./tmdb"; +import { Trakt, formatTTVMeta } from "./trakttv"; import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, MWMediaMeta, MWMediaType, + TMDBMovieData, + TMDBShowData, + TTVSeasonMetaResult, } from "./types"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; @@ -37,6 +42,56 @@ export async function getMetaFromId( type: MWMediaType, id: string, seasonId?: string +): Promise { + const result = await Trakt.searchById(id, mediaTypeToJW(type)); + if (!result) return null; + const details = await Tmdb.getMediaDetails(id, type); + + if (!details) return null; + + let imdbId; + if (type === MWMediaType.MOVIE) { + imdbId = (details as TMDBMovieData).imdb_id ?? undefined; + } + + let seasonData: TTVSeasonMetaResult | undefined; + + if (type === MWMediaType.SERIES) { + const seasons = (details as TMDBShowData).seasons; + const season = + seasons?.find((v) => v.id.toString() === seasonId) ?? seasons?.[0]; + + const episodes = await Trakt.getEpisodes( + result.ttv_entity_id, + season?.season_number ?? 1 + ); + + if (season && episodes) { + seasonData = { + id: season.id.toString(), + season_number: season.season_number, + title: season.name, + episodes, + }; + } + } + + const meta = formatTTVMeta(result, seasonData); + if (!meta) return null; + + console.log(meta); + + return { + meta, + imdbId, + tmdbId: id, + }; +} + +export async function getLegacyMetaFromId( + type: MWMediaType, + id: string, + seasonId?: string ): Promise { const queryType = mediaTypeToJW(type); diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 4506514a..8eb246b7 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -16,6 +16,7 @@ export async function searchForMedia(query: MWQuery): Promise { const contentType = mediaTypeToTTV(type); const results = await Trakt.search(searchQuery, contentType); + console.log(results[0]); cache.set(query, results, 3600); return results; } diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 460e13b4..3aa1821f 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -47,31 +47,4 @@ export abstract class Tmdb { public static getMediaPoster(posterPath: string | null): string | undefined { if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; } - - /* public static async getMetaFromId( - type: MWMediaType, - id: string, - seasonId?: string - ): Promise { - console.log("getMetaFromId", type, id, seasonId); - - const details = await Tmdb.getMediaDetails(id, type); - - if (!details) return null; - - let imdbId; - if (type === MWMediaType.MOVIE) { - imdbId = (details as TMDBMovieData).imdb_id ?? undefined; - } - - if (!meta.length) return null; - - console.log(meta); - - return { - meta, - imdbId, - tmdbId: id, - }; - } */ } diff --git a/src/backend/metadata/trakttv.ts b/src/backend/metadata/trakttv.ts index 5fb67a17..ae50aefe 100644 --- a/src/backend/metadata/trakttv.ts +++ b/src/backend/metadata/trakttv.ts @@ -2,12 +2,13 @@ import { conf } from "@/setup/config"; import { Tmdb } from "./tmdb"; import { - DetailedMeta, MWMediaMeta, MWMediaType, MWSeasonMeta, TMDBShowData, TTVContentTypes, + TTVEpisodeResult, + TTVEpisodeShort, TTVMediaResult, TTVSearchResult, TTVSeasonMetaResult, @@ -69,14 +70,14 @@ export function formatTTVMeta( } export function TTVMediaToId(media: MWMediaMeta): string { - return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); + return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); } export function decodeTTVId( paramId: string ): { id: string; type: MWMediaType } | null { const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "TTV") return null; + if (prefix !== "MW") return null; let mediaType; try { mediaType = TTVMediaToMediaType(type); @@ -101,7 +102,6 @@ export async function formatTTVSearchResult( media.ids.tmdb.toString(), TTVMediaToMediaType(result.type) ); - console.log(details); const seasons = type === MWMediaType.SERIES @@ -115,7 +115,7 @@ export async function formatTTVSearchResult( return { title: media.title, poster: Tmdb.getMediaPoster(details.poster_path), - id: media.ids.trakt, + id: media.ids.tmdb, original_release_year: media.year, ttv_entity_id: media.ids.slug, object_type: mediaTypeToTTV(type), @@ -155,12 +155,33 @@ export abstract class Trakt { return formatted.map((v) => formatTTVMeta(v)); } - public static async getMetaFromId( - type: MWMediaType, - id: string, - seasonId?: string - ): Promise { - console.log("getMetaFromId", type, id, seasonId); - return null; + public static async searchById( + tmdbId: string, + type: "movie" | "show" + ): Promise { + const data = await Trakt.get( + `/search/tmdb/${tmdbId}?type=${type}` + ); + + const formatted = await Promise.all( + // eslint-disable-next-line no-return-await + data.map(async (v) => await formatTTVSearchResult(v)) + ); + return formatted[0]; + } + + public static async getEpisodes( + slug: string, + season: number + ): Promise { + const data = await Trakt.get( + `/shows/${slug}/seasons/${season}` + ); + + return data.map((e) => ({ + id: e.ids.tmdb, + episode_number: e.number, + title: e.title, + })); } } diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index 9e87d49d..07671e39 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -311,3 +311,15 @@ export type JWSeasonMetaResult = { season_number: number; episodes: JWEpisodeShort[]; }; + +export interface TTVEpisodeResult { + season: number; + number: number; + title: string; + ids: { + trakt: number; + tvdb: number; + imdb: string; + tmdb: number; + }; +} From baf744b5d6d82bbded5fb902027d35c61d283703 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 11:01:07 +0200 Subject: [PATCH 24/96] refactor url prefix --- src/backend/metadata/getmeta.ts | 28 ++++++++++++++++++- src/components/media/MediaCard.tsx | 4 +-- .../popouts/EpisodeSelectionPopout.tsx | 5 ++-- src/views/media/MediaView.tsx | 9 ++++-- 4 files changed, 37 insertions(+), 9 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index e428199e..26464299 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -2,7 +2,12 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; import { Tmdb } from "./tmdb"; -import { Trakt, formatTTVMeta } from "./trakttv"; +import { + TTVMediaToMediaType, + Trakt, + formatTTVMeta, + mediaTypeToTTV, +} from "./trakttv"; import { JWMediaResult, JWSeasonMetaResult, @@ -137,3 +142,24 @@ export async function getLegacyMetaFromId( tmdbId, }; } + +export function MWMediaToId(media: MWMediaMeta): string { + return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); +} + +export function decodeMWId( + paramId: string +): { id: string; type: MWMediaType } | null { + const [prefix, type, id] = paramId.split("-", 3); + if (prefix !== "MW") return null; + let mediaType; + try { + mediaType = TTVMediaToMediaType(type); + } catch { + return null; + } + return { + type: mediaType, + id, + }; +} diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index 22865717..3bac4d08 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -1,7 +1,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { JWMediaToId } from "@/backend/metadata/justwatch"; +import { MWMediaToId } from "@/backend/metadata/getmeta"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -132,7 +132,7 @@ export function MediaCard(props: MediaCardProps) { const canLink = props.linkable && !props.closable; let link = canLink - ? `/media/${encodeURIComponent(JWMediaToId(props.media))}` + ? `/media/${encodeURIComponent(MWMediaToId(props.media))}` : "#"; if (canLink && props.series) link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index bd152378..63ab1c81 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -2,8 +2,7 @@ import { useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; -import { getMetaFromId } from "@/backend/metadata/getmeta"; -import { decodeJWId } from "@/backend/metadata/justwatch"; +import { decodeMWId, getMetaFromId } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; @@ -45,7 +44,7 @@ export function EpisodeSelectionPopout() { seasonId: sId, season: undefined, }); - reqSeasonMeta(decodeJWId(params.media)?.id as string, sId).then((v) => { + reqSeasonMeta(decodeMWId(params.media)?.id as string, sId).then((v) => { if (v?.meta.type !== MWMediaType.SERIES) return; setCurrentVisibleSeason({ seasonId: sId, diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index c55211c7..1a4709e3 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -4,8 +4,11 @@ import { useTranslation } from "react-i18next"; import { useHistory, useParams } from "react-router-dom"; import { MWStream } from "@/backend/helpers/streams"; -import { DetailedMeta, getMetaFromId } from "@/backend/metadata/getmeta"; -import { decodeJWId } from "@/backend/metadata/justwatch"; +import { + DetailedMeta, + decodeMWId, + getMetaFromId, +} from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icons } from "@/components/Icon"; @@ -181,7 +184,7 @@ export function MediaView() { const [selected, setSelected] = useState(null); const [exec, loading, error] = useLoading( async (mediaParams: string, seasonId?: string) => { - const data = decodeJWId(mediaParams); + const data = decodeMWId(mediaParams); if (!data) return null; return getMetaFromId(data.type, data.id, seasonId); } From e889eaebaa60560556ce38f2c22ee71fbd31c453 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 14:06:37 +0200 Subject: [PATCH 25/96] implement legacy url conversion --- src/backend/metadata/getmeta.ts | 15 +++++++++++++++ src/setup/App.tsx | 18 +++++++++++++++++- 2 files changed, 32 insertions(+), 1 deletion(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 26464299..b6f90f26 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -163,3 +163,18 @@ export function decodeMWId( id, }; } + +export async function convertLegacyUrl( + url: string +): Promise { + if (url.startsWith("/media/JW")) { + const urlParts = url.split("/").slice(2); + const [, type, id] = urlParts[0].split("-", 3); + const meta = await getLegacyMetaFromId(TTVMediaToMediaType(type), id); + if (!meta) return undefined; + const tmdbId = meta.tmdbId; + if (!tmdbId) return undefined; + return `/media/MW-${type}-${tmdbId}`; + } + return undefined; +} diff --git a/src/setup/App.tsx b/src/setup/App.tsx index 992549e0..7be4d581 100644 --- a/src/setup/App.tsx +++ b/src/setup/App.tsx @@ -1,6 +1,13 @@ import { lazy } from "react"; -import { Redirect, Route, Switch } from "react-router-dom"; +import { + Redirect, + Route, + Switch, + useHistory, + useLocation, +} from "react-router-dom"; +import { convertLegacyUrl } from "@/backend/metadata/getmeta"; import { MWMediaType } from "@/backend/metadata/types"; import { BannerContextProvider } from "@/hooks/useBanner"; import { Layout } from "@/setup/Layout"; @@ -13,6 +20,15 @@ import { V2MigrationView } from "@/views/other/v2Migration"; import { SearchView } from "@/views/search/SearchView"; function App() { + const location = useLocation(); + const history = useHistory(); + + // Call the conversion function and redirect if necessary + convertLegacyUrl(location.pathname).then((convertedUrl) => { + if (convertedUrl) { + history.replace(convertedUrl); + } + }); return ( From a7af04530805d079f19aea5618c8989e8cab3105 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 14:20:33 +0200 Subject: [PATCH 26/96] refactor to initial prefix choice --- src/backend/metadata/getmeta.ts | 10 +++++----- src/components/media/MediaCard.tsx | 4 ++-- .../components/popouts/EpisodeSelectionPopout.tsx | 4 ++-- src/views/media/MediaView.tsx | 4 ++-- 4 files changed, 11 insertions(+), 11 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index b6f90f26..82b3c20b 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -143,15 +143,15 @@ export async function getLegacyMetaFromId( }; } -export function MWMediaToId(media: MWMediaMeta): string { - return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); +export function TTVMediaToId(media: MWMediaMeta): string { + return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); } -export function decodeMWId( +export function decodeTTVId( paramId: string ): { id: string; type: MWMediaType } | null { const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "MW") return null; + if (prefix !== "TTV") return null; let mediaType; try { mediaType = TTVMediaToMediaType(type); @@ -174,7 +174,7 @@ export async function convertLegacyUrl( if (!meta) return undefined; const tmdbId = meta.tmdbId; if (!tmdbId) return undefined; - return `/media/MW-${type}-${tmdbId}`; + return `/media/TTV-${type}-${tmdbId}`; } return undefined; } diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index 3bac4d08..b87654a9 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -1,7 +1,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { MWMediaToId } from "@/backend/metadata/getmeta"; +import { TTVMediaToId } from "@/backend/metadata/getmeta"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -132,7 +132,7 @@ export function MediaCard(props: MediaCardProps) { const canLink = props.linkable && !props.closable; let link = canLink - ? `/media/${encodeURIComponent(MWMediaToId(props.media))}` + ? `/media/${encodeURIComponent(TTVMediaToId(props.media))}` : "#"; if (canLink && props.series) link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index 63ab1c81..6cf5d2d6 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -2,7 +2,7 @@ import { useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; -import { decodeMWId, getMetaFromId } from "@/backend/metadata/getmeta"; +import { decodeTTVId, getMetaFromId } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; @@ -44,7 +44,7 @@ export function EpisodeSelectionPopout() { seasonId: sId, season: undefined, }); - reqSeasonMeta(decodeMWId(params.media)?.id as string, sId).then((v) => { + reqSeasonMeta(decodeTTVId(params.media)?.id as string, sId).then((v) => { if (v?.meta.type !== MWMediaType.SERIES) return; setCurrentVisibleSeason({ seasonId: sId, diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index 1a4709e3..c0e81455 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -6,7 +6,7 @@ import { useHistory, useParams } from "react-router-dom"; import { MWStream } from "@/backend/helpers/streams"; import { DetailedMeta, - decodeMWId, + decodeTTVId, getMetaFromId, } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; @@ -184,7 +184,7 @@ export function MediaView() { const [selected, setSelected] = useState(null); const [exec, loading, error] = useLoading( async (mediaParams: string, seasonId?: string) => { - const data = decodeMWId(mediaParams); + const data = decodeTTVId(mediaParams); if (!data) return null; return getMetaFromId(data.type, data.id, seasonId); } From b22e3ff8c14e4eb78e52dcdeb7a61814a37c6a75 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 14:25:31 +0200 Subject: [PATCH 27/96] cleanup --- src/backend/metadata/justwatch.ts | 4 ++- src/backend/metadata/search_old.ts | 54 ------------------------------ src/backend/metadata/types_old.ts | 47 -------------------------- 3 files changed, 3 insertions(+), 102 deletions(-) delete mode 100644 src/backend/metadata/search_old.ts delete mode 100644 src/backend/metadata/types_old.ts diff --git a/src/backend/metadata/justwatch.ts b/src/backend/metadata/justwatch.ts index 27c5aa4c..857ff006 100644 --- a/src/backend/metadata/justwatch.ts +++ b/src/backend/metadata/justwatch.ts @@ -3,8 +3,10 @@ import { JWMediaResult, JWSeasonMetaResult, JW_IMAGE_BASE, + MWMediaMeta, + MWMediaType, + MWSeasonMeta, } from "./types"; -import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types_old"; export function mediaTypeToJW(type: MWMediaType): JWContentTypes { if (type === MWMediaType.MOVIE) return "movie"; diff --git a/src/backend/metadata/search_old.ts b/src/backend/metadata/search_old.ts deleted file mode 100644 index f12f62d2..00000000 --- a/src/backend/metadata/search_old.ts +++ /dev/null @@ -1,54 +0,0 @@ -import { SimpleCache } from "@/utils/cache"; - -import { formatJWMeta, mediaTypeToJW } from "./justwatch"; -import { JWContentTypes, JWMediaResult, JW_API_BASE } from "./types"; -import { MWMediaMeta, MWQuery } from "./types_old"; -import { proxiedFetch } from "../helpers/fetch"; - -const cache = new SimpleCache(); -cache.setCompare((a, b) => { - return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); -}); -cache.initialize(); - -type JWSearchQuery = { - content_types: JWContentTypes[]; - page: number; - page_size: number; - query: string; -}; - -type JWPage = { - items: T[]; - page: number; - page_size: number; - total_pages: number; - total_results: number; -}; - -export async function searchForMedia(query: MWQuery): Promise { - if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; - const { searchQuery, type } = query; - - const contentType = mediaTypeToJW(type); - const body: JWSearchQuery = { - content_types: [contentType], - page: 1, - query: searchQuery, - page_size: 40, - }; - - const data = await proxiedFetch>( - "/content/titles/en_US/popular", - { - baseURL: JW_API_BASE, - params: { - body: JSON.stringify(body), - }, - } - ); - - const returnData = data.items.map((v) => formatJWMeta(v)); - cache.set(query, returnData, 3600); // cache for an hour - return returnData; -} diff --git a/src/backend/metadata/types_old.ts b/src/backend/metadata/types_old.ts deleted file mode 100644 index 2723fbe7..00000000 --- a/src/backend/metadata/types_old.ts +++ /dev/null @@ -1,47 +0,0 @@ -export enum MWMediaType { - MOVIE = "movie", - SERIES = "series", - ANIME = "anime", -} - -export type MWSeasonMeta = { - id: string; - number: number; - title: string; -}; - -export type MWSeasonWithEpisodeMeta = { - id: string; - number: number; - title: string; - episodes: { - id: string; - number: number; - title: string; - }[]; -}; - -type MWMediaMetaBase = { - title: string; - id: string; - year?: string; - poster?: string; -}; - -type MWMediaMetaSpecific = - | { - type: MWMediaType.MOVIE | MWMediaType.ANIME; - seasons: undefined; - } - | { - type: MWMediaType.SERIES; - seasons: MWSeasonMeta[]; - seasonData: MWSeasonWithEpisodeMeta; - }; - -export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; - -export interface MWQuery { - searchQuery: string; - type: MWMediaType; -} From 3ee9ee43a54cf5f0a21eac66b333fed749663644 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 21:23:47 +0200 Subject: [PATCH 28/96] refactor everything to use tmdb exclusively --- src/backend/metadata/getmeta.ts | 61 +++--- src/backend/metadata/search.ts | 20 +- src/backend/metadata/tmdb.ts | 159 ++++++++++++++- src/backend/metadata/trakttv.ts | 187 ------------------ src/backend/metadata/types.ts | 140 +++++++++---- src/components/media/MediaCard.tsx | 4 +- .../popouts/EpisodeSelectionPopout.tsx | 4 +- src/views/media/MediaView.tsx | 4 +- 8 files changed, 315 insertions(+), 264 deletions(-) delete mode 100644 src/backend/metadata/trakttv.ts diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 82b3c20b..777fae42 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -1,22 +1,22 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; -import { Tmdb } from "./tmdb"; import { - TTVMediaToMediaType, - Trakt, - formatTTVMeta, - mediaTypeToTTV, -} from "./trakttv"; + TMDBMediaToMediaType, + Tmdb, + formatTMDBMeta, + mediaTypeToTMDB, +} from "./tmdb"; import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, MWMediaMeta, MWMediaType, + TMDBMediaResult, TMDBMovieData, + TMDBSeasonMetaResult, TMDBShowData, - TTVSeasonMetaResult, } from "./types"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; @@ -48,9 +48,7 @@ export async function getMetaFromId( id: string, seasonId?: string ): Promise { - const result = await Trakt.searchById(id, mediaTypeToJW(type)); - if (!result) return null; - const details = await Tmdb.getMediaDetails(id, type); + const details = await Tmdb.getMediaDetails(id, mediaTypeToTMDB(type)); if (!details) return null; @@ -59,15 +57,15 @@ export async function getMetaFromId( imdbId = (details as TMDBMovieData).imdb_id ?? undefined; } - let seasonData: TTVSeasonMetaResult | undefined; + let seasonData: TMDBSeasonMetaResult | undefined; if (type === MWMediaType.SERIES) { const seasons = (details as TMDBShowData).seasons; const season = seasons?.find((v) => v.id.toString() === seasonId) ?? seasons?.[0]; - const episodes = await Trakt.getEpisodes( - result.ttv_entity_id, + const episodes = await Tmdb.getEpisodes( + details.id.toString(), season?.season_number ?? 1 ); @@ -81,10 +79,27 @@ export async function getMetaFromId( } } - const meta = formatTTVMeta(result, seasonData); - if (!meta) return null; + const tmdbmeta: TMDBMediaResult = { + id: details.id, + title: + type === MWMediaType.MOVIE + ? (details as TMDBMovieData).title + : (details as TMDBShowData).name, + object_type: mediaTypeToTMDB(type), + seasons: (details as TMDBShowData).seasons.map((v) => ({ + id: v.id, + season_number: v.season_number, + title: v.name, + })), + poster: (details as TMDBMovieData).poster_path ?? undefined, + original_release_year: + type === MWMediaType.MOVIE + ? Number((details as TMDBMovieData).release_date?.split("-")[0]) + : Number((details as TMDBShowData).first_air_date?.split("-")[0]), + }; - console.log(meta); + const meta = formatTMDBMeta(tmdbmeta, seasonData); + if (!meta) return null; return { meta, @@ -143,18 +158,18 @@ export async function getLegacyMetaFromId( }; } -export function TTVMediaToId(media: MWMediaMeta): string { - return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); +export function TMDBMediaToId(media: MWMediaMeta): string { + return ["tmdb", mediaTypeToTMDB(media.type), media.id].join("-"); } -export function decodeTTVId( +export function decodeTMDBId( paramId: string ): { id: string; type: MWMediaType } | null { const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "TTV") return null; + if (prefix !== "tmdb") return null; let mediaType; try { - mediaType = TTVMediaToMediaType(type); + mediaType = TMDBMediaToMediaType(type); } catch { return null; } @@ -170,11 +185,11 @@ export async function convertLegacyUrl( if (url.startsWith("/media/JW")) { const urlParts = url.split("/").slice(2); const [, type, id] = urlParts[0].split("-", 3); - const meta = await getLegacyMetaFromId(TTVMediaToMediaType(type), id); + const meta = await getLegacyMetaFromId(TMDBMediaToMediaType(type), id); if (!meta) return undefined; const tmdbId = meta.tmdbId; if (!tmdbId) return undefined; - return `/media/TTV-${type}-${tmdbId}`; + return `/media/tmdb-${type}-${tmdbId}`; } return undefined; } diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 8eb246b7..31b2c682 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -1,6 +1,11 @@ import { SimpleCache } from "@/utils/cache"; -import { Trakt, mediaTypeToTTV } from "./trakttv"; +import { + Tmdb, + formatTMDBMeta, + formatTMDBSearchResult, + mediaTypeToTMDB, +} from "./tmdb"; import { MWMediaMeta, MWQuery } from "./types"; const cache = new SimpleCache(); @@ -13,10 +18,17 @@ export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const contentType = mediaTypeToTTV(type); + const data = await Tmdb.searchMedia(searchQuery, mediaTypeToTMDB(type)); + const results = await Promise.all( + data.results.map(async (v) => { + const formattedResult = await formatTMDBSearchResult( + v, + mediaTypeToTMDB(type) + ); + return formatTMDBMeta(formattedResult); + }) + ); - const results = await Trakt.search(searchQuery, contentType); - console.log(results[0]); cache.set(query, results, 3600); return results; } diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 3aa1821f..f01709bb 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -1,13 +1,100 @@ import { conf } from "@/setup/config"; import { + MWMediaMeta, MWMediaType, + MWSeasonMeta, + TMDBContentTypes, + TMDBEpisodeShort, + TMDBMediaResult, TMDBMediaStatic, TMDBMovieData, + TMDBMovieResponse, + TMDBMovieResult, + TMDBSearchResultStatic, + TMDBSeason, + TMDBSeasonMetaResult, TMDBShowData, + TMDBShowResponse, + TMDBShowResult, } from "./types"; import { mwFetch } from "../helpers/fetch"; +export function mediaTypeToTMDB(type: MWMediaType): TMDBContentTypes { + if (type === MWMediaType.MOVIE) return "movie"; + if (type === MWMediaType.SERIES) return "show"; + throw new Error("unsupported type"); +} + +export function TMDBMediaToMediaType(type: string): MWMediaType { + if (type === "movie") return MWMediaType.MOVIE; + if (type === "show") return MWMediaType.SERIES; + throw new Error("unsupported type"); +} + +export function formatTMDBMeta( + media: TMDBMediaResult, + season?: TMDBSeasonMetaResult +): MWMediaMeta { + const type = TMDBMediaToMediaType(media.object_type); + let seasons: undefined | MWSeasonMeta[]; + if (type === MWMediaType.SERIES) { + seasons = media.seasons + ?.sort((a, b) => a.season_number - b.season_number) + .map( + (v): MWSeasonMeta => ({ + title: v.title, + id: v.id.toString(), + number: v.season_number, + }) + ); + } + + return { + title: media.title, + id: media.id.toString(), + year: media.original_release_year?.toString(), + poster: media.poster, + type, + seasons: seasons as any, + seasonData: season + ? ({ + id: season.id.toString(), + number: season.season_number, + title: season.title, + episodes: season.episodes + .sort((a, b) => a.episode_number - b.episode_number) + .map((v) => ({ + id: v.id.toString(), + number: v.episode_number, + title: v.title, + })), + } as any) + : (undefined as any), + }; +} + +export function TMDBMediaToId(media: MWMediaMeta): string { + return ["tmdb", mediaTypeToTMDB(media.type), media.id].join("-"); +} + +export function decodeTMDBId( + paramId: string +): { id: string; type: MWMediaType } | null { + const [prefix, type, id] = paramId.split("-", 3); + if (prefix !== "tmdb") return null; + let mediaType; + try { + mediaType = TMDBMediaToMediaType(type); + } catch { + return null; + } + return { + type: mediaType, + id, + }; +} + export abstract class Tmdb { private static baseURL = "https://api.themoviedb.org/3"; @@ -24,9 +111,33 @@ export abstract class Tmdb { return res; } + public static searchMedia: TMDBSearchResultStatic["searchMedia"] = async ( + query: string, + type: TMDBContentTypes + ) => { + let data; + + switch (type) { + case "movie": + data = await Tmdb.get( + `search/movie?query=${query}&include_adult=true&language=en-US&page=1` + ); + break; + case "show": + data = await Tmdb.get( + `search/tv?query=${query}&include_adult=true&language=en-US&page=1` + ); + break; + default: + throw new Error("Invalid media type"); + } + + return data; + }; + public static getMediaDetails: TMDBMediaStatic["getMediaDetails"] = async ( id: string, - type: MWMediaType + type: TMDBContentTypes ) => { let data; @@ -34,7 +145,7 @@ export abstract class Tmdb { case "movie": data = await Tmdb.get(`/movie/${id}`); break; - case "series": + case "show": data = await Tmdb.get(`/tv/${id}`); break; default: @@ -47,4 +158,48 @@ export abstract class Tmdb { public static getMediaPoster(posterPath: string | null): string | undefined { if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; } + + public static async getEpisodes( + id: string, + season: number + ): Promise { + const data = await Tmdb.get(`/tv/${id}/season/${season}`); + return data.episodes.map((e) => ({ + id: e.id, + episode_number: e.episode_number, + title: e.name, + })); + } +} + +export async function formatTMDBSearchResult( + result: TMDBShowResult | TMDBMovieResult, + mediatype: TMDBContentTypes +): Promise { + const type = TMDBMediaToMediaType(mediatype); + const details = await Tmdb.getMediaDetails(result.id.toString(), mediatype); + + const seasons = + type === MWMediaType.SERIES + ? (details as TMDBShowData).seasons?.map((v) => ({ + id: v.id, + title: v.name, + season_number: v.season_number, + })) + : undefined; + + return { + title: + type === MWMediaType.SERIES + ? (result as TMDBShowResult).name + : (result as TMDBMovieResult).title, + poster: Tmdb.getMediaPoster(details.poster_path), + id: result.id, + original_release_year: + type === MWMediaType.SERIES + ? Number((result as TMDBShowResult).first_air_date?.split("-")[0]) + : Number((result as TMDBMovieResult).release_date?.split("-")[0]), + object_type: mediaTypeToTMDB(type), + seasons, + }; } diff --git a/src/backend/metadata/trakttv.ts b/src/backend/metadata/trakttv.ts deleted file mode 100644 index ae50aefe..00000000 --- a/src/backend/metadata/trakttv.ts +++ /dev/null @@ -1,187 +0,0 @@ -import { conf } from "@/setup/config"; - -import { Tmdb } from "./tmdb"; -import { - MWMediaMeta, - MWMediaType, - MWSeasonMeta, - TMDBShowData, - TTVContentTypes, - TTVEpisodeResult, - TTVEpisodeShort, - TTVMediaResult, - TTVSearchResult, - TTVSeasonMetaResult, -} from "./types"; -import { mwFetch } from "../helpers/fetch"; - -export function mediaTypeToTTV(type: MWMediaType): TTVContentTypes { - if (type === MWMediaType.MOVIE) return "movie"; - if (type === MWMediaType.SERIES) return "show"; - throw new Error("unsupported type"); -} - -export function TTVMediaToMediaType(type: string): MWMediaType { - if (type === "movie") return MWMediaType.MOVIE; - if (type === "show") return MWMediaType.SERIES; - throw new Error("unsupported type"); -} - -export function formatTTVMeta( - media: TTVMediaResult, - season?: TTVSeasonMetaResult -): MWMediaMeta { - const type = TTVMediaToMediaType(media.object_type); - let seasons: undefined | MWSeasonMeta[]; - if (type === MWMediaType.SERIES) { - seasons = media.seasons - ?.sort((a, b) => a.season_number - b.season_number) - .map( - (v): MWSeasonMeta => ({ - title: v.title, - id: v.id.toString(), - number: v.season_number, - }) - ); - } - - return { - title: media.title, - id: media.id.toString(), - year: media.original_release_year?.toString(), - poster: media.poster, - type, - seasons: seasons as any, - seasonData: season - ? ({ - id: season.id.toString(), - number: season.season_number, - title: season.title, - episodes: season.episodes - .sort((a, b) => a.episode_number - b.episode_number) - .map((v) => ({ - id: v.id.toString(), - number: v.episode_number, - title: v.title, - })), - } as any) - : (undefined as any), - }; -} - -export function TTVMediaToId(media: MWMediaMeta): string { - return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); -} - -export function decodeTTVId( - paramId: string -): { id: string; type: MWMediaType } | null { - const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "MW") return null; - let mediaType; - try { - mediaType = TTVMediaToMediaType(type); - } catch { - return null; - } - return { - type: mediaType, - id, - }; -} - -export async function formatTTVSearchResult( - result: TTVSearchResult -): Promise { - const type = TTVMediaToMediaType(result.type); - const media = result[result.type]; - - if (!media) throw new Error("invalid result"); - - const details = await Tmdb.getMediaDetails( - media.ids.tmdb.toString(), - TTVMediaToMediaType(result.type) - ); - - const seasons = - type === MWMediaType.SERIES - ? (details as TMDBShowData).seasons?.map((v) => ({ - id: v.id, - title: v.name, - season_number: v.season_number, - })) - : undefined; - - return { - title: media.title, - poster: Tmdb.getMediaPoster(details.poster_path), - id: media.ids.tmdb, - original_release_year: media.year, - ttv_entity_id: media.ids.slug, - object_type: mediaTypeToTTV(type), - seasons, - }; -} - -export abstract class Trakt { - private static baseURL = "https://api.trakt.tv"; - - private static headers = { - "Content-Type": "application/json", - "trakt-api-version": "2", - "trakt-api-key": conf().TRAKT_CLIENT_ID, - }; - - private static async get(url: string): Promise { - const res = await mwFetch(url, { - headers: Trakt.headers, - baseURL: Trakt.baseURL, - }); - return res; - } - - public static async search( - query: string, - type: "movie" | "show" - ): Promise { - const data = await Trakt.get( - `/search/${type}?query=${encodeURIComponent(query)}` - ); - - const formatted = await Promise.all( - // eslint-disable-next-line no-return-await - data.map(async (v) => await formatTTVSearchResult(v)) - ); - return formatted.map((v) => formatTTVMeta(v)); - } - - public static async searchById( - tmdbId: string, - type: "movie" | "show" - ): Promise { - const data = await Trakt.get( - `/search/tmdb/${tmdbId}?type=${type}` - ); - - const formatted = await Promise.all( - // eslint-disable-next-line no-return-await - data.map(async (v) => await formatTTVSearchResult(v)) - ); - return formatted[0]; - } - - public static async getEpisodes( - slug: string, - season: number - ): Promise { - const data = await Trakt.get( - `/shows/${slug}/seasons/${season}` - ); - - return data.map((e) => ({ - id: e.ids.tmdb, - episode_number: e.number, - title: e.title, - })); - } -} diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index 07671e39..e23d9a5b 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -46,63 +46,36 @@ export interface MWQuery { type: MWMediaType; } -export type TTVContentTypes = "movie" | "show"; +export type TMDBContentTypes = "movie" | "show"; -export type TTVSeasonShort = { +export type TMDBSeasonShort = { title: string; id: number; season_number: number; }; -export type TTVEpisodeShort = { +export type TMDBEpisodeShort = { title: string; id: number; episode_number: number; }; -export type TTVMediaResult = { +export type TMDBMediaResult = { title: string; poster?: string; id: number; original_release_year?: number; - ttv_entity_id: string; - object_type: TTVContentTypes; - seasons?: TTVSeasonShort[]; + object_type: TMDBContentTypes; + seasons?: TMDBSeasonShort[]; }; -export type TTVSeasonMetaResult = { +export type TMDBSeasonMetaResult = { title: string; id: string; season_number: number; - episodes: TTVEpisodeShort[]; + episodes: TMDBEpisodeShort[]; }; -export interface TTVSearchResult { - type: "movie" | "show"; - score: number; - movie?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - imdb: string; - tmdb: number; - }; - }; - show?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - tvdb: number; - imdb: string; - tmdb: number; - }; - }; -} - export interface DetailedMeta { meta: MWMediaMeta; imdbId?: string; @@ -255,12 +228,9 @@ export interface TMDBMovieData { export type TMDBMediaDetailsPromise = Promise; export interface TMDBMediaStatic { - getMediaDetails( - id: string, - type: MWMediaType.SERIES - ): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: "show"): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: "movie"): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: TMDBContentTypes): TMDBMediaDetailsPromise; } export type JWContentTypes = "movie" | "show"; @@ -312,7 +282,7 @@ export type JWSeasonMetaResult = { episodes: JWEpisodeShort[]; }; -export interface TTVEpisodeResult { +export interface TMDBEpisodeResult { season: number; number: number; title: string; @@ -323,3 +293,89 @@ export interface TTVEpisodeResult { tmdb: number; }; } + +export interface TMDBShowResult { + adult: boolean; + backdrop_path: string | null; + genre_ids: number[]; + id: number; + origin_country: string[]; + original_language: string; + original_name: string; + overview: string; + popularity: number; + poster_path: string | null; + first_air_date: string; + name: string; + vote_average: number; + vote_count: number; +} + +export interface TMDBShowResponse { + page: number; + results: TMDBShowResult[]; + total_pages: number; + total_results: number; +} + +export interface TMDBMovieResult { + adult: boolean; + backdrop_path: string | null; + genre_ids: number[]; + id: number; + original_language: string; + original_title: string; + overview: string; + popularity: number; + poster_path: string | null; + release_date: string; + title: string; + video: boolean; + vote_average: number; + vote_count: number; +} + +export interface TMDBMovieResponse { + page: number; + results: TMDBMovieResult[]; + total_pages: number; + total_results: number; +} + +export type TMDBSearchResultsPromise = Promise< + TMDBShowResponse | TMDBMovieResponse +>; + +export interface TMDBSearchResultStatic { + searchMedia(query: string, type: TMDBContentTypes): TMDBSearchResultsPromise; + searchMedia(query: string, type: "movie"): TMDBSearchResultsPromise; + searchMedia(query: string, type: "show"): TMDBSearchResultsPromise; +} + +export interface TMDBEpisode { + air_date: string; + episode_number: number; + id: number; + name: string; + overview: string; + production_code: string; + runtime: number; + season_number: number; + show_id: number; + still_path: string | null; + vote_average: number; + vote_count: number; + crew: any[]; + guest_stars: any[]; +} + +export interface TMDBSeason { + _id: string; + air_date: string; + episodes: TMDBEpisode[]; + name: string; + overview: string; + id: number; + poster_path: string | null; + season_number: number; +} diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index b87654a9..695027a2 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -1,7 +1,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { TTVMediaToId } from "@/backend/metadata/getmeta"; +import { TMDBMediaToId } from "@/backend/metadata/getmeta"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -132,7 +132,7 @@ export function MediaCard(props: MediaCardProps) { const canLink = props.linkable && !props.closable; let link = canLink - ? `/media/${encodeURIComponent(TTVMediaToId(props.media))}` + ? `/media/${encodeURIComponent(TMDBMediaToId(props.media))}` : "#"; if (canLink && props.series) link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index 6cf5d2d6..bc050bf2 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -2,7 +2,7 @@ import { useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; -import { decodeTTVId, getMetaFromId } from "@/backend/metadata/getmeta"; +import { decodeTMDBId, getMetaFromId } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; @@ -44,7 +44,7 @@ export function EpisodeSelectionPopout() { seasonId: sId, season: undefined, }); - reqSeasonMeta(decodeTTVId(params.media)?.id as string, sId).then((v) => { + reqSeasonMeta(decodeTMDBId(params.media)?.id as string, sId).then((v) => { if (v?.meta.type !== MWMediaType.SERIES) return; setCurrentVisibleSeason({ seasonId: sId, diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index c0e81455..7ae1c01b 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -6,7 +6,7 @@ import { useHistory, useParams } from "react-router-dom"; import { MWStream } from "@/backend/helpers/streams"; import { DetailedMeta, - decodeTTVId, + decodeTMDBId, getMetaFromId, } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; @@ -184,7 +184,7 @@ export function MediaView() { const [selected, setSelected] = useState(null); const [exec, loading, error] = useLoading( async (mediaParams: string, seasonId?: string) => { - const data = decodeTTVId(mediaParams); + const data = decodeTMDBId(mediaParams); if (!data) return null; return getMetaFromId(data.type, data.id, seasonId); } From c4afc372178669c0641049fb29c5da3e96f8feb8 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 21:26:58 +0200 Subject: [PATCH 29/96] cleanup --- src/setup/config.ts | 4 ---- 1 file changed, 4 deletions(-) diff --git a/src/setup/config.ts b/src/setup/config.ts index c24117bb..f1db01da 100644 --- a/src/setup/config.ts +++ b/src/setup/config.ts @@ -8,7 +8,6 @@ interface Config { TMDB_API_KEY: string; CORS_PROXY_URL: string; NORMAL_ROUTER: boolean; - TRAKT_CLIENT_ID: string; } export interface RuntimeConfig { @@ -19,7 +18,6 @@ export interface RuntimeConfig { TMDB_API_KEY: string; NORMAL_ROUTER: boolean; PROXY_URLS: string[]; - TRAKT_CLIENT_ID: string; } const env: Record = { @@ -30,7 +28,6 @@ const env: Record = { DISCORD_LINK: undefined, CORS_PROXY_URL: import.meta.env.VITE_CORS_PROXY_URL, NORMAL_ROUTER: import.meta.env.VITE_NORMAL_ROUTER, - TRAKT_CLIENT_ID: import.meta.env.VITE_TRAKT_CLIENT_ID, }; const alerts = [] as string[]; @@ -65,6 +62,5 @@ export function conf(): RuntimeConfig { .split(",") .map((v) => v.trim()), NORMAL_ROUTER: getKey("NORMAL_ROUTER", "false") === "true", - TRAKT_CLIENT_ID: getKey("TRAKT_CLIENT_ID"), }; } From 20c4b14799d6724ae0b34d694dbc6cfa53df950e Mon Sep 17 00:00:00 2001 From: castdrian Date: Wed, 14 Jun 2023 07:48:31 +0200 Subject: [PATCH 30/96] fix movie metadata --- src/backend/metadata/getmeta.ts | 56 ++++++++++++++++++++++----------- src/backend/metadata/search.ts | 2 ++ 2 files changed, 39 insertions(+), 19 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 777fae42..41e67772 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -43,6 +43,41 @@ export interface DetailedMeta { tmdbId?: string; } +export function fromatTMDBMetaResult( + details: TMDBShowData | TMDBMovieData, + type: MWMediaType +): TMDBMediaResult | undefined { + let tmdbmeta; + if (type === MWMediaType.MOVIE) { + tmdbmeta = { + id: details.id, + title: (details as TMDBMovieData).title, + object_type: mediaTypeToTMDB(type), + poster: (details as TMDBMovieData).poster_path ?? undefined, + original_release_year: Number( + (details as TMDBMovieData).release_date?.split("-")[0] + ), + }; + } + if (type === MWMediaType.SERIES) { + tmdbmeta = { + id: details.id, + title: (details as TMDBShowData).name, + object_type: mediaTypeToTMDB(type), + seasons: (details as TMDBShowData).seasons.map((v) => ({ + id: v.id, + season_number: v.season_number, + title: v.name, + })), + poster: (details as TMDBMovieData).poster_path ?? undefined, + original_release_year: Number( + (details as TMDBShowData).first_air_date?.split("-")[0] + ), + }; + } + return tmdbmeta; +} + export async function getMetaFromId( type: MWMediaType, id: string, @@ -79,25 +114,8 @@ export async function getMetaFromId( } } - const tmdbmeta: TMDBMediaResult = { - id: details.id, - title: - type === MWMediaType.MOVIE - ? (details as TMDBMovieData).title - : (details as TMDBShowData).name, - object_type: mediaTypeToTMDB(type), - seasons: (details as TMDBShowData).seasons.map((v) => ({ - id: v.id, - season_number: v.season_number, - title: v.name, - })), - poster: (details as TMDBMovieData).poster_path ?? undefined, - original_release_year: - type === MWMediaType.MOVIE - ? Number((details as TMDBMovieData).release_date?.split("-")[0]) - : Number((details as TMDBShowData).first_air_date?.split("-")[0]), - }; - + const tmdbmeta = fromatTMDBMetaResult(details, type); + if (!tmdbmeta) return null; const meta = formatTMDBMeta(tmdbmeta, seasonData); if (!meta) return null; diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 31b2c682..7d06ab2c 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -29,6 +29,8 @@ export async function searchForMedia(query: MWQuery): Promise { }) ); + console.log(results[0]); + cache.set(query, results, 3600); return results; } From 5d56b847c690a4a8c988026a9f33b6e2c4a072a3 Mon Sep 17 00:00:00 2001 From: castdrian Date: Wed, 14 Jun 2023 07:52:04 +0200 Subject: [PATCH 31/96] cleanup --- src/backend/metadata/search.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 7d06ab2c..31b2c682 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -29,8 +29,6 @@ export async function searchForMedia(query: MWQuery): Promise { }) ); - console.log(results[0]); - cache.set(query, results, 3600); return results; } From 424ee6fe7791b356ad43765d747dfd6420664f6e Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 08:55:40 +0700 Subject: [PATCH 32/96] Update i18n.ts --- src/setup/i18n.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index 6116434a..a225c91e 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -11,6 +11,7 @@ import nl from "./locales/nl/translation.json"; import pirate from "./locales/pirate/translation.json"; import tr from "./locales/tr/translation.json"; import zh from "./locales/zh/translation.json"; +import vi from "./locales/vi/translation.json"; const locales = { en: { From f9d756e0efe4321e53e30c9b8ba7df88570d8d1a Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 09:06:19 +0700 Subject: [PATCH 33/96] Update i18n.ts --- src/setup/i18n.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index a225c91e..21388b9f 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -10,8 +10,8 @@ import fr from "./locales/fr/translation.json"; import nl from "./locales/nl/translation.json"; import pirate from "./locales/pirate/translation.json"; import tr from "./locales/tr/translation.json"; -import zh from "./locales/zh/translation.json"; import vi from "./locales/vi/translation.json"; +import zh from "./locales/zh/translation.json"; const locales = { en: { From db75f2320d78c721d3754e1162ad49497d6a1a79 Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 10:46:05 +0700 Subject: [PATCH 34/96] Add files via upload add translation --- src/setup/locales/translation.json | 128 +++++++++++++++++++++++++++++ 1 file changed, 128 insertions(+) create mode 100644 src/setup/locales/translation.json diff --git a/src/setup/locales/translation.json b/src/setup/locales/translation.json new file mode 100644 index 00000000..dae9c21d --- /dev/null +++ b/src/setup/locales/translation.json @@ -0,0 +1,128 @@ +{ + "global": { + "name": "movie-web" + }, + "search": { + "loading_series": "Đang tìm chương trình yêu thích của bạn...", + "loading_movie": "Đang tìm bộ phim yêu thích của bạn...", + "loading": "Đang tải...", + "allResults": "Đó là tất cả chúng tối có!", + "noResults": "Chúng tôi không thể tìm thấy gì!", + "allFailed": "Không thể tìm thấy nội dung, hãy thử lại!", + "headingTitle": "Kết quả tìm kiếm", + "bookmarks": "Đánh dấu", + "continueWatching": "Tiếp tục xem", + "title": "Bạn muốn xem gì?", + "placeholder": "Bạn muốn xem gì?" + }, + "media": { + "movie": "Phim", + "series": "Chương trình truyền hình", + "stopEditing": "Hãy dừng chỉnh sửa", + "errors": { + "genericTitle": "Rất tiếc, nó đã hỏng!", + "failedMeta": "Không thể tải meta", + "mediaFailed": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu, hãy kiểm tra kết nối internet của bạn và thử lại.", + "videoFailed": "Chúng tôi đã gặp lỗi khi phát nội dung mà bạn yêu cầu. Nếu điều này tiếp tục xảy ra, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." + } + }, + "seasons": { + "seasonAndEpisode": "M{{season}} T{{episode}}" + }, + "notFound": { + "genericTitle": "Không tìm thấy", + "backArrow": "Quay lại trang chính", + "media": { + "title": "Không thể tìm thấy nội dung", + "description": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu. Hoặc là nó đã bị xóa, hoặc bạn đã xáo trộn URL" + }, + "provider": { + "title": "Nhà cung cấp này đã bị vô hiệu hóa", + "description": "Chúng tôi gặp vấn đề với nhà cung cấp hoặc nó quá bất ổn để sử dụng, cho nên chúng tôi đã phải vô hiệu hóa nó." + }, + "page": { + "title": "Không thể tìm thấy trang", + "description": "Chúng tôi đã tìm kiếm khắp nơi: dưới thùng rác, trong tủ quần áo, đằng sau máy chủ proxy nhưng vẫn không thể tìm thấy trang bạn đang tìm kiếm." + } + }, + "searchBar": { + "movie": "Phim", + "series": "Chương trình truyền hình", + "Search": "Tìm kiếm" + }, + "videoPlayer": { + "findingBestVideo": "Đang tìm nội dung tốt nhất cho bạn", + "noVideos": "Rất tiếc, không tìm thấy nội dung nào cho bạn", + "loading": "Đang tải...", + "backToHome": "Quay lại trang chính", + "backToHomeShort": "Quay lại", + "seasonAndEpisode": "M{{season}} T{{episode}}", + "timeLeft": "Còn {{timeLeft}}", + "finishAt": "Kết thúc vào {{timeFinished, datetime}}", + "buttons": { + "episodes": "Tập", + "source": "Source", + "captions": "Phụ đề", + "download": "Tải xuống", + "settings": "Cài đặt", + "pictureInPicture": "Hình trong hình", + "playbackSpeed": "Tốc độ phát" + }, + "popouts": { + "back": "Quay lại", + "sources": "Nguồn", + "seasons": "Mùa", + "captions": "Phụ đề", + "playbackSpeed": "Tốc độ phát", + "customPlaybackSpeed": "Tủy chỉnh tốc độ phát", + "captionPreferences": { + "title": "Tùy chỉnh", + "delay": "Trì hoãn", + "fontSize": "Kích cỡ", + "opacity": "Độ mờ", + "color": "Màu sắc" + }, + "episode": "T{{index}} - {{title}}", + "noCaptions": "Không phụ đề", + "linkedCaptions": "Phụ đề được liên kết", + "customCaption": "Phụ đề tùy chỉnh", + "uploadCustomCaption": "Tải phụ đề lên", + "noEmbeds": "Không tìm thấy nội dung nhúng nào cho nguồn này", + + "errors": { + "loadingWentWong": "Đã xảy ra lỗi khi tải các tập phim cho {{seasonTitle}}", + "embedsError": "Đã xảy ra lỗi khi tải nội dung nhúng cho nội dung bạn thích này" + }, + "descriptions": { + "sources": "Bạn muốn sử dụng nhà cung cấp nào?", + "embeds": "Chọn video để xem", + "seasons": "Chọn mùa bạn muốn xem", + "episode": "Chọn một tập", + "captions": "Chọn ngôn ngữ của phụ đề", + "captionPreferences": "Làm cho phụ đề trông như thế nào bạn muốn", + "playbackSpeed": "Thay đổi tốc độ phát" + } + }, + "errors": { + "fatalError": "Trình phát video đã gặp phải lỗi nghiêm trọng, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." + } + }, + "settings": { + "title": "Cài đặt", + "language": "Ngôn ngữ", + "captionLanguage": "Ngôn ngữ phụ đề" + }, + "v3": { + "newSiteTitle": "Phiên bản mới đã được phát hành!", + "newDomain": "https://movie-web.app", + "newDomainText": "movie-web sẽ sớm chuyển sang trang mới: <0>https://movie-web.app. Hãy đảm bảo rằng các đánh dấu đã được cập nhật vì <1>trang web cũ sẽ dừng hoạt động vào {{date}}.", + "tireless": "Chúng tôi đã làm việc vất vả để tạo phiên bản mới này, chúng tôi hy vọng bạn sẽ thích những gì chúng tôi đã nung nấu trong những tháng qua.", + "leaveAnnouncement": "Hãy đưa tôi đến đó!" + }, + "casting": { + "casting": "Đang truyền tới thiết bị..." + }, + "errors": { + "offline": "Hãy kiểm tra kết nối Internet của bạn" + } +} From 80dd2158dffbc9d33e154971bd44d828b5a61b38 Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 10:48:26 +0700 Subject: [PATCH 35/96] Create translation.json --- src/setup/locales/vi/translation.json | 128 ++++++++++++++++++++++++++ 1 file changed, 128 insertions(+) create mode 100644 src/setup/locales/vi/translation.json diff --git a/src/setup/locales/vi/translation.json b/src/setup/locales/vi/translation.json new file mode 100644 index 00000000..45494388 --- /dev/null +++ b/src/setup/locales/vi/translation.json @@ -0,0 +1,128 @@ +{ + "global": { + "name": "movie-web" + }, + "search": { + "loading_series": "Đang tìm chương trình yêu thích của bạn...", + "loading_movie": "Đang tìm bộ phim yêu thích của bạn...", + "loading": "Đang tải...", + "allResults": "Đó là tất cả chúng tôi có!", + "noResults": "Chúng tôi không thể tìm thấy gì!", + "allFailed": "Không thể tìm thấy nội dung, hãy thử lại!", + "headingTitle": "Kết quả tìm kiếm", + "bookmarks": "Đánh dấu", + "continueWatching": "Tiếp tục xem", + "title": "Bạn muốn xem gì?", + "placeholder": "Bạn muốn xem gì?" + }, + "media": { + "movie": "Phim", + "series": "Chương trình truyền hình", + "stopEditing": "Hãy dừng chỉnh sửa", + "errors": { + "genericTitle": "Rất tiếc, nó đã hỏng!", + "failedMeta": "Không thể tải meta", + "mediaFailed": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu, hãy kiểm tra kết nối internet của bạn và thử lại.", + "videoFailed": "Chúng tôi đã gặp lỗi khi phát nội dung mà bạn yêu cầu. Nếu điều này tiếp tục xảy ra, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." + } + }, + "seasons": { + "seasonAndEpisode": "M{{season}} T{{episode}}" + }, + "notFound": { + "genericTitle": "Không tìm thấy", + "backArrow": "Quay lại trang chính", + "media": { + "title": "Không thể tìm thấy nội dung", + "description": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu. Hoặc là nó đã bị xóa, hoặc bạn đã xáo trộn URL" + }, + "provider": { + "title": "Nhà cung cấp này đã bị vô hiệu hóa", + "description": "Chúng tôi đã gặp vấn đề với nhà cung cấp hoặc nó quá bất ổn để sử dụng, cho nên chúng tôi đã phải vô hiệu hóa nó." + }, + "page": { + "title": "Không thể tìm thấy trang", + "description": "Chúng tôi đã tìm kiếm khắp nơi: dưới thùng rác, trong tủ quần áo, đằng sau máy chủ proxy nhưng vẫn không thể tìm thấy trang bạn đang tìm kiếm." + } + }, + "searchBar": { + "movie": "Phim", + "series": "Chương trình truyền hình", + "Search": "Tìm kiếm" + }, + "videoPlayer": { + "findingBestVideo": "Đang tìm nội dung tốt nhất cho bạn", + "noVideos": "Rất tiếc, không tìm thấy nội dung nào cho bạn", + "loading": "Đang tải...", + "backToHome": "Quay lại trang chính", + "backToHomeShort": "Quay lại", + "seasonAndEpisode": "M{{season}} T{{episode}}", + "timeLeft": "Còn {{timeLeft}}", + "finishAt": "Kết thúc vào {{timeFinished, datetime}}", + "buttons": { + "episodes": "Tập", + "source": "Source", + "captions": "Phụ đề", + "download": "Tải xuống", + "settings": "Cài đặt", + "pictureInPicture": "Hình trong hình", + "playbackSpeed": "Tốc độ phát" + }, + "popouts": { + "back": "Quay lại", + "sources": "Nguồn", + "seasons": "Mùa", + "captions": "Phụ đề", + "playbackSpeed": "Tốc độ phát", + "customPlaybackSpeed": "Tủy chỉnh tốc độ phát", + "captionPreferences": { + "title": "Tùy chỉnh", + "delay": "Trì hoãn", + "fontSize": "Kích cỡ", + "opacity": "Độ mờ", + "color": "Màu sắc" + }, + "episode": "T{{index}} - {{title}}", + "noCaptions": "Không phụ đề", + "linkedCaptions": "Phụ đề được liên kết", + "customCaption": "Phụ đề tùy chỉnh", + "uploadCustomCaption": "Tải phụ đề lên", + "noEmbeds": "Không tìm thấy nội dung nhúng nào cho nguồn này", + + "errors": { + "loadingWentWong": "Đã xảy ra lỗi khi tải các tập phim cho {{seasonTitle}}", + "embedsError": "Đã xảy ra lỗi khi tải nội dung nhúng cho nội dung bạn thích này" + }, + "descriptions": { + "sources": "Bạn muốn sử dụng nhà cung cấp nào?", + "embeds": "Chọn video để xem", + "seasons": "Chọn mùa bạn muốn xem", + "episode": "Chọn một tập", + "captions": "Chọn ngôn ngữ của phụ đề", + "captionPreferences": "Làm cho phụ đề trông như thế nào bạn muốn", + "playbackSpeed": "Thay đổi tốc độ phát" + } + }, + "errors": { + "fatalError": "Trình phát video đã gặp phải lỗi nghiêm trọng, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." + } + }, + "settings": { + "title": "Cài đặt", + "language": "Ngôn ngữ", + "captionLanguage": "Ngôn ngữ phụ đề" + }, + "v3": { + "newSiteTitle": "Phiên bản mới đã được phát hành!", + "newDomain": "https://movie-web.app", + "newDomainText": "movie-web sẽ sớm chuyển sang trang mới: <0>https://movie-web.app. Hãy đảm bảo rằng các đánh dấu đã được cập nhật vì <1>trang web cũ sẽ dừng hoạt động vào {{date}}.", + "tireless": "Chúng tôi đã làm việc vất vả để tạo phiên bản mới này, chúng tôi hy vọng bạn sẽ thích những gì chúng tôi đã nung nấu trong những tháng qua.", + "leaveAnnouncement": "Hãy đưa tôi đến đó!" + }, + "casting": { + "casting": "Đang truyền tới thiết bị..." + }, + "errors": { + "offline": "Hãy kiểm tra kết nối Internet của bạn" + } +} From 61c3ed076f778549a34f3ec92dfbe4dde65951dc Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 10:48:45 +0700 Subject: [PATCH 36/96] Delete translation.json --- src/setup/locales/translation.json | 128 ----------------------------- 1 file changed, 128 deletions(-) delete mode 100644 src/setup/locales/translation.json diff --git a/src/setup/locales/translation.json b/src/setup/locales/translation.json deleted file mode 100644 index dae9c21d..00000000 --- a/src/setup/locales/translation.json +++ /dev/null @@ -1,128 +0,0 @@ -{ - "global": { - "name": "movie-web" - }, - "search": { - "loading_series": "Đang tìm chương trình yêu thích của bạn...", - "loading_movie": "Đang tìm bộ phim yêu thích của bạn...", - "loading": "Đang tải...", - "allResults": "Đó là tất cả chúng tối có!", - "noResults": "Chúng tôi không thể tìm thấy gì!", - "allFailed": "Không thể tìm thấy nội dung, hãy thử lại!", - "headingTitle": "Kết quả tìm kiếm", - "bookmarks": "Đánh dấu", - "continueWatching": "Tiếp tục xem", - "title": "Bạn muốn xem gì?", - "placeholder": "Bạn muốn xem gì?" - }, - "media": { - "movie": "Phim", - "series": "Chương trình truyền hình", - "stopEditing": "Hãy dừng chỉnh sửa", - "errors": { - "genericTitle": "Rất tiếc, nó đã hỏng!", - "failedMeta": "Không thể tải meta", - "mediaFailed": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu, hãy kiểm tra kết nối internet của bạn và thử lại.", - "videoFailed": "Chúng tôi đã gặp lỗi khi phát nội dung mà bạn yêu cầu. Nếu điều này tiếp tục xảy ra, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." - } - }, - "seasons": { - "seasonAndEpisode": "M{{season}} T{{episode}}" - }, - "notFound": { - "genericTitle": "Không tìm thấy", - "backArrow": "Quay lại trang chính", - "media": { - "title": "Không thể tìm thấy nội dung", - "description": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu. Hoặc là nó đã bị xóa, hoặc bạn đã xáo trộn URL" - }, - "provider": { - "title": "Nhà cung cấp này đã bị vô hiệu hóa", - "description": "Chúng tôi gặp vấn đề với nhà cung cấp hoặc nó quá bất ổn để sử dụng, cho nên chúng tôi đã phải vô hiệu hóa nó." - }, - "page": { - "title": "Không thể tìm thấy trang", - "description": "Chúng tôi đã tìm kiếm khắp nơi: dưới thùng rác, trong tủ quần áo, đằng sau máy chủ proxy nhưng vẫn không thể tìm thấy trang bạn đang tìm kiếm." - } - }, - "searchBar": { - "movie": "Phim", - "series": "Chương trình truyền hình", - "Search": "Tìm kiếm" - }, - "videoPlayer": { - "findingBestVideo": "Đang tìm nội dung tốt nhất cho bạn", - "noVideos": "Rất tiếc, không tìm thấy nội dung nào cho bạn", - "loading": "Đang tải...", - "backToHome": "Quay lại trang chính", - "backToHomeShort": "Quay lại", - "seasonAndEpisode": "M{{season}} T{{episode}}", - "timeLeft": "Còn {{timeLeft}}", - "finishAt": "Kết thúc vào {{timeFinished, datetime}}", - "buttons": { - "episodes": "Tập", - "source": "Source", - "captions": "Phụ đề", - "download": "Tải xuống", - "settings": "Cài đặt", - "pictureInPicture": "Hình trong hình", - "playbackSpeed": "Tốc độ phát" - }, - "popouts": { - "back": "Quay lại", - "sources": "Nguồn", - "seasons": "Mùa", - "captions": "Phụ đề", - "playbackSpeed": "Tốc độ phát", - "customPlaybackSpeed": "Tủy chỉnh tốc độ phát", - "captionPreferences": { - "title": "Tùy chỉnh", - "delay": "Trì hoãn", - "fontSize": "Kích cỡ", - "opacity": "Độ mờ", - "color": "Màu sắc" - }, - "episode": "T{{index}} - {{title}}", - "noCaptions": "Không phụ đề", - "linkedCaptions": "Phụ đề được liên kết", - "customCaption": "Phụ đề tùy chỉnh", - "uploadCustomCaption": "Tải phụ đề lên", - "noEmbeds": "Không tìm thấy nội dung nhúng nào cho nguồn này", - - "errors": { - "loadingWentWong": "Đã xảy ra lỗi khi tải các tập phim cho {{seasonTitle}}", - "embedsError": "Đã xảy ra lỗi khi tải nội dung nhúng cho nội dung bạn thích này" - }, - "descriptions": { - "sources": "Bạn muốn sử dụng nhà cung cấp nào?", - "embeds": "Chọn video để xem", - "seasons": "Chọn mùa bạn muốn xem", - "episode": "Chọn một tập", - "captions": "Chọn ngôn ngữ của phụ đề", - "captionPreferences": "Làm cho phụ đề trông như thế nào bạn muốn", - "playbackSpeed": "Thay đổi tốc độ phát" - } - }, - "errors": { - "fatalError": "Trình phát video đã gặp phải lỗi nghiêm trọng, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." - } - }, - "settings": { - "title": "Cài đặt", - "language": "Ngôn ngữ", - "captionLanguage": "Ngôn ngữ phụ đề" - }, - "v3": { - "newSiteTitle": "Phiên bản mới đã được phát hành!", - "newDomain": "https://movie-web.app", - "newDomainText": "movie-web sẽ sớm chuyển sang trang mới: <0>https://movie-web.app. Hãy đảm bảo rằng các đánh dấu đã được cập nhật vì <1>trang web cũ sẽ dừng hoạt động vào {{date}}.", - "tireless": "Chúng tôi đã làm việc vất vả để tạo phiên bản mới này, chúng tôi hy vọng bạn sẽ thích những gì chúng tôi đã nung nấu trong những tháng qua.", - "leaveAnnouncement": "Hãy đưa tôi đến đó!" - }, - "casting": { - "casting": "Đang truyền tới thiết bị..." - }, - "errors": { - "offline": "Hãy kiểm tra kết nối Internet của bạn" - } -} From be90b020431d017c4fef23111ed7809a8dceda33 Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 10:53:08 +0700 Subject: [PATCH 37/96] Update translation.json --- src/setup/locales/vi/translation.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/setup/locales/vi/translation.json b/src/setup/locales/vi/translation.json index 45494388..d56b3595 100644 --- a/src/setup/locales/vi/translation.json +++ b/src/setup/locales/vi/translation.json @@ -20,7 +20,7 @@ "series": "Chương trình truyền hình", "stopEditing": "Hãy dừng chỉnh sửa", "errors": { - "genericTitle": "Rất tiếc, nó đã hỏng!", + "genericTitle": "Rất tiếc, đã hỏng!", "failedMeta": "Không thể tải meta", "mediaFailed": "Chúng tôi không thể tìm thấy nội dung mà bạn yêu cầu, hãy kiểm tra kết nối internet của bạn và thử lại.", "videoFailed": "Chúng tôi đã gặp lỗi khi phát nội dung mà bạn yêu cầu. Nếu điều này tiếp tục xảy ra, vui lòng báo cáo sự cố trên <0>máy chủ Discord hoặc trên <1>GitHub." From 07deb1897d1113460101e6464b61e37d5997ac13 Mon Sep 17 00:00:00 2001 From: spinixster <135334442+spinixster@users.noreply.github.com> Date: Thu, 15 Jun 2023 10:55:02 +0700 Subject: [PATCH 38/96] Update i18n.ts --- src/setup/i18n.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index 21388b9f..d0ec7035 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -38,6 +38,9 @@ const locales = { pirate: { translation: pirate, }, + vi: { + translation: vi, + }, }; i18n // pass the i18n instance to react-i18next. From 74cc50cfa2d64013e485b48862d3f3f3a673fd9f Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 08:30:05 +0200 Subject: [PATCH 39/96] show poster in bookmarks --- src/backend/metadata/getmeta.ts | 8 ++++++-- src/components/media/MediaCard.tsx | 5 ++++- 2 files changed, 10 insertions(+), 3 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 41e67772..bba3948e 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -53,7 +53,9 @@ export function fromatTMDBMetaResult( id: details.id, title: (details as TMDBMovieData).title, object_type: mediaTypeToTMDB(type), - poster: (details as TMDBMovieData).poster_path ?? undefined, + poster: + Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? + undefined, original_release_year: Number( (details as TMDBMovieData).release_date?.split("-")[0] ), @@ -69,7 +71,9 @@ export function fromatTMDBMetaResult( season_number: v.season_number, title: v.name, })), - poster: (details as TMDBMovieData).poster_path ?? undefined, + poster: + Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? + undefined, original_release_year: Number( (details as TMDBShowData).first_air_date?.split("-")[0] ), diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index 695027a2..ece6d293 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -2,6 +2,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; import { TMDBMediaToId } from "@/backend/metadata/getmeta"; +import { Tmdb } from "@/backend/metadata/tmdb"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -55,7 +56,9 @@ function MediaCardContent({ closable ? "" : "group-hover:rounded-lg", ].join(" ")} style={{ - backgroundImage: media.poster ? `url(${media.poster})` : undefined, + backgroundImage: media.poster + ? `url(${Tmdb.getMediaPoster(media.poster)})` + : undefined, }} > {series ? ( From 28d2dd0e89393fb29c6b363dd6445fb66c546959 Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 08:30:57 +0200 Subject: [PATCH 40/96] set adult false in query --- src/backend/metadata/tmdb.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index f01709bb..aed31c1d 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -120,12 +120,12 @@ export abstract class Tmdb { switch (type) { case "movie": data = await Tmdb.get( - `search/movie?query=${query}&include_adult=true&language=en-US&page=1` + `search/movie?query=${query}&include_adult=false&language=en-US&page=1` ); break; case "show": data = await Tmdb.get( - `search/tv?query=${query}&include_adult=true&language=en-US&page=1` + `search/tv?query=${query}&include_adult=false&language=en-US&page=1` ); break; default: From 330cbf2d9e5dfbe00797b118c621252be14aa42d Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 11:06:24 +0200 Subject: [PATCH 41/96] undo duplicate path --- src/backend/metadata/getmeta.ts | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index bba3948e..41e67772 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -53,9 +53,7 @@ export function fromatTMDBMetaResult( id: details.id, title: (details as TMDBMovieData).title, object_type: mediaTypeToTMDB(type), - poster: - Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? - undefined, + poster: (details as TMDBMovieData).poster_path ?? undefined, original_release_year: Number( (details as TMDBMovieData).release_date?.split("-")[0] ), @@ -71,9 +69,7 @@ export function fromatTMDBMetaResult( season_number: v.season_number, title: v.name, })), - poster: - Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? - undefined, + poster: (details as TMDBMovieData).poster_path ?? undefined, original_release_year: Number( (details as TMDBShowData).first_air_date?.split("-")[0] ), From d96165518670d2c3ce8e1ef1a753ecc75956ba98 Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 22:13:19 +0200 Subject: [PATCH 42/96] fix typo 'cause I can't type --- src/backend/metadata/getmeta.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 41e67772..a5246fcf 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -43,7 +43,7 @@ export interface DetailedMeta { tmdbId?: string; } -export function fromatTMDBMetaResult( +export function formatTMDBMetaResult( details: TMDBShowData | TMDBMovieData, type: MWMediaType ): TMDBMediaResult | undefined { @@ -114,7 +114,7 @@ export async function getMetaFromId( } } - const tmdbmeta = fromatTMDBMetaResult(details, type); + const tmdbmeta = formatTMDBMetaResult(details, type); if (!tmdbmeta) return null; const meta = formatTMDBMeta(tmdbmeta, seasonData); if (!meta) return null; From 4bd00eb47a23464611820c6a2cdb87e5bf06ce75 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:37:07 +0530 Subject: [PATCH 43/96] feat(embed): add upcloud and streamsb embed scrapers --- src/backend/embeds/streamsb.ts | 212 +++++++++++++++++++++++++++++++++ src/backend/embeds/upcloud.ts | 93 +++++++++++++++ 2 files changed, 305 insertions(+) create mode 100644 src/backend/embeds/streamsb.ts create mode 100644 src/backend/embeds/upcloud.ts diff --git a/src/backend/embeds/streamsb.ts b/src/backend/embeds/streamsb.ts new file mode 100644 index 00000000..c755a0b0 --- /dev/null +++ b/src/backend/embeds/streamsb.ts @@ -0,0 +1,212 @@ +import Base64 from "crypto-js/enc-base64"; +import Utf8 from "crypto-js/enc-utf8"; + +import { MWEmbedType } from "@/backend/helpers/embed"; +import { proxiedFetch } from "@/backend/helpers/fetch"; +import { registerEmbedScraper } from "@/backend/helpers/register"; +import { + MWCaptionType, + MWStreamQuality, + MWStreamType, +} from "@/backend/helpers/streams"; + +const qualityOrder = [ + MWStreamQuality.Q1080P, + MWStreamQuality.Q720P, + MWStreamQuality.Q480P, + MWStreamQuality.Q360P, +]; + +async function fetchCaptchaToken(domain: string, recaptchaKey: string) { + const domainHash = Base64.stringify(Utf8.parse(domain)).replace(/=/g, "."); + + const recaptchaRender = await proxiedFetch( + `https://www.google.com/recaptcha/api.js?render=${recaptchaKey}` + ); + + const vToken = recaptchaRender.substring( + recaptchaRender.indexOf("/releases/") + 10, + recaptchaRender.indexOf("/recaptcha__en.js") + ); + + const recaptchaAnchor = await proxiedFetch( + `https://www.google.com/recaptcha/api2/anchor?ar=1&hl=en&size=invisible&cb=flicklax&k=${recaptchaKey}&co=${domainHash}&v=${vToken}` + ); + + const cToken = new DOMParser() + .parseFromString(recaptchaAnchor, "text/html") + .getElementById("recaptcha-token") + ?.getAttribute("value"); + + if (!cToken) throw new Error("Unable to find cToken"); + + const payload = { + v: vToken, + reason: "q", + k: recaptchaKey, + c: cToken, + sa: "", + co: domain, + }; + + const tokenData = await proxiedFetch( + `https://www.google.com/recaptcha/api2/reload?${new URLSearchParams( + payload + ).toString()}`, + { + headers: { referer: "https://www.google.com/recaptcha/api2/" }, + method: "POST", + } + ); + + const token = tokenData.match('rresp","(.+?)"'); + return token ? token[1] : null; +} + +registerEmbedScraper({ + id: "streamsb", + displayName: "StreamSB", + for: MWEmbedType.STREAMSB, + rank: 150, + async getStream({ url, progress }) { + /* Url variations + - domain.com/{id}?.html + - domain.com/{id} + - domain.com/embed-{id} + - domain.com/d/{id} + - domain.com/e/{id} + - domain.com/e/{id}-embed + */ + const streamsbUrl = url + .replace(".html", "") + .replace("embed-", "") + .replace("e/", "") + .replace("d/", ""); + + const parsedUrl = new URL(streamsbUrl); + const base = await proxiedFetch( + `${parsedUrl.origin}/d${parsedUrl.pathname}` + ); + + progress(20); + + // Parse captions from url + const captionUrl = parsedUrl.searchParams.get("caption_1"); + const captionLang = parsedUrl.searchParams.get("sub_1"); + + const basePage = new DOMParser().parseFromString(base, "text/html"); + + const downloadVideoFunctions = basePage.querySelectorAll( + "[onclick^=download_video]" + ); + + const dlDetails = []; + for (const func of downloadVideoFunctions) { + const funcContents = func.getAttribute("onclick"); + const regExpFunc = /download_video\('(.+?)','(.+?)','(.+?)'\)/; + const matchesFunc = regExpFunc.exec(funcContents ?? ""); + if (matchesFunc !== null) { + const quality = func.querySelector("span")?.textContent; + const regExpQuality = /(.+?) \((.+?)\)/; + const matchesQuality = regExpQuality.exec(quality ?? ""); + if (matchesQuality !== null) { + dlDetails.push({ + parameters: [matchesFunc[1], matchesFunc[2], matchesFunc[3]], + quality: { + label: matchesQuality[1].trim(), + size: matchesQuality[2], + }, + }); + } + } + } + + progress(40); + + let dls = await Promise.all( + dlDetails.map(async (dl) => { + const getDownload = await proxiedFetch( + `/dl?op=download_orig&id=${dl.parameters[0]}&mode=${dl.parameters[1]}&hash=${dl.parameters[2]}`, + { + baseURL: parsedUrl.origin, + } + ); + + const downloadPage = new DOMParser().parseFromString( + getDownload, + "text/html" + ); + + const recaptchaKey = downloadPage + .querySelector(".g-recaptcha") + ?.getAttribute("data-sitekey"); + if (!recaptchaKey) throw new Error("Unable to get captcha key"); + + const captchaToken = await fetchCaptchaToken( + parsedUrl.origin, + recaptchaKey + ); + if (!captchaToken) throw new Error("Unable to get captcha token"); + + const dlForm = new FormData(); + dlForm.append("op", "download_orig"); + dlForm.append("id", dl.parameters[0]); + dlForm.append("mode", dl.parameters[1]); + dlForm.append("hash", dl.parameters[2]); + dlForm.append("g-recaptcha-response", captchaToken); + + const download = await proxiedFetch( + `/dl?op=download_orig&id=${dl.parameters[0]}&mode=${dl.parameters[1]}&hash=${dl.parameters[2]}`, + { + baseURL: parsedUrl.origin, + method: "POST", + body: dlForm, + } + ); + + const dlLink = new DOMParser() + .parseFromString(download, "text/html") + .querySelector(".btn.btn-light.btn-lg") + ?.getAttribute("href"); + + console.log(dlLink); + + return { + quality: dl.quality.label as MWStreamQuality, + url: dlLink, + size: dl.quality.size, + captions: + captionUrl && captionLang + ? [ + { + url: captionUrl, + langIso: captionLang, + type: MWCaptionType.VTT, + }, + ] + : [], + }; + }) + ); + dls = dls.filter((d) => !!d.url); + dls = dls.sort((a, b) => { + const aQuality = qualityOrder.indexOf(a.quality); + const bQuality = qualityOrder.indexOf(b.quality); + return aQuality - bQuality; + }); + + progress(60); + + // TODO: Quality selection for embed scrapers + const dl = dls[0]; + if (!dl.url) throw new Error("No stream url found"); + + return { + embedId: MWEmbedType.STREAMSB, + streamUrl: dl.url, + quality: dl.quality, + captions: dl.captions, + type: MWStreamType.MP4, + }; + }, +}); diff --git a/src/backend/embeds/upcloud.ts b/src/backend/embeds/upcloud.ts new file mode 100644 index 00000000..b2877bb3 --- /dev/null +++ b/src/backend/embeds/upcloud.ts @@ -0,0 +1,93 @@ +import { AES, enc } from "crypto-js"; + +import { MWEmbedType } from "@/backend/helpers/embed"; +import { registerEmbedScraper } from "@/backend/helpers/register"; +import { + MWCaptionType, + MWStreamQuality, + MWStreamType, +} from "@/backend/helpers/streams"; + +import { proxiedFetch } from "../helpers/fetch"; + +interface StreamRes { + server: number; + sources: string; + tracks: { + file: string; + kind: "captions" | "thumbnails"; + label: string; + }[]; +} + +function isJSON(json: string) { + try { + JSON.parse(json); + return true; + } catch { + return false; + } +} + +registerEmbedScraper({ + id: "upcloud", + displayName: "UpCloud", + for: MWEmbedType.UPCLOUD, + rank: 200, + async getStream({ url }) { + // Example url: https://dokicloud.one/embed-4/{id}?z= + const parsedUrl = new URL(url.replace("embed-5", "embed-4")); + + const dataPath = parsedUrl.pathname.split("/"); + const dataId = dataPath[dataPath.length - 1]; + + const streamRes = await proxiedFetch( + `${parsedUrl.origin}/ajax/embed-4/getSources?id=${dataId}`, + { + headers: { + Referer: parsedUrl.origin, + "X-Requested-With": "XMLHttpRequest", + }, + } + ); + + let sources: + | { + file: string; + type: string; + } + | string = streamRes.sources; + + if (!isJSON(sources) || typeof sources === "string") { + const decryptionKey = await proxiedFetch( + `https://raw.githubusercontent.com/enimax-anime/key/e4/key.txt` + ); + + const decryptedStream = AES.decrypt(sources, decryptionKey).toString( + enc.Utf8 + ); + + const parsedStream = JSON.parse(decryptedStream)[0]; + if (!parsedStream) throw new Error("No stream found"); + sources = parsedStream as { file: string; type: string }; + } + + return { + embedId: MWEmbedType.UPCLOUD, + streamUrl: sources.file, + quality: MWStreamQuality.Q1080P, + type: MWStreamType.HLS, + captions: streamRes.tracks + .filter((sub) => sub.kind === "captions") + .map((sub) => { + return { + langIso: sub.label, + url: sub.file, + type: sub.file.endsWith("vtt") + ? MWCaptionType.VTT + : MWCaptionType.UNKNOWN, + }; + }), + }; + }, +}); From 7e696d5c2cad0e8c4f236cad4d7e1ca47f0f59ea Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:37:41 +0530 Subject: [PATCH 44/96] feat(provider): add gomovies provider --- src/backend/providers/gomovies.ts | 162 ++++++++++++++++++++++++++++++ 1 file changed, 162 insertions(+) create mode 100644 src/backend/providers/gomovies.ts diff --git a/src/backend/providers/gomovies.ts b/src/backend/providers/gomovies.ts new file mode 100644 index 00000000..9e22d095 --- /dev/null +++ b/src/backend/providers/gomovies.ts @@ -0,0 +1,162 @@ +import { MWEmbedType } from "../helpers/embed"; +import { proxiedFetch } from "../helpers/fetch"; +import { registerProvider } from "../helpers/register"; +import { MWMediaType } from "../metadata/types"; + +const gomoviesBase = "https://gomovies.sx"; + +registerProvider({ + id: "gomovies", + displayName: "GOmovies", + rank: 300, + type: [MWMediaType.MOVIE, MWMediaType.SERIES], + + async scrape({ media, episode }) { + const search = await proxiedFetch("/ajax/search", { + baseURL: gomoviesBase, + method: "POST", + body: JSON.stringify({ + keyword: media.meta.title, + }), + headers: { + "X-Requested-With": "XMLHttpRequest", + }, + }); + + const searchPage = new DOMParser().parseFromString(search, "text/html"); + const mediaElements = searchPage.querySelectorAll("a.nav-item"); + + const mediaData = Array.from(mediaElements).map((movieEl) => { + const name = movieEl?.querySelector("h3.film-name")?.textContent; + const year = movieEl?.querySelector( + "div.film-infor span:first-of-type" + )?.textContent; + const path = movieEl.getAttribute("href"); + return { name, year, path }; + }); + + const targetMedia = mediaData.find( + (m) => + m.name === media.meta.title && + (media.meta.type === MWMediaType.MOVIE + ? m.year === media.meta.year + : true) + ); + if (!targetMedia?.path) throw new Error("Media not found"); + + // Example movie path: /movie/watch-{slug}-{id} + // Example series path: /tv/watch-{slug}-{id} + let mediaId = targetMedia.path.split("-").pop()?.replace("/", ""); + + let sources = null; + if (media.meta.type === MWMediaType.SERIES) { + const seasons = await proxiedFetch( + `/ajax/v2/tv/seasons/${mediaId}`, + { + baseURL: gomoviesBase, + headers: { + "X-Requested-With": "XMLHttpRequest", + }, + } + ); + + const seasonsEl = new DOMParser() + .parseFromString(seasons, "text/html") + .querySelectorAll(".ss-item"); + + const seasonsData = [...seasonsEl].map((season) => ({ + number: season.innerHTML.replace("Season ", ""), + dataId: season.getAttribute("data-id"), + })); + + const seasonNumber = media.meta.seasonData.number; + const targetSeason = seasonsData.find( + (season) => +season.number === seasonNumber + ); + if (!targetSeason) throw new Error("Season not found"); + + const episodes = await proxiedFetch( + `/ajax/v2/season/episodes/${targetSeason.dataId}`, + { + baseURL: gomoviesBase, + headers: { + "X-Requested-With": "XMLHttpRequest", + }, + } + ); + + const episodesEl = new DOMParser() + .parseFromString(episodes, "text/html") + .querySelectorAll(".eps-item"); + + const episodesData = Array.from(episodesEl).map((ep) => ({ + dataId: ep.getAttribute("data-id"), + number: ep + .querySelector("strong") + ?.textContent?.replace("Eps", "") + .replace(":", "") + .trim(), + })); + + const episodeNumber = media.meta.seasonData.episodes.find( + (e) => e.id === episode + )?.number; + + const targetEpisode = episodesData.find((ep) => + ep.number ? +ep.number : ep.number === episodeNumber + ); + + if (!targetEpisode?.dataId) throw new Error("Episode not found"); + + mediaId = targetEpisode.dataId; + + sources = await proxiedFetch(`/ajax/v2/episode/servers/${mediaId}`, { + baseURL: gomoviesBase, + headers: { + "X-Requested-With": "XMLHttpRequest", + }, + }); + } else { + sources = await proxiedFetch(`/ajax/movie/episodes/${mediaId}`, { + baseURL: gomoviesBase, + headers: { + "X-Requested-With": "XMLHttpRequest", + }, + }); + } + + const upcloud = new DOMParser() + .parseFromString(sources, "text/html") + .querySelector('a[title*="upcloud" i]'); + + const upcloudDataId = + upcloud?.getAttribute("data-id") ?? upcloud?.getAttribute("data-linkid"); + + if (!upcloudDataId) throw new Error("Upcloud source not available"); + + const upcloudSource = await proxiedFetch<{ + type: "iframe" | string; + link: string; + sources: []; + title: string; + tracks: []; + }>(`/ajax/sources/${upcloudDataId}`, { + baseURL: gomoviesBase, + headers: { + "X-Requested-With": "XMLHttpRequest", + }, + }); + + if (!upcloudSource.link || upcloudSource.type !== "iframe") + throw new Error("No upcloud stream found"); + + return { + embeds: [ + { + type: MWEmbedType.UPCLOUD, + url: upcloudSource.link, + }, + ], + }; + }, +}); From d198760f9c0051c4cf91cab4ad2b84762e09c6ae Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:37:57 +0530 Subject: [PATCH 45/96] feat(provider): add kissasian provider --- src/backend/providers/kissasian.ts | 103 +++++++++++++++++++++++++++++ 1 file changed, 103 insertions(+) create mode 100644 src/backend/providers/kissasian.ts diff --git a/src/backend/providers/kissasian.ts b/src/backend/providers/kissasian.ts new file mode 100644 index 00000000..01fa0a2e --- /dev/null +++ b/src/backend/providers/kissasian.ts @@ -0,0 +1,103 @@ +import { MWEmbedType } from "../helpers/embed"; +import { proxiedFetch } from "../helpers/fetch"; +import { registerProvider } from "../helpers/register"; +import { MWMediaType } from "../metadata/types"; + +const kissasianBase = "https://kissasian.li"; + +registerProvider({ + id: "kissasian", + displayName: "KissAsian", + rank: 10000, + type: [MWMediaType.MOVIE, MWMediaType.SERIES], + + async scrape({ media, episode, progress }) { + let seasonNumber = ""; + let episodeNumber = ""; + + if (media.meta.type === MWMediaType.SERIES) { + seasonNumber = + media.meta.seasonData.number === 1 + ? "" + : `${media.meta.seasonData.number}`; + episodeNumber = `${ + media.meta.seasonData.episodes.find((e) => e.id === episode)?.number ?? + "" + }`; + } + + const searchForm = new FormData(); + searchForm.append("keyword", `${media.meta.title} ${seasonNumber}`.trim()); + searchForm.append("type", "Drama"); + + const search = await proxiedFetch("/Search/SearchSuggest", { + baseURL: kissasianBase, + method: "POST", + body: searchForm, + }); + + const searchPage = new DOMParser().parseFromString(search, "text/html"); + + const dramas = Array.from(searchPage.querySelectorAll("a")).map((drama) => { + return { + name: drama.textContent, + url: drama.href, + }; + }); + + const targetDrama = + dramas.find( + (d) => d.name?.toLowerCase() === media.meta.title.toLowerCase() + ) ?? dramas[0]; + if (!targetDrama) throw new Error("Drama not found"); + + progress(30); + + const drama = await proxiedFetch(targetDrama.url); + + const dramaPage = new DOMParser().parseFromString(drama, "text/html"); + + const episodesEl = dramaPage.querySelectorAll("tbody tr:not(:first-child)"); + + const episodes = Array.from(episodesEl) + .map((ep) => { + const number = ep + ?.querySelector("td.episodeSub a") + ?.textContent?.split("Episode")[1] + ?.trim(); + const href = ep?.querySelector("td.episodeSub a")?.getAttribute("href"); + return { number, href }; + }) + .filter((e) => !!e.href); + + const targetEpisode = + media.meta.type === MWMediaType.MOVIE + ? episodes[0] + : episodes.find((e) => e.number === `${episodeNumber}`); + if (!targetEpisode?.href) throw new Error("Episode not found"); + + progress(70); + + const watch = await proxiedFetch(`${targetEpisode.href}&s=sb`, { + baseURL: kissasianBase, + }); + + const watchPage = new DOMParser().parseFromString(watch, "text/html"); + + const streamsbUrl = watchPage + .querySelector("iframe[id=my_video_1]") + ?.getAttribute("src"); + if (!streamsbUrl) throw new Error("Streamsb embed not found"); + + console.log(streamsbUrl); + + return { + embeds: [ + { + type: MWEmbedType.STREAMSB, + url: streamsbUrl, + }, + ], + }; + }, +}); From 2db7e0bef8ffbdf41f74481a21c2c11cc6d6dd2f Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:41:30 +0530 Subject: [PATCH 46/96] feat(enum): add upcloud and streamsb enum --- src/backend/helpers/embed.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/backend/helpers/embed.ts b/src/backend/helpers/embed.ts index 0dec6422..a584a0c7 100644 --- a/src/backend/helpers/embed.ts +++ b/src/backend/helpers/embed.ts @@ -4,6 +4,8 @@ export enum MWEmbedType { M4UFREE = "m4ufree", STREAMM4U = "streamm4u", PLAYM4U = "playm4u", + UPCLOUD = "upcloud", + STREAMSB = "streamsb", } export type MWEmbed = { From d4c6dac9f2edecf9c186d8fed18160945e4e7122 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:43:36 +0530 Subject: [PATCH 47/96] disable 2embed --- src/backend/providers/2embed.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/backend/providers/2embed.ts b/src/backend/providers/2embed.ts index 48056020..7cc8938e 100644 --- a/src/backend/providers/2embed.ts +++ b/src/backend/providers/2embed.ts @@ -191,6 +191,7 @@ registerProvider({ displayName: "2Embed", rank: 125, type: [MWMediaType.MOVIE, MWMediaType.SERIES], + disabled: true, // Disabled, not working async scrape({ media, episode, progress }) { let embedUrl = `${twoEmbedBase}/embed/tmdb/movie?id=${media.tmdbId}`; From f6b830d06df314b442ccafdfd6cc9cc303db54ad Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:44:54 +0530 Subject: [PATCH 48/96] feat(register): new providers and embed scrapers --- src/backend/index.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/backend/index.ts b/src/backend/index.ts index eb0ad897..065f1c62 100644 --- a/src/backend/index.ts +++ b/src/backend/index.ts @@ -9,9 +9,13 @@ import "./providers/m4ufree"; import "./providers/hdwatched"; import "./providers/2embed"; import "./providers/sflix"; +import "./providers/gomovies"; +import "./providers/kissasian"; // embeds import "./embeds/streamm4u"; import "./embeds/playm4u"; +import "./embeds/upcloud"; +import "./embeds/streamsb"; initializeScraperStore(); From ad263916454d31847a56456f040218f340817747 Mon Sep 17 00:00:00 2001 From: castdrian Date: Fri, 16 Jun 2023 11:18:32 +0200 Subject: [PATCH 49/96] use external ids endpoint for imdb ids --- src/backend/metadata/getmeta.ts | 6 ++---- src/backend/metadata/tmdb.ts | 25 +++++++++++++++++++++++++ src/backend/metadata/types.ts | 24 ++++++++++++++++++++++++ 3 files changed, 51 insertions(+), 4 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index a5246fcf..6868c7e4 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -87,10 +87,8 @@ export async function getMetaFromId( if (!details) return null; - let imdbId; - if (type === MWMediaType.MOVIE) { - imdbId = (details as TMDBMovieData).imdb_id ?? undefined; - } + const externalIds = await Tmdb.getExternalIds(id, mediaTypeToTMDB(type)); + const imdbId = externalIds.imdb_id ?? undefined; let seasonData: TMDBSeasonMetaResult | undefined; diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index aed31c1d..e22e86e7 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -6,15 +6,18 @@ import { MWSeasonMeta, TMDBContentTypes, TMDBEpisodeShort, + TMDBExternalIds, TMDBMediaResult, TMDBMediaStatic, TMDBMovieData, + TMDBMovieExternalIds, TMDBMovieResponse, TMDBMovieResult, TMDBSearchResultStatic, TMDBSeason, TMDBSeasonMetaResult, TMDBShowData, + TMDBShowExternalIds, TMDBShowResponse, TMDBShowResult, } from "./types"; @@ -170,6 +173,28 @@ export abstract class Tmdb { title: e.name, })); } + + public static async getExternalIds( + id: string, + type: TMDBContentTypes + ): Promise { + let data; + + switch (type) { + case "movie": + data = await Tmdb.get( + `/movie/${id}/external_ids` + ); + break; + case "show": + data = await Tmdb.get(`/tv/${id}/external_ids`); + break; + default: + throw new Error("Invalid media type"); + } + + return data; + } } export async function formatTMDBSearchResult( diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index e23d9a5b..fa7a7ef0 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -379,3 +379,27 @@ export interface TMDBSeason { poster_path: string | null; season_number: number; } + +export interface TMDBShowExternalIds { + id: number; + imdb_id: null | string; + freebase_mid: null | string; + freebase_id: null | string; + tvdb_id: number; + tvrage_id: null | string; + wikidata_id: null | string; + facebook_id: null | string; + instagram_id: null | string; + twitter_id: null | string; +} + +export interface TMDBMovieExternalIds { + id: number; + imdb_id: null | string; + wikidata_id: null | string; + facebook_id: null | string; + instagram_id: null | string; + twitter_id: null | string; +} + +export type TMDBExternalIds = TMDBShowExternalIds | TMDBMovieExternalIds; From 58ca372a49ce1ae8009b1e5f33a4809ef9b4c6a4 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 14:52:42 +0530 Subject: [PATCH 50/96] refactor(kissasian): change rank --- src/backend/providers/kissasian.ts | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/backend/providers/kissasian.ts b/src/backend/providers/kissasian.ts index 01fa0a2e..8d4e50c4 100644 --- a/src/backend/providers/kissasian.ts +++ b/src/backend/providers/kissasian.ts @@ -8,7 +8,7 @@ const kissasianBase = "https://kissasian.li"; registerProvider({ id: "kissasian", displayName: "KissAsian", - rank: 10000, + rank: 130, type: [MWMediaType.MOVIE, MWMediaType.SERIES], async scrape({ media, episode, progress }) { @@ -89,8 +89,6 @@ registerProvider({ ?.getAttribute("src"); if (!streamsbUrl) throw new Error("Streamsb embed not found"); - console.log(streamsbUrl); - return { embeds: [ { From e912ea4715035b9305a22cbabad3db0f5a4795e3 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 15:05:42 +0530 Subject: [PATCH 51/96] cleanup --- src/backend/embeds/streamsb.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/backend/embeds/streamsb.ts b/src/backend/embeds/streamsb.ts index c755a0b0..99ed563e 100644 --- a/src/backend/embeds/streamsb.ts +++ b/src/backend/embeds/streamsb.ts @@ -169,8 +169,6 @@ registerEmbedScraper({ .querySelector(".btn.btn-light.btn-lg") ?.getAttribute("href"); - console.log(dlLink); - return { quality: dl.quality.label as MWStreamQuality, url: dlLink, From 9003bf67887e4750e47a093db1a2b7a3ca8e4703 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 16:12:07 +0530 Subject: [PATCH 52/96] feat(embed): add mp4upload embed scraper --- src/backend/embeds/mp4upload.ts | 32 ++++++++++++++++++++++++++++++++ src/backend/index.ts | 1 + 2 files changed, 33 insertions(+) create mode 100644 src/backend/embeds/mp4upload.ts diff --git a/src/backend/embeds/mp4upload.ts b/src/backend/embeds/mp4upload.ts new file mode 100644 index 00000000..3902e20b --- /dev/null +++ b/src/backend/embeds/mp4upload.ts @@ -0,0 +1,32 @@ +import { MWEmbedType } from "@/backend/helpers/embed"; +import { registerEmbedScraper } from "@/backend/helpers/register"; +import { MWStreamQuality, MWStreamType } from "@/backend/helpers/streams"; + +import { proxiedFetch } from "../helpers/fetch"; + +registerEmbedScraper({ + id: "mp4upload", + displayName: "mp4upload", + for: MWEmbedType.MP4UPLOAD, + rank: 170, + async getStream({ url }) { + const embed = await proxiedFetch(url); + + const playerSrcRegex = + /(?<=player\.src\()\s*{\s*type:\s*"[^"]+",\s*src:\s*"([^"]+)"\s*}\s*(?=\);)/s; + + const playerSrc = embed.match(playerSrcRegex); + + const streamUrl = playerSrc[1]; + + if (!streamUrl) throw new Error("Stream url not found"); + + return { + embedId: MWEmbedType.MP4UPLOAD, + streamUrl, + quality: MWStreamQuality.Q1080P, + captions: [], + type: MWStreamType.MP4, + }; + }, +}); diff --git a/src/backend/index.ts b/src/backend/index.ts index 065f1c62..a2beaa2a 100644 --- a/src/backend/index.ts +++ b/src/backend/index.ts @@ -17,5 +17,6 @@ import "./embeds/streamm4u"; import "./embeds/playm4u"; import "./embeds/upcloud"; import "./embeds/streamsb"; +import "./embeds/mp4upload"; initializeScraperStore(); From 7e948c60c1473dd4bda958add89c62dacdef74a3 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 16:12:53 +0530 Subject: [PATCH 53/96] feat(enum): add mp4upload enum --- src/backend/helpers/embed.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/backend/helpers/embed.ts b/src/backend/helpers/embed.ts index a584a0c7..1ec3362c 100644 --- a/src/backend/helpers/embed.ts +++ b/src/backend/helpers/embed.ts @@ -6,6 +6,7 @@ export enum MWEmbedType { PLAYM4U = "playm4u", UPCLOUD = "upcloud", STREAMSB = "streamsb", + MP4UPLOAD = "mp4upload", } export type MWEmbed = { From a0bb03790a37fad2eed1ab15f4ada9814a84dc45 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 16:14:05 +0530 Subject: [PATCH 54/96] refactor(streamsb): improve quality sorting --- src/backend/embeds/streamsb.ts | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/src/backend/embeds/streamsb.ts b/src/backend/embeds/streamsb.ts index 99ed563e..e91b43c7 100644 --- a/src/backend/embeds/streamsb.ts +++ b/src/backend/embeds/streamsb.ts @@ -100,7 +100,7 @@ registerEmbedScraper({ "[onclick^=download_video]" ); - const dlDetails = []; + let dlDetails = []; for (const func of downloadVideoFunctions) { const funcContents = func.getAttribute("onclick"); const regExpFunc = /download_video\('(.+?)','(.+?)','(.+?)'\)/; @@ -121,6 +121,12 @@ registerEmbedScraper({ } } + dlDetails = dlDetails.sort((a, b) => { + const aQuality = qualityOrder.indexOf(a.quality.label as MWStreamQuality); + const bQuality = qualityOrder.indexOf(b.quality.label as MWStreamQuality); + return aQuality - bQuality; + }); + progress(40); let dls = await Promise.all( @@ -187,11 +193,6 @@ registerEmbedScraper({ }) ); dls = dls.filter((d) => !!d.url); - dls = dls.sort((a, b) => { - const aQuality = qualityOrder.indexOf(a.quality); - const bQuality = qualityOrder.indexOf(b.quality); - return aQuality - bQuality; - }); progress(60); From bc0f9a6abff9ab90b7106456f852de9fffbf8276 Mon Sep 17 00:00:00 2001 From: Jordaar <69628820+Jordaar@users.noreply.github.com> Date: Fri, 16 Jun 2023 16:15:41 +0530 Subject: [PATCH 55/96] feat(kissasian): additional mp4upload embed scraper --- src/backend/providers/kissasian.ts | 54 ++++++++++++++++++++---------- 1 file changed, 36 insertions(+), 18 deletions(-) diff --git a/src/backend/providers/kissasian.ts b/src/backend/providers/kissasian.ts index 8d4e50c4..90708970 100644 --- a/src/backend/providers/kissasian.ts +++ b/src/backend/providers/kissasian.ts @@ -5,6 +5,17 @@ import { MWMediaType } from "../metadata/types"; const kissasianBase = "https://kissasian.li"; +const embedProviders = [ + { + type: MWEmbedType.MP4UPLOAD, + id: "mp", + }, + { + type: MWEmbedType.STREAMSB, + id: "sb", + }, +]; + registerProvider({ id: "kissasian", displayName: "KissAsian", @@ -65,37 +76,44 @@ registerProvider({ ?.querySelector("td.episodeSub a") ?.textContent?.split("Episode")[1] ?.trim(); - const href = ep?.querySelector("td.episodeSub a")?.getAttribute("href"); - return { number, href }; + const url = ep?.querySelector("td.episodeSub a")?.getAttribute("href"); + return { number, url }; }) - .filter((e) => !!e.href); + .filter((e) => !!e.url); const targetEpisode = media.meta.type === MWMediaType.MOVIE ? episodes[0] : episodes.find((e) => e.number === `${episodeNumber}`); - if (!targetEpisode?.href) throw new Error("Episode not found"); + if (!targetEpisode?.url) throw new Error("Episode not found"); progress(70); - const watch = await proxiedFetch(`${targetEpisode.href}&s=sb`, { - baseURL: kissasianBase, - }); + let embeds = await Promise.all( + embedProviders.map(async (provider) => { + const watch = await proxiedFetch( + `${targetEpisode.url}&s=${provider.id}`, + { + baseURL: kissasianBase, + } + ); - const watchPage = new DOMParser().parseFromString(watch, "text/html"); + const watchPage = new DOMParser().parseFromString(watch, "text/html"); - const streamsbUrl = watchPage - .querySelector("iframe[id=my_video_1]") - ?.getAttribute("src"); - if (!streamsbUrl) throw new Error("Streamsb embed not found"); + const embedUrl = watchPage + .querySelector("iframe[id=my_video_1]") + ?.getAttribute("src"); + + return { + type: provider.type, + url: embedUrl ?? "", + }; + }) + ); + embeds = embeds.filter((e) => e.url !== ""); return { - embeds: [ - { - type: MWEmbedType.STREAMSB, - url: streamsbUrl, - }, - ], + embeds, }; }, }); From d47acada58f660cf4bc0788fd6b6a80d5e6c2729 Mon Sep 17 00:00:00 2001 From: mrjvs Date: Sat, 17 Jun 2023 20:20:38 +0200 Subject: [PATCH 56/96] Update i18n.ts --- src/setup/i18n.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/setup/i18n.ts b/src/setup/i18n.ts index 919754ae..ccb77d22 100644 --- a/src/setup/i18n.ts +++ b/src/setup/i18n.ts @@ -45,7 +45,7 @@ const locales = { }, vi: { translation: vi, - } + }, pl: { translation: pl, }, From a4bd9bb87a09da7e1880089816e578053746de30 Mon Sep 17 00:00:00 2001 From: frost768 Date: Sun, 18 Jun 2023 15:10:26 +0300 Subject: [PATCH 57/96] fix: language preference persistence --- src/index.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/index.tsx b/src/index.tsx index 1bf99f70..36b1fb14 100644 --- a/src/index.tsx +++ b/src/index.tsx @@ -8,13 +8,14 @@ import { registerSW } from "virtual:pwa-register"; import { ErrorBoundary } from "@/components/layout/ErrorBoundary"; import App from "@/setup/App"; import { conf } from "@/setup/config"; +import i18n from "@/setup/i18n"; import "@/setup/ga"; import "@/setup/sentry"; -import "@/setup/i18n"; import "@/setup/index.css"; import "@/backend"; import { initializeChromecast } from "./setup/chromecast"; +import { SettingsStore } from "./state/settings/store"; import { initializeStores } from "./utils/storage"; // initialize @@ -30,6 +31,7 @@ registerSW({ const LazyLoadedApp = React.lazy(async () => { await initializeStores(); + i18n.changeLanguage(SettingsStore.get().language ?? "en"); return { default: App, }; From 70852773f94c30ab47d9737bff8a9875f514c564 Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 12 Jun 2023 20:06:46 +0200 Subject: [PATCH 58/96] partial refactor --- src/backend/metadata/search_new.ts | 21 +++ src/backend/metadata/tmdb.ts | 78 +++++++++ src/backend/metadata/trakttv.ts | 166 ++++++++++++++++++ src/backend/metadata/types_new.ts | 264 +++++++++++++++++++++++++++++ 4 files changed, 529 insertions(+) create mode 100644 src/backend/metadata/search_new.ts create mode 100644 src/backend/metadata/tmdb.ts create mode 100644 src/backend/metadata/trakttv.ts create mode 100644 src/backend/metadata/types_new.ts diff --git a/src/backend/metadata/search_new.ts b/src/backend/metadata/search_new.ts new file mode 100644 index 00000000..4506514a --- /dev/null +++ b/src/backend/metadata/search_new.ts @@ -0,0 +1,21 @@ +import { SimpleCache } from "@/utils/cache"; + +import { Trakt, mediaTypeToTTV } from "./trakttv"; +import { MWMediaMeta, MWQuery } from "./types"; + +const cache = new SimpleCache(); +cache.setCompare((a, b) => { + return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); +}); +cache.initialize(); + +export async function searchForMedia(query: MWQuery): Promise { + if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; + const { searchQuery, type } = query; + + const contentType = mediaTypeToTTV(type); + + const results = await Trakt.search(searchQuery, contentType); + cache.set(query, results, 3600); + return results; +} diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts new file mode 100644 index 00000000..0700945b --- /dev/null +++ b/src/backend/metadata/tmdb.ts @@ -0,0 +1,78 @@ +import { conf } from "@/setup/config"; + +import { + DetailedMeta, + MWMediaType, + TMDBMediaStatic, + TMDBMovieData, + TMDBShowData, +} from "./types"; +import { mwFetch } from "../helpers/fetch"; + +export abstract class Tmdb { + private static baseURL = "https://api.themoviedb.org/3"; + + private static headers = { + accept: "application/json", + Authorization: `Bearer ${conf().TMDB_API_KEY}`, + }; + + private static async get(url: string): Promise { + const res = await mwFetch(url, { + headers: Tmdb.headers, + baseURL: Tmdb.baseURL, + }); + return res; + } + + public static getMediaDetails: TMDBMediaStatic["getMediaDetails"] = async ( + id: string, + type: MWMediaType + ) => { + let data; + + switch (type) { + case "movie": + data = await Tmdb.get(`/movie/${id}`); + break; + case "series": + data = await Tmdb.get(`/tv/${id}`); + break; + default: + throw new Error("Invalid media type"); + } + + return data; + }; + + public static getMediaPoster(posterPath: string | null): string | undefined { + if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; + } + + /* public static async getMetaFromId( + type: MWMediaType, + id: string, + seasonId?: string + ): Promise { + console.log("getMetaFromId", type, id, seasonId); + + const details = await Tmdb.getMediaDetails(id, type); + + if (!details) return null; + + let imdbId; + if (type === MWMediaType.MOVIE) { + imdbId = (details as TMDBMovieData).imdb_id ?? undefined; + } + + if (!meta.length) return null; + + console.log(meta); + + return { + meta, + imdbId, + tmdbId: id, + }; + } */ +} diff --git a/src/backend/metadata/trakttv.ts b/src/backend/metadata/trakttv.ts new file mode 100644 index 00000000..5fb67a17 --- /dev/null +++ b/src/backend/metadata/trakttv.ts @@ -0,0 +1,166 @@ +import { conf } from "@/setup/config"; + +import { Tmdb } from "./tmdb"; +import { + DetailedMeta, + MWMediaMeta, + MWMediaType, + MWSeasonMeta, + TMDBShowData, + TTVContentTypes, + TTVMediaResult, + TTVSearchResult, + TTVSeasonMetaResult, +} from "./types"; +import { mwFetch } from "../helpers/fetch"; + +export function mediaTypeToTTV(type: MWMediaType): TTVContentTypes { + if (type === MWMediaType.MOVIE) return "movie"; + if (type === MWMediaType.SERIES) return "show"; + throw new Error("unsupported type"); +} + +export function TTVMediaToMediaType(type: string): MWMediaType { + if (type === "movie") return MWMediaType.MOVIE; + if (type === "show") return MWMediaType.SERIES; + throw new Error("unsupported type"); +} + +export function formatTTVMeta( + media: TTVMediaResult, + season?: TTVSeasonMetaResult +): MWMediaMeta { + const type = TTVMediaToMediaType(media.object_type); + let seasons: undefined | MWSeasonMeta[]; + if (type === MWMediaType.SERIES) { + seasons = media.seasons + ?.sort((a, b) => a.season_number - b.season_number) + .map( + (v): MWSeasonMeta => ({ + title: v.title, + id: v.id.toString(), + number: v.season_number, + }) + ); + } + + return { + title: media.title, + id: media.id.toString(), + year: media.original_release_year?.toString(), + poster: media.poster, + type, + seasons: seasons as any, + seasonData: season + ? ({ + id: season.id.toString(), + number: season.season_number, + title: season.title, + episodes: season.episodes + .sort((a, b) => a.episode_number - b.episode_number) + .map((v) => ({ + id: v.id.toString(), + number: v.episode_number, + title: v.title, + })), + } as any) + : (undefined as any), + }; +} + +export function TTVMediaToId(media: MWMediaMeta): string { + return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); +} + +export function decodeTTVId( + paramId: string +): { id: string; type: MWMediaType } | null { + const [prefix, type, id] = paramId.split("-", 3); + if (prefix !== "TTV") return null; + let mediaType; + try { + mediaType = TTVMediaToMediaType(type); + } catch { + return null; + } + return { + type: mediaType, + id, + }; +} + +export async function formatTTVSearchResult( + result: TTVSearchResult +): Promise { + const type = TTVMediaToMediaType(result.type); + const media = result[result.type]; + + if (!media) throw new Error("invalid result"); + + const details = await Tmdb.getMediaDetails( + media.ids.tmdb.toString(), + TTVMediaToMediaType(result.type) + ); + console.log(details); + + const seasons = + type === MWMediaType.SERIES + ? (details as TMDBShowData).seasons?.map((v) => ({ + id: v.id, + title: v.name, + season_number: v.season_number, + })) + : undefined; + + return { + title: media.title, + poster: Tmdb.getMediaPoster(details.poster_path), + id: media.ids.trakt, + original_release_year: media.year, + ttv_entity_id: media.ids.slug, + object_type: mediaTypeToTTV(type), + seasons, + }; +} + +export abstract class Trakt { + private static baseURL = "https://api.trakt.tv"; + + private static headers = { + "Content-Type": "application/json", + "trakt-api-version": "2", + "trakt-api-key": conf().TRAKT_CLIENT_ID, + }; + + private static async get(url: string): Promise { + const res = await mwFetch(url, { + headers: Trakt.headers, + baseURL: Trakt.baseURL, + }); + return res; + } + + public static async search( + query: string, + type: "movie" | "show" + ): Promise { + const data = await Trakt.get( + `/search/${type}?query=${encodeURIComponent(query)}` + ); + + const formatted = await Promise.all( + // eslint-disable-next-line no-return-await + data.map(async (v) => await formatTTVSearchResult(v)) + ); + return formatted.map((v) => formatTTVMeta(v)); + } + + public static async getMetaFromId( + type: MWMediaType, + id: string, + seasonId?: string + ): Promise { + console.log("getMetaFromId", type, id, seasonId); + return null; + } +} diff --git a/src/backend/metadata/types_new.ts b/src/backend/metadata/types_new.ts new file mode 100644 index 00000000..f954ff6a --- /dev/null +++ b/src/backend/metadata/types_new.ts @@ -0,0 +1,264 @@ +export enum MWMediaType { + MOVIE = "movie", + SERIES = "series", + ANIME = "anime", +} + +export type MWSeasonMeta = { + id: string; + number: number; + title: string; +}; + +export type MWSeasonWithEpisodeMeta = { + id: string; + number: number; + title: string; + episodes: { + id: string; + number: number; + title: string; + }[]; +}; + +type MWMediaMetaBase = { + title: string; + id: string; + year?: string; + poster?: string; +}; + +type MWMediaMetaSpecific = + | { + type: MWMediaType.MOVIE | MWMediaType.ANIME; + seasons: undefined; + } + | { + type: MWMediaType.SERIES; + seasons: MWSeasonMeta[]; + seasonData: MWSeasonWithEpisodeMeta; + }; + +export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; + +export interface MWQuery { + searchQuery: string; + type: MWMediaType; +} + +export type TTVContentTypes = "movie" | "show"; + +export type TTVSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type TTVEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type TTVMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + ttv_entity_id: string; + object_type: TTVContentTypes; + seasons?: TTVSeasonShort[]; +}; + +export type TTVSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: TTVEpisodeShort[]; +}; + +export interface TTVSearchResult { + type: "movie" | "show"; + score: number; + movie?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + imdb: string; + tmdb: number; + }; + }; + show?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + tvdb: number; + imdb: string; + tmdb: number; + }; + }; +} + +export interface DetailedMeta { + meta: MWMediaMeta; + imdbId?: string; + tmdbId?: string; +} + +export interface TMDBShowData { + adult: boolean; + backdrop_path: string | null; + created_by: { + id: number; + credit_id: string; + name: string; + gender: number; + profile_path: string | null; + }[]; + episode_run_time: number[]; + first_air_date: string; + genres: { + id: number; + name: string; + }[]; + homepage: string; + id: number; + in_production: boolean; + languages: string[]; + last_air_date: string; + last_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + name: string; + next_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + networks: { + id: number; + logo_path: string; + name: string; + origin_country: string; + }[]; + number_of_episodes: number; + number_of_seasons: number; + origin_country: string[]; + original_language: string; + original_name: string; + overview: string; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + seasons: { + air_date: string; + episode_count: number; + id: number; + name: string; + overview: string; + poster_path: string | null; + season_number: number; + }[]; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string; + type: string; + vote_average: number; + vote_count: number; +} + +export interface TMDBMovieData { + adult: boolean; + backdrop_path: string | null; + belongs_to_collection: { + id: number; + name: string; + poster_path: string | null; + backdrop_path: string | null; + } | null; + budget: number; + genres: { + id: number; + name: string; + }[]; + homepage: string | null; + id: number; + imdb_id: string | null; + original_language: string; + original_title: string; + overview: string | null; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + release_date: string; + revenue: number; + runtime: number | null; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string | null; + title: string; + video: boolean; + vote_average: number; + vote_count: number; +} + +export type TMDBMediaDetailsPromise = Promise; + +export interface TMDBMediaStatic { + getMediaDetails( + id: string, + type: MWMediaType.SERIES + ): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; +} From 63f26b81dec2bab16f6d19c0d502a05753d85497 Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 12 Jun 2023 20:17:42 +0200 Subject: [PATCH 59/96] preliminary refactor --- src/backend/metadata/search.ts | 48 +----- src/backend/metadata/search_new.ts | 21 --- src/backend/metadata/search_old.ts | 59 +++++++ src/backend/metadata/tmdb.ts | 1 - src/backend/metadata/types.ts | 217 ++++++++++++++++++++++++ src/backend/metadata/types_new.ts | 264 ----------------------------- src/backend/metadata/types_old.ts | 47 +++++ src/setup/config.ts | 4 + 8 files changed, 332 insertions(+), 329 deletions(-) delete mode 100644 src/backend/metadata/search_new.ts create mode 100644 src/backend/metadata/search_old.ts delete mode 100644 src/backend/metadata/types_new.ts create mode 100644 src/backend/metadata/types_old.ts diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 10cbb285..4506514a 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -1,14 +1,7 @@ import { SimpleCache } from "@/utils/cache"; -import { - JWContentTypes, - JWMediaResult, - JW_API_BASE, - formatJWMeta, - mediaTypeToJW, -} from "./justwatch"; +import { Trakt, mediaTypeToTTV } from "./trakttv"; import { MWMediaMeta, MWQuery } from "./types"; -import { proxiedFetch } from "../helpers/fetch"; const cache = new SimpleCache(); cache.setCompare((a, b) => { @@ -16,44 +9,13 @@ cache.setCompare((a, b) => { }); cache.initialize(); -type JWSearchQuery = { - content_types: JWContentTypes[]; - page: number; - page_size: number; - query: string; -}; - -type JWPage = { - items: T[]; - page: number; - page_size: number; - total_pages: number; - total_results: number; -}; - export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const contentType = mediaTypeToJW(type); - const body: JWSearchQuery = { - content_types: [contentType], - page: 1, - query: searchQuery, - page_size: 40, - }; + const contentType = mediaTypeToTTV(type); - const data = await proxiedFetch>( - "/content/titles/en_US/popular", - { - baseURL: JW_API_BASE, - params: { - body: JSON.stringify(body), - }, - } - ); - - const returnData = data.items.map((v) => formatJWMeta(v)); - cache.set(query, returnData, 3600); // cache for an hour - return returnData; + const results = await Trakt.search(searchQuery, contentType); + cache.set(query, results, 3600); + return results; } diff --git a/src/backend/metadata/search_new.ts b/src/backend/metadata/search_new.ts deleted file mode 100644 index 4506514a..00000000 --- a/src/backend/metadata/search_new.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { SimpleCache } from "@/utils/cache"; - -import { Trakt, mediaTypeToTTV } from "./trakttv"; -import { MWMediaMeta, MWQuery } from "./types"; - -const cache = new SimpleCache(); -cache.setCompare((a, b) => { - return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); -}); -cache.initialize(); - -export async function searchForMedia(query: MWQuery): Promise { - if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; - const { searchQuery, type } = query; - - const contentType = mediaTypeToTTV(type); - - const results = await Trakt.search(searchQuery, contentType); - cache.set(query, results, 3600); - return results; -} diff --git a/src/backend/metadata/search_old.ts b/src/backend/metadata/search_old.ts new file mode 100644 index 00000000..05be994d --- /dev/null +++ b/src/backend/metadata/search_old.ts @@ -0,0 +1,59 @@ +import { SimpleCache } from "@/utils/cache"; + +import { + JWContentTypes, + JWMediaResult, + JW_API_BASE, + formatJWMeta, + mediaTypeToJW, +} from "./justwatch"; +import { MWMediaMeta, MWQuery } from "./types_old"; +import { proxiedFetch } from "../helpers/fetch"; + +const cache = new SimpleCache(); +cache.setCompare((a, b) => { + return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); +}); +cache.initialize(); + +type JWSearchQuery = { + content_types: JWContentTypes[]; + page: number; + page_size: number; + query: string; +}; + +type JWPage = { + items: T[]; + page: number; + page_size: number; + total_pages: number; + total_results: number; +}; + +export async function searchForMedia(query: MWQuery): Promise { + if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; + const { searchQuery, type } = query; + + const contentType = mediaTypeToJW(type); + const body: JWSearchQuery = { + content_types: [contentType], + page: 1, + query: searchQuery, + page_size: 40, + }; + + const data = await proxiedFetch>( + "/content/titles/en_US/popular", + { + baseURL: JW_API_BASE, + params: { + body: JSON.stringify(body), + }, + } + ); + + const returnData = data.items.map((v) => formatJWMeta(v)); + cache.set(query, returnData, 3600); // cache for an hour + return returnData; +} diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 0700945b..460e13b4 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -1,7 +1,6 @@ import { conf } from "@/setup/config"; import { - DetailedMeta, MWMediaType, TMDBMediaStatic, TMDBMovieData, diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index 2723fbe7..f954ff6a 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -45,3 +45,220 @@ export interface MWQuery { searchQuery: string; type: MWMediaType; } + +export type TTVContentTypes = "movie" | "show"; + +export type TTVSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type TTVEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type TTVMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + ttv_entity_id: string; + object_type: TTVContentTypes; + seasons?: TTVSeasonShort[]; +}; + +export type TTVSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: TTVEpisodeShort[]; +}; + +export interface TTVSearchResult { + type: "movie" | "show"; + score: number; + movie?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + imdb: string; + tmdb: number; + }; + }; + show?: { + title: string; + year: number; + ids: { + trakt: number; + slug: string; + tvdb: number; + imdb: string; + tmdb: number; + }; + }; +} + +export interface DetailedMeta { + meta: MWMediaMeta; + imdbId?: string; + tmdbId?: string; +} + +export interface TMDBShowData { + adult: boolean; + backdrop_path: string | null; + created_by: { + id: number; + credit_id: string; + name: string; + gender: number; + profile_path: string | null; + }[]; + episode_run_time: number[]; + first_air_date: string; + genres: { + id: number; + name: string; + }[]; + homepage: string; + id: number; + in_production: boolean; + languages: string[]; + last_air_date: string; + last_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + name: string; + next_episode_to_air: { + id: number; + name: string; + overview: string; + vote_average: number; + vote_count: number; + air_date: string; + episode_number: number; + production_code: string; + runtime: number | null; + season_number: number; + show_id: number; + still_path: string | null; + } | null; + networks: { + id: number; + logo_path: string; + name: string; + origin_country: string; + }[]; + number_of_episodes: number; + number_of_seasons: number; + origin_country: string[]; + original_language: string; + original_name: string; + overview: string; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + seasons: { + air_date: string; + episode_count: number; + id: number; + name: string; + overview: string; + poster_path: string | null; + season_number: number; + }[]; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string; + type: string; + vote_average: number; + vote_count: number; +} + +export interface TMDBMovieData { + adult: boolean; + backdrop_path: string | null; + belongs_to_collection: { + id: number; + name: string; + poster_path: string | null; + backdrop_path: string | null; + } | null; + budget: number; + genres: { + id: number; + name: string; + }[]; + homepage: string | null; + id: number; + imdb_id: string | null; + original_language: string; + original_title: string; + overview: string | null; + popularity: number; + poster_path: string | null; + production_companies: { + id: number; + logo_path: string | null; + name: string; + origin_country: string; + }[]; + production_countries: { + iso_3166_1: string; + name: string; + }[]; + release_date: string; + revenue: number; + runtime: number | null; + spoken_languages: { + english_name: string; + iso_639_1: string; + name: string; + }[]; + status: string; + tagline: string | null; + title: string; + video: boolean; + vote_average: number; + vote_count: number; +} + +export type TMDBMediaDetailsPromise = Promise; + +export interface TMDBMediaStatic { + getMediaDetails( + id: string, + type: MWMediaType.SERIES + ): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; +} diff --git a/src/backend/metadata/types_new.ts b/src/backend/metadata/types_new.ts deleted file mode 100644 index f954ff6a..00000000 --- a/src/backend/metadata/types_new.ts +++ /dev/null @@ -1,264 +0,0 @@ -export enum MWMediaType { - MOVIE = "movie", - SERIES = "series", - ANIME = "anime", -} - -export type MWSeasonMeta = { - id: string; - number: number; - title: string; -}; - -export type MWSeasonWithEpisodeMeta = { - id: string; - number: number; - title: string; - episodes: { - id: string; - number: number; - title: string; - }[]; -}; - -type MWMediaMetaBase = { - title: string; - id: string; - year?: string; - poster?: string; -}; - -type MWMediaMetaSpecific = - | { - type: MWMediaType.MOVIE | MWMediaType.ANIME; - seasons: undefined; - } - | { - type: MWMediaType.SERIES; - seasons: MWSeasonMeta[]; - seasonData: MWSeasonWithEpisodeMeta; - }; - -export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; - -export interface MWQuery { - searchQuery: string; - type: MWMediaType; -} - -export type TTVContentTypes = "movie" | "show"; - -export type TTVSeasonShort = { - title: string; - id: number; - season_number: number; -}; - -export type TTVEpisodeShort = { - title: string; - id: number; - episode_number: number; -}; - -export type TTVMediaResult = { - title: string; - poster?: string; - id: number; - original_release_year?: number; - ttv_entity_id: string; - object_type: TTVContentTypes; - seasons?: TTVSeasonShort[]; -}; - -export type TTVSeasonMetaResult = { - title: string; - id: string; - season_number: number; - episodes: TTVEpisodeShort[]; -}; - -export interface TTVSearchResult { - type: "movie" | "show"; - score: number; - movie?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - imdb: string; - tmdb: number; - }; - }; - show?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - tvdb: number; - imdb: string; - tmdb: number; - }; - }; -} - -export interface DetailedMeta { - meta: MWMediaMeta; - imdbId?: string; - tmdbId?: string; -} - -export interface TMDBShowData { - adult: boolean; - backdrop_path: string | null; - created_by: { - id: number; - credit_id: string; - name: string; - gender: number; - profile_path: string | null; - }[]; - episode_run_time: number[]; - first_air_date: string; - genres: { - id: number; - name: string; - }[]; - homepage: string; - id: number; - in_production: boolean; - languages: string[]; - last_air_date: string; - last_episode_to_air: { - id: number; - name: string; - overview: string; - vote_average: number; - vote_count: number; - air_date: string; - episode_number: number; - production_code: string; - runtime: number | null; - season_number: number; - show_id: number; - still_path: string | null; - } | null; - name: string; - next_episode_to_air: { - id: number; - name: string; - overview: string; - vote_average: number; - vote_count: number; - air_date: string; - episode_number: number; - production_code: string; - runtime: number | null; - season_number: number; - show_id: number; - still_path: string | null; - } | null; - networks: { - id: number; - logo_path: string; - name: string; - origin_country: string; - }[]; - number_of_episodes: number; - number_of_seasons: number; - origin_country: string[]; - original_language: string; - original_name: string; - overview: string; - popularity: number; - poster_path: string | null; - production_companies: { - id: number; - logo_path: string | null; - name: string; - origin_country: string; - }[]; - production_countries: { - iso_3166_1: string; - name: string; - }[]; - seasons: { - air_date: string; - episode_count: number; - id: number; - name: string; - overview: string; - poster_path: string | null; - season_number: number; - }[]; - spoken_languages: { - english_name: string; - iso_639_1: string; - name: string; - }[]; - status: string; - tagline: string; - type: string; - vote_average: number; - vote_count: number; -} - -export interface TMDBMovieData { - adult: boolean; - backdrop_path: string | null; - belongs_to_collection: { - id: number; - name: string; - poster_path: string | null; - backdrop_path: string | null; - } | null; - budget: number; - genres: { - id: number; - name: string; - }[]; - homepage: string | null; - id: number; - imdb_id: string | null; - original_language: string; - original_title: string; - overview: string | null; - popularity: number; - poster_path: string | null; - production_companies: { - id: number; - logo_path: string | null; - name: string; - origin_country: string; - }[]; - production_countries: { - iso_3166_1: string; - name: string; - }[]; - release_date: string; - revenue: number; - runtime: number | null; - spoken_languages: { - english_name: string; - iso_639_1: string; - name: string; - }[]; - status: string; - tagline: string | null; - title: string; - video: boolean; - vote_average: number; - vote_count: number; -} - -export type TMDBMediaDetailsPromise = Promise; - -export interface TMDBMediaStatic { - getMediaDetails( - id: string, - type: MWMediaType.SERIES - ): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; -} diff --git a/src/backend/metadata/types_old.ts b/src/backend/metadata/types_old.ts new file mode 100644 index 00000000..2723fbe7 --- /dev/null +++ b/src/backend/metadata/types_old.ts @@ -0,0 +1,47 @@ +export enum MWMediaType { + MOVIE = "movie", + SERIES = "series", + ANIME = "anime", +} + +export type MWSeasonMeta = { + id: string; + number: number; + title: string; +}; + +export type MWSeasonWithEpisodeMeta = { + id: string; + number: number; + title: string; + episodes: { + id: string; + number: number; + title: string; + }[]; +}; + +type MWMediaMetaBase = { + title: string; + id: string; + year?: string; + poster?: string; +}; + +type MWMediaMetaSpecific = + | { + type: MWMediaType.MOVIE | MWMediaType.ANIME; + seasons: undefined; + } + | { + type: MWMediaType.SERIES; + seasons: MWSeasonMeta[]; + seasonData: MWSeasonWithEpisodeMeta; + }; + +export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; + +export interface MWQuery { + searchQuery: string; + type: MWMediaType; +} diff --git a/src/setup/config.ts b/src/setup/config.ts index f1db01da..c24117bb 100644 --- a/src/setup/config.ts +++ b/src/setup/config.ts @@ -8,6 +8,7 @@ interface Config { TMDB_API_KEY: string; CORS_PROXY_URL: string; NORMAL_ROUTER: boolean; + TRAKT_CLIENT_ID: string; } export interface RuntimeConfig { @@ -18,6 +19,7 @@ export interface RuntimeConfig { TMDB_API_KEY: string; NORMAL_ROUTER: boolean; PROXY_URLS: string[]; + TRAKT_CLIENT_ID: string; } const env: Record = { @@ -28,6 +30,7 @@ const env: Record = { DISCORD_LINK: undefined, CORS_PROXY_URL: import.meta.env.VITE_CORS_PROXY_URL, NORMAL_ROUTER: import.meta.env.VITE_NORMAL_ROUTER, + TRAKT_CLIENT_ID: import.meta.env.VITE_TRAKT_CLIENT_ID, }; const alerts = [] as string[]; @@ -62,5 +65,6 @@ export function conf(): RuntimeConfig { .split(",") .map((v) => v.trim()), NORMAL_ROUTER: getKey("NORMAL_ROUTER", "false") === "true", + TRAKT_CLIENT_ID: getKey("TRAKT_CLIENT_ID"), }; } From c17f8a15e8d8b89a488630b267efc86ab181d04a Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 12 Jun 2023 21:25:24 +0200 Subject: [PATCH 60/96] more refactorings --- src/backend/metadata/getmeta.ts | 8 ++--- src/backend/metadata/justwatch.ts | 42 +++++-------------------- src/backend/metadata/search_old.ts | 9 ++---- src/backend/metadata/types.ts | 49 ++++++++++++++++++++++++++++++ 4 files changed, 62 insertions(+), 46 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 6b3b9a30..c4771451 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -1,13 +1,13 @@ import { FetchError } from "ofetch"; +import { formatJWMeta, mediaTypeToJW } from "./justwatch"; import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, - formatJWMeta, - mediaTypeToJW, -} from "./justwatch"; -import { MWMediaMeta, MWMediaType } from "./types"; + MWMediaMeta, + MWMediaType, +} from "./types"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; type JWExternalIdType = diff --git a/src/backend/metadata/justwatch.ts b/src/backend/metadata/justwatch.ts index 5c79c1e3..27c5aa4c 100644 --- a/src/backend/metadata/justwatch.ts +++ b/src/backend/metadata/justwatch.ts @@ -1,38 +1,10 @@ -import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types"; - -export const JW_API_BASE = "https://apis.justwatch.com"; -export const JW_IMAGE_BASE = "https://images.justwatch.com"; - -export type JWContentTypes = "movie" | "show"; - -export type JWSeasonShort = { - title: string; - id: number; - season_number: number; -}; - -export type JWEpisodeShort = { - title: string; - id: number; - episode_number: number; -}; - -export type JWMediaResult = { - title: string; - poster?: string; - id: number; - original_release_year?: number; - jw_entity_id: string; - object_type: JWContentTypes; - seasons?: JWSeasonShort[]; -}; - -export type JWSeasonMetaResult = { - title: string; - id: string; - season_number: number; - episodes: JWEpisodeShort[]; -}; +import { + JWContentTypes, + JWMediaResult, + JWSeasonMetaResult, + JW_IMAGE_BASE, +} from "./types"; +import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types_old"; export function mediaTypeToJW(type: MWMediaType): JWContentTypes { if (type === MWMediaType.MOVIE) return "movie"; diff --git a/src/backend/metadata/search_old.ts b/src/backend/metadata/search_old.ts index 05be994d..f12f62d2 100644 --- a/src/backend/metadata/search_old.ts +++ b/src/backend/metadata/search_old.ts @@ -1,12 +1,7 @@ import { SimpleCache } from "@/utils/cache"; -import { - JWContentTypes, - JWMediaResult, - JW_API_BASE, - formatJWMeta, - mediaTypeToJW, -} from "./justwatch"; +import { formatJWMeta, mediaTypeToJW } from "./justwatch"; +import { JWContentTypes, JWMediaResult, JW_API_BASE } from "./types"; import { MWMediaMeta, MWQuery } from "./types_old"; import { proxiedFetch } from "../helpers/fetch"; diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index f954ff6a..9e87d49d 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -262,3 +262,52 @@ export interface TMDBMediaStatic { getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; } + +export type JWContentTypes = "movie" | "show"; + +export type JWSearchQuery = { + content_types: JWContentTypes[]; + page: number; + page_size: number; + query: string; +}; + +export type JWPage = { + items: T[]; + page: number; + page_size: number; + total_pages: number; + total_results: number; +}; + +export const JW_API_BASE = "https://apis.justwatch.com"; +export const JW_IMAGE_BASE = "https://images.justwatch.com"; + +export type JWSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type JWEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type JWMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + jw_entity_id: string; + object_type: JWContentTypes; + seasons?: JWSeasonShort[]; +}; + +export type JWSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: JWEpisodeShort[]; +}; From 3af98373fbafc7d979492fb87d21033bd2241890 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 10:41:54 +0200 Subject: [PATCH 61/96] finish initial refactor --- src/backend/metadata/getmeta.ts | 55 +++++++++++++++++++++++++++++++++ src/backend/metadata/search.ts | 1 + src/backend/metadata/tmdb.ts | 27 ---------------- src/backend/metadata/trakttv.ts | 45 ++++++++++++++++++++------- src/backend/metadata/types.ts | 12 +++++++ 5 files changed, 101 insertions(+), 39 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index c4771451..e428199e 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -1,12 +1,17 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; +import { Tmdb } from "./tmdb"; +import { Trakt, formatTTVMeta } from "./trakttv"; import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, MWMediaMeta, MWMediaType, + TMDBMovieData, + TMDBShowData, + TTVSeasonMetaResult, } from "./types"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; @@ -37,6 +42,56 @@ export async function getMetaFromId( type: MWMediaType, id: string, seasonId?: string +): Promise { + const result = await Trakt.searchById(id, mediaTypeToJW(type)); + if (!result) return null; + const details = await Tmdb.getMediaDetails(id, type); + + if (!details) return null; + + let imdbId; + if (type === MWMediaType.MOVIE) { + imdbId = (details as TMDBMovieData).imdb_id ?? undefined; + } + + let seasonData: TTVSeasonMetaResult | undefined; + + if (type === MWMediaType.SERIES) { + const seasons = (details as TMDBShowData).seasons; + const season = + seasons?.find((v) => v.id.toString() === seasonId) ?? seasons?.[0]; + + const episodes = await Trakt.getEpisodes( + result.ttv_entity_id, + season?.season_number ?? 1 + ); + + if (season && episodes) { + seasonData = { + id: season.id.toString(), + season_number: season.season_number, + title: season.name, + episodes, + }; + } + } + + const meta = formatTTVMeta(result, seasonData); + if (!meta) return null; + + console.log(meta); + + return { + meta, + imdbId, + tmdbId: id, + }; +} + +export async function getLegacyMetaFromId( + type: MWMediaType, + id: string, + seasonId?: string ): Promise { const queryType = mediaTypeToJW(type); diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 4506514a..8eb246b7 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -16,6 +16,7 @@ export async function searchForMedia(query: MWQuery): Promise { const contentType = mediaTypeToTTV(type); const results = await Trakt.search(searchQuery, contentType); + console.log(results[0]); cache.set(query, results, 3600); return results; } diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 460e13b4..3aa1821f 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -47,31 +47,4 @@ export abstract class Tmdb { public static getMediaPoster(posterPath: string | null): string | undefined { if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; } - - /* public static async getMetaFromId( - type: MWMediaType, - id: string, - seasonId?: string - ): Promise { - console.log("getMetaFromId", type, id, seasonId); - - const details = await Tmdb.getMediaDetails(id, type); - - if (!details) return null; - - let imdbId; - if (type === MWMediaType.MOVIE) { - imdbId = (details as TMDBMovieData).imdb_id ?? undefined; - } - - if (!meta.length) return null; - - console.log(meta); - - return { - meta, - imdbId, - tmdbId: id, - }; - } */ } diff --git a/src/backend/metadata/trakttv.ts b/src/backend/metadata/trakttv.ts index 5fb67a17..ae50aefe 100644 --- a/src/backend/metadata/trakttv.ts +++ b/src/backend/metadata/trakttv.ts @@ -2,12 +2,13 @@ import { conf } from "@/setup/config"; import { Tmdb } from "./tmdb"; import { - DetailedMeta, MWMediaMeta, MWMediaType, MWSeasonMeta, TMDBShowData, TTVContentTypes, + TTVEpisodeResult, + TTVEpisodeShort, TTVMediaResult, TTVSearchResult, TTVSeasonMetaResult, @@ -69,14 +70,14 @@ export function formatTTVMeta( } export function TTVMediaToId(media: MWMediaMeta): string { - return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); + return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); } export function decodeTTVId( paramId: string ): { id: string; type: MWMediaType } | null { const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "TTV") return null; + if (prefix !== "MW") return null; let mediaType; try { mediaType = TTVMediaToMediaType(type); @@ -101,7 +102,6 @@ export async function formatTTVSearchResult( media.ids.tmdb.toString(), TTVMediaToMediaType(result.type) ); - console.log(details); const seasons = type === MWMediaType.SERIES @@ -115,7 +115,7 @@ export async function formatTTVSearchResult( return { title: media.title, poster: Tmdb.getMediaPoster(details.poster_path), - id: media.ids.trakt, + id: media.ids.tmdb, original_release_year: media.year, ttv_entity_id: media.ids.slug, object_type: mediaTypeToTTV(type), @@ -155,12 +155,33 @@ export abstract class Trakt { return formatted.map((v) => formatTTVMeta(v)); } - public static async getMetaFromId( - type: MWMediaType, - id: string, - seasonId?: string - ): Promise { - console.log("getMetaFromId", type, id, seasonId); - return null; + public static async searchById( + tmdbId: string, + type: "movie" | "show" + ): Promise { + const data = await Trakt.get( + `/search/tmdb/${tmdbId}?type=${type}` + ); + + const formatted = await Promise.all( + // eslint-disable-next-line no-return-await + data.map(async (v) => await formatTTVSearchResult(v)) + ); + return formatted[0]; + } + + public static async getEpisodes( + slug: string, + season: number + ): Promise { + const data = await Trakt.get( + `/shows/${slug}/seasons/${season}` + ); + + return data.map((e) => ({ + id: e.ids.tmdb, + episode_number: e.number, + title: e.title, + })); } } diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index 9e87d49d..07671e39 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -311,3 +311,15 @@ export type JWSeasonMetaResult = { season_number: number; episodes: JWEpisodeShort[]; }; + +export interface TTVEpisodeResult { + season: number; + number: number; + title: string; + ids: { + trakt: number; + tvdb: number; + imdb: string; + tmdb: number; + }; +} From 70f835538683e3dba953b443a2b17538e9567764 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 11:01:07 +0200 Subject: [PATCH 62/96] refactor url prefix --- src/backend/metadata/getmeta.ts | 28 ++++++++++++++++++- src/components/media/MediaCard.tsx | 4 +-- .../popouts/EpisodeSelectionPopout.tsx | 5 ++-- src/views/media/MediaView.tsx | 9 ++++-- 4 files changed, 37 insertions(+), 9 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index e428199e..26464299 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -2,7 +2,12 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; import { Tmdb } from "./tmdb"; -import { Trakt, formatTTVMeta } from "./trakttv"; +import { + TTVMediaToMediaType, + Trakt, + formatTTVMeta, + mediaTypeToTTV, +} from "./trakttv"; import { JWMediaResult, JWSeasonMetaResult, @@ -137,3 +142,24 @@ export async function getLegacyMetaFromId( tmdbId, }; } + +export function MWMediaToId(media: MWMediaMeta): string { + return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); +} + +export function decodeMWId( + paramId: string +): { id: string; type: MWMediaType } | null { + const [prefix, type, id] = paramId.split("-", 3); + if (prefix !== "MW") return null; + let mediaType; + try { + mediaType = TTVMediaToMediaType(type); + } catch { + return null; + } + return { + type: mediaType, + id, + }; +} diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index 22865717..3bac4d08 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -1,7 +1,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { JWMediaToId } from "@/backend/metadata/justwatch"; +import { MWMediaToId } from "@/backend/metadata/getmeta"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -132,7 +132,7 @@ export function MediaCard(props: MediaCardProps) { const canLink = props.linkable && !props.closable; let link = canLink - ? `/media/${encodeURIComponent(JWMediaToId(props.media))}` + ? `/media/${encodeURIComponent(MWMediaToId(props.media))}` : "#"; if (canLink && props.series) link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index c80045bd..3d0b431b 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -2,8 +2,7 @@ import { useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; -import { getMetaFromId } from "@/backend/metadata/getmeta"; -import { decodeJWId } from "@/backend/metadata/justwatch"; +import { decodeMWId, getMetaFromId } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; @@ -45,7 +44,7 @@ export function EpisodeSelectionPopout() { seasonId: sId, season: undefined, }); - reqSeasonMeta(decodeJWId(params.media)?.id as string, sId).then((v) => { + reqSeasonMeta(decodeMWId(params.media)?.id as string, sId).then((v) => { if (v?.meta.type !== MWMediaType.SERIES) return; setCurrentVisibleSeason({ seasonId: sId, diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index c55211c7..1a4709e3 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -4,8 +4,11 @@ import { useTranslation } from "react-i18next"; import { useHistory, useParams } from "react-router-dom"; import { MWStream } from "@/backend/helpers/streams"; -import { DetailedMeta, getMetaFromId } from "@/backend/metadata/getmeta"; -import { decodeJWId } from "@/backend/metadata/justwatch"; +import { + DetailedMeta, + decodeMWId, + getMetaFromId, +} from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icons } from "@/components/Icon"; @@ -181,7 +184,7 @@ export function MediaView() { const [selected, setSelected] = useState(null); const [exec, loading, error] = useLoading( async (mediaParams: string, seasonId?: string) => { - const data = decodeJWId(mediaParams); + const data = decodeMWId(mediaParams); if (!data) return null; return getMetaFromId(data.type, data.id, seasonId); } From 879271c23976126b41a73f439f01cacc5140edbf Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 14:06:37 +0200 Subject: [PATCH 63/96] implement legacy url conversion --- src/backend/metadata/getmeta.ts | 15 +++++++++++++++ src/setup/App.tsx | 18 +++++++++++++++++- 2 files changed, 32 insertions(+), 1 deletion(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 26464299..b6f90f26 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -163,3 +163,18 @@ export function decodeMWId( id, }; } + +export async function convertLegacyUrl( + url: string +): Promise { + if (url.startsWith("/media/JW")) { + const urlParts = url.split("/").slice(2); + const [, type, id] = urlParts[0].split("-", 3); + const meta = await getLegacyMetaFromId(TTVMediaToMediaType(type), id); + if (!meta) return undefined; + const tmdbId = meta.tmdbId; + if (!tmdbId) return undefined; + return `/media/MW-${type}-${tmdbId}`; + } + return undefined; +} diff --git a/src/setup/App.tsx b/src/setup/App.tsx index 992549e0..7be4d581 100644 --- a/src/setup/App.tsx +++ b/src/setup/App.tsx @@ -1,6 +1,13 @@ import { lazy } from "react"; -import { Redirect, Route, Switch } from "react-router-dom"; +import { + Redirect, + Route, + Switch, + useHistory, + useLocation, +} from "react-router-dom"; +import { convertLegacyUrl } from "@/backend/metadata/getmeta"; import { MWMediaType } from "@/backend/metadata/types"; import { BannerContextProvider } from "@/hooks/useBanner"; import { Layout } from "@/setup/Layout"; @@ -13,6 +20,15 @@ import { V2MigrationView } from "@/views/other/v2Migration"; import { SearchView } from "@/views/search/SearchView"; function App() { + const location = useLocation(); + const history = useHistory(); + + // Call the conversion function and redirect if necessary + convertLegacyUrl(location.pathname).then((convertedUrl) => { + if (convertedUrl) { + history.replace(convertedUrl); + } + }); return ( From b5c330d4e3cd4b4aa77b3f3dc7cfcba02a785f7b Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 14:20:33 +0200 Subject: [PATCH 64/96] refactor to initial prefix choice --- src/backend/metadata/getmeta.ts | 10 +++++----- src/components/media/MediaCard.tsx | 4 ++-- .../components/popouts/EpisodeSelectionPopout.tsx | 4 ++-- src/views/media/MediaView.tsx | 4 ++-- 4 files changed, 11 insertions(+), 11 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index b6f90f26..82b3c20b 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -143,15 +143,15 @@ export async function getLegacyMetaFromId( }; } -export function MWMediaToId(media: MWMediaMeta): string { - return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); +export function TTVMediaToId(media: MWMediaMeta): string { + return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); } -export function decodeMWId( +export function decodeTTVId( paramId: string ): { id: string; type: MWMediaType } | null { const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "MW") return null; + if (prefix !== "TTV") return null; let mediaType; try { mediaType = TTVMediaToMediaType(type); @@ -174,7 +174,7 @@ export async function convertLegacyUrl( if (!meta) return undefined; const tmdbId = meta.tmdbId; if (!tmdbId) return undefined; - return `/media/MW-${type}-${tmdbId}`; + return `/media/TTV-${type}-${tmdbId}`; } return undefined; } diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index 3bac4d08..b87654a9 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -1,7 +1,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { MWMediaToId } from "@/backend/metadata/getmeta"; +import { TTVMediaToId } from "@/backend/metadata/getmeta"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -132,7 +132,7 @@ export function MediaCard(props: MediaCardProps) { const canLink = props.linkable && !props.closable; let link = canLink - ? `/media/${encodeURIComponent(MWMediaToId(props.media))}` + ? `/media/${encodeURIComponent(TTVMediaToId(props.media))}` : "#"; if (canLink && props.series) link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index 3d0b431b..7ea6c5a3 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -2,7 +2,7 @@ import { useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; -import { decodeMWId, getMetaFromId } from "@/backend/metadata/getmeta"; +import { decodeTTVId, getMetaFromId } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; @@ -44,7 +44,7 @@ export function EpisodeSelectionPopout() { seasonId: sId, season: undefined, }); - reqSeasonMeta(decodeMWId(params.media)?.id as string, sId).then((v) => { + reqSeasonMeta(decodeTTVId(params.media)?.id as string, sId).then((v) => { if (v?.meta.type !== MWMediaType.SERIES) return; setCurrentVisibleSeason({ seasonId: sId, diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index 1a4709e3..c0e81455 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -6,7 +6,7 @@ import { useHistory, useParams } from "react-router-dom"; import { MWStream } from "@/backend/helpers/streams"; import { DetailedMeta, - decodeMWId, + decodeTTVId, getMetaFromId, } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; @@ -184,7 +184,7 @@ export function MediaView() { const [selected, setSelected] = useState(null); const [exec, loading, error] = useLoading( async (mediaParams: string, seasonId?: string) => { - const data = decodeMWId(mediaParams); + const data = decodeTTVId(mediaParams); if (!data) return null; return getMetaFromId(data.type, data.id, seasonId); } From 8da155ba2bfad94160469118804665e34a80c8a1 Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 14:25:31 +0200 Subject: [PATCH 65/96] cleanup --- src/backend/metadata/justwatch.ts | 4 ++- src/backend/metadata/search_old.ts | 54 ------------------------------ src/backend/metadata/types_old.ts | 47 -------------------------- 3 files changed, 3 insertions(+), 102 deletions(-) delete mode 100644 src/backend/metadata/search_old.ts delete mode 100644 src/backend/metadata/types_old.ts diff --git a/src/backend/metadata/justwatch.ts b/src/backend/metadata/justwatch.ts index 27c5aa4c..857ff006 100644 --- a/src/backend/metadata/justwatch.ts +++ b/src/backend/metadata/justwatch.ts @@ -3,8 +3,10 @@ import { JWMediaResult, JWSeasonMetaResult, JW_IMAGE_BASE, + MWMediaMeta, + MWMediaType, + MWSeasonMeta, } from "./types"; -import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types_old"; export function mediaTypeToJW(type: MWMediaType): JWContentTypes { if (type === MWMediaType.MOVIE) return "movie"; diff --git a/src/backend/metadata/search_old.ts b/src/backend/metadata/search_old.ts deleted file mode 100644 index f12f62d2..00000000 --- a/src/backend/metadata/search_old.ts +++ /dev/null @@ -1,54 +0,0 @@ -import { SimpleCache } from "@/utils/cache"; - -import { formatJWMeta, mediaTypeToJW } from "./justwatch"; -import { JWContentTypes, JWMediaResult, JW_API_BASE } from "./types"; -import { MWMediaMeta, MWQuery } from "./types_old"; -import { proxiedFetch } from "../helpers/fetch"; - -const cache = new SimpleCache(); -cache.setCompare((a, b) => { - return a.type === b.type && a.searchQuery.trim() === b.searchQuery.trim(); -}); -cache.initialize(); - -type JWSearchQuery = { - content_types: JWContentTypes[]; - page: number; - page_size: number; - query: string; -}; - -type JWPage = { - items: T[]; - page: number; - page_size: number; - total_pages: number; - total_results: number; -}; - -export async function searchForMedia(query: MWQuery): Promise { - if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; - const { searchQuery, type } = query; - - const contentType = mediaTypeToJW(type); - const body: JWSearchQuery = { - content_types: [contentType], - page: 1, - query: searchQuery, - page_size: 40, - }; - - const data = await proxiedFetch>( - "/content/titles/en_US/popular", - { - baseURL: JW_API_BASE, - params: { - body: JSON.stringify(body), - }, - } - ); - - const returnData = data.items.map((v) => formatJWMeta(v)); - cache.set(query, returnData, 3600); // cache for an hour - return returnData; -} diff --git a/src/backend/metadata/types_old.ts b/src/backend/metadata/types_old.ts deleted file mode 100644 index 2723fbe7..00000000 --- a/src/backend/metadata/types_old.ts +++ /dev/null @@ -1,47 +0,0 @@ -export enum MWMediaType { - MOVIE = "movie", - SERIES = "series", - ANIME = "anime", -} - -export type MWSeasonMeta = { - id: string; - number: number; - title: string; -}; - -export type MWSeasonWithEpisodeMeta = { - id: string; - number: number; - title: string; - episodes: { - id: string; - number: number; - title: string; - }[]; -}; - -type MWMediaMetaBase = { - title: string; - id: string; - year?: string; - poster?: string; -}; - -type MWMediaMetaSpecific = - | { - type: MWMediaType.MOVIE | MWMediaType.ANIME; - seasons: undefined; - } - | { - type: MWMediaType.SERIES; - seasons: MWSeasonMeta[]; - seasonData: MWSeasonWithEpisodeMeta; - }; - -export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; - -export interface MWQuery { - searchQuery: string; - type: MWMediaType; -} From 46bd20f71866241a09756d46b111f1ef416fe02b Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 21:23:47 +0200 Subject: [PATCH 66/96] refactor everything to use tmdb exclusively --- src/backend/metadata/getmeta.ts | 61 +++--- src/backend/metadata/search.ts | 20 +- src/backend/metadata/tmdb.ts | 159 ++++++++++++++- src/backend/metadata/trakttv.ts | 187 ------------------ src/backend/metadata/types.ts | 140 +++++++++---- src/components/media/MediaCard.tsx | 4 +- .../popouts/EpisodeSelectionPopout.tsx | 4 +- src/views/media/MediaView.tsx | 4 +- 8 files changed, 315 insertions(+), 264 deletions(-) delete mode 100644 src/backend/metadata/trakttv.ts diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 82b3c20b..777fae42 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -1,22 +1,22 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; -import { Tmdb } from "./tmdb"; import { - TTVMediaToMediaType, - Trakt, - formatTTVMeta, - mediaTypeToTTV, -} from "./trakttv"; + TMDBMediaToMediaType, + Tmdb, + formatTMDBMeta, + mediaTypeToTMDB, +} from "./tmdb"; import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, MWMediaMeta, MWMediaType, + TMDBMediaResult, TMDBMovieData, + TMDBSeasonMetaResult, TMDBShowData, - TTVSeasonMetaResult, } from "./types"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; @@ -48,9 +48,7 @@ export async function getMetaFromId( id: string, seasonId?: string ): Promise { - const result = await Trakt.searchById(id, mediaTypeToJW(type)); - if (!result) return null; - const details = await Tmdb.getMediaDetails(id, type); + const details = await Tmdb.getMediaDetails(id, mediaTypeToTMDB(type)); if (!details) return null; @@ -59,15 +57,15 @@ export async function getMetaFromId( imdbId = (details as TMDBMovieData).imdb_id ?? undefined; } - let seasonData: TTVSeasonMetaResult | undefined; + let seasonData: TMDBSeasonMetaResult | undefined; if (type === MWMediaType.SERIES) { const seasons = (details as TMDBShowData).seasons; const season = seasons?.find((v) => v.id.toString() === seasonId) ?? seasons?.[0]; - const episodes = await Trakt.getEpisodes( - result.ttv_entity_id, + const episodes = await Tmdb.getEpisodes( + details.id.toString(), season?.season_number ?? 1 ); @@ -81,10 +79,27 @@ export async function getMetaFromId( } } - const meta = formatTTVMeta(result, seasonData); - if (!meta) return null; + const tmdbmeta: TMDBMediaResult = { + id: details.id, + title: + type === MWMediaType.MOVIE + ? (details as TMDBMovieData).title + : (details as TMDBShowData).name, + object_type: mediaTypeToTMDB(type), + seasons: (details as TMDBShowData).seasons.map((v) => ({ + id: v.id, + season_number: v.season_number, + title: v.name, + })), + poster: (details as TMDBMovieData).poster_path ?? undefined, + original_release_year: + type === MWMediaType.MOVIE + ? Number((details as TMDBMovieData).release_date?.split("-")[0]) + : Number((details as TMDBShowData).first_air_date?.split("-")[0]), + }; - console.log(meta); + const meta = formatTMDBMeta(tmdbmeta, seasonData); + if (!meta) return null; return { meta, @@ -143,18 +158,18 @@ export async function getLegacyMetaFromId( }; } -export function TTVMediaToId(media: MWMediaMeta): string { - return ["TTV", mediaTypeToTTV(media.type), media.id].join("-"); +export function TMDBMediaToId(media: MWMediaMeta): string { + return ["tmdb", mediaTypeToTMDB(media.type), media.id].join("-"); } -export function decodeTTVId( +export function decodeTMDBId( paramId: string ): { id: string; type: MWMediaType } | null { const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "TTV") return null; + if (prefix !== "tmdb") return null; let mediaType; try { - mediaType = TTVMediaToMediaType(type); + mediaType = TMDBMediaToMediaType(type); } catch { return null; } @@ -170,11 +185,11 @@ export async function convertLegacyUrl( if (url.startsWith("/media/JW")) { const urlParts = url.split("/").slice(2); const [, type, id] = urlParts[0].split("-", 3); - const meta = await getLegacyMetaFromId(TTVMediaToMediaType(type), id); + const meta = await getLegacyMetaFromId(TMDBMediaToMediaType(type), id); if (!meta) return undefined; const tmdbId = meta.tmdbId; if (!tmdbId) return undefined; - return `/media/TTV-${type}-${tmdbId}`; + return `/media/tmdb-${type}-${tmdbId}`; } return undefined; } diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 8eb246b7..31b2c682 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -1,6 +1,11 @@ import { SimpleCache } from "@/utils/cache"; -import { Trakt, mediaTypeToTTV } from "./trakttv"; +import { + Tmdb, + formatTMDBMeta, + formatTMDBSearchResult, + mediaTypeToTMDB, +} from "./tmdb"; import { MWMediaMeta, MWQuery } from "./types"; const cache = new SimpleCache(); @@ -13,10 +18,17 @@ export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const contentType = mediaTypeToTTV(type); + const data = await Tmdb.searchMedia(searchQuery, mediaTypeToTMDB(type)); + const results = await Promise.all( + data.results.map(async (v) => { + const formattedResult = await formatTMDBSearchResult( + v, + mediaTypeToTMDB(type) + ); + return formatTMDBMeta(formattedResult); + }) + ); - const results = await Trakt.search(searchQuery, contentType); - console.log(results[0]); cache.set(query, results, 3600); return results; } diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 3aa1821f..f01709bb 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -1,13 +1,100 @@ import { conf } from "@/setup/config"; import { + MWMediaMeta, MWMediaType, + MWSeasonMeta, + TMDBContentTypes, + TMDBEpisodeShort, + TMDBMediaResult, TMDBMediaStatic, TMDBMovieData, + TMDBMovieResponse, + TMDBMovieResult, + TMDBSearchResultStatic, + TMDBSeason, + TMDBSeasonMetaResult, TMDBShowData, + TMDBShowResponse, + TMDBShowResult, } from "./types"; import { mwFetch } from "../helpers/fetch"; +export function mediaTypeToTMDB(type: MWMediaType): TMDBContentTypes { + if (type === MWMediaType.MOVIE) return "movie"; + if (type === MWMediaType.SERIES) return "show"; + throw new Error("unsupported type"); +} + +export function TMDBMediaToMediaType(type: string): MWMediaType { + if (type === "movie") return MWMediaType.MOVIE; + if (type === "show") return MWMediaType.SERIES; + throw new Error("unsupported type"); +} + +export function formatTMDBMeta( + media: TMDBMediaResult, + season?: TMDBSeasonMetaResult +): MWMediaMeta { + const type = TMDBMediaToMediaType(media.object_type); + let seasons: undefined | MWSeasonMeta[]; + if (type === MWMediaType.SERIES) { + seasons = media.seasons + ?.sort((a, b) => a.season_number - b.season_number) + .map( + (v): MWSeasonMeta => ({ + title: v.title, + id: v.id.toString(), + number: v.season_number, + }) + ); + } + + return { + title: media.title, + id: media.id.toString(), + year: media.original_release_year?.toString(), + poster: media.poster, + type, + seasons: seasons as any, + seasonData: season + ? ({ + id: season.id.toString(), + number: season.season_number, + title: season.title, + episodes: season.episodes + .sort((a, b) => a.episode_number - b.episode_number) + .map((v) => ({ + id: v.id.toString(), + number: v.episode_number, + title: v.title, + })), + } as any) + : (undefined as any), + }; +} + +export function TMDBMediaToId(media: MWMediaMeta): string { + return ["tmdb", mediaTypeToTMDB(media.type), media.id].join("-"); +} + +export function decodeTMDBId( + paramId: string +): { id: string; type: MWMediaType } | null { + const [prefix, type, id] = paramId.split("-", 3); + if (prefix !== "tmdb") return null; + let mediaType; + try { + mediaType = TMDBMediaToMediaType(type); + } catch { + return null; + } + return { + type: mediaType, + id, + }; +} + export abstract class Tmdb { private static baseURL = "https://api.themoviedb.org/3"; @@ -24,9 +111,33 @@ export abstract class Tmdb { return res; } + public static searchMedia: TMDBSearchResultStatic["searchMedia"] = async ( + query: string, + type: TMDBContentTypes + ) => { + let data; + + switch (type) { + case "movie": + data = await Tmdb.get( + `search/movie?query=${query}&include_adult=true&language=en-US&page=1` + ); + break; + case "show": + data = await Tmdb.get( + `search/tv?query=${query}&include_adult=true&language=en-US&page=1` + ); + break; + default: + throw new Error("Invalid media type"); + } + + return data; + }; + public static getMediaDetails: TMDBMediaStatic["getMediaDetails"] = async ( id: string, - type: MWMediaType + type: TMDBContentTypes ) => { let data; @@ -34,7 +145,7 @@ export abstract class Tmdb { case "movie": data = await Tmdb.get(`/movie/${id}`); break; - case "series": + case "show": data = await Tmdb.get(`/tv/${id}`); break; default: @@ -47,4 +158,48 @@ export abstract class Tmdb { public static getMediaPoster(posterPath: string | null): string | undefined { if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; } + + public static async getEpisodes( + id: string, + season: number + ): Promise { + const data = await Tmdb.get(`/tv/${id}/season/${season}`); + return data.episodes.map((e) => ({ + id: e.id, + episode_number: e.episode_number, + title: e.name, + })); + } +} + +export async function formatTMDBSearchResult( + result: TMDBShowResult | TMDBMovieResult, + mediatype: TMDBContentTypes +): Promise { + const type = TMDBMediaToMediaType(mediatype); + const details = await Tmdb.getMediaDetails(result.id.toString(), mediatype); + + const seasons = + type === MWMediaType.SERIES + ? (details as TMDBShowData).seasons?.map((v) => ({ + id: v.id, + title: v.name, + season_number: v.season_number, + })) + : undefined; + + return { + title: + type === MWMediaType.SERIES + ? (result as TMDBShowResult).name + : (result as TMDBMovieResult).title, + poster: Tmdb.getMediaPoster(details.poster_path), + id: result.id, + original_release_year: + type === MWMediaType.SERIES + ? Number((result as TMDBShowResult).first_air_date?.split("-")[0]) + : Number((result as TMDBMovieResult).release_date?.split("-")[0]), + object_type: mediaTypeToTMDB(type), + seasons, + }; } diff --git a/src/backend/metadata/trakttv.ts b/src/backend/metadata/trakttv.ts deleted file mode 100644 index ae50aefe..00000000 --- a/src/backend/metadata/trakttv.ts +++ /dev/null @@ -1,187 +0,0 @@ -import { conf } from "@/setup/config"; - -import { Tmdb } from "./tmdb"; -import { - MWMediaMeta, - MWMediaType, - MWSeasonMeta, - TMDBShowData, - TTVContentTypes, - TTVEpisodeResult, - TTVEpisodeShort, - TTVMediaResult, - TTVSearchResult, - TTVSeasonMetaResult, -} from "./types"; -import { mwFetch } from "../helpers/fetch"; - -export function mediaTypeToTTV(type: MWMediaType): TTVContentTypes { - if (type === MWMediaType.MOVIE) return "movie"; - if (type === MWMediaType.SERIES) return "show"; - throw new Error("unsupported type"); -} - -export function TTVMediaToMediaType(type: string): MWMediaType { - if (type === "movie") return MWMediaType.MOVIE; - if (type === "show") return MWMediaType.SERIES; - throw new Error("unsupported type"); -} - -export function formatTTVMeta( - media: TTVMediaResult, - season?: TTVSeasonMetaResult -): MWMediaMeta { - const type = TTVMediaToMediaType(media.object_type); - let seasons: undefined | MWSeasonMeta[]; - if (type === MWMediaType.SERIES) { - seasons = media.seasons - ?.sort((a, b) => a.season_number - b.season_number) - .map( - (v): MWSeasonMeta => ({ - title: v.title, - id: v.id.toString(), - number: v.season_number, - }) - ); - } - - return { - title: media.title, - id: media.id.toString(), - year: media.original_release_year?.toString(), - poster: media.poster, - type, - seasons: seasons as any, - seasonData: season - ? ({ - id: season.id.toString(), - number: season.season_number, - title: season.title, - episodes: season.episodes - .sort((a, b) => a.episode_number - b.episode_number) - .map((v) => ({ - id: v.id.toString(), - number: v.episode_number, - title: v.title, - })), - } as any) - : (undefined as any), - }; -} - -export function TTVMediaToId(media: MWMediaMeta): string { - return ["MW", mediaTypeToTTV(media.type), media.id].join("-"); -} - -export function decodeTTVId( - paramId: string -): { id: string; type: MWMediaType } | null { - const [prefix, type, id] = paramId.split("-", 3); - if (prefix !== "MW") return null; - let mediaType; - try { - mediaType = TTVMediaToMediaType(type); - } catch { - return null; - } - return { - type: mediaType, - id, - }; -} - -export async function formatTTVSearchResult( - result: TTVSearchResult -): Promise { - const type = TTVMediaToMediaType(result.type); - const media = result[result.type]; - - if (!media) throw new Error("invalid result"); - - const details = await Tmdb.getMediaDetails( - media.ids.tmdb.toString(), - TTVMediaToMediaType(result.type) - ); - - const seasons = - type === MWMediaType.SERIES - ? (details as TMDBShowData).seasons?.map((v) => ({ - id: v.id, - title: v.name, - season_number: v.season_number, - })) - : undefined; - - return { - title: media.title, - poster: Tmdb.getMediaPoster(details.poster_path), - id: media.ids.tmdb, - original_release_year: media.year, - ttv_entity_id: media.ids.slug, - object_type: mediaTypeToTTV(type), - seasons, - }; -} - -export abstract class Trakt { - private static baseURL = "https://api.trakt.tv"; - - private static headers = { - "Content-Type": "application/json", - "trakt-api-version": "2", - "trakt-api-key": conf().TRAKT_CLIENT_ID, - }; - - private static async get(url: string): Promise { - const res = await mwFetch(url, { - headers: Trakt.headers, - baseURL: Trakt.baseURL, - }); - return res; - } - - public static async search( - query: string, - type: "movie" | "show" - ): Promise { - const data = await Trakt.get( - `/search/${type}?query=${encodeURIComponent(query)}` - ); - - const formatted = await Promise.all( - // eslint-disable-next-line no-return-await - data.map(async (v) => await formatTTVSearchResult(v)) - ); - return formatted.map((v) => formatTTVMeta(v)); - } - - public static async searchById( - tmdbId: string, - type: "movie" | "show" - ): Promise { - const data = await Trakt.get( - `/search/tmdb/${tmdbId}?type=${type}` - ); - - const formatted = await Promise.all( - // eslint-disable-next-line no-return-await - data.map(async (v) => await formatTTVSearchResult(v)) - ); - return formatted[0]; - } - - public static async getEpisodes( - slug: string, - season: number - ): Promise { - const data = await Trakt.get( - `/shows/${slug}/seasons/${season}` - ); - - return data.map((e) => ({ - id: e.ids.tmdb, - episode_number: e.number, - title: e.title, - })); - } -} diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index 07671e39..e23d9a5b 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -46,63 +46,36 @@ export interface MWQuery { type: MWMediaType; } -export type TTVContentTypes = "movie" | "show"; +export type TMDBContentTypes = "movie" | "show"; -export type TTVSeasonShort = { +export type TMDBSeasonShort = { title: string; id: number; season_number: number; }; -export type TTVEpisodeShort = { +export type TMDBEpisodeShort = { title: string; id: number; episode_number: number; }; -export type TTVMediaResult = { +export type TMDBMediaResult = { title: string; poster?: string; id: number; original_release_year?: number; - ttv_entity_id: string; - object_type: TTVContentTypes; - seasons?: TTVSeasonShort[]; + object_type: TMDBContentTypes; + seasons?: TMDBSeasonShort[]; }; -export type TTVSeasonMetaResult = { +export type TMDBSeasonMetaResult = { title: string; id: string; season_number: number; - episodes: TTVEpisodeShort[]; + episodes: TMDBEpisodeShort[]; }; -export interface TTVSearchResult { - type: "movie" | "show"; - score: number; - movie?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - imdb: string; - tmdb: number; - }; - }; - show?: { - title: string; - year: number; - ids: { - trakt: number; - slug: string; - tvdb: number; - imdb: string; - tmdb: number; - }; - }; -} - export interface DetailedMeta { meta: MWMediaMeta; imdbId?: string; @@ -255,12 +228,9 @@ export interface TMDBMovieData { export type TMDBMediaDetailsPromise = Promise; export interface TMDBMediaStatic { - getMediaDetails( - id: string, - type: MWMediaType.SERIES - ): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType.MOVIE): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: MWMediaType): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: "show"): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: "movie"): TMDBMediaDetailsPromise; + getMediaDetails(id: string, type: TMDBContentTypes): TMDBMediaDetailsPromise; } export type JWContentTypes = "movie" | "show"; @@ -312,7 +282,7 @@ export type JWSeasonMetaResult = { episodes: JWEpisodeShort[]; }; -export interface TTVEpisodeResult { +export interface TMDBEpisodeResult { season: number; number: number; title: string; @@ -323,3 +293,89 @@ export interface TTVEpisodeResult { tmdb: number; }; } + +export interface TMDBShowResult { + adult: boolean; + backdrop_path: string | null; + genre_ids: number[]; + id: number; + origin_country: string[]; + original_language: string; + original_name: string; + overview: string; + popularity: number; + poster_path: string | null; + first_air_date: string; + name: string; + vote_average: number; + vote_count: number; +} + +export interface TMDBShowResponse { + page: number; + results: TMDBShowResult[]; + total_pages: number; + total_results: number; +} + +export interface TMDBMovieResult { + adult: boolean; + backdrop_path: string | null; + genre_ids: number[]; + id: number; + original_language: string; + original_title: string; + overview: string; + popularity: number; + poster_path: string | null; + release_date: string; + title: string; + video: boolean; + vote_average: number; + vote_count: number; +} + +export interface TMDBMovieResponse { + page: number; + results: TMDBMovieResult[]; + total_pages: number; + total_results: number; +} + +export type TMDBSearchResultsPromise = Promise< + TMDBShowResponse | TMDBMovieResponse +>; + +export interface TMDBSearchResultStatic { + searchMedia(query: string, type: TMDBContentTypes): TMDBSearchResultsPromise; + searchMedia(query: string, type: "movie"): TMDBSearchResultsPromise; + searchMedia(query: string, type: "show"): TMDBSearchResultsPromise; +} + +export interface TMDBEpisode { + air_date: string; + episode_number: number; + id: number; + name: string; + overview: string; + production_code: string; + runtime: number; + season_number: number; + show_id: number; + still_path: string | null; + vote_average: number; + vote_count: number; + crew: any[]; + guest_stars: any[]; +} + +export interface TMDBSeason { + _id: string; + air_date: string; + episodes: TMDBEpisode[]; + name: string; + overview: string; + id: number; + poster_path: string | null; + season_number: number; +} diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index b87654a9..695027a2 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -1,7 +1,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { TTVMediaToId } from "@/backend/metadata/getmeta"; +import { TMDBMediaToId } from "@/backend/metadata/getmeta"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -132,7 +132,7 @@ export function MediaCard(props: MediaCardProps) { const canLink = props.linkable && !props.closable; let link = canLink - ? `/media/${encodeURIComponent(TTVMediaToId(props.media))}` + ? `/media/${encodeURIComponent(TMDBMediaToId(props.media))}` : "#"; if (canLink && props.series) link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index 7ea6c5a3..ce45c318 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -2,7 +2,7 @@ import { useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; -import { decodeTTVId, getMetaFromId } from "@/backend/metadata/getmeta"; +import { decodeTMDBId, getMetaFromId } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; @@ -44,7 +44,7 @@ export function EpisodeSelectionPopout() { seasonId: sId, season: undefined, }); - reqSeasonMeta(decodeTTVId(params.media)?.id as string, sId).then((v) => { + reqSeasonMeta(decodeTMDBId(params.media)?.id as string, sId).then((v) => { if (v?.meta.type !== MWMediaType.SERIES) return; setCurrentVisibleSeason({ seasonId: sId, diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index c0e81455..7ae1c01b 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -6,7 +6,7 @@ import { useHistory, useParams } from "react-router-dom"; import { MWStream } from "@/backend/helpers/streams"; import { DetailedMeta, - decodeTTVId, + decodeTMDBId, getMetaFromId, } from "@/backend/metadata/getmeta"; import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; @@ -184,7 +184,7 @@ export function MediaView() { const [selected, setSelected] = useState(null); const [exec, loading, error] = useLoading( async (mediaParams: string, seasonId?: string) => { - const data = decodeTTVId(mediaParams); + const data = decodeTMDBId(mediaParams); if (!data) return null; return getMetaFromId(data.type, data.id, seasonId); } From 763de37e9eaa4f6933e631fcd4b06c5703d37a2e Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 13 Jun 2023 21:26:58 +0200 Subject: [PATCH 67/96] cleanup --- src/setup/config.ts | 4 ---- 1 file changed, 4 deletions(-) diff --git a/src/setup/config.ts b/src/setup/config.ts index c24117bb..f1db01da 100644 --- a/src/setup/config.ts +++ b/src/setup/config.ts @@ -8,7 +8,6 @@ interface Config { TMDB_API_KEY: string; CORS_PROXY_URL: string; NORMAL_ROUTER: boolean; - TRAKT_CLIENT_ID: string; } export interface RuntimeConfig { @@ -19,7 +18,6 @@ export interface RuntimeConfig { TMDB_API_KEY: string; NORMAL_ROUTER: boolean; PROXY_URLS: string[]; - TRAKT_CLIENT_ID: string; } const env: Record = { @@ -30,7 +28,6 @@ const env: Record = { DISCORD_LINK: undefined, CORS_PROXY_URL: import.meta.env.VITE_CORS_PROXY_URL, NORMAL_ROUTER: import.meta.env.VITE_NORMAL_ROUTER, - TRAKT_CLIENT_ID: import.meta.env.VITE_TRAKT_CLIENT_ID, }; const alerts = [] as string[]; @@ -65,6 +62,5 @@ export function conf(): RuntimeConfig { .split(",") .map((v) => v.trim()), NORMAL_ROUTER: getKey("NORMAL_ROUTER", "false") === "true", - TRAKT_CLIENT_ID: getKey("TRAKT_CLIENT_ID"), }; } From 0e9263b6192353b13640da33fbd61e1b990d9aba Mon Sep 17 00:00:00 2001 From: castdrian Date: Wed, 14 Jun 2023 07:48:31 +0200 Subject: [PATCH 68/96] fix movie metadata --- src/backend/metadata/getmeta.ts | 56 ++++++++++++++++++++++----------- src/backend/metadata/search.ts | 2 ++ 2 files changed, 39 insertions(+), 19 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 777fae42..41e67772 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -43,6 +43,41 @@ export interface DetailedMeta { tmdbId?: string; } +export function fromatTMDBMetaResult( + details: TMDBShowData | TMDBMovieData, + type: MWMediaType +): TMDBMediaResult | undefined { + let tmdbmeta; + if (type === MWMediaType.MOVIE) { + tmdbmeta = { + id: details.id, + title: (details as TMDBMovieData).title, + object_type: mediaTypeToTMDB(type), + poster: (details as TMDBMovieData).poster_path ?? undefined, + original_release_year: Number( + (details as TMDBMovieData).release_date?.split("-")[0] + ), + }; + } + if (type === MWMediaType.SERIES) { + tmdbmeta = { + id: details.id, + title: (details as TMDBShowData).name, + object_type: mediaTypeToTMDB(type), + seasons: (details as TMDBShowData).seasons.map((v) => ({ + id: v.id, + season_number: v.season_number, + title: v.name, + })), + poster: (details as TMDBMovieData).poster_path ?? undefined, + original_release_year: Number( + (details as TMDBShowData).first_air_date?.split("-")[0] + ), + }; + } + return tmdbmeta; +} + export async function getMetaFromId( type: MWMediaType, id: string, @@ -79,25 +114,8 @@ export async function getMetaFromId( } } - const tmdbmeta: TMDBMediaResult = { - id: details.id, - title: - type === MWMediaType.MOVIE - ? (details as TMDBMovieData).title - : (details as TMDBShowData).name, - object_type: mediaTypeToTMDB(type), - seasons: (details as TMDBShowData).seasons.map((v) => ({ - id: v.id, - season_number: v.season_number, - title: v.name, - })), - poster: (details as TMDBMovieData).poster_path ?? undefined, - original_release_year: - type === MWMediaType.MOVIE - ? Number((details as TMDBMovieData).release_date?.split("-")[0]) - : Number((details as TMDBShowData).first_air_date?.split("-")[0]), - }; - + const tmdbmeta = fromatTMDBMetaResult(details, type); + if (!tmdbmeta) return null; const meta = formatTMDBMeta(tmdbmeta, seasonData); if (!meta) return null; diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 31b2c682..7d06ab2c 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -29,6 +29,8 @@ export async function searchForMedia(query: MWQuery): Promise { }) ); + console.log(results[0]); + cache.set(query, results, 3600); return results; } From 06eb8e6b6d2b40778e7ae93797d919a08fa8ed52 Mon Sep 17 00:00:00 2001 From: castdrian Date: Wed, 14 Jun 2023 07:52:04 +0200 Subject: [PATCH 69/96] cleanup --- src/backend/metadata/search.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 7d06ab2c..31b2c682 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -29,8 +29,6 @@ export async function searchForMedia(query: MWQuery): Promise { }) ); - console.log(results[0]); - cache.set(query, results, 3600); return results; } From c9bac3ed68f2688ceabbeaac68bb845443b551b7 Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 08:30:05 +0200 Subject: [PATCH 70/96] show poster in bookmarks --- src/backend/metadata/getmeta.ts | 8 ++++++-- src/components/media/MediaCard.tsx | 5 ++++- 2 files changed, 10 insertions(+), 3 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 41e67772..bba3948e 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -53,7 +53,9 @@ export function fromatTMDBMetaResult( id: details.id, title: (details as TMDBMovieData).title, object_type: mediaTypeToTMDB(type), - poster: (details as TMDBMovieData).poster_path ?? undefined, + poster: + Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? + undefined, original_release_year: Number( (details as TMDBMovieData).release_date?.split("-")[0] ), @@ -69,7 +71,9 @@ export function fromatTMDBMetaResult( season_number: v.season_number, title: v.name, })), - poster: (details as TMDBMovieData).poster_path ?? undefined, + poster: + Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? + undefined, original_release_year: Number( (details as TMDBShowData).first_air_date?.split("-")[0] ), diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index 695027a2..ece6d293 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -2,6 +2,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; import { TMDBMediaToId } from "@/backend/metadata/getmeta"; +import { Tmdb } from "@/backend/metadata/tmdb"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -55,7 +56,9 @@ function MediaCardContent({ closable ? "" : "group-hover:rounded-lg", ].join(" ")} style={{ - backgroundImage: media.poster ? `url(${media.poster})` : undefined, + backgroundImage: media.poster + ? `url(${Tmdb.getMediaPoster(media.poster)})` + : undefined, }} > {series ? ( From c08a6c7e54b2dff6d2955eb6ee7f561d8379aa42 Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 08:30:57 +0200 Subject: [PATCH 71/96] set adult false in query --- src/backend/metadata/tmdb.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index f01709bb..aed31c1d 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -120,12 +120,12 @@ export abstract class Tmdb { switch (type) { case "movie": data = await Tmdb.get( - `search/movie?query=${query}&include_adult=true&language=en-US&page=1` + `search/movie?query=${query}&include_adult=false&language=en-US&page=1` ); break; case "show": data = await Tmdb.get( - `search/tv?query=${query}&include_adult=true&language=en-US&page=1` + `search/tv?query=${query}&include_adult=false&language=en-US&page=1` ); break; default: From 4d51de3bd126feaa4585b3041242f73029cf600f Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 11:06:24 +0200 Subject: [PATCH 72/96] undo duplicate path --- src/backend/metadata/getmeta.ts | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index bba3948e..41e67772 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -53,9 +53,7 @@ export function fromatTMDBMetaResult( id: details.id, title: (details as TMDBMovieData).title, object_type: mediaTypeToTMDB(type), - poster: - Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? - undefined, + poster: (details as TMDBMovieData).poster_path ?? undefined, original_release_year: Number( (details as TMDBMovieData).release_date?.split("-")[0] ), @@ -71,9 +69,7 @@ export function fromatTMDBMetaResult( season_number: v.season_number, title: v.name, })), - poster: - Tmdb.getMediaPoster((details as TMDBMovieData).poster_path) ?? - undefined, + poster: (details as TMDBMovieData).poster_path ?? undefined, original_release_year: Number( (details as TMDBShowData).first_air_date?.split("-")[0] ), From 0d249a3e27fee182b33980c72837804fa5579a13 Mon Sep 17 00:00:00 2001 From: castdrian Date: Thu, 15 Jun 2023 22:13:19 +0200 Subject: [PATCH 73/96] fix typo 'cause I can't type --- src/backend/metadata/getmeta.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 41e67772..a5246fcf 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -43,7 +43,7 @@ export interface DetailedMeta { tmdbId?: string; } -export function fromatTMDBMetaResult( +export function formatTMDBMetaResult( details: TMDBShowData | TMDBMovieData, type: MWMediaType ): TMDBMediaResult | undefined { @@ -114,7 +114,7 @@ export async function getMetaFromId( } } - const tmdbmeta = fromatTMDBMetaResult(details, type); + const tmdbmeta = formatTMDBMetaResult(details, type); if (!tmdbmeta) return null; const meta = formatTMDBMeta(tmdbmeta, seasonData); if (!meta) return null; From 205248a376d9347dac3c0f075bf05b3ff280a1f2 Mon Sep 17 00:00:00 2001 From: castdrian Date: Fri, 16 Jun 2023 11:18:32 +0200 Subject: [PATCH 74/96] use external ids endpoint for imdb ids --- src/backend/metadata/getmeta.ts | 6 ++---- src/backend/metadata/tmdb.ts | 25 +++++++++++++++++++++++++ src/backend/metadata/types.ts | 24 ++++++++++++++++++++++++ 3 files changed, 51 insertions(+), 4 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index a5246fcf..6868c7e4 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -87,10 +87,8 @@ export async function getMetaFromId( if (!details) return null; - let imdbId; - if (type === MWMediaType.MOVIE) { - imdbId = (details as TMDBMovieData).imdb_id ?? undefined; - } + const externalIds = await Tmdb.getExternalIds(id, mediaTypeToTMDB(type)); + const imdbId = externalIds.imdb_id ?? undefined; let seasonData: TMDBSeasonMetaResult | undefined; diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index aed31c1d..e22e86e7 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -6,15 +6,18 @@ import { MWSeasonMeta, TMDBContentTypes, TMDBEpisodeShort, + TMDBExternalIds, TMDBMediaResult, TMDBMediaStatic, TMDBMovieData, + TMDBMovieExternalIds, TMDBMovieResponse, TMDBMovieResult, TMDBSearchResultStatic, TMDBSeason, TMDBSeasonMetaResult, TMDBShowData, + TMDBShowExternalIds, TMDBShowResponse, TMDBShowResult, } from "./types"; @@ -170,6 +173,28 @@ export abstract class Tmdb { title: e.name, })); } + + public static async getExternalIds( + id: string, + type: TMDBContentTypes + ): Promise { + let data; + + switch (type) { + case "movie": + data = await Tmdb.get( + `/movie/${id}/external_ids` + ); + break; + case "show": + data = await Tmdb.get(`/tv/${id}/external_ids`); + break; + default: + throw new Error("Invalid media type"); + } + + return data; + } } export async function formatTMDBSearchResult( diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types.ts index e23d9a5b..fa7a7ef0 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types.ts @@ -379,3 +379,27 @@ export interface TMDBSeason { poster_path: string | null; season_number: number; } + +export interface TMDBShowExternalIds { + id: number; + imdb_id: null | string; + freebase_mid: null | string; + freebase_id: null | string; + tvdb_id: number; + tvrage_id: null | string; + wikidata_id: null | string; + facebook_id: null | string; + instagram_id: null | string; + twitter_id: null | string; +} + +export interface TMDBMovieExternalIds { + id: number; + imdb_id: null | string; + wikidata_id: null | string; + facebook_id: null | string; + instagram_id: null | string; + twitter_id: null | string; +} + +export type TMDBExternalIds = TMDBShowExternalIds | TMDBMovieExternalIds; From 5661a7873a67437a70b3233fe637acbf9d235fd7 Mon Sep 17 00:00:00 2001 From: castdrian Date: Mon, 19 Jun 2023 17:03:12 +0200 Subject: [PATCH 75/96] remove seasons from search result --- src/backend/metadata/tmdb.ts | 13 +------------ 1 file changed, 1 insertion(+), 12 deletions(-) diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index e22e86e7..0df2df7e 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -202,29 +202,18 @@ export async function formatTMDBSearchResult( mediatype: TMDBContentTypes ): Promise { const type = TMDBMediaToMediaType(mediatype); - const details = await Tmdb.getMediaDetails(result.id.toString(), mediatype); - - const seasons = - type === MWMediaType.SERIES - ? (details as TMDBShowData).seasons?.map((v) => ({ - id: v.id, - title: v.name, - season_number: v.season_number, - })) - : undefined; return { title: type === MWMediaType.SERIES ? (result as TMDBShowResult).name : (result as TMDBMovieResult).title, - poster: Tmdb.getMediaPoster(details.poster_path), + poster: Tmdb.getMediaPoster(result.poster_path), id: result.id, original_release_year: type === MWMediaType.SERIES ? Number((result as TMDBShowResult).first_air_date?.split("-")[0]) : Number((result as TMDBMovieResult).release_date?.split("-")[0]), object_type: mediaTypeToTMDB(type), - seasons, }; } From 3f241c2d072d0a05b9ba8a197e62a774fa538c7b Mon Sep 17 00:00:00 2001 From: castdrian Date: Tue, 20 Jun 2023 19:39:16 +0200 Subject: [PATCH 76/96] fix idiotism --- src/backend/metadata/getmeta.ts | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 6868c7e4..4fd39e7b 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -99,13 +99,18 @@ export async function getMetaFromId( const episodes = await Tmdb.getEpisodes( details.id.toString(), - season?.season_number ?? 1 + season.season_number === null || season.season_number === 0 + ? 1 + : season.season_number ); if (season && episodes) { seasonData = { id: season.id.toString(), - season_number: season.season_number, + season_number: + season.season_number === null || season.season_number === 0 + ? 1 + : season.season_number, title: season.name, episodes, }; From 33b67f32b14bff402d3162f5c46b3f34d727fe3e Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 12:43:36 +0200 Subject: [PATCH 77/96] no undef for tmdbmetaresult --- src/backend/metadata/getmeta.ts | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 4fd39e7b..d8eca10e 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -46,7 +46,7 @@ export interface DetailedMeta { export function formatTMDBMetaResult( details: TMDBShowData | TMDBMovieData, type: MWMediaType -): TMDBMediaResult | undefined { +): TMDBMediaResult { let tmdbmeta; if (type === MWMediaType.MOVIE) { tmdbmeta = { @@ -75,6 +75,8 @@ export function formatTMDBMetaResult( ), }; } + + if (!tmdbmeta) throw new Error("unsupported type"); return tmdbmeta; } From 9495a3bf413d61d72ebb35cd7f17a6069a47e544 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 12:47:09 +0200 Subject: [PATCH 78/96] reduce casts --- src/backend/metadata/getmeta.ts | 18 ++++++++---------- 1 file changed, 8 insertions(+), 10 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index d8eca10e..0ef84474 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -49,30 +49,28 @@ export function formatTMDBMetaResult( ): TMDBMediaResult { let tmdbmeta; if (type === MWMediaType.MOVIE) { + const movie = details as TMDBMovieData; tmdbmeta = { id: details.id, - title: (details as TMDBMovieData).title, + title: movie.title, object_type: mediaTypeToTMDB(type), - poster: (details as TMDBMovieData).poster_path ?? undefined, - original_release_year: Number( - (details as TMDBMovieData).release_date?.split("-")[0] - ), + poster: movie.poster_path ?? undefined, + original_release_year: Number(movie.release_date?.split("-")[0]), }; } if (type === MWMediaType.SERIES) { + const show = details as TMDBShowData; tmdbmeta = { id: details.id, - title: (details as TMDBShowData).name, + title: show.name, object_type: mediaTypeToTMDB(type), - seasons: (details as TMDBShowData).seasons.map((v) => ({ + seasons: show.seasons.map((v) => ({ id: v.id, season_number: v.season_number, title: v.name, })), poster: (details as TMDBMovieData).poster_path ?? undefined, - original_release_year: Number( - (details as TMDBShowData).first_air_date?.split("-")[0] - ), + original_release_year: Number(show.first_air_date?.split("-")[0]), }; } From 430486a9b9f09cdc0515d9c3513ca49f48dee649 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 12:48:33 +0200 Subject: [PATCH 79/96] direct return --- src/backend/metadata/getmeta.ts | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 0ef84474..8010c89a 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -47,10 +47,9 @@ export function formatTMDBMetaResult( details: TMDBShowData | TMDBMovieData, type: MWMediaType ): TMDBMediaResult { - let tmdbmeta; if (type === MWMediaType.MOVIE) { const movie = details as TMDBMovieData; - tmdbmeta = { + return { id: details.id, title: movie.title, object_type: mediaTypeToTMDB(type), @@ -60,7 +59,7 @@ export function formatTMDBMetaResult( } if (type === MWMediaType.SERIES) { const show = details as TMDBShowData; - tmdbmeta = { + return { id: details.id, title: show.name, object_type: mediaTypeToTMDB(type), @@ -74,8 +73,7 @@ export function formatTMDBMetaResult( }; } - if (!tmdbmeta) throw new Error("unsupported type"); - return tmdbmeta; + throw new Error("unsupported type"); } export async function getMetaFromId( From 984d215312d5cb616af0c03a5c7e76e4d0e9284b Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 12:50:41 +0200 Subject: [PATCH 80/96] parse dates instead of cringe string manipulation --- src/backend/metadata/getmeta.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 8010c89a..4548a436 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -54,7 +54,7 @@ export function formatTMDBMetaResult( title: movie.title, object_type: mediaTypeToTMDB(type), poster: movie.poster_path ?? undefined, - original_release_year: Number(movie.release_date?.split("-")[0]), + original_release_year: new Date(movie.release_date).getFullYear(), }; } if (type === MWMediaType.SERIES) { @@ -69,7 +69,7 @@ export function formatTMDBMetaResult( title: v.name, })), poster: (details as TMDBMovieData).poster_path ?? undefined, - original_release_year: Number(show.first_air_date?.split("-")[0]), + original_release_year: new Date(show.first_air_date).getFullYear(), }; } From 89cdf74b2fa8e2a1d47978620d3ebf852b0613d1 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 12:51:30 +0200 Subject: [PATCH 81/96] readd vanished comment --- src/backend/metadata/search.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 31b2c682..549d7ba4 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -29,6 +29,6 @@ export async function searchForMedia(query: MWQuery): Promise { }) ); - cache.set(query, results, 3600); + cache.set(query, results, 3600); // cache results for 1 hour return results; } From 1408fcde9300def772923c83fb6bd203344451c9 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 13:07:33 +0200 Subject: [PATCH 82/96] export functions directly --- src/backend/metadata/getmeta.ts | 10 +- src/backend/metadata/search.ts | 4 +- src/backend/metadata/tmdb.ts | 168 ++++++++++++++--------------- src/components/media/MediaCard.tsx | 4 +- 4 files changed, 88 insertions(+), 98 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 4548a436..aa4267c5 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -3,8 +3,10 @@ import { FetchError } from "ofetch"; import { formatJWMeta, mediaTypeToJW } from "./justwatch"; import { TMDBMediaToMediaType, - Tmdb, formatTMDBMeta, + getEpisodes, + getExternalIds, + getMediaDetails, mediaTypeToTMDB, } from "./tmdb"; import { @@ -81,11 +83,11 @@ export async function getMetaFromId( id: string, seasonId?: string ): Promise { - const details = await Tmdb.getMediaDetails(id, mediaTypeToTMDB(type)); + const details = await getMediaDetails(id, mediaTypeToTMDB(type)); if (!details) return null; - const externalIds = await Tmdb.getExternalIds(id, mediaTypeToTMDB(type)); + const externalIds = await getExternalIds(id, mediaTypeToTMDB(type)); const imdbId = externalIds.imdb_id ?? undefined; let seasonData: TMDBSeasonMetaResult | undefined; @@ -95,7 +97,7 @@ export async function getMetaFromId( const season = seasons?.find((v) => v.id.toString() === seasonId) ?? seasons?.[0]; - const episodes = await Tmdb.getEpisodes( + const episodes = await getEpisodes( details.id.toString(), season.season_number === null || season.season_number === 0 ? 1 diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 549d7ba4..9e2883d4 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -1,10 +1,10 @@ import { SimpleCache } from "@/utils/cache"; import { - Tmdb, formatTMDBMeta, formatTMDBSearchResult, mediaTypeToTMDB, + searchMedia, } from "./tmdb"; import { MWMediaMeta, MWQuery } from "./types"; @@ -18,7 +18,7 @@ export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const data = await Tmdb.searchMedia(searchQuery, mediaTypeToTMDB(type)); + const data = await searchMedia(searchQuery, mediaTypeToTMDB(type)); const results = await Promise.all( data.results.map(async (v) => { const formattedResult = await formatTMDBSearchResult( diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 0df2df7e..cf327070 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -8,12 +8,10 @@ import { TMDBEpisodeShort, TMDBExternalIds, TMDBMediaResult, - TMDBMediaStatic, TMDBMovieData, TMDBMovieExternalIds, TMDBMovieResponse, TMDBMovieResult, - TMDBSearchResultStatic, TMDBSeason, TMDBSeasonMetaResult, TMDBShowData, @@ -98,103 +96,93 @@ export function decodeTMDBId( }; } -export abstract class Tmdb { - private static baseURL = "https://api.themoviedb.org/3"; +const baseURL = "https://api.themoviedb.org/3"; - private static headers = { - accept: "application/json", - Authorization: `Bearer ${conf().TMDB_API_KEY}`, - }; +const headers = { + accept: "application/json", + Authorization: `Bearer ${conf().TMDB_API_KEY}`, +}; - private static async get(url: string): Promise { - const res = await mwFetch(url, { - headers: Tmdb.headers, - baseURL: Tmdb.baseURL, - }); - return res; +async function get(url: string): Promise { + const res = await mwFetch(url, { + headers, + baseURL, + }); + return res; +} + +export async function searchMedia(query: string, type: TMDBContentTypes) { + let data; + + switch (type) { + case "movie": + data = await get( + `search/movie?query=${query}&include_adult=false&language=en-US&page=1` + ); + break; + case "show": + data = await get( + `search/tv?query=${query}&include_adult=false&language=en-US&page=1` + ); + break; + default: + throw new Error("Invalid media type"); } - public static searchMedia: TMDBSearchResultStatic["searchMedia"] = async ( - query: string, - type: TMDBContentTypes - ) => { - let data; + return data; +} - switch (type) { - case "movie": - data = await Tmdb.get( - `search/movie?query=${query}&include_adult=false&language=en-US&page=1` - ); - break; - case "show": - data = await Tmdb.get( - `search/tv?query=${query}&include_adult=false&language=en-US&page=1` - ); - break; - default: - throw new Error("Invalid media type"); - } +export async function getMediaDetails(id: string, type: TMDBContentTypes) { + let data; - return data; - }; - - public static getMediaDetails: TMDBMediaStatic["getMediaDetails"] = async ( - id: string, - type: TMDBContentTypes - ) => { - let data; - - switch (type) { - case "movie": - data = await Tmdb.get(`/movie/${id}`); - break; - case "show": - data = await Tmdb.get(`/tv/${id}`); - break; - default: - throw new Error("Invalid media type"); - } - - return data; - }; - - public static getMediaPoster(posterPath: string | null): string | undefined { - if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; + switch (type) { + case "movie": + data = await get(`/movie/${id}`); + break; + case "show": + data = await get(`/tv/${id}`); + break; + default: + throw new Error("Invalid media type"); } - public static async getEpisodes( - id: string, - season: number - ): Promise { - const data = await Tmdb.get(`/tv/${id}/season/${season}`); - return data.episodes.map((e) => ({ - id: e.id, - episode_number: e.episode_number, - title: e.name, - })); + return data; +} + +export function getMediaPoster(posterPath: string | null): string | undefined { + if (posterPath) return `https://image.tmdb.org/t/p/w185/${posterPath}`; +} + +export async function getEpisodes( + id: string, + season: number +): Promise { + const data = await get(`/tv/${id}/season/${season}`); + return data.episodes.map((e) => ({ + id: e.id, + episode_number: e.episode_number, + title: e.name, + })); +} + +export async function getExternalIds( + id: string, + type: TMDBContentTypes +): Promise { + let data; + + switch (type) { + case "movie": + data = await get(`/movie/${id}/external_ids`); + break; + case "show": + data = await get(`/tv/${id}/external_ids`); + break; + default: + throw new Error("Invalid media type"); } - public static async getExternalIds( - id: string, - type: TMDBContentTypes - ): Promise { - let data; - - switch (type) { - case "movie": - data = await Tmdb.get( - `/movie/${id}/external_ids` - ); - break; - case "show": - data = await Tmdb.get(`/tv/${id}/external_ids`); - break; - default: - throw new Error("Invalid media type"); - } - - return data; - } + return data; } export async function formatTMDBSearchResult( @@ -208,7 +196,7 @@ export async function formatTMDBSearchResult( type === MWMediaType.SERIES ? (result as TMDBShowResult).name : (result as TMDBMovieResult).title, - poster: Tmdb.getMediaPoster(result.poster_path), + poster: getMediaPoster(result.poster_path), id: result.id, original_release_year: type === MWMediaType.SERIES diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index ece6d293..fd460bb7 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -2,7 +2,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; import { TMDBMediaToId } from "@/backend/metadata/getmeta"; -import { Tmdb } from "@/backend/metadata/tmdb"; +import { getMediaPoster } from "@/backend/metadata/tmdb"; import { MWMediaMeta } from "@/backend/metadata/types"; import { DotList } from "@/components/text/DotList"; @@ -57,7 +57,7 @@ function MediaCardContent({ ].join(" ")} style={{ backgroundImage: media.poster - ? `url(${Tmdb.getMediaPoster(media.poster)})` + ? `url(${getMediaPoster(media.poster)})` : undefined, }} > From 7c3d4aac272a77debe89a02a77e4c54c9b09ed5e Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 13:23:39 +0200 Subject: [PATCH 83/96] refactor typedefs --- src/backend/metadata/getmeta.ts | 7 +- src/backend/metadata/justwatch.ts | 6 +- src/backend/metadata/search.ts | 7 +- src/backend/metadata/tmdb.ts | 11 +- src/backend/metadata/types/justwatch.ts | 48 +++++++ src/backend/metadata/types/mw.ts | 53 ++++++++ .../metadata/{types.ts => types/tmdb.ts} | 121 ------------------ 7 files changed, 118 insertions(+), 135 deletions(-) create mode 100644 src/backend/metadata/types/justwatch.ts create mode 100644 src/backend/metadata/types/mw.ts rename src/backend/metadata/{types.ts => types/tmdb.ts} (69%) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index aa4267c5..b347e720 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -13,13 +13,14 @@ import { JWMediaResult, JWSeasonMetaResult, JW_API_BASE, - MWMediaMeta, - MWMediaType, +} from "./types/justwatch"; +import { MWMediaMeta, MWMediaType } from "./types/mw"; +import { TMDBMediaResult, TMDBMovieData, TMDBSeasonMetaResult, TMDBShowData, -} from "./types"; +} from "./types/tmdb"; import { makeUrl, proxiedFetch } from "../helpers/fetch"; type JWExternalIdType = diff --git a/src/backend/metadata/justwatch.ts b/src/backend/metadata/justwatch.ts index 857ff006..724c4acf 100644 --- a/src/backend/metadata/justwatch.ts +++ b/src/backend/metadata/justwatch.ts @@ -3,10 +3,8 @@ import { JWMediaResult, JWSeasonMetaResult, JW_IMAGE_BASE, - MWMediaMeta, - MWMediaType, - MWSeasonMeta, -} from "./types"; +} from "./types/justwatch"; +import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types/mw"; export function mediaTypeToJW(type: MWMediaType): JWContentTypes { if (type === MWMediaType.MOVIE) return "movie"; diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 9e2883d4..99cb51ba 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -6,7 +6,8 @@ import { mediaTypeToTMDB, searchMedia, } from "./tmdb"; -import { MWMediaMeta, MWQuery } from "./types"; +import { MWMediaMeta, MWQuery } from "./types/mw"; +import { TMDBMovieResponse, TMDBShowResponse } from "./types/tmdb"; const cache = new SimpleCache(); cache.setCompare((a, b) => { @@ -18,7 +19,9 @@ export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const data = await searchMedia(searchQuery, mediaTypeToTMDB(type)); + const data = (await searchMedia(searchQuery, mediaTypeToTMDB(type))) as + | TMDBMovieResponse + | TMDBShowResponse; const results = await Promise.all( data.results.map(async (v) => { const formattedResult = await formatTMDBSearchResult( diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index cf327070..4a9271ba 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -1,9 +1,7 @@ import { conf } from "@/setup/config"; +import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types/mw"; import { - MWMediaMeta, - MWMediaType, - MWSeasonMeta, TMDBContentTypes, TMDBEpisodeShort, TMDBExternalIds, @@ -18,7 +16,7 @@ import { TMDBShowExternalIds, TMDBShowResponse, TMDBShowResult, -} from "./types"; +} from "./types/tmdb"; import { mwFetch } from "../helpers/fetch"; export function mediaTypeToTMDB(type: MWMediaType): TMDBContentTypes { @@ -111,7 +109,10 @@ async function get(url: string): Promise { return res; } -export async function searchMedia(query: string, type: TMDBContentTypes) { +export async function searchMedia( + query: string, + type: TMDBContentTypes +): Promise { let data; switch (type) { diff --git a/src/backend/metadata/types/justwatch.ts b/src/backend/metadata/types/justwatch.ts new file mode 100644 index 00000000..cb3ac092 --- /dev/null +++ b/src/backend/metadata/types/justwatch.ts @@ -0,0 +1,48 @@ +export type JWContentTypes = "movie" | "show"; + +export type JWSearchQuery = { + content_types: JWContentTypes[]; + page: number; + page_size: number; + query: string; +}; + +export type JWPage = { + items: T[]; + page: number; + page_size: number; + total_pages: number; + total_results: number; +}; + +export const JW_API_BASE = "https://apis.justwatch.com"; +export const JW_IMAGE_BASE = "https://images.justwatch.com"; + +export type JWSeasonShort = { + title: string; + id: number; + season_number: number; +}; + +export type JWEpisodeShort = { + title: string; + id: number; + episode_number: number; +}; + +export type JWMediaResult = { + title: string; + poster?: string; + id: number; + original_release_year?: number; + jw_entity_id: string; + object_type: JWContentTypes; + seasons?: JWSeasonShort[]; +}; + +export type JWSeasonMetaResult = { + title: string; + id: string; + season_number: number; + episodes: JWEpisodeShort[]; +}; diff --git a/src/backend/metadata/types/mw.ts b/src/backend/metadata/types/mw.ts new file mode 100644 index 00000000..e7cc26fe --- /dev/null +++ b/src/backend/metadata/types/mw.ts @@ -0,0 +1,53 @@ +export enum MWMediaType { + MOVIE = "movie", + SERIES = "series", + ANIME = "anime", +} + +export type MWSeasonMeta = { + id: string; + number: number; + title: string; +}; + +export type MWSeasonWithEpisodeMeta = { + id: string; + number: number; + title: string; + episodes: { + id: string; + number: number; + title: string; + }[]; +}; + +type MWMediaMetaBase = { + title: string; + id: string; + year?: string; + poster?: string; +}; + +type MWMediaMetaSpecific = + | { + type: MWMediaType.MOVIE | MWMediaType.ANIME; + seasons: undefined; + } + | { + type: MWMediaType.SERIES; + seasons: MWSeasonMeta[]; + seasonData: MWSeasonWithEpisodeMeta; + }; + +export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; + +export interface MWQuery { + searchQuery: string; + type: MWMediaType; +} + +export interface DetailedMeta { + meta: MWMediaMeta; + imdbId?: string; + tmdbId?: string; +} diff --git a/src/backend/metadata/types.ts b/src/backend/metadata/types/tmdb.ts similarity index 69% rename from src/backend/metadata/types.ts rename to src/backend/metadata/types/tmdb.ts index fa7a7ef0..cb5e9aa4 100644 --- a/src/backend/metadata/types.ts +++ b/src/backend/metadata/types/tmdb.ts @@ -1,51 +1,3 @@ -export enum MWMediaType { - MOVIE = "movie", - SERIES = "series", - ANIME = "anime", -} - -export type MWSeasonMeta = { - id: string; - number: number; - title: string; -}; - -export type MWSeasonWithEpisodeMeta = { - id: string; - number: number; - title: string; - episodes: { - id: string; - number: number; - title: string; - }[]; -}; - -type MWMediaMetaBase = { - title: string; - id: string; - year?: string; - poster?: string; -}; - -type MWMediaMetaSpecific = - | { - type: MWMediaType.MOVIE | MWMediaType.ANIME; - seasons: undefined; - } - | { - type: MWMediaType.SERIES; - seasons: MWSeasonMeta[]; - seasonData: MWSeasonWithEpisodeMeta; - }; - -export type MWMediaMeta = MWMediaMetaBase & MWMediaMetaSpecific; - -export interface MWQuery { - searchQuery: string; - type: MWMediaType; -} - export type TMDBContentTypes = "movie" | "show"; export type TMDBSeasonShort = { @@ -76,12 +28,6 @@ export type TMDBSeasonMetaResult = { episodes: TMDBEpisodeShort[]; }; -export interface DetailedMeta { - meta: MWMediaMeta; - imdbId?: string; - tmdbId?: string; -} - export interface TMDBShowData { adult: boolean; backdrop_path: string | null; @@ -225,63 +171,6 @@ export interface TMDBMovieData { vote_count: number; } -export type TMDBMediaDetailsPromise = Promise; - -export interface TMDBMediaStatic { - getMediaDetails(id: string, type: "show"): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: "movie"): TMDBMediaDetailsPromise; - getMediaDetails(id: string, type: TMDBContentTypes): TMDBMediaDetailsPromise; -} - -export type JWContentTypes = "movie" | "show"; - -export type JWSearchQuery = { - content_types: JWContentTypes[]; - page: number; - page_size: number; - query: string; -}; - -export type JWPage = { - items: T[]; - page: number; - page_size: number; - total_pages: number; - total_results: number; -}; - -export const JW_API_BASE = "https://apis.justwatch.com"; -export const JW_IMAGE_BASE = "https://images.justwatch.com"; - -export type JWSeasonShort = { - title: string; - id: number; - season_number: number; -}; - -export type JWEpisodeShort = { - title: string; - id: number; - episode_number: number; -}; - -export type JWMediaResult = { - title: string; - poster?: string; - id: number; - original_release_year?: number; - jw_entity_id: string; - object_type: JWContentTypes; - seasons?: JWSeasonShort[]; -}; - -export type JWSeasonMetaResult = { - title: string; - id: string; - season_number: number; - episodes: JWEpisodeShort[]; -}; - export interface TMDBEpisodeResult { season: number; number: number; @@ -342,16 +231,6 @@ export interface TMDBMovieResponse { total_results: number; } -export type TMDBSearchResultsPromise = Promise< - TMDBShowResponse | TMDBMovieResponse ->; - -export interface TMDBSearchResultStatic { - searchMedia(query: string, type: TMDBContentTypes): TMDBSearchResultsPromise; - searchMedia(query: string, type: "movie"): TMDBSearchResultsPromise; - searchMedia(query: string, type: "show"): TMDBSearchResultsPromise; -} - export interface TMDBEpisode { air_date: string; episode_number: number; From dccab9b0bf084b974db4d2fd5ae64179c7ed2eb9 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 13:26:03 +0200 Subject: [PATCH 84/96] directly get poster url --- src/backend/metadata/getmeta.ts | 3 ++- src/components/media/MediaCard.tsx | 7 ++----- 2 files changed, 4 insertions(+), 6 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index b347e720..21b6843a 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -7,6 +7,7 @@ import { getEpisodes, getExternalIds, getMediaDetails, + getMediaPoster, mediaTypeToTMDB, } from "./tmdb"; import { @@ -56,7 +57,7 @@ export function formatTMDBMetaResult( id: details.id, title: movie.title, object_type: mediaTypeToTMDB(type), - poster: movie.poster_path ?? undefined, + poster: getMediaPoster(movie.poster_path) ?? undefined, original_release_year: new Date(movie.release_date).getFullYear(), }; } diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index fd460bb7..c38b4a2b 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -2,8 +2,7 @@ import { useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; import { TMDBMediaToId } from "@/backend/metadata/getmeta"; -import { getMediaPoster } from "@/backend/metadata/tmdb"; -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; import { DotList } from "@/components/text/DotList"; import { IconPatch } from "../buttons/IconPatch"; @@ -56,9 +55,7 @@ function MediaCardContent({ closable ? "" : "group-hover:rounded-lg", ].join(" ")} style={{ - backgroundImage: media.poster - ? `url(${getMediaPoster(media.poster)})` - : undefined, + backgroundImage: media.poster ? `url(${media.poster})` : undefined, }} > {series ? ( From a46cfa43d3597516cb37e97e7592ba9b74079c7d Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 13:31:50 +0200 Subject: [PATCH 85/96] fix test imports --- src/__tests__/providers/providers.test.ts | 2 +- src/__tests__/providers/testdata.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/__tests__/providers/providers.test.ts b/src/__tests__/providers/providers.test.ts index 35c77d5d..350d4255 100644 --- a/src/__tests__/providers/providers.test.ts +++ b/src/__tests__/providers/providers.test.ts @@ -4,7 +4,7 @@ import "@/backend"; import { testData } from "@/__tests__/providers/testdata"; import { getProviders } from "@/backend/helpers/register"; import { runProvider } from "@/backend/helpers/run"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; describe("providers", () => { const providers = getProviders(); diff --git a/src/__tests__/providers/testdata.ts b/src/__tests__/providers/testdata.ts index 37e63e06..6db686e3 100644 --- a/src/__tests__/providers/testdata.ts +++ b/src/__tests__/providers/testdata.ts @@ -1,5 +1,5 @@ import { DetailedMeta } from "@/backend/metadata/getmeta"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; export const testData: DetailedMeta[] = [ { From 436fb2707b4ed03c60e32735ba2a5f68150eae38 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 13:38:48 +0200 Subject: [PATCH 86/96] update all remaining imports --- src/backend/helpers/provider.ts | 2 +- src/backend/helpers/scrape.ts | 2 +- src/backend/providers/2embed.ts | 2 +- src/backend/providers/flixhq.ts | 2 +- src/backend/providers/gdriveplayer.ts | 2 +- src/backend/providers/gomovies.ts | 2 +- src/backend/providers/hdwatched.ts | 2 +- src/backend/providers/kissasian.ts | 2 +- src/backend/providers/m4ufree.ts | 2 +- src/backend/providers/netfilm.ts | 2 +- src/backend/providers/remotestream.ts | 2 +- src/backend/providers/sflix.ts | 2 +- src/backend/providers/streamflix.ts | 2 +- src/backend/providers/superstream/index.ts | 2 +- src/components/SearchBar.tsx | 2 +- src/components/media/WatchedMediaCard.tsx | 2 +- src/hooks/useScrape.ts | 2 +- src/hooks/useSearchQuery.ts | 2 +- src/setup/App.tsx | 2 +- src/state/bookmark/context.tsx | 2 +- src/state/bookmark/types.ts | 2 +- src/state/watched/context.tsx | 2 +- src/state/watched/migrations/v2.ts | 2 +- src/state/watched/types.ts | 2 +- src/video/components/actions/DividerAction.tsx | 2 +- src/video/components/actions/SeriesSelectionAction.tsx | 2 +- src/video/components/controllers/MetaController.tsx | 2 +- src/video/components/hooks/useCurrentSeriesEpisodeInfo.ts | 2 +- src/video/components/parts/VideoErrorBoundary.tsx | 2 +- src/video/components/parts/VideoPlayerHeader.tsx | 2 +- src/video/components/popouts/EpisodeSelectionPopout.tsx | 5 ++++- src/views/developer/VideoTesterView.tsx | 2 +- src/views/media/MediaView.tsx | 5 ++++- src/views/other/v2Migration.tsx | 2 +- src/views/search/SearchResultsPartial.tsx | 2 +- src/views/search/SearchResultsView.tsx | 2 +- 36 files changed, 42 insertions(+), 36 deletions(-) diff --git a/src/backend/helpers/provider.ts b/src/backend/helpers/provider.ts index 6eed4560..58dea7d4 100644 --- a/src/backend/helpers/provider.ts +++ b/src/backend/helpers/provider.ts @@ -1,7 +1,7 @@ import { MWEmbed } from "./embed"; import { MWStream } from "./streams"; import { DetailedMeta } from "../metadata/getmeta"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; export type MWProviderScrapeResult = { stream?: MWStream; diff --git a/src/backend/helpers/scrape.ts b/src/backend/helpers/scrape.ts index 70e20348..5f1a100c 100644 --- a/src/backend/helpers/scrape.ts +++ b/src/backend/helpers/scrape.ts @@ -3,7 +3,7 @@ import { getEmbedScraperByType, getProviders } from "./register"; import { runEmbedScraper, runProvider } from "./run"; import { MWStream } from "./streams"; import { DetailedMeta } from "../metadata/getmeta"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; interface MWProgressData { type: "embed" | "provider"; diff --git a/src/backend/providers/2embed.ts b/src/backend/providers/2embed.ts index 7cc8938e..507d5a2d 100644 --- a/src/backend/providers/2embed.ts +++ b/src/backend/providers/2embed.ts @@ -8,7 +8,7 @@ import { MWStreamQuality, MWStreamType, } from "../helpers/streams"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const twoEmbedBase = "https://www.2embed.to"; diff --git a/src/backend/providers/flixhq.ts b/src/backend/providers/flixhq.ts index 376abd08..fd905019 100644 --- a/src/backend/providers/flixhq.ts +++ b/src/backend/providers/flixhq.ts @@ -7,7 +7,7 @@ import { import { mwFetch } from "../helpers/fetch"; import { registerProvider } from "../helpers/register"; import { MWCaption, MWStreamQuality, MWStreamType } from "../helpers/streams"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const flixHqBase = "https://consumet-api-clone.vercel.app/meta/tmdb"; // instance stolen from streaminal :) diff --git a/src/backend/providers/gdriveplayer.ts b/src/backend/providers/gdriveplayer.ts index 5478b6ed..c184fea7 100644 --- a/src/backend/providers/gdriveplayer.ts +++ b/src/backend/providers/gdriveplayer.ts @@ -3,7 +3,7 @@ import { unpack } from "unpacker"; import { registerProvider } from "@/backend/helpers/register"; import { MWStreamQuality } from "@/backend/helpers/streams"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { proxiedFetch } from "../helpers/fetch"; diff --git a/src/backend/providers/gomovies.ts b/src/backend/providers/gomovies.ts index 9e22d095..fdce289b 100644 --- a/src/backend/providers/gomovies.ts +++ b/src/backend/providers/gomovies.ts @@ -1,7 +1,7 @@ import { MWEmbedType } from "../helpers/embed"; import { proxiedFetch } from "../helpers/fetch"; import { registerProvider } from "../helpers/register"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const gomoviesBase = "https://gomovies.sx"; diff --git a/src/backend/providers/hdwatched.ts b/src/backend/providers/hdwatched.ts index 2096e160..458c3424 100644 --- a/src/backend/providers/hdwatched.ts +++ b/src/backend/providers/hdwatched.ts @@ -2,7 +2,7 @@ import { proxiedFetch } from "../helpers/fetch"; import { MWProviderContext } from "../helpers/provider"; import { registerProvider } from "../helpers/register"; import { MWStreamQuality, MWStreamType } from "../helpers/streams"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const hdwatchedBase = "https://www.hdwatched.xyz"; diff --git a/src/backend/providers/kissasian.ts b/src/backend/providers/kissasian.ts index 90708970..a95e05ab 100644 --- a/src/backend/providers/kissasian.ts +++ b/src/backend/providers/kissasian.ts @@ -1,7 +1,7 @@ import { MWEmbedType } from "../helpers/embed"; import { proxiedFetch } from "../helpers/fetch"; import { registerProvider } from "../helpers/register"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const kissasianBase = "https://kissasian.li"; diff --git a/src/backend/providers/m4ufree.ts b/src/backend/providers/m4ufree.ts index 0fe5303d..b9d5aef0 100644 --- a/src/backend/providers/m4ufree.ts +++ b/src/backend/providers/m4ufree.ts @@ -2,7 +2,7 @@ import { MWEmbed, MWEmbedType } from "@/backend/helpers/embed"; import { proxiedFetch } from "../helpers/fetch"; import { registerProvider } from "../helpers/register"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const HOST = "m4ufree.com"; const URL_BASE = `https://${HOST}`; diff --git a/src/backend/providers/netfilm.ts b/src/backend/providers/netfilm.ts index f7efcfbe..54016733 100644 --- a/src/backend/providers/netfilm.ts +++ b/src/backend/providers/netfilm.ts @@ -5,7 +5,7 @@ import { MWStreamQuality, MWStreamType, } from "../helpers/streams"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const netfilmBase = "https://net-film.vercel.app"; diff --git a/src/backend/providers/remotestream.ts b/src/backend/providers/remotestream.ts index 02c0f199..093069e8 100644 --- a/src/backend/providers/remotestream.ts +++ b/src/backend/providers/remotestream.ts @@ -1,7 +1,7 @@ import { mwFetch } from "@/backend/helpers/fetch"; import { registerProvider } from "@/backend/helpers/register"; import { MWStreamQuality, MWStreamType } from "@/backend/helpers/streams"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; const remotestreamBase = `https://fsa.remotestre.am`; diff --git a/src/backend/providers/sflix.ts b/src/backend/providers/sflix.ts index 4121046b..2cb1c598 100644 --- a/src/backend/providers/sflix.ts +++ b/src/backend/providers/sflix.ts @@ -1,7 +1,7 @@ import { proxiedFetch } from "../helpers/fetch"; import { registerProvider } from "../helpers/register"; import { MWStreamQuality, MWStreamType } from "../helpers/streams"; -import { MWMediaType } from "../metadata/types"; +import { MWMediaType } from "../metadata/types/mw"; const sflixBase = "https://sflix.video"; diff --git a/src/backend/providers/streamflix.ts b/src/backend/providers/streamflix.ts index 90dd4975..d4488b03 100644 --- a/src/backend/providers/streamflix.ts +++ b/src/backend/providers/streamflix.ts @@ -5,7 +5,7 @@ import { MWStreamQuality, MWStreamType, } from "@/backend/helpers/streams"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; const streamflixBase = "https://us-west2-compute-proxied.streamflix.one"; diff --git a/src/backend/providers/superstream/index.ts b/src/backend/providers/superstream/index.ts index 585d8d8a..75a8b844 100644 --- a/src/backend/providers/superstream/index.ts +++ b/src/backend/providers/superstream/index.ts @@ -13,7 +13,7 @@ import { MWStreamQuality, MWStreamType, } from "@/backend/helpers/streams"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { compareTitle } from "@/utils/titleMatch"; const nanoid = customAlphabet("0123456789abcdef", 32); diff --git a/src/components/SearchBar.tsx b/src/components/SearchBar.tsx index 4940cbc7..431de337 100644 --- a/src/components/SearchBar.tsx +++ b/src/components/SearchBar.tsx @@ -1,7 +1,7 @@ import { useState } from "react"; import { useTranslation } from "react-i18next"; -import { MWMediaType, MWQuery } from "@/backend/metadata/types"; +import { MWMediaType, MWQuery } from "@/backend/metadata/types/mw"; import { DropdownButton } from "./buttons/DropdownButton"; import { Icon, Icons } from "./Icon"; diff --git a/src/components/media/WatchedMediaCard.tsx b/src/components/media/WatchedMediaCard.tsx index 346c77b6..ade1612a 100644 --- a/src/components/media/WatchedMediaCard.tsx +++ b/src/components/media/WatchedMediaCard.tsx @@ -1,6 +1,6 @@ import { useMemo } from "react"; -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; import { useWatchedContext } from "@/state/watched"; import { MediaCard } from "./MediaCard"; diff --git a/src/hooks/useScrape.ts b/src/hooks/useScrape.ts index a375e618..3cffa4ee 100644 --- a/src/hooks/useScrape.ts +++ b/src/hooks/useScrape.ts @@ -3,7 +3,7 @@ import { useEffect, useState } from "react"; import { findBestStream } from "@/backend/helpers/scrape"; import { MWStream } from "@/backend/helpers/streams"; import { DetailedMeta } from "@/backend/metadata/getmeta"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; export interface ScrapeEventLog { type: "provider" | "embed"; diff --git a/src/hooks/useSearchQuery.ts b/src/hooks/useSearchQuery.ts index d431a0d0..cb8c3171 100644 --- a/src/hooks/useSearchQuery.ts +++ b/src/hooks/useSearchQuery.ts @@ -1,7 +1,7 @@ import { useState } from "react"; import { generatePath, useHistory, useRouteMatch } from "react-router-dom"; -import { MWMediaType, MWQuery } from "@/backend/metadata/types"; +import { MWMediaType, MWQuery } from "@/backend/metadata/types/mw"; function getInitialValue(params: { type: string; query: string }) { const type = diff --git a/src/setup/App.tsx b/src/setup/App.tsx index 7be4d581..d0a0887f 100644 --- a/src/setup/App.tsx +++ b/src/setup/App.tsx @@ -8,7 +8,7 @@ import { } from "react-router-dom"; import { convertLegacyUrl } from "@/backend/metadata/getmeta"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { BannerContextProvider } from "@/hooks/useBanner"; import { Layout } from "@/setup/Layout"; import { BookmarkContextProvider } from "@/state/bookmark"; diff --git a/src/state/bookmark/context.tsx b/src/state/bookmark/context.tsx index 9dca821f..692d4e76 100644 --- a/src/state/bookmark/context.tsx +++ b/src/state/bookmark/context.tsx @@ -1,6 +1,6 @@ import { ReactNode, createContext, useContext, useMemo } from "react"; -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; import { useStore } from "@/utils/storage"; import { BookmarkStore } from "./store"; diff --git a/src/state/bookmark/types.ts b/src/state/bookmark/types.ts index 05cb3641..79b92a5c 100644 --- a/src/state/bookmark/types.ts +++ b/src/state/bookmark/types.ts @@ -1,4 +1,4 @@ -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; export interface BookmarkStoreData { bookmarks: MWMediaMeta[]; diff --git a/src/state/watched/context.tsx b/src/state/watched/context.tsx index 3ce17b2a..661b0ed3 100644 --- a/src/state/watched/context.tsx +++ b/src/state/watched/context.tsx @@ -8,7 +8,7 @@ import { } from "react"; import { DetailedMeta } from "@/backend/metadata/getmeta"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { useStore } from "@/utils/storage"; import { VideoProgressStore } from "./store"; diff --git a/src/state/watched/migrations/v2.ts b/src/state/watched/migrations/v2.ts index 8f7a56b6..94f1141b 100644 --- a/src/state/watched/migrations/v2.ts +++ b/src/state/watched/migrations/v2.ts @@ -1,6 +1,6 @@ import { DetailedMeta, getMetaFromId } from "@/backend/metadata/getmeta"; import { searchForMedia } from "@/backend/metadata/search"; -import { MWMediaMeta, MWMediaType } from "@/backend/metadata/types"; +import { MWMediaMeta, MWMediaType } from "@/backend/metadata/types/mw"; import { compareTitle } from "@/utils/titleMatch"; import { WatchedStoreData, WatchedStoreItem } from "../types"; diff --git a/src/state/watched/types.ts b/src/state/watched/types.ts index a3246c38..0854b90b 100644 --- a/src/state/watched/types.ts +++ b/src/state/watched/types.ts @@ -1,4 +1,4 @@ -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; export interface StoreMediaItem { meta: MWMediaMeta; diff --git a/src/video/components/actions/DividerAction.tsx b/src/video/components/actions/DividerAction.tsx index 5778e16f..3aeaeaef 100644 --- a/src/video/components/actions/DividerAction.tsx +++ b/src/video/components/actions/DividerAction.tsx @@ -1,4 +1,4 @@ -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { useVideoPlayerDescriptor } from "@/video/state/hooks"; import { useMeta } from "@/video/state/logic/meta"; diff --git a/src/video/components/actions/SeriesSelectionAction.tsx b/src/video/components/actions/SeriesSelectionAction.tsx index d228b047..9eff0bb6 100644 --- a/src/video/components/actions/SeriesSelectionAction.tsx +++ b/src/video/components/actions/SeriesSelectionAction.tsx @@ -1,6 +1,6 @@ import { useTranslation } from "react-i18next"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { Icons } from "@/components/Icon"; import { FloatingAnchor } from "@/components/popout/FloatingAnchor"; import { VideoPlayerIconButton } from "@/video/components/parts/VideoPlayerIconButton"; diff --git a/src/video/components/controllers/MetaController.tsx b/src/video/components/controllers/MetaController.tsx index ee6bc696..25757e25 100644 --- a/src/video/components/controllers/MetaController.tsx +++ b/src/video/components/controllers/MetaController.tsx @@ -1,7 +1,7 @@ import { useEffect } from "react"; import { MWCaption } from "@/backend/helpers/streams"; -import { MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; +import { MWSeasonWithEpisodeMeta } from "@/backend/metadata/types/mw"; import { useVideoPlayerDescriptor } from "@/video/state/hooks"; import { useControls } from "@/video/state/logic/controls"; import { VideoPlayerMeta } from "@/video/state/types"; diff --git a/src/video/components/hooks/useCurrentSeriesEpisodeInfo.ts b/src/video/components/hooks/useCurrentSeriesEpisodeInfo.ts index 6eb51170..11dfdc88 100644 --- a/src/video/components/hooks/useCurrentSeriesEpisodeInfo.ts +++ b/src/video/components/hooks/useCurrentSeriesEpisodeInfo.ts @@ -1,7 +1,7 @@ import { useMemo } from "react"; import { useTranslation } from "react-i18next"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { useMeta } from "@/video/state/logic/meta"; export function useCurrentSeriesEpisodeInfo(descriptor: string) { diff --git a/src/video/components/parts/VideoErrorBoundary.tsx b/src/video/components/parts/VideoErrorBoundary.tsx index 5786aa7a..061bf2b7 100644 --- a/src/video/components/parts/VideoErrorBoundary.tsx +++ b/src/video/components/parts/VideoErrorBoundary.tsx @@ -2,7 +2,7 @@ import { Component } from "react"; import { Trans } from "react-i18next"; import type { ReactNode } from "react-router-dom/node_modules/@types/react/index"; -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; import { ErrorMessage } from "@/components/layout/ErrorBoundary"; import { Link } from "@/components/text/Link"; import { conf } from "@/setup/config"; diff --git a/src/video/components/parts/VideoPlayerHeader.tsx b/src/video/components/parts/VideoPlayerHeader.tsx index 8c026c49..3a333ee3 100644 --- a/src/video/components/parts/VideoPlayerHeader.tsx +++ b/src/video/components/parts/VideoPlayerHeader.tsx @@ -1,6 +1,6 @@ import { useTranslation } from "react-i18next"; -import { MWMediaMeta } from "@/backend/metadata/types"; +import { MWMediaMeta } from "@/backend/metadata/types/mw"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; import { BrandPill } from "@/components/layout/BrandPill"; diff --git a/src/video/components/popouts/EpisodeSelectionPopout.tsx b/src/video/components/popouts/EpisodeSelectionPopout.tsx index ce45c318..66c9ae49 100644 --- a/src/video/components/popouts/EpisodeSelectionPopout.tsx +++ b/src/video/components/popouts/EpisodeSelectionPopout.tsx @@ -3,7 +3,10 @@ import { useTranslation } from "react-i18next"; import { useParams } from "react-router-dom"; import { decodeTMDBId, getMetaFromId } from "@/backend/metadata/getmeta"; -import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; +import { + MWMediaType, + MWSeasonWithEpisodeMeta, +} from "@/backend/metadata/types/mw"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icon, Icons } from "@/components/Icon"; import { Loading } from "@/components/layout/Loading"; diff --git a/src/views/developer/VideoTesterView.tsx b/src/views/developer/VideoTesterView.tsx index b192cd40..e1b3fa35 100644 --- a/src/views/developer/VideoTesterView.tsx +++ b/src/views/developer/VideoTesterView.tsx @@ -3,7 +3,7 @@ import { Helmet } from "react-helmet"; import { MWStreamQuality, MWStreamType } from "@/backend/helpers/streams"; import { DetailedMeta } from "@/backend/metadata/getmeta"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { Button } from "@/components/Button"; import { Dropdown } from "@/components/Dropdown"; import { Navigation } from "@/components/layout/Navigation"; diff --git a/src/views/media/MediaView.tsx b/src/views/media/MediaView.tsx index 7ae1c01b..6e1659a6 100644 --- a/src/views/media/MediaView.tsx +++ b/src/views/media/MediaView.tsx @@ -9,7 +9,10 @@ import { decodeTMDBId, getMetaFromId, } from "@/backend/metadata/getmeta"; -import { MWMediaType, MWSeasonWithEpisodeMeta } from "@/backend/metadata/types"; +import { + MWMediaType, + MWSeasonWithEpisodeMeta, +} from "@/backend/metadata/types/mw"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icons } from "@/components/Icon"; import { Loading } from "@/components/layout/Loading"; diff --git a/src/views/other/v2Migration.tsx b/src/views/other/v2Migration.tsx index 1334ae26..d0b05e42 100644 --- a/src/views/other/v2Migration.tsx +++ b/src/views/other/v2Migration.tsx @@ -1,7 +1,7 @@ import pako from "pako"; import { useEffect, useState } from "react"; -import { MWMediaType } from "@/backend/metadata/types"; +import { MWMediaType } from "@/backend/metadata/types/mw"; import { conf } from "@/setup/config"; function fromBinary(str: string): Uint8Array { diff --git a/src/views/search/SearchResultsPartial.tsx b/src/views/search/SearchResultsPartial.tsx index 5769338b..e7cfc509 100644 --- a/src/views/search/SearchResultsPartial.tsx +++ b/src/views/search/SearchResultsPartial.tsx @@ -1,6 +1,6 @@ import { useEffect, useMemo, useState } from "react"; -import { MWQuery } from "@/backend/metadata/types"; +import { MWQuery } from "@/backend/metadata/types/mw"; import { useDebounce } from "@/hooks/useDebounce"; import { HomeView } from "./HomeView"; diff --git a/src/views/search/SearchResultsView.tsx b/src/views/search/SearchResultsView.tsx index 331d4f2d..f6507ef1 100644 --- a/src/views/search/SearchResultsView.tsx +++ b/src/views/search/SearchResultsView.tsx @@ -2,7 +2,7 @@ import { useEffect, useState } from "react"; import { useTranslation } from "react-i18next"; import { searchForMedia } from "@/backend/metadata/search"; -import { MWMediaMeta, MWQuery } from "@/backend/metadata/types"; +import { MWMediaMeta, MWQuery } from "@/backend/metadata/types/mw"; import { IconPatch } from "@/components/buttons/IconPatch"; import { Icons } from "@/components/Icon"; import { SectionHeading } from "@/components/layout/SectionHeading"; From 09f6a3125b7b3a6c0841f4c8408a724cfa153e97 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 13:54:34 +0200 Subject: [PATCH 87/96] clean up remnants from details fetch --- src/backend/metadata/search.ts | 18 +++++------------- src/backend/metadata/tmdb.ts | 31 ++++++++++++++++++------------- 2 files changed, 23 insertions(+), 26 deletions(-) diff --git a/src/backend/metadata/search.ts b/src/backend/metadata/search.ts index 99cb51ba..0d8f561f 100644 --- a/src/backend/metadata/search.ts +++ b/src/backend/metadata/search.ts @@ -7,7 +7,6 @@ import { searchMedia, } from "./tmdb"; import { MWMediaMeta, MWQuery } from "./types/mw"; -import { TMDBMovieResponse, TMDBShowResponse } from "./types/tmdb"; const cache = new SimpleCache(); cache.setCompare((a, b) => { @@ -19,18 +18,11 @@ export async function searchForMedia(query: MWQuery): Promise { if (cache.has(query)) return cache.get(query) as MWMediaMeta[]; const { searchQuery, type } = query; - const data = (await searchMedia(searchQuery, mediaTypeToTMDB(type))) as - | TMDBMovieResponse - | TMDBShowResponse; - const results = await Promise.all( - data.results.map(async (v) => { - const formattedResult = await formatTMDBSearchResult( - v, - mediaTypeToTMDB(type) - ); - return formatTMDBMeta(formattedResult); - }) - ); + const data = await searchMedia(searchQuery, mediaTypeToTMDB(type)); + const results = data.results.map((v) => { + const formattedResult = formatTMDBSearchResult(v, mediaTypeToTMDB(type)); + return formatTMDBMeta(formattedResult); + }); cache.set(query, results, 3600); // cache results for 1 hour return results; diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 4a9271ba..4c3259a3 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -186,23 +186,28 @@ export async function getExternalIds( return data; } -export async function formatTMDBSearchResult( +export function formatTMDBSearchResult( result: TMDBShowResult | TMDBMovieResult, mediatype: TMDBContentTypes -): Promise { +): TMDBMediaResult { const type = TMDBMediaToMediaType(mediatype); + if (type === MWMediaType.SERIES) { + const show = result as TMDBShowResult; + return { + title: show.name, + poster: getMediaPoster(show.poster_path), + id: show.id, + original_release_year: new Date(show.first_air_date).getFullYear(), + object_type: mediatype, + }; + } + const movie = result as TMDBMovieResult; return { - title: - type === MWMediaType.SERIES - ? (result as TMDBShowResult).name - : (result as TMDBMovieResult).title, - poster: getMediaPoster(result.poster_path), - id: result.id, - original_release_year: - type === MWMediaType.SERIES - ? Number((result as TMDBShowResult).first_air_date?.split("-")[0]) - : Number((result as TMDBMovieResult).release_date?.split("-")[0]), - object_type: mediaTypeToTMDB(type), + title: movie.title, + poster: getMediaPoster(movie.poster_path), + id: movie.id, + original_release_year: new Date(movie.release_date).getFullYear(), + object_type: mediatype, }; } From 1c17ef679ddb72889771ba488b416590ba1674cb Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 14:04:37 +0200 Subject: [PATCH 88/96] clean up requests --- src/backend/metadata/tmdb.ts | 25 +++++++++++++++++-------- 1 file changed, 17 insertions(+), 8 deletions(-) diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index 4c3259a3..f5d1e370 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -101,10 +101,13 @@ const headers = { Authorization: `Bearer ${conf().TMDB_API_KEY}`, }; -async function get(url: string): Promise { - const res = await mwFetch(url, { +async function get(url: string, params?: object): Promise { + const res = await mwFetch(encodeURI(url), { headers, baseURL, + params: { + ...params, + }, }); return res; } @@ -117,14 +120,20 @@ export async function searchMedia( switch (type) { case "movie": - data = await get( - `search/movie?query=${query}&include_adult=false&language=en-US&page=1` - ); + data = await get("search/movie", { + query, + include_adult: false, + language: "en-US", + page: 1, + }); break; case "show": - data = await get( - `search/tv?query=${query}&include_adult=false&language=en-US&page=1` - ); + data = await get("search/tv", { + query, + include_adult: false, + language: "en-US", + page: 1, + }); break; default: throw new Error("Invalid media type"); From f5f69ca7d4b923e997d1221b1acd9faa3836b811 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 15:14:48 +0200 Subject: [PATCH 89/96] default to season 1, with specials still playable --- src/backend/metadata/getmeta.ts | 28 +++++++++++++--------------- src/components/media/MediaCard.tsx | 17 +++++++++++------ 2 files changed, 24 insertions(+), 21 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index 21b6843a..b2166c34 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -96,24 +96,22 @@ export async function getMetaFromId( if (type === MWMediaType.SERIES) { const seasons = (details as TMDBShowData).seasons; - const season = - seasons?.find((v) => v.id.toString() === seasonId) ?? seasons?.[0]; - const episodes = await getEpisodes( - details.id.toString(), - season.season_number === null || season.season_number === 0 - ? 1 - : season.season_number - ); + let selectedSeason = seasons.find((v) => v.id.toString() === seasonId); + if (!selectedSeason) { + selectedSeason = seasons.find((v) => v.season_number === 1); + } + + if (selectedSeason) { + const episodes = await getEpisodes( + details.id.toString(), + selectedSeason.season_number + ); - if (season && episodes) { seasonData = { - id: season.id.toString(), - season_number: - season.season_number === null || season.season_number === 0 - ? 1 - : season.season_number, - title: season.name, + id: selectedSeason.id.toString(), + season_number: selectedSeason.season_number, + title: selectedSeason.name, episodes, }; } diff --git a/src/components/media/MediaCard.tsx b/src/components/media/MediaCard.tsx index c38b4a2b..a153d8b4 100644 --- a/src/components/media/MediaCard.tsx +++ b/src/components/media/MediaCard.tsx @@ -13,7 +13,7 @@ export interface MediaCardProps { linkable?: boolean; series?: { episode: number; - season: number; + season?: number; episodeId: string; seasonId: string; }; @@ -72,7 +72,7 @@ function MediaCardContent({ ].join(" ")} > {t("seasons.seasonAndEpisode", { - season: series.season, + season: series.season || 1, episode: series.episode, })}

@@ -134,10 +134,15 @@ export function MediaCard(props: MediaCardProps) { let link = canLink ? `/media/${encodeURIComponent(TMDBMediaToId(props.media))}` : "#"; - if (canLink && props.series) - link += `/${encodeURIComponent(props.series.seasonId)}/${encodeURIComponent( - props.series.episodeId - )}`; + if (canLink && props.series) { + if (props.series.season === 0 && !props.series.episodeId) { + link += `/${encodeURIComponent(props.series.seasonId)}`; + } else { + link += `/${encodeURIComponent( + props.series.seasonId + )}/${encodeURIComponent(props.series.episodeId)}`; + } + } if (!props.linkable) return {content}; return ( From 394271857f9fb78eda78ef0b26745680a0d8d2b3 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Wed, 21 Jun 2023 18:16:41 +0200 Subject: [PATCH 90/96] refactor and improve legacy redirect --- src/backend/metadata/getmeta.ts | 21 +++- src/backend/metadata/tmdb.ts | 14 +++ src/backend/metadata/types/tmdb.ts | 24 +++++ src/setup/App.tsx | 167 ++++++++++++++++------------- 4 files changed, 145 insertions(+), 81 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index b2166c34..fe2ea62b 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -8,6 +8,7 @@ import { getExternalIds, getMediaDetails, getMediaPoster, + getMovieFromExternalId, mediaTypeToTMDB, } from "./tmdb"; import { @@ -206,11 +207,23 @@ export async function convertLegacyUrl( if (url.startsWith("/media/JW")) { const urlParts = url.split("/").slice(2); const [, type, id] = urlParts[0].split("-", 3); - const meta = await getLegacyMetaFromId(TMDBMediaToMediaType(type), id); + + const mediaType = TMDBMediaToMediaType(type); + const meta = await getLegacyMetaFromId(mediaType, id); + if (!meta) return undefined; - const tmdbId = meta.tmdbId; - if (!tmdbId) return undefined; - return `/media/tmdb-${type}-${tmdbId}`; + const { tmdbId, imdbId } = meta; + if (!tmdbId && !imdbId) return undefined; + + // movies always have an imdb id on tmdb + if (imdbId && mediaType === MWMediaType.MOVIE) { + const movieId = await getMovieFromExternalId(imdbId); + if (movieId) return `/media/tmdb-movie-${movieId}`; + } + + if (tmdbId) { + return `/media/tmdb-${type}-${tmdbId}`; + } } return undefined; } diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index f5d1e370..db665528 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -2,6 +2,7 @@ import { conf } from "@/setup/config"; import { MWMediaMeta, MWMediaType, MWSeasonMeta } from "./types/mw"; import { + ExternalIdMovieSearchResult, TMDBContentTypes, TMDBEpisodeShort, TMDBExternalIds, @@ -195,6 +196,19 @@ export async function getExternalIds( return data; } +export async function getMovieFromExternalId( + imdbId: string +): Promise { + const data = await get(`/find/${imdbId}`, { + external_source: "imdb_id", + }); + + const movie = data.movie_results[0]; + if (!movie) return undefined; + + return movie.id.toString(); +} + export function formatTMDBSearchResult( result: TMDBShowResult | TMDBMovieResult, mediatype: TMDBContentTypes diff --git a/src/backend/metadata/types/tmdb.ts b/src/backend/metadata/types/tmdb.ts index cb5e9aa4..843786f4 100644 --- a/src/backend/metadata/types/tmdb.ts +++ b/src/backend/metadata/types/tmdb.ts @@ -282,3 +282,27 @@ export interface TMDBMovieExternalIds { } export type TMDBExternalIds = TMDBShowExternalIds | TMDBMovieExternalIds; + +export interface ExternalIdMovieSearchResult { + movie_results: { + adult: boolean; + backdrop_path: string; + id: number; + title: string; + original_language: string; + original_title: string; + overview: string; + poster_path: string; + media_type: string; + genre_ids: number[]; + popularity: number; + release_date: string; + video: boolean; + vote_average: number; + vote_count: number; + }[]; + person_results: any[]; + tv_results: any[]; + tv_episode_results: any[]; + tv_season_results: any[]; +} diff --git a/src/setup/App.tsx b/src/setup/App.tsx index d0a0887f..0516eb48 100644 --- a/src/setup/App.tsx +++ b/src/setup/App.tsx @@ -1,11 +1,5 @@ -import { lazy } from "react"; -import { - Redirect, - Route, - Switch, - useHistory, - useLocation, -} from "react-router-dom"; +import { lazy, useEffect, useState } from "react"; +import { Redirect, Route, Switch, useLocation } from "react-router-dom"; import { convertLegacyUrl } from "@/backend/metadata/getmeta"; import { MWMediaType } from "@/backend/metadata/types/mw"; @@ -19,86 +13,105 @@ import { NotFoundPage } from "@/views/notfound/NotFoundView"; import { V2MigrationView } from "@/views/other/v2Migration"; import { SearchView } from "@/views/search/SearchView"; -function App() { +// eslint-disable-next-line react/function-component-definition, react/prop-types +const LegacyUrlView: React.FC = ({ children }) => { const location = useLocation(); - const history = useHistory(); + const [redirectUrl, setRedirectUrl] = useState(null); - // Call the conversion function and redirect if necessary - convertLegacyUrl(location.pathname).then((convertedUrl) => { - if (convertedUrl) { - history.replace(convertedUrl); - } - }); + useEffect(() => { + // Call the conversion function and set the redirect URL if necessary + convertLegacyUrl(location.pathname).then((convertedUrl) => { + if (convertedUrl) { + setRedirectUrl(convertedUrl); + } + }); + }, [location.pathname]); + + if (redirectUrl) { + return ; + } + + // eslint-disable-next-line react/jsx-no-useless-fragment + return <>{children}; +}; + +function App() { return ( - - {/* functional routes */} - - - - + + + {/* functional routes */} + + + + - {/* pages */} - - - + {/* pages */} + + + - {/* other */} - import("@/views/developer/DeveloperView") - )} - /> - import("@/views/developer/VideoTesterView") - )} - /> - {/* developer routes that can abuse workers are disabled in production */} - {process.env.NODE_ENV === "development" ? ( - <> - import("@/views/developer/TestView") - )} - /> + {/* other */} + import("@/views/developer/DeveloperView") + )} + /> + import("@/views/developer/VideoTesterView") + )} + /> + {/* developer routes that can abuse workers are disabled in production */} + {process.env.NODE_ENV === "development" ? ( + <> + import("@/views/developer/TestView") + )} + /> - import("@/views/developer/ProviderTesterView") - )} - /> - import("@/views/developer/EmbedTesterView") - )} - /> - - ) : null} - - + import("@/views/developer/ProviderTesterView") + )} + /> + import("@/views/developer/EmbedTesterView") + )} + /> + + ) : null} + + + From f892a3037f39a1a685f5f1bf8db8babaf458c72d Mon Sep 17 00:00:00 2001 From: mrjvs Date: Wed, 21 Jun 2023 21:35:25 +0200 Subject: [PATCH 91/96] fix redirection issues --- src/backend/metadata/getmeta.ts | 38 ++++---- src/setup/App.tsx | 168 ++++++++++++++++---------------- 2 files changed, 104 insertions(+), 102 deletions(-) diff --git a/src/backend/metadata/getmeta.ts b/src/backend/metadata/getmeta.ts index fe2ea62b..c09d8292 100644 --- a/src/backend/metadata/getmeta.ts +++ b/src/backend/metadata/getmeta.ts @@ -201,29 +201,33 @@ export function decodeTMDBId( }; } +export function isLegacyUrl(url: string): boolean { + if (url.startsWith("/media/JW")) return true; + return false; +} + export async function convertLegacyUrl( url: string ): Promise { - if (url.startsWith("/media/JW")) { - const urlParts = url.split("/").slice(2); - const [, type, id] = urlParts[0].split("-", 3); + if (!isLegacyUrl(url)) return undefined; - const mediaType = TMDBMediaToMediaType(type); - const meta = await getLegacyMetaFromId(mediaType, id); + const urlParts = url.split("/").slice(2); + const [, type, id] = urlParts[0].split("-", 3); - if (!meta) return undefined; - const { tmdbId, imdbId } = meta; - if (!tmdbId && !imdbId) return undefined; + const mediaType = TMDBMediaToMediaType(type); + const meta = await getLegacyMetaFromId(mediaType, id); - // movies always have an imdb id on tmdb - if (imdbId && mediaType === MWMediaType.MOVIE) { - const movieId = await getMovieFromExternalId(imdbId); - if (movieId) return `/media/tmdb-movie-${movieId}`; - } + if (!meta) return undefined; + const { tmdbId, imdbId } = meta; + if (!tmdbId && !imdbId) return undefined; - if (tmdbId) { - return `/media/tmdb-${type}-${tmdbId}`; - } + // movies always have an imdb id on tmdb + if (imdbId && mediaType === MWMediaType.MOVIE) { + const movieId = await getMovieFromExternalId(imdbId); + if (movieId) return `/media/tmdb-movie-${movieId}`; + } + + if (tmdbId) { + return `/media/tmdb-${type}-${tmdbId}`; } - return undefined; } diff --git a/src/setup/App.tsx b/src/setup/App.tsx index 0516eb48..7d1847ae 100644 --- a/src/setup/App.tsx +++ b/src/setup/App.tsx @@ -1,7 +1,13 @@ -import { lazy, useEffect, useState } from "react"; -import { Redirect, Route, Switch, useLocation } from "react-router-dom"; +import { ReactElement, lazy, useEffect } from "react"; +import { + Redirect, + Route, + Switch, + useHistory, + useLocation, +} from "react-router-dom"; -import { convertLegacyUrl } from "@/backend/metadata/getmeta"; +import { convertLegacyUrl, isLegacyUrl } from "@/backend/metadata/getmeta"; import { MWMediaType } from "@/backend/metadata/types/mw"; import { BannerContextProvider } from "@/hooks/useBanner"; import { Layout } from "@/setup/Layout"; @@ -13,27 +19,21 @@ import { NotFoundPage } from "@/views/notfound/NotFoundView"; import { V2MigrationView } from "@/views/other/v2Migration"; import { SearchView } from "@/views/search/SearchView"; -// eslint-disable-next-line react/function-component-definition, react/prop-types -const LegacyUrlView: React.FC = ({ children }) => { +function LegacyUrlView({ children }: { children: ReactElement }) { const location = useLocation(); - const [redirectUrl, setRedirectUrl] = useState(null); + const { replace } = useHistory(); useEffect(() => { - // Call the conversion function and set the redirect URL if necessary + const url = location.pathname; + if (!isLegacyUrl(url)) return; convertLegacyUrl(location.pathname).then((convertedUrl) => { - if (convertedUrl) { - setRedirectUrl(convertedUrl); - } + replace(convertedUrl ?? "/"); }); - }, [location.pathname]); + }, [location.pathname, replace]); - if (redirectUrl) { - return ; - } - - // eslint-disable-next-line react/jsx-no-useless-fragment - return <>{children}; -}; + if (isLegacyUrl(location.pathname)) return null; + return children; +} function App() { return ( @@ -42,76 +42,74 @@ function App() { - - - {/* functional routes */} - - - - + + {/* functional routes */} + + + + - {/* pages */} - - - + {/* pages */} + + + + + + + + + + + - {/* other */} - import("@/views/developer/DeveloperView") - )} - /> - import("@/views/developer/VideoTesterView") - )} - /> - {/* developer routes that can abuse workers are disabled in production */} - {process.env.NODE_ENV === "development" ? ( - <> - import("@/views/developer/TestView") - )} - /> + {/* other */} + import("@/views/developer/DeveloperView") + )} + /> + import("@/views/developer/VideoTesterView") + )} + /> + {/* developer routes that can abuse workers are disabled in production */} + {process.env.NODE_ENV === "development" ? ( + <> + import("@/views/developer/TestView") + )} + /> - import("@/views/developer/ProviderTesterView") - )} - /> - import("@/views/developer/EmbedTesterView") - )} - /> - - ) : null} - - - + import("@/views/developer/ProviderTesterView") + )} + /> + import("@/views/developer/EmbedTesterView") + )} + /> + + ) : null} + + From 9fbba7ea55f2a8eedb193b402f56a17edd6ac51e Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Thu, 22 Jun 2023 10:47:14 +0200 Subject: [PATCH 92/96] localstorage migration --- src/state/bookmark/store.ts | 7 +++ src/state/watched/migrations/v3.ts | 87 ++++++++++++++++++++++++++++++ src/state/watched/store.ts | 7 +++ 3 files changed, 101 insertions(+) create mode 100644 src/state/watched/migrations/v3.ts diff --git a/src/state/bookmark/store.ts b/src/state/bookmark/store.ts index 1b7a2053..51de0ed0 100644 --- a/src/state/bookmark/store.ts +++ b/src/state/bookmark/store.ts @@ -2,6 +2,7 @@ import { createVersionedStore } from "@/utils/storage"; import { BookmarkStoreData } from "./types"; import { OldBookmarks, migrateV1Bookmarks } from "../watched/migrations/v2"; +import { migrateV2Bookmarks } from "../watched/migrations/v3"; export const BookmarkStore = createVersionedStore() .setKey("mw-bookmarks") @@ -13,6 +14,12 @@ export const BookmarkStore = createVersionedStore() }) .addVersion({ version: 1, + migrate(old: OldBookmarks) { + return migrateV2Bookmarks(old); + }, + }) + .addVersion({ + version: 2, create() { return { bookmarks: [], diff --git a/src/state/watched/migrations/v3.ts b/src/state/watched/migrations/v3.ts new file mode 100644 index 00000000..971dacf1 --- /dev/null +++ b/src/state/watched/migrations/v3.ts @@ -0,0 +1,87 @@ +import { getLegacyMetaFromId } from "@/backend/metadata/getmeta"; +import { getMovieFromExternalId } from "@/backend/metadata/tmdb"; +import { MWMediaType } from "@/backend/metadata/types/mw"; + +import { WatchedStoreData } from "../types"; + +async function migrateId( + id: number, + type: MWMediaType +): Promise { + console.log("migrating id", id, type); + const meta = await getLegacyMetaFromId(type, id.toString()); + console.log("migrating id", meta); + + if (!meta) return undefined; + const { tmdbId, imdbId } = meta; + if (!tmdbId && !imdbId) return undefined; + + // movies always have an imdb id on tmdb + if (imdbId && type === MWMediaType.MOVIE) { + const movieId = await getMovieFromExternalId(imdbId); + if (movieId) return movieId; + } + + if (tmdbId) { + return tmdbId; + } +} + +export async function migrateV2Bookmarks(old: any) { + const oldData = old; + if (!oldData) return; + + const updatedBookmarks = oldData.bookmarks.map( + async (item: { id: number; type: MWMediaType }) => ({ + ...item, + mediaId: await migrateId(item.id, item.type), + }) + ); + + return { + bookmarks: (await Promise.all(updatedBookmarks)).filter( + (item) => item.mediaId + ), + }; +} + +export async function migrateV3Videos(old: any) { + console.log("migrating watched"); + const oldData = old; + if (!oldData) return; + console.log(oldData); + + const updatedItems = await Promise.all( + oldData.items.map(async (item: any) => { + const migratedId = await migrateId( + item.item.meta.id, + item.item.meta.type + ); + + const migratedItem = { + ...item, + item: { + ...item.item, + meta: { + ...item.item.meta, + id: migratedId, + }, + }, + }; + + return { + ...item, + item: migratedId ? migratedItem : item.item, + }; + }) + ); + + const newData: WatchedStoreData = { + items: updatedItems.map((item) => item.item), // Extract the "item" object + }; + + return { + ...oldData, + items: newData.items, + }; +} diff --git a/src/state/watched/store.ts b/src/state/watched/store.ts index 95adef28..b59c37dc 100644 --- a/src/state/watched/store.ts +++ b/src/state/watched/store.ts @@ -1,6 +1,7 @@ import { createVersionedStore } from "@/utils/storage"; import { OldData, migrateV2Videos } from "./migrations/v2"; +import { migrateV3Videos } from "./migrations/v3"; import { WatchedStoreData } from "./types"; export const VideoProgressStore = createVersionedStore() @@ -21,6 +22,12 @@ export const VideoProgressStore = createVersionedStore() }) .addVersion({ version: 2, + migrate(old: OldData) { + return migrateV3Videos(old); + }, + }) + .addVersion({ + version: 3, create() { return { items: [], From e0bf711a79b6e2a3fc9e6c1b634b18dbad369e18 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Thu, 22 Jun 2023 10:48:00 +0200 Subject: [PATCH 93/96] cleanup --- src/state/watched/migrations/v3.ts | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/src/state/watched/migrations/v3.ts b/src/state/watched/migrations/v3.ts index 971dacf1..cbf081e2 100644 --- a/src/state/watched/migrations/v3.ts +++ b/src/state/watched/migrations/v3.ts @@ -8,9 +8,7 @@ async function migrateId( id: number, type: MWMediaType ): Promise { - console.log("migrating id", id, type); const meta = await getLegacyMetaFromId(type, id.toString()); - console.log("migrating id", meta); if (!meta) return undefined; const { tmdbId, imdbId } = meta; @@ -46,10 +44,8 @@ export async function migrateV2Bookmarks(old: any) { } export async function migrateV3Videos(old: any) { - console.log("migrating watched"); const oldData = old; if (!oldData) return; - console.log(oldData); const updatedItems = await Promise.all( oldData.items.map(async (item: any) => { @@ -77,7 +73,7 @@ export async function migrateV3Videos(old: any) { ); const newData: WatchedStoreData = { - items: updatedItems.map((item) => item.item), // Extract the "item" object + items: updatedItems.map((item) => item.item), }; return { From 845fd935979aa3006384fe53a83a460a0716fa18 Mon Sep 17 00:00:00 2001 From: adrifcastr Date: Thu, 22 Jun 2023 20:29:10 +0200 Subject: [PATCH 94/96] fix small oversight --- src/state/watched/migrations/v3.ts | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/state/watched/migrations/v3.ts b/src/state/watched/migrations/v3.ts index cbf081e2..71e0b182 100644 --- a/src/state/watched/migrations/v3.ts +++ b/src/state/watched/migrations/v3.ts @@ -32,14 +32,12 @@ export async function migrateV2Bookmarks(old: any) { const updatedBookmarks = oldData.bookmarks.map( async (item: { id: number; type: MWMediaType }) => ({ ...item, - mediaId: await migrateId(item.id, item.type), + id: await migrateId(item.id, item.type), }) ); return { - bookmarks: (await Promise.all(updatedBookmarks)).filter( - (item) => item.mediaId - ), + bookmarks: (await Promise.all(updatedBookmarks)).filter((item) => item.id), }; } From 545120d5cc6c5fc4fcbad00a2ea6bc7a99dc4970 Mon Sep 17 00:00:00 2001 From: mrjvs Date: Thu, 22 Jun 2023 20:58:44 +0200 Subject: [PATCH 95/96] bump version --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 43436852..3f7c4bf9 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "movie-web", - "version": "3.0.15", + "version": "3.1.0", "private": true, "homepage": "https://movie-web.app", "dependencies": { From c4c7816543900173e17bde50826bd4f397ae21d6 Mon Sep 17 00:00:00 2001 From: mrjvs Date: Thu, 22 Jun 2023 22:37:16 +0200 Subject: [PATCH 96/96] migrations but better Co-authored-by: William Oldham --- src/backend/metadata/tmdb.ts | 29 ++++--- src/backend/providers/gomovies.ts | 2 +- src/backend/providers/superstream/index.ts | 2 +- src/state/bookmark/store.ts | 2 +- src/state/watched/migrations/v3.ts | 92 ++++++++++++---------- src/state/watched/store.ts | 2 +- src/utils/storage.ts | 7 +- src/utils/typeguard.ts | 3 + 8 files changed, 79 insertions(+), 60 deletions(-) create mode 100644 src/utils/typeguard.ts diff --git a/src/backend/metadata/tmdb.ts b/src/backend/metadata/tmdb.ts index db665528..1c442028 100644 --- a/src/backend/metadata/tmdb.ts +++ b/src/backend/metadata/tmdb.ts @@ -143,21 +143,24 @@ export async function searchMedia( return data; } -export async function getMediaDetails(id: string, type: TMDBContentTypes) { - let data; +// Conditional type which for inferring the return type based on the content type +type MediaDetailReturn = T extends "movie" + ? TMDBMovieData + : T extends "show" + ? TMDBShowData + : never; - switch (type) { - case "movie": - data = await get(`/movie/${id}`); - break; - case "show": - data = await get(`/tv/${id}`); - break; - default: - throw new Error("Invalid media type"); +export function getMediaDetails< + T extends TMDBContentTypes, + TReturn = MediaDetailReturn +>(id: string, type: T): Promise { + if (type === "movie") { + return get(`/movie/${id}`); } - - return data; + if (type === "show") { + return get(`/tv/${id}`); + } + throw new Error("Invalid media type"); } export function getMediaPoster(posterPath: string | null): string | undefined { diff --git a/src/backend/providers/gomovies.ts b/src/backend/providers/gomovies.ts index fdce289b..ddd43509 100644 --- a/src/backend/providers/gomovies.ts +++ b/src/backend/providers/gomovies.ts @@ -8,7 +8,7 @@ const gomoviesBase = "https://gomovies.sx"; registerProvider({ id: "gomovies", displayName: "GOmovies", - rank: 300, + rank: 200, type: [MWMediaType.MOVIE, MWMediaType.SERIES], async scrape({ media, episode }) { diff --git a/src/backend/providers/superstream/index.ts b/src/backend/providers/superstream/index.ts index 75a8b844..5af85cb9 100644 --- a/src/backend/providers/superstream/index.ts +++ b/src/backend/providers/superstream/index.ts @@ -142,7 +142,7 @@ const convertSubtitles = (subtitleGroup: any): MWCaption | null => { registerProvider({ id: "superstream", displayName: "Superstream", - rank: 200, + rank: 300, type: [MWMediaType.MOVIE, MWMediaType.SERIES], async scrape({ media, episode, progress }) { diff --git a/src/state/bookmark/store.ts b/src/state/bookmark/store.ts index 51de0ed0..b2020020 100644 --- a/src/state/bookmark/store.ts +++ b/src/state/bookmark/store.ts @@ -14,7 +14,7 @@ export const BookmarkStore = createVersionedStore() }) .addVersion({ version: 1, - migrate(old: OldBookmarks) { + migrate(old: BookmarkStoreData) { return migrateV2Bookmarks(old); }, }) diff --git a/src/state/watched/migrations/v3.ts b/src/state/watched/migrations/v3.ts index 71e0b182..dffae637 100644 --- a/src/state/watched/migrations/v3.ts +++ b/src/state/watched/migrations/v3.ts @@ -1,14 +1,20 @@ import { getLegacyMetaFromId } from "@/backend/metadata/getmeta"; -import { getMovieFromExternalId } from "@/backend/metadata/tmdb"; +import { + getEpisodes, + getMediaDetails, + getMovieFromExternalId, +} from "@/backend/metadata/tmdb"; import { MWMediaType } from "@/backend/metadata/types/mw"; +import { BookmarkStoreData } from "@/state/bookmark/types"; +import { isNotNull } from "@/utils/typeguard"; import { WatchedStoreData } from "../types"; async function migrateId( - id: number, + id: string, type: MWMediaType ): Promise { - const meta = await getLegacyMetaFromId(type, id.toString()); + const meta = await getLegacyMetaFromId(type, id); if (!meta) return undefined; const { tmdbId, imdbId } = meta; @@ -25,57 +31,59 @@ async function migrateId( } } -export async function migrateV2Bookmarks(old: any) { - const oldData = old; - if (!oldData) return; - - const updatedBookmarks = oldData.bookmarks.map( - async (item: { id: number; type: MWMediaType }) => ({ - ...item, - id: await migrateId(item.id, item.type), - }) - ); +export async function migrateV2Bookmarks(old: BookmarkStoreData) { + const updatedBookmarks = old.bookmarks.map(async (item) => ({ + ...item, + id: await migrateId(item.id, item.type).catch(() => undefined), + })); return { bookmarks: (await Promise.all(updatedBookmarks)).filter((item) => item.id), }; } -export async function migrateV3Videos(old: any) { - const oldData = old; - if (!oldData) return; - +export async function migrateV3Videos( + old: WatchedStoreData +): Promise { const updatedItems = await Promise.all( - oldData.items.map(async (item: any) => { - const migratedId = await migrateId( - item.item.meta.id, - item.item.meta.type - ); + old.items.map(async (progress) => { + try { + const migratedId = await migrateId( + progress.item.meta.id, + progress.item.meta.type + ); - const migratedItem = { - ...item, - item: { - ...item.item, - meta: { - ...item.item.meta, - id: migratedId, - }, - }, - }; + if (!migratedId) return null; - return { - ...item, - item: migratedId ? migratedItem : item.item, - }; + const clone = structuredClone(progress); + clone.item.meta.id = migratedId; + if (clone.item.series) { + const series = clone.item.series; + const details = await getMediaDetails(migratedId, "show"); + + const season = details.seasons.find( + (v) => v.season_number === series.season + ); + if (!season) return null; + + const episodes = await getEpisodes(migratedId, season.season_number); + const episode = episodes.find( + (v) => v.episode_number === series.episode + ); + if (!episode) return null; + + clone.item.series.episodeId = episode.id.toString(); + clone.item.series.seasonId = season.id.toString(); + } + + return clone; + } catch (err) { + return null; + } }) ); - const newData: WatchedStoreData = { - items: updatedItems.map((item) => item.item), - }; - return { - ...oldData, - items: newData.items, + items: updatedItems.filter(isNotNull), }; } diff --git a/src/state/watched/store.ts b/src/state/watched/store.ts index b59c37dc..c11e3f59 100644 --- a/src/state/watched/store.ts +++ b/src/state/watched/store.ts @@ -22,7 +22,7 @@ export const VideoProgressStore = createVersionedStore() }) .addVersion({ version: 2, - migrate(old: OldData) { + migrate(old: WatchedStoreData) { return migrateV3Videos(old); }, }) diff --git a/src/utils/storage.ts b/src/utils/storage.ts index f48e0245..83057d54 100644 --- a/src/utils/storage.ts +++ b/src/utils/storage.ts @@ -46,8 +46,13 @@ export async function initializeStores() { let mostRecentData = data; try { for (const version of relevantVersions) { - if (version.migrate) + if (version.migrate) { + localStorage.setItem( + `BACKUP-v${version.version}-${internal.key}`, + JSON.stringify(mostRecentData) + ); mostRecentData = await version.migrate(mostRecentData); + } } } catch (err) { console.error(`FAILED TO MIGRATE STORE ${internal.key}`, err); diff --git a/src/utils/typeguard.ts b/src/utils/typeguard.ts new file mode 100644 index 00000000..95dd81a1 --- /dev/null +++ b/src/utils/typeguard.ts @@ -0,0 +1,3 @@ +export function isNotNull(obj: T | null): obj is T { + return obj != null; +}