2023-01-15 15:01:07 +00:00
|
|
|
import { FetchError } from "ofetch";
|
2023-01-22 19:51:58 +00:00
|
|
|
import { makeUrl, proxiedFetch } from "../helpers/fetch";
|
2023-01-15 15:01:07 +00:00
|
|
|
import {
|
|
|
|
formatJWMeta,
|
|
|
|
JWMediaResult,
|
2023-01-22 18:26:08 +00:00
|
|
|
JWSeasonMetaResult,
|
2023-01-15 15:01:07 +00:00
|
|
|
JW_API_BASE,
|
|
|
|
mediaTypeToJW,
|
|
|
|
} from "./justwatch";
|
2023-01-13 23:12:56 +00:00
|
|
|
import { MWMediaMeta, MWMediaType } from "./types";
|
|
|
|
|
|
|
|
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;
|
|
|
|
tmdbId: string;
|
|
|
|
imdbId: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getMetaFromId(
|
|
|
|
type: MWMediaType,
|
2023-01-22 18:26:08 +00:00
|
|
|
id: string,
|
|
|
|
seasonId?: string
|
2023-01-15 15:01:07 +00: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 19:51:58 +00:00
|
|
|
data = await proxiedFetch<JWDetailedMeta>(url, { baseURL: JW_API_BASE });
|
2023-01-15 15:01:07 +00: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-13 23:12:56 +00:00
|
|
|
|
2023-03-10 19:54:56 +00:00
|
|
|
let imdbId = data.external_ids.find(
|
2023-01-13 23:12:56 +00:00
|
|
|
(v) => v.provider === "imdb_latest"
|
2023-03-10 19:54:56 +00:00
|
|
|
)?.external_id
|
|
|
|
if (!imdbId)
|
|
|
|
imdbId = data.external_ids.find(
|
|
|
|
(v) => v.provider === "imdb"
|
|
|
|
)?.external_id;
|
|
|
|
|
|
|
|
let tmdbId = data.external_ids.find(
|
2023-01-13 23:12:56 +00:00
|
|
|
(v) => v.provider === "tmdb_latest"
|
2023-03-10 19:54:56 +00:00
|
|
|
)?.external_id
|
|
|
|
if (!tmdbId)
|
|
|
|
tmdbId = data.external_ids.find(
|
|
|
|
(v) => v.provider === "tmdb"
|
|
|
|
)?.external_id;
|
2023-01-13 23:12:56 +00:00
|
|
|
|
|
|
|
if (!imdbId || !tmdbId) throw new Error("not enough info");
|
|
|
|
|
2023-01-22 18:26:08 +00: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 19:51:58 +00:00
|
|
|
seasonData = await proxiedFetch<any>(url, { baseURL: JW_API_BASE });
|
2023-01-22 18:26:08 +00:00
|
|
|
}
|
|
|
|
|
2023-01-13 23:12:56 +00:00
|
|
|
return {
|
2023-01-22 18:26:08 +00:00
|
|
|
meta: formatJWMeta(data, seasonData),
|
2023-01-13 23:12:56 +00:00
|
|
|
imdbId,
|
|
|
|
tmdbId,
|
|
|
|
};
|
|
|
|
}
|