feat: cache marka api responses

This commit is contained in:
Max Richter
2025-11-04 12:09:17 +01:00
parent bb4e895770
commit fea9b69d4d
9 changed files with 163 additions and 98 deletions

View File

@@ -26,7 +26,7 @@ function generateJsonLd(resource: GenericResource): string {
if (resource.content?.datePublished) { if (resource.content?.datePublished) {
try { try {
baseSchema.datePublished = formatDate( baseSchema.datePublished = formatDate(
new Date(resource.content.datePublished), resource.content.datePublished,
); );
} catch (_) { } catch (_) {
// Ignore invalid date // Ignore invalid date

View File

@@ -50,6 +50,7 @@ export const createNewSeries: MenuEntry = {
return { return {
title: `${r.name} - ${r.first_air_date}`, title: `${r.name} - ${r.first_air_date}`,
cb: async () => { cb: async () => {
try {
state.activeState.value = "loading"; state.activeState.value = "loading";
const response = await fetch("/api/series/" + r.id, { const response = await fetch("/api/series/" + r.id, {
method: "POST", method: "POST",
@@ -57,6 +58,9 @@ export const createNewSeries: MenuEntry = {
const series = await response.json() as ReviewResource; const series = await response.json() as ReviewResource;
unsub(); unsub();
globalThis.location.href = "/series/" + series.name; globalThis.location.href = "/series/" + series.name;
} catch (_e) {
state.activeState.value = "normal";
}
}, },
}; };
}), }),

View File

@@ -1,3 +1,4 @@
import { createCache } from "../cache.ts";
import { MARKA_API_KEY } from "../env.ts"; import { MARKA_API_KEY } from "../env.ts";
import { getImage } from "../image.ts"; import { getImage } from "../image.ts";
import { GenericResource } from "./schema.ts"; import { GenericResource } from "./schema.ts";
@@ -24,32 +25,62 @@ async function addImageToResource<T extends GenericResource>(
return resource as T; return resource as T;
} }
export async function fetchResource<T extends GenericResource>( type Resource = GenericResource & {
content: GenericResource["content"] & Array<GenericResource>;
};
const fetchCache = createCache<Resource>("marka");
const cacheLock = new Map<string, Promise<Resource>>();
async function cachedFetch(
url: string,
): Promise<Resource | undefined> {
if (fetchCache.has(url)) {
return fetchCache.get(url);
}
if (cacheLock.has(url)) {
return cacheLock.get(url);
}
const response = (async () => {
const response = await fetch(url);
const res = await response.json();
fetchCache.set(url, res);
return res;
})();
cacheLock.set(
url,
response,
);
const res = await response;
if (!res) {
cacheLock.delete(url);
}
return res;
}
export async function fetchResource<T extends Resource>(
resource: string, resource: string,
): Promise<T | undefined> { ): Promise<T | undefined> {
try { try {
const response = await fetch( const d = `${url}/resources/${resource}`;
`${url}/resources/${resource}`, const res = await cachedFetch(d);
); if (!res) return;
const res = await response.json();
return addImageToResource<T>(res); return addImageToResource<T>(res);
} catch (_e) { } catch (_e) {
return; return;
} }
} }
export async function listResources<T = GenericResource>( export async function listResources<T extends GenericResource>(
resource: string, resource: string,
): Promise<T[]> { ): Promise<T[]> {
try { try {
const response = await fetch( const d = `${url}/resources/${resource}`;
`${url}/resources/${resource}`, const list = await cachedFetch(d);
); if (!list) return [];
const list = await response.json();
return Promise.all( return Promise.all(
list?.content list?.content
.filter((a: GenericResource) => a?.content?._type) .filter((a) => a?.content?._type)
.map((res: GenericResource) => addImageToResource(res)), .map((res) => addImageToResource(res) as Promise<T>),
); );
} catch (_e) { } catch (_e) {
return []; return [];

View File

@@ -1,4 +1,13 @@
export function formatDate(date: Date): string { export function formatDate(date?: string | Date): string {
if (!date) return "";
if (typeof date === "string") {
try {
const d = new Date(date);
return formatDate(d);
} catch (_e) {
return "";
}
}
const options = { year: "numeric", month: "long", day: "numeric" } as const; const options = { year: "numeric", month: "long", day: "numeric" } as const;
return new Intl.DateTimeFormat("en-US", options).format(date); return new Intl.DateTimeFormat("en-US", options).format(date);
} }

View File

@@ -6,13 +6,15 @@ import * as openai from "@lib/openai.ts";
import { getYoutubeVideoDetails } from "@lib/youtube.ts"; import { getYoutubeVideoDetails } from "@lib/youtube.ts";
import { import {
extractYoutubeId, extractYoutubeId,
formatDate,
isYoutubeLink, isYoutubeLink,
toUrlSafeString, safeFileName,
} from "@lib/string.ts"; } from "@lib/string.ts";
import { createLogger } from "@lib/log/index.ts"; import { createLogger } from "@lib/log/index.ts";
import { createResource } from "@lib/marka/index.ts"; import { createResource } from "@lib/marka/index.ts";
import { webScrape } from "@lib/webScraper.ts"; import { webScrape } from "@lib/webScraper.ts";
import { ArticleResource } from "@lib/marka/schema.ts"; import { ArticleResource } from "@lib/marka/schema.ts";
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
const log = createLogger("api/article"); const log = createLogger("api/article");
@@ -44,16 +46,34 @@ async function processCreateArticle(
streamResponse.enqueue("postprocessing article"); streamResponse.enqueue("postprocessing article");
const title = result?.title || aiMeta?.headline || ""; const title = result?.title || aiMeta?.headline || "";
const id = toUrlSafeString(title);
let finalPath = result.image;
if (result?.image) {
const extension = fileExtension(result?.image);
const imagePath = `resources/articles/images/${
safeFileName(title)
}_cover.${extension}`;
try {
streamResponse.enqueue("downloading image");
const res = await fetch(result.image);
streamResponse.enqueue("saving image");
const buffer = await res.arrayBuffer();
await createResource(imagePath, buffer);
finalPath = imagePath;
} catch (err) {
console.log("Failed to save image", err);
}
}
const newArticle: ArticleResource["content"] = { const newArticle: ArticleResource["content"] = {
_type: "Article", _type: "Article",
headline: title, headline: title,
articleBody: result.content, articleBody: result.content,
url: fetchUrl, url: fetchUrl,
datePublished: result?.published || aiMeta?.datePublished || datePublished: formatDate(
new Date().toISOString(), result?.published || aiMeta?.datePublished || undefined,
image: result?.image, ),
image: finalPath,
author: { author: {
_type: "Person", _type: "Person",
name: (result.schemaOrgData?.author?.name || aiMeta?.author || "") name: (result.schemaOrgData?.author?.name || aiMeta?.author || "")
@@ -66,9 +86,9 @@ async function processCreateArticle(
streamResponse.enqueue("writing to disk"); streamResponse.enqueue("writing to disk");
await createResource(`articles/${id}.md`, newArticle); await createResource(`articles/${title}.md`, newArticle);
streamResponse.enqueue("id: " + id); streamResponse.enqueue("id: " + title);
} }
async function processCreateYoutubeVideo( async function processCreateYoutubeVideo(

View File

@@ -2,8 +2,8 @@ import { Handlers } from "$fresh/server.ts";
import { json } from "@lib/helpers.ts"; import { json } from "@lib/helpers.ts";
import * as tmdb from "@lib/tmdb.ts"; import * as tmdb from "@lib/tmdb.ts";
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts"; import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
import { isString, safeFileName } from "@lib/string.ts"; import { formatDate, isString, safeFileName } from "@lib/string.ts";
import { AccessDeniedError } from "@lib/errors.ts"; import { AccessDeniedError, BadRequestError } from "@lib/errors.ts";
import { createResource, fetchResource } from "@lib/marka/index.ts"; import { createResource, fetchResource } from "@lib/marka/index.ts";
import { ReviewResource } from "@lib/marka/schema.ts"; import { ReviewResource } from "@lib/marka/schema.ts";
@@ -14,26 +14,22 @@ export const handler: Handlers = {
}, },
async POST(_, ctx) { async POST(_, ctx) {
const session = ctx.state.session; const session = ctx.state.session;
if (!session) { if (!session) throw new AccessDeniedError();
throw new AccessDeniedError();
}
const tmdbId = parseInt(ctx.params.name); const tmdbId = parseInt(ctx.params.name);
if (Number.isNaN(tmdbId)) throw new BadRequestError();
const movieDetails = await tmdb.getMovie(tmdbId); const [movieDetails, movieCredits] = await Promise.all([
const movieCredits = await tmdb.getMovieCredits(tmdbId); tmdb.getMovie(tmdbId),
tmdb.getMovieCredits(tmdbId),
]);
const releaseDate = movieDetails.release_date; const name = movieDetails.title ||
const posterPath = movieDetails.poster_path; movieDetails.original_title ||
const director = movieCredits?.crew?.filter?.((person) =>
person.job === "Director"
)[0];
movieDetails.overview;
let finalPath = "";
const name = movieDetails.title || movieDetails.original_title ||
ctx.params.name; ctx.params.name;
const posterPath = movieDetails.poster_path;
let finalPath = "";
if (posterPath) { if (posterPath) {
const poster = await tmdb.getMoviePoster(posterPath); const poster = await tmdb.getMoviePoster(posterPath);
const extension = fileExtension(posterPath); const extension = fileExtension(posterPath);
@@ -41,33 +37,32 @@ export const handler: Handlers = {
await createResource(finalPath, poster); await createResource(finalPath, poster);
} }
const tags: string[] = []; const keywords = movieDetails.genres
if (movieDetails.genres) { ?.map((g) => g.name?.toLowerCase())
tags.push( .filter(isString) || [];
...movieDetails.genres.map((g) => g.name?.toLowerCase()).filter(
isString,
),
);
}
const movie: ReviewResource["content"] = { const movie: ReviewResource["content"] = {
_type: "Review", _type: "Review",
image: `resources/${finalPath}`, image: `resources/${finalPath}`,
datePublished: releaseDate, datePublished: formatDate(movieDetails.release_date),
tmdbId, tmdbId,
author: { author: {
_type: "Person", _type: "Person",
name: director?.name, name: movieCredits.crew?.filter?.((person) =>
person.job === "Director"
)[0]?.name,
}, },
itemReviewed: { itemReviewed: {
name: name, name,
}, },
reviewBody: "", reviewBody: "",
keywords: tags, keywords,
}; };
await createResource(`movies/${safeFileName(name)}.md`, movie); const fileName = `${safeFileName(name)}.md`;
return json(movie); await createResource(`movies/${fileName}`, movie);
return json({ name: fileName });
}, },
}; };

View File

@@ -39,15 +39,16 @@ const POST = async (
const movieCredits = !movie.content?.author && const movieCredits = !movie.content?.author &&
await tmdb.getMovieCredits(tmdbId); await tmdb.getMovieCredits(tmdbId);
const releaseDate = movieDetails.release_date;
if (releaseDate && !movie.content?.datePublished) {
movie.content = movie.content || {};
movie.content.datePublished = formatDate(new Date(releaseDate));
}
const director = movieCredits && const director = movieCredits &&
movieCredits?.crew?.filter?.((person) => person.job === "Director")[0]; movieCredits?.crew?.filter?.((person) => person.job === "Director")[0];
movie.content ??= {
_type: "Review",
};
movie.content.datePublished ??= formatDate(movieDetails.release_date);
if (director && !movie.content?.author) { if (director && !movie.content?.author) {
movie.content = movie.content || {};
movie.content.author = { movie.content.author = {
_type: "Person", _type: "Person",
name: director.name, name: director.name,
@@ -65,18 +66,15 @@ const POST = async (
]; ];
} }
if (!movie.name) { movie.content.tmdbId ??= tmdbId;
movie.name = tmdbId;
}
let finalPath = ""; let finalPath = "";
const posterPath = movieDetails.poster_path; const posterPath = movieDetails.poster_path;
if (posterPath && !movie.content?.image) { if (posterPath && !movie.content.image) {
const poster = await tmdb.getMoviePoster(posterPath); const poster = await tmdb.getMoviePoster(posterPath);
const extension = fileExtension(posterPath); const extension = fileExtension(posterPath);
finalPath = `movies/images/${safeFileName(name)}_cover.${extension}`; finalPath = `movies/images/${safeFileName(name)}_cover.${extension}`;
await createResource(finalPath, poster); await createResource(finalPath, poster);
movie.content = movie.content || {};
movie.content.image = finalPath; movie.content.image = finalPath;
} }

View File

@@ -76,7 +76,7 @@ async function processCreateRecipeFromUrl(
}; };
if (newRecipe?.image && newRecipe.image.length > 5) { if (newRecipe?.image && newRecipe.image.length > 5) {
const extension = fileExtension(new URL(newRecipe.image).pathname); const extension = fileExtension(newRecipe.image);
const finalPath = `resources/recipes/images/${ const finalPath = `resources/recipes/images/${
safeFileName(id) safeFileName(id)
}_cover.${extension}`; }_cover.${extension}`;

View File

@@ -2,11 +2,19 @@ import { Handlers } from "$fresh/server.ts";
import { json } from "@lib/helpers.ts"; import { json } from "@lib/helpers.ts";
import * as tmdb from "@lib/tmdb.ts"; import * as tmdb from "@lib/tmdb.ts";
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts"; import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
import { isString, safeFileName } from "@lib/string.ts"; import { formatDate, isString, safeFileName } from "@lib/string.ts";
import { AccessDeniedError } from "@lib/errors.ts"; import { AccessDeniedError, BadRequestError } from "@lib/errors.ts";
import { createResource, fetchResource } from "@lib/marka/index.ts"; import { createResource, fetchResource } from "@lib/marka/index.ts";
import { ReviewResource } from "@lib/marka/schema.ts"; import { ReviewResource } from "@lib/marka/schema.ts";
function pickDirector(
credits: Awaited<ReturnType<typeof tmdb.getSeriesCredits>>,
createdBy?: { name?: string }[],
): string | undefined {
const crewDirector = credits?.crew?.find?.((p) => p.job === "Director");
return crewDirector?.name ?? createdBy?.[0]?.name;
}
export const handler: Handlers = { export const handler: Handlers = {
async GET(_, ctx) { async GET(_, ctx) {
const series = await fetchResource(`series/${ctx.params.name}`); const series = await fetchResource(`series/${ctx.params.name}`);
@@ -19,54 +27,54 @@ export const handler: Handlers = {
} }
const tmdbId = parseInt(ctx.params.name); const tmdbId = parseInt(ctx.params.name);
if (Number.isNaN(tmdbId)) throw new BadRequestError();
const seriesDetails = await tmdb.getSeries(tmdbId); const [seriesDetails, seriesCredits] = await Promise.all([
const seriesCredits = await tmdb.getSeriesCredits(tmdbId); tmdb.getSeries(tmdbId),
tmdb.getSeriesCredits(tmdbId),
]);
const releaseDate = seriesDetails.first_air_date; const name = seriesDetails.name ||
const posterPath = seriesDetails.poster_path; seriesDetails.original_name ||
const director = ctx.params.name;
seriesCredits?.crew?.filter?.((person) => person.job === "Director")[0] ||
seriesDetails.created_by?.[0];
let finalPath = ""; let finalPath = "";
const name = seriesDetails.name || seriesDetails.original_name || const posterPath = seriesDetails.poster_path;
ctx.params.name;
if (posterPath) { if (posterPath) {
const poster = await tmdb.getMoviePoster(posterPath); const poster = await tmdb.getMoviePoster(posterPath);
const extension = fileExtension(posterPath); const extension = fileExtension(posterPath);
const imagePath = `series/images/${
finalPath = `series/images/${safeFileName(name)}_cover.${extension}`; safeFileName(name)
await createResource(finalPath, poster); }_cover.${extension}`;
await createResource(imagePath, poster);
finalPath = imagePath;
} }
const tags: string[] = []; const keywords = seriesDetails.genres
if (seriesDetails.genres) { ?.map((g) => g.name?.toLowerCase())
tags.push( .filter(isString) ??
...seriesDetails.genres.map((g) => g.name?.toLowerCase()).filter( [];
isString,
),
);
}
const series: ReviewResource["content"] = { const series: ReviewResource["content"] = {
_type: "Review", _type: "Review",
image: `resources/${finalPath}`, image: `resources/${finalPath}`,
datePublished: releaseDate, datePublished: formatDate(seriesDetails.first_air_date),
tmdbId, tmdbId,
author: { author: {
_type: "Person", _type: "Person",
name: director?.name, name: pickDirector(seriesCredits, seriesDetails?.created_by),
}, },
itemReviewed: { itemReviewed: {
name: name, name: name,
}, },
reviewBody: "", reviewBody: "",
keywords: tags, keywords: keywords,
}; };
await createResource(`series/${safeFileName(name)}.md`, series); const fileName = `${safeFileName(name)}.md`;
return json(series); await createResource(`series/${fileName}`, series);
return json({ name: fileName });
}, },
}; };