Kyoo/api/src/controllers/seasons.ts
2026-02-09 14:21:29 +01:00

183 lines
4.0 KiB
TypeScript

import { and, eq, type SQL, sql } from "drizzle-orm";
import { Elysia, t } from "elysia";
import { db } from "~/db";
import { seasons, seasonTranslations, shows } from "~/db/schema";
import { getColumns, sqlarr } from "~/db/utils";
import { KError } from "~/models/error";
import { madeInAbyss } from "~/models/examples";
import {
AcceptLanguage,
createPage,
Filter,
type FilterDef,
isUuid,
keysetPaginate,
Page,
processLanguages,
Sort,
sortToSql,
} from "~/models/utils";
import { desc } from "~/models/utils/descriptions";
import { Season, SeasonTranslation } from "../models/season";
const seasonFilters: FilterDef = {
seasonNumber: { column: seasons.seasonNumber, type: "int" },
startAir: { column: seasons.startAir, type: "date" },
endAir: { column: seasons.endAir, type: "date" },
entriesCount: { column: seasons.entriesCount, type: "int" },
availableCount: { column: seasons.availableCount, type: "int" },
};
const seasonSort = Sort(
{
seasonNumber: seasons.seasonNumber,
startAir: seasons.startAir,
endAir: seasons.endAir,
entriesCount: seasons.entriesCount,
availableCount: seasons.availableCount,
nextRefresh: seasons.nextRefresh,
},
{
default: ["seasonNumber"],
tablePk: seasons.pk,
},
);
export async function getSeasons({
after,
limit,
query,
sort,
filter,
languages,
}: {
after?: string;
limit: number;
query?: string;
sort?: Sort;
filter: SQL | undefined;
languages: string[];
}) {
sort ??= {
tablePk: seasons.pk,
sort: [
{
sql: seasons.seasonNumber,
isNullable: false,
accessor: (x) => x.seasonNumber,
desc: false,
},
],
};
const transQ = db
.selectDistinctOn([seasonTranslations.pk])
.from(seasonTranslations)
.orderBy(
seasonTranslations.pk,
sql`array_position(${sqlarr(languages)}, ${seasonTranslations.language})`,
)
.as("t");
const { pk, ...transCol } = getColumns(transQ);
return await db
.select({
...getColumns(seasons),
...transCol,
})
.from(seasons)
.leftJoin(transQ, eq(seasons.pk, transQ.pk))
.where(
and(
filter,
query ? sql`${transQ.name} %> ${query}::text` : undefined,
keysetPaginate({ after, sort }),
),
)
.orderBy(
...(query
? [sql`word_similarity(${query}::text, ${transQ.name}) desc`]
: sortToSql(sort)),
seasons.pk,
)
.limit(limit);
}
export const seasonsH = new Elysia({ tags: ["series"] })
.model({
season: Season,
"season-translation": SeasonTranslation,
})
.get(
"/series/:id/seasons",
async ({
params: { id },
query: { limit, after, query, sort, filter },
headers: { "accept-language": languages, ...headers },
request: { url },
status,
}) => {
const [serie] = await db
.select({ pk: shows.pk })
.from(shows)
.where(
and(
eq(shows.kind, "serie"),
isUuid(id) ? eq(shows.id, id) : eq(shows.slug, id),
),
)
.limit(1);
if (!serie) {
return status(404, {
status: 404,
message: `No serie with the id or slug: '${id}'.`,
});
}
const langs = processLanguages(languages);
const items = await getSeasons({
limit,
after,
query,
sort,
filter: and(eq(seasons.showPk, serie.pk), filter),
languages: langs,
});
return createPage(items, { url, sort, limit, headers });
},
{
detail: { description: "Get seasons of a serie" },
params: t.Object({
id: t.String({
description: "The id or slug of the serie.",
example: madeInAbyss.slug,
}),
}),
query: t.Object({
sort: seasonSort,
filter: t.Optional(Filter({ def: seasonFilters })),
query: t.Optional(t.String({ description: desc.query })),
limit: t.Integer({
minimum: 1,
maximum: 250,
default: 50,
description: "Max page size.",
}),
after: t.Optional(t.String({ description: desc.after })),
}),
headers: t.Object({
"accept-language": AcceptLanguage({ autoFallback: true }),
}),
response: {
200: Page(Season),
404: {
...KError,
description: "No serie found with the given id or slug.",
},
422: KError,
},
},
);