2023-01-15 16:01:07 +01:00
|
|
|
import { FetchError } from "ofetch";
|
2023-04-24 18:41:54 +03:00
|
|
|
|
2023-06-12 21:25:24 +02:00
|
|
|
import { formatJWMeta, mediaTypeToJW } from "./justwatch";
|
2023-06-13 11:01:07 +02:00
|
|
|
import {
|
2023-06-13 21:23:47 +02:00
|
|
|
TMDBMediaToMediaType,
|
|
|
|
formatTMDBMeta,
|
2023-06-21 13:07:33 +02:00
|
|
|
getEpisodes,
|
|
|
|
getExternalIds,
|
|
|
|
getMediaDetails,
|
2023-06-21 13:26:03 +02:00
|
|
|
getMediaPoster,
|
2023-06-21 18:16:41 +02:00
|
|
|
getMovieFromExternalId,
|
2023-06-13 21:23:47 +02:00
|
|
|
mediaTypeToTMDB,
|
|
|
|
} from "./tmdb";
|
2023-01-15 16:01:07 +01:00
|
|
|
import {
|
|
|
|
JWMediaResult,
|
2023-01-22 19:26:08 +01:00
|
|
|
JWSeasonMetaResult,
|
2023-01-15 16:01:07 +01:00
|
|
|
JW_API_BASE,
|
2023-06-21 13:23:39 +02:00
|
|
|
} from "./types/justwatch";
|
|
|
|
import { MWMediaMeta, MWMediaType } from "./types/mw";
|
|
|
|
import {
|
2023-06-13 21:23:47 +02:00
|
|
|
TMDBMediaResult,
|
2023-06-13 10:41:54 +02:00
|
|
|
TMDBMovieData,
|
2023-06-13 21:23:47 +02:00
|
|
|
TMDBSeasonMetaResult,
|
2023-06-13 10:41:54 +02:00
|
|
|
TMDBShowData,
|
2023-06-21 13:23:39 +02:00
|
|
|
} from "./types/tmdb";
|
2023-04-24 18:41:54 +03:00
|
|
|
import { makeUrl, proxiedFetch } from "../helpers/fetch";
|
2023-01-14 00:12:56 +01:00
|
|
|
|
|
|
|
type JWExternalIdType =
|
|
|
|
| "eidr"
|
|
|
|
| "imdb_latest"
|
|
|
|
| "imdb"
|
|
|
|
| "tmdb_latest"
|
|
|
|
| "tmdb"
|
|
|
|
| "tms";
|
|
|
|
|
|
|
|
interface JWExternalId {
|
|
|
|
provider: JWExternalIdType;
|
|
|
|
external_id: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface JWDetailedMeta extends JWMediaResult {
|
|
|
|
external_ids: JWExternalId[];
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface DetailedMeta {
|
|
|
|
meta: MWMediaMeta;
|
2023-05-21 18:12:45 +02:00
|
|
|
imdbId?: string;
|
2023-05-21 21:00:35 +02:00
|
|
|
tmdbId?: string;
|
2023-01-14 00:12:56 +01:00
|
|
|
}
|
|
|
|
|
2023-06-15 22:13:19 +02:00
|
|
|
export function formatTMDBMetaResult(
|
2023-06-14 07:48:31 +02:00
|
|
|
details: TMDBShowData | TMDBMovieData,
|
|
|
|
type: MWMediaType
|
2023-06-21 12:43:36 +02:00
|
|
|
): TMDBMediaResult {
|
2023-06-14 07:48:31 +02:00
|
|
|
if (type === MWMediaType.MOVIE) {
|
2023-06-21 12:47:09 +02:00
|
|
|
const movie = details as TMDBMovieData;
|
2023-06-21 12:48:33 +02:00
|
|
|
return {
|
2023-06-14 07:48:31 +02:00
|
|
|
id: details.id,
|
2023-06-21 12:47:09 +02:00
|
|
|
title: movie.title,
|
2023-06-14 07:48:31 +02:00
|
|
|
object_type: mediaTypeToTMDB(type),
|
2023-06-21 13:26:03 +02:00
|
|
|
poster: getMediaPoster(movie.poster_path) ?? undefined,
|
2023-06-21 12:50:41 +02:00
|
|
|
original_release_year: new Date(movie.release_date).getFullYear(),
|
2023-06-14 07:48:31 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
if (type === MWMediaType.SERIES) {
|
2023-06-21 12:47:09 +02:00
|
|
|
const show = details as TMDBShowData;
|
2023-06-21 12:48:33 +02:00
|
|
|
return {
|
2023-06-14 07:48:31 +02:00
|
|
|
id: details.id,
|
2023-06-21 12:47:09 +02:00
|
|
|
title: show.name,
|
2023-06-14 07:48:31 +02:00
|
|
|
object_type: mediaTypeToTMDB(type),
|
2023-06-21 12:47:09 +02:00
|
|
|
seasons: show.seasons.map((v) => ({
|
2023-06-14 07:48:31 +02:00
|
|
|
id: v.id,
|
|
|
|
season_number: v.season_number,
|
|
|
|
title: v.name,
|
|
|
|
})),
|
2023-06-15 11:06:24 +02:00
|
|
|
poster: (details as TMDBMovieData).poster_path ?? undefined,
|
2023-06-21 12:50:41 +02:00
|
|
|
original_release_year: new Date(show.first_air_date).getFullYear(),
|
2023-06-14 07:48:31 +02:00
|
|
|
};
|
|
|
|
}
|
2023-06-21 12:43:36 +02:00
|
|
|
|
2023-06-21 12:48:33 +02:00
|
|
|
throw new Error("unsupported type");
|
2023-06-14 07:48:31 +02:00
|
|
|
}
|
|
|
|
|
2023-01-14 00:12:56 +01:00
|
|
|
export async function getMetaFromId(
|
|
|
|
type: MWMediaType,
|
2023-01-22 19:26:08 +01:00
|
|
|
id: string,
|
|
|
|
seasonId?: string
|
2023-06-13 10:41:54 +02:00
|
|
|
): Promise<DetailedMeta | null> {
|
2023-06-21 13:07:33 +02:00
|
|
|
const details = await getMediaDetails(id, mediaTypeToTMDB(type));
|
2023-06-13 10:41:54 +02:00
|
|
|
|
|
|
|
if (!details) return null;
|
|
|
|
|
2023-06-21 13:07:33 +02:00
|
|
|
const externalIds = await getExternalIds(id, mediaTypeToTMDB(type));
|
2023-06-16 11:18:32 +02:00
|
|
|
const imdbId = externalIds.imdb_id ?? undefined;
|
2023-06-13 10:41:54 +02:00
|
|
|
|
2023-06-13 21:23:47 +02:00
|
|
|
let seasonData: TMDBSeasonMetaResult | undefined;
|
2023-06-13 10:41:54 +02:00
|
|
|
|
|
|
|
if (type === MWMediaType.SERIES) {
|
|
|
|
const seasons = (details as TMDBShowData).seasons;
|
|
|
|
|
2023-06-21 15:14:48 +02:00
|
|
|
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
|
|
|
|
);
|
2023-06-13 10:41:54 +02:00
|
|
|
|
|
|
|
seasonData = {
|
2023-06-21 15:14:48 +02:00
|
|
|
id: selectedSeason.id.toString(),
|
|
|
|
season_number: selectedSeason.season_number,
|
|
|
|
title: selectedSeason.name,
|
2023-06-13 10:41:54 +02:00
|
|
|
episodes,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-15 22:13:19 +02:00
|
|
|
const tmdbmeta = formatTMDBMetaResult(details, type);
|
2023-06-14 07:48:31 +02:00
|
|
|
if (!tmdbmeta) return null;
|
2023-06-13 21:23:47 +02:00
|
|
|
const meta = formatTMDBMeta(tmdbmeta, seasonData);
|
|
|
|
if (!meta) return null;
|
2023-06-13 10:41:54 +02:00
|
|
|
|
|
|
|
return {
|
|
|
|
meta,
|
|
|
|
imdbId,
|
|
|
|
tmdbId: id,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getLegacyMetaFromId(
|
|
|
|
type: MWMediaType,
|
|
|
|
id: string,
|
|
|
|
seasonId?: string
|
2023-01-15 16:01:07 +01:00
|
|
|
): Promise<DetailedMeta | null> {
|
|
|
|
const queryType = mediaTypeToJW(type);
|
|
|
|
|
|
|
|
let data: JWDetailedMeta;
|
|
|
|
try {
|
|
|
|
const url = makeUrl("/content/titles/{type}/{id}/locale/en_US", {
|
|
|
|
type: queryType,
|
|
|
|
id,
|
|
|
|
});
|
2023-01-22 20:51:58 +01:00
|
|
|
data = await proxiedFetch<JWDetailedMeta>(url, { baseURL: JW_API_BASE });
|
2023-01-15 16:01:07 +01:00
|
|
|
} catch (err) {
|
|
|
|
if (err instanceof FetchError) {
|
|
|
|
// 400 and 404 are treated as not found
|
|
|
|
if (err.statusCode === 400 || err.statusCode === 404) return null;
|
|
|
|
}
|
|
|
|
throw err;
|
|
|
|
}
|
2023-01-14 00:12:56 +01:00
|
|
|
|
2023-03-10 20:54:56 +01:00
|
|
|
let imdbId = data.external_ids.find(
|
2023-01-14 00:12:56 +01:00
|
|
|
(v) => v.provider === "imdb_latest"
|
2023-03-10 20:59:10 +01:00
|
|
|
)?.external_id;
|
2023-03-10 20:54:56 +01:00
|
|
|
if (!imdbId)
|
2023-03-10 20:59:10 +01:00
|
|
|
imdbId = data.external_ids.find((v) => v.provider === "imdb")?.external_id;
|
2023-03-10 20:54:56 +01:00
|
|
|
|
2023-05-21 21:00:35 +02:00
|
|
|
let tmdbId = data.external_ids.find(
|
|
|
|
(v) => v.provider === "tmdb_latest"
|
|
|
|
)?.external_id;
|
|
|
|
if (!tmdbId)
|
|
|
|
tmdbId = data.external_ids.find((v) => v.provider === "tmdb")?.external_id;
|
|
|
|
|
2023-01-22 19:26:08 +01:00
|
|
|
let seasonData: JWSeasonMetaResult | undefined;
|
|
|
|
if (data.object_type === "show") {
|
|
|
|
const seasonToScrape = seasonId ?? data.seasons?.[0].id.toString() ?? "";
|
|
|
|
const url = makeUrl("/content/titles/show_season/{id}/locale/en_US", {
|
|
|
|
id: seasonToScrape,
|
|
|
|
});
|
2023-01-22 20:51:58 +01:00
|
|
|
seasonData = await proxiedFetch<any>(url, { baseURL: JW_API_BASE });
|
2023-01-22 19:26:08 +01:00
|
|
|
}
|
|
|
|
|
2023-01-14 00:12:56 +01:00
|
|
|
return {
|
2023-01-22 19:26:08 +01:00
|
|
|
meta: formatJWMeta(data, seasonData),
|
2023-01-14 00:12:56 +01:00
|
|
|
imdbId,
|
2023-05-21 21:00:35 +02:00
|
|
|
tmdbId,
|
2023-01-14 00:12:56 +01:00
|
|
|
};
|
|
|
|
}
|
2023-06-13 11:01:07 +02:00
|
|
|
|
2023-06-13 21:23:47 +02:00
|
|
|
export function TMDBMediaToId(media: MWMediaMeta): string {
|
|
|
|
return ["tmdb", mediaTypeToTMDB(media.type), media.id].join("-");
|
2023-06-13 11:01:07 +02:00
|
|
|
}
|
|
|
|
|
2023-06-13 21:23:47 +02:00
|
|
|
export function decodeTMDBId(
|
2023-06-13 11:01:07 +02:00
|
|
|
paramId: string
|
|
|
|
): { id: string; type: MWMediaType } | null {
|
|
|
|
const [prefix, type, id] = paramId.split("-", 3);
|
2023-06-13 21:23:47 +02:00
|
|
|
if (prefix !== "tmdb") return null;
|
2023-06-13 11:01:07 +02:00
|
|
|
let mediaType;
|
|
|
|
try {
|
2023-06-13 21:23:47 +02:00
|
|
|
mediaType = TMDBMediaToMediaType(type);
|
2023-06-13 11:01:07 +02:00
|
|
|
} catch {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
type: mediaType,
|
|
|
|
id,
|
|
|
|
};
|
|
|
|
}
|
2023-06-13 14:06:37 +02:00
|
|
|
|
|
|
|
export async function convertLegacyUrl(
|
|
|
|
url: string
|
|
|
|
): Promise<string | undefined> {
|
|
|
|
if (url.startsWith("/media/JW")) {
|
|
|
|
const urlParts = url.split("/").slice(2);
|
|
|
|
const [, type, id] = urlParts[0].split("-", 3);
|
2023-06-21 18:16:41 +02:00
|
|
|
|
|
|
|
const mediaType = TMDBMediaToMediaType(type);
|
|
|
|
const meta = await getLegacyMetaFromId(mediaType, id);
|
|
|
|
|
2023-06-13 14:06:37 +02:00
|
|
|
if (!meta) return undefined;
|
2023-06-21 18:16:41 +02:00
|
|
|
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}`;
|
|
|
|
}
|
2023-06-13 14:06:37 +02:00
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|