Skip to content

Commit 956ab51

Browse files
committed
Add a flag to retrieve all translations
1 parent 02ddd14 commit 956ab51

File tree

4 files changed

+57
-32
lines changed

4 files changed

+57
-32
lines changed

api/src/controllers/movies.ts

+29-27
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,12 @@ import {
1111
} from "~/db/schema";
1212
import { getColumns, sqlarr } from "~/db/schema/utils";
1313
import { bubble } from "~/models/examples";
14-
import { Movie, MovieStatus, MovieTranslation } from "~/models/movie";
14+
import {
15+
FullMovie,
16+
Movie,
17+
MovieStatus,
18+
MovieTranslation,
19+
} from "~/models/movie";
1520
import {
1621
Filter,
1722
type Image,
@@ -49,7 +54,7 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
4954
async ({
5055
params: { id },
5156
headers: { "accept-language": languages },
52-
query: { preferOriginal },
57+
query: { preferOriginal, with: relations },
5358
error,
5459
set,
5560
}) => {
@@ -86,7 +91,7 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
8691
isUuid(id) ? eq(shows.id, id) : eq(shows.slug, id),
8792
),
8893
with: {
89-
translations: {
94+
selectedTranslation: {
9095
columns: {
9196
pk: false,
9297
},
@@ -113,6 +118,13 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
113118
),
114119
},
115120
},
121+
...(relations.includes("translations") && {
122+
translations: {
123+
columns: {
124+
pk: false,
125+
},
126+
},
127+
}),
116128
},
117129
});
118130

@@ -122,7 +134,7 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
122134
message: "Movie not found",
123135
});
124136
}
125-
const translation = ret.translations[0];
137+
const translation = ret.selectedTranslation[0];
126138
if (!translation) {
127139
return error(422, {
128140
status: 422,
@@ -140,6 +152,14 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
140152
...(ot.banner && { banner: ot.banner }),
141153
...(ot.logo && { logo: ot.logo }),
142154
}),
155+
...(ret.translations && {
156+
translations: Object.fromEntries(
157+
ret.translations.map(
158+
({ language, ...translation }) =>
159+
[language, translation] as const,
160+
),
161+
),
162+
}),
143163
};
144164
},
145165
{
@@ -162,6 +182,10 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
162182
`,
163183
}),
164184
),
185+
with: t.Array(t.UnionEnum(["translations", "videos"]), {
186+
default: [],
187+
description: "Include related resources in the response.",
188+
}),
165189
}),
166190
headers: t.Object({
167191
"accept-language": t.String({
@@ -174,13 +198,10 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
174198
}),
175199
}),
176200
response: {
177-
200: { ...Movie, description: "Found" },
201+
200: { ...FullMovie, description: "Found" },
178202
404: {
179203
...KError,
180204
description: "No movie found with the given id or slug.",
181-
examples: [
182-
{ status: 404, message: "Movie not found", details: undefined },
183-
],
184205
},
185206
422: {
186207
...KError,
@@ -189,12 +210,6 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
189210
unavailable.) Try with another languages or add * to the list of languages
190211
to fallback to any language.
191212
`,
192-
examples: [
193-
{
194-
status: 422,
195-
message: "Accept-Language header could not be satisfied.",
196-
},
197-
],
198213
},
199214
},
200215
},
@@ -227,9 +242,6 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
227242
404: {
228243
...KError,
229244
description: "No movie found with the given id or slug.",
230-
examples: [
231-
{ status: 404, message: "Movie not found", details: undefined },
232-
],
233245
},
234246
},
235247
},
@@ -362,16 +374,6 @@ export const movies = new Elysia({ prefix: "/movies", tags: ["movies"] })
362374
422: {
363375
...KError,
364376
description: "Invalid query parameters.",
365-
examples: [
366-
{
367-
status: 422,
368-
message:
369-
"Invalid property: slug. Expected one of genres, rating, status, runtime, airDate, originalLanguage.",
370-
details: {
371-
in: "slug eq bubble",
372-
},
373-
},
374-
],
375377
},
376378
},
377379
},

api/src/db/schema/shows.ts

+8
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,9 @@ export const showTranslations = schema.table(
110110
);
111111

112112
export const showsRelations = relations(shows, ({ many, one }) => ({
113+
selectedTranslation: many(showTranslations, {
114+
relationName: "selectedTranslation",
115+
}),
113116
translations: many(showTranslations, { relationName: "showTranslations" }),
114117
originalTranslation: one(showTranslations, {
115118
relationName: "originalTranslation",
@@ -123,6 +126,11 @@ export const showsTrRelations = relations(showTranslations, ({ one }) => ({
123126
fields: [showTranslations.pk],
124127
references: [shows.pk],
125128
}),
129+
selectedTranslation: one(shows, {
130+
relationName: "selectedTranslation",
131+
fields: [showTranslations.pk],
132+
references: [shows.pk],
133+
}),
126134
originalTranslation: one(shows, {
127135
relationName: "originalTranslation",
128136
fields: [showTranslations.pk, showTranslations.language],

api/src/models/error.ts

+10-5
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,15 @@
11
import { t } from "elysia";
22

3-
export const KError = t.Object({
4-
status: t.Integer(),
5-
message: t.String(),
6-
details: t.Optional(t.Any()),
7-
});
3+
export const KError = t.Object(
4+
{
5+
status: t.Integer(),
6+
message: t.String(),
7+
details: t.Optional(t.Any()),
8+
},
9+
{
10+
examples: [{ status: 404, message: "Movie not found" }],
11+
},
12+
);
813
export type KError = typeof KError.static;
914

1015
export class KErrorT extends Error {

api/src/models/movie.ts

+10
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import {
99
} from "./utils";
1010
import { bubble, registerExamples } from "./examples";
1111
import { bubbleImages } from "./examples/bubble";
12+
import { Video } from "./video";
1213

1314
export const MovieStatus = t.UnionEnum(["unknown", "finished", "planned"]);
1415
export type MovieStatus = typeof MovieStatus.static;
@@ -58,6 +59,15 @@ export const Movie = t.Intersect([
5859
]);
5960
export type Movie = typeof Movie.static;
6061

62+
export const FullMovie = t.Intersect([
63+
Movie,
64+
t.Object({
65+
translations: t.Optional(TranslationRecord(MovieTranslation)),
66+
videos: t.Optional(t.Array(Video)),
67+
}),
68+
]);
69+
export type FullMovie = typeof FullMovie.static;
70+
6171
export const SeedMovie = t.Intersect([
6272
t.Omit(BaseMovie, ["id", "createdAt", "nextRefresh"]),
6373
t.Object({

0 commit comments

Comments
 (0)