mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-05-31 20:24:27 -04:00
Extract get image logic to a function
This commit is contained in:
parent
9905587c83
commit
7d3413a3d5
@ -1,7 +1,8 @@
|
|||||||
import { stat } from "node:fs/promises";
|
import { stat } from "node:fs/promises";
|
||||||
import type { BunFile } from "bun";
|
import type { BunFile } from "bun";
|
||||||
import { and, eq, sql } from "drizzle-orm";
|
import { type SQL, and, eq, sql } from "drizzle-orm";
|
||||||
import Elysia, { t } from "elysia";
|
import type { PgColumn } from "drizzle-orm/pg-core";
|
||||||
|
import Elysia, { type InferContext, t } from "elysia";
|
||||||
import { db } from "~/db";
|
import { db } from "~/db";
|
||||||
import { showTranslations, shows } from "~/db/schema";
|
import { showTranslations, shows } from "~/db/schema";
|
||||||
import { sqlarr } from "~/db/utils";
|
import { sqlarr } from "~/db/utils";
|
||||||
@ -10,6 +11,77 @@ import { bubble } from "~/models/examples";
|
|||||||
import { AcceptLanguage, isUuid, processLanguages } from "~/models/utils";
|
import { AcceptLanguage, isUuid, processLanguages } from "~/models/utils";
|
||||||
import { imageDir } from "./seed/images";
|
import { imageDir } from "./seed/images";
|
||||||
|
|
||||||
|
async function redirectToImage({
|
||||||
|
image,
|
||||||
|
filter,
|
||||||
|
id,
|
||||||
|
languages,
|
||||||
|
quality,
|
||||||
|
set,
|
||||||
|
error,
|
||||||
|
redirect,
|
||||||
|
}: {
|
||||||
|
image: typeof showTranslations.poster;
|
||||||
|
filter: SQL;
|
||||||
|
id: string;
|
||||||
|
languages: string;
|
||||||
|
quality?: "high" | "medium" | "low";
|
||||||
|
set: InferContext<typeof imagesH>["set"];
|
||||||
|
error: InferContext<typeof imagesH>["error"];
|
||||||
|
redirect: InferContext<typeof imagesH>["redirect"];
|
||||||
|
}) {
|
||||||
|
const lang = processLanguages(languages);
|
||||||
|
const item = db.$with("item").as(
|
||||||
|
db
|
||||||
|
.select({ pk: shows.pk })
|
||||||
|
.from(shows)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
filter,
|
||||||
|
id !== "random"
|
||||||
|
? isUuid(id)
|
||||||
|
? eq(shows.id, id)
|
||||||
|
: eq(shows.slug, id)
|
||||||
|
: undefined,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.orderBy(sql`random()`)
|
||||||
|
.limit(1),
|
||||||
|
);
|
||||||
|
const [ret] = await db
|
||||||
|
.with(item)
|
||||||
|
.select({
|
||||||
|
image,
|
||||||
|
language: showTranslations.language,
|
||||||
|
})
|
||||||
|
.from(item)
|
||||||
|
.leftJoin(showTranslations, eq(item.pk, showTranslations.pk))
|
||||||
|
.where(
|
||||||
|
!lang.includes("*")
|
||||||
|
? eq(showTranslations.language, sql`any(${sqlarr(lang)})`)
|
||||||
|
: undefined,
|
||||||
|
)
|
||||||
|
.orderBy(sql`array_position(${sqlarr(lang)}, ${showTranslations.language})`)
|
||||||
|
.limit(1);
|
||||||
|
|
||||||
|
if (!ret) {
|
||||||
|
return error(404, {
|
||||||
|
status: 404,
|
||||||
|
message: `No movie found with id or slug: '${id}'.`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!ret.language) {
|
||||||
|
return error(422, {
|
||||||
|
status: 422,
|
||||||
|
message: "Accept-Language header could not be satisfied.",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
set.headers["content-language"] = ret.language;
|
||||||
|
return quality
|
||||||
|
? redirect(`/images/${ret.image!.id}?quality=${quality}`)
|
||||||
|
: redirect(`/images/${ret.image!.id}`);
|
||||||
|
}
|
||||||
|
|
||||||
export const imagesH = new Elysia({ tags: ["images"] })
|
export const imagesH = new Elysia({ tags: ["images"] })
|
||||||
.get(
|
.get(
|
||||||
"/images/:id",
|
"/images/:id",
|
||||||
@ -55,83 +127,58 @@ export const imagesH = new Elysia({ tags: ["images"] })
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
.guard({
|
||||||
|
params: t.Object({
|
||||||
|
id: t.String({
|
||||||
|
description: "The id or slug of the item to retrieve.",
|
||||||
|
example: bubble.slug,
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
query: t.Object({
|
||||||
|
quality: t.Optional(
|
||||||
|
t.UnionEnum(["high", "medium", "low"], {
|
||||||
|
default: "high",
|
||||||
|
description: "The quality you want your image to be in.",
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
headers: t.Object({
|
||||||
|
"accept-language": AcceptLanguage(),
|
||||||
|
}),
|
||||||
|
response: {
|
||||||
|
302: t.Void({
|
||||||
|
description:
|
||||||
|
"Redirected to the [/images/{id}](#tag/images/GET/images/{id}) route.",
|
||||||
|
}),
|
||||||
|
404: {
|
||||||
|
...KError,
|
||||||
|
description: "No item found with the given id or slug.",
|
||||||
|
},
|
||||||
|
422: KError,
|
||||||
|
},
|
||||||
|
})
|
||||||
.get(
|
.get(
|
||||||
"/movies/:id/poster",
|
"/movies/:id/poster",
|
||||||
async ({
|
({
|
||||||
params: { id },
|
params: { id },
|
||||||
headers: { "accept-language": languages },
|
headers: { "accept-language": languages },
|
||||||
query: { quality },
|
query: { quality },
|
||||||
set,
|
set,
|
||||||
error,
|
error,
|
||||||
redirect,
|
redirect,
|
||||||
}) => {
|
}) =>
|
||||||
const lang = processLanguages(languages);
|
redirectToImage({
|
||||||
const [movie] = await db
|
filter: eq(shows.kind, "movie"),
|
||||||
.select({
|
image: showTranslations.poster,
|
||||||
poster: showTranslations.poster,
|
id,
|
||||||
language: showTranslations.language,
|
languages,
|
||||||
})
|
quality,
|
||||||
.from(shows)
|
set,
|
||||||
.leftJoin(showTranslations, eq(shows.pk, showTranslations.pk))
|
error,
|
||||||
.where(
|
redirect,
|
||||||
and(
|
}),
|
||||||
eq(shows.kind, "movie"),
|
|
||||||
isUuid(id) ? eq(shows.id, id) : eq(shows.slug, id),
|
|
||||||
!lang.includes("*")
|
|
||||||
? eq(showTranslations.language, sql`any(${sqlarr(lang)})`)
|
|
||||||
: undefined,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
.orderBy(
|
|
||||||
sql`array_position(${sqlarr(lang)}, ${showTranslations.language})`,
|
|
||||||
)
|
|
||||||
.limit(1);
|
|
||||||
|
|
||||||
if (!movie) {
|
|
||||||
return error(404, {
|
|
||||||
status: 404,
|
|
||||||
message: `No movie found with id or slug: '${id}'.`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (!movie.language) {
|
|
||||||
return error(422, {
|
|
||||||
status: 422,
|
|
||||||
message: "Accept-Language header could not be satisfied.",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
set.headers["content-language"] = movie.language;
|
|
||||||
return redirect(`/images/${movie.poster!.id}?quality=${quality}`);
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
detail: { description: "Get the poster of a movie" },
|
detail: { description: "Get the poster of a movie" },
|
||||||
params: t.Object({
|
|
||||||
id: t.String({
|
|
||||||
description: "The id or slug of the movie to retrieve.",
|
|
||||||
example: bubble.slug,
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
query: t.Object({
|
|
||||||
quality: t.Optional(
|
|
||||||
t.UnionEnum(["high", "medium", "low"], {
|
|
||||||
default: "high",
|
|
||||||
description: "The quality you want your image to be in.",
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
}),
|
|
||||||
headers: t.Object({
|
|
||||||
"accept-language": AcceptLanguage(),
|
|
||||||
}),
|
|
||||||
response: {
|
|
||||||
302: t.Void({
|
|
||||||
description:
|
|
||||||
"Redirected to the [/movies/{id}](#tag/movies/GET/movies/{id}) route.",
|
|
||||||
}),
|
|
||||||
404: {
|
|
||||||
...KError,
|
|
||||||
description: "No movie found with the given id or slug.",
|
|
||||||
},
|
|
||||||
422: KError,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user