From 1dd7bdbf77d19800b2a97cb9d10594fc91c63346 Mon Sep 17 00:00:00 2001 From: Daniel Bachler Date: Mon, 19 Aug 2024 09:36:00 +0000 Subject: [PATCH] =?UTF-8?q?=E2=9C=A8=20Render=20thumbnails=20for=20grapher?= =?UTF-8?q?s=20by=20uuid?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- functions/_common/grapherRenderer.ts | 26 +++++++----------- functions/_common/reusableHandlers.ts | 37 ++++++++++++++++++++++++++ functions/grapher/[slug].ts | 38 ++++++++++++++++++++++++--- functions/grapher/by-uuid/[uuid].ts | 28 +++++++++++++++++++- functions/grapher/thumbnail/[slug].ts | 23 +++++++++++++--- 5 files changed, 128 insertions(+), 24 deletions(-) create mode 100644 functions/_common/reusableHandlers.ts diff --git a/functions/_common/grapherRenderer.ts b/functions/_common/grapherRenderer.ts index c0bbdc0eca1..a33fef1bdb9 100644 --- a/functions/_common/grapherRenderer.ts +++ b/functions/_common/grapherRenderer.ts @@ -23,6 +23,8 @@ declare global { var window: any } +export type Etag = string + const grapherBaseUrl = "https://ourworldindata.org/grapher" // Lots of defaults; these are mostly the same as they are in owid-grapher. @@ -164,17 +166,17 @@ interface FetchGrapherConfigResult { etag: string | undefined } -interface GrapherSlug { +export interface GrapherSlug { type: "slug" id: string } -interface GrapherUuid { +export interface GrapherUuid { type: "uuid" id: string } -type GrapherIdentifier = GrapherSlug | GrapherUuid +export type GrapherIdentifier = GrapherSlug | GrapherUuid export async function fetchUnparsedGrapherConfig( identifier: GrapherIdentifier, @@ -258,17 +260,14 @@ export async function fetchGrapherConfig( } async function fetchAndRenderGrapherToSvg( - slug: string, + id: GrapherIdentifier, options: ImageOptions, searchParams: URLSearchParams, env: Env ) { const grapherLogger = new TimeLogger("grapher") - const grapherConfigResponse = await fetchGrapherConfig( - { type: "slug", id: slug }, - env - ) + const grapherConfigResponse = await fetchGrapherConfig(id, env) if (grapherConfigResponse.status !== 200) { return null @@ -308,20 +307,15 @@ async function fetchAndRenderGrapherToSvg( } export const fetchAndRenderGrapher = async ( - slug: string, + id: GrapherIdentifier, searchParams: URLSearchParams, outType: "png" | "svg", env: Env ) => { const options = extractOptions(searchParams) - console.log("Rendering", slug, outType, options) - const svg = await fetchAndRenderGrapherToSvg( - slug, - options, - searchParams, - env - ) + console.log("Rendering", id.id, outType, options) + const svg = await fetchAndRenderGrapherToSvg(id, options, searchParams, env) console.log("fetched svg") if (!svg) { diff --git a/functions/_common/reusableHandlers.ts b/functions/_common/reusableHandlers.ts new file mode 100644 index 00000000000..4381545b549 --- /dev/null +++ b/functions/_common/reusableHandlers.ts @@ -0,0 +1,37 @@ +import { Env } from "../grapher/thumbnail/[slug].js" +import { + Etag, + GrapherIdentifier, + fetchAndRenderGrapher, +} from "./grapherRenderer.js" + +export async function handleThumbnailRequest( + id: GrapherIdentifier, + searchParams: URLSearchParams, + env: Env, + _etag: Etag, + ctx: EventContext>, + extension: "png" | "svg" +) { + const url = new URL(env.url) + const shouldCache = !url.searchParams.has("nocache") + + const cache = caches.default + console.log("Handling", env.url, ctx.request.headers.get("User-Agent")) + if (shouldCache) { + console.log("Checking cache") + const maybeCached = await cache.match(ctx.request) + console.log("Cache check result", maybeCached ? "hit" : "miss") + if (maybeCached) return maybeCached + } + const resp = await fetchAndRenderGrapher(id, searchParams, extension, env) + if (shouldCache) { + resp.headers.set("Cache-Control", "public, s-maxage=3600, max-age=3600") + ctx.waitUntil(caches.default.put(ctx.request, resp.clone())) + } else + resp.headers.set( + "Cache-Control", + "public, s-maxage=0, max-age=0, must-revalidate" + ) + return resp +} diff --git a/functions/grapher/[slug].ts b/functions/grapher/[slug].ts index d6b8b41b165..05f44367b3a 100644 --- a/functions/grapher/[slug].ts +++ b/functions/grapher/[slug].ts @@ -2,15 +2,20 @@ import { Env } from "../_common/env.js" import { getOptionalRedirectForSlug, createRedirectResponse, + Etag, fetchUnparsedGrapherConfig, } from "../_common/grapherRenderer.js" import { IRequestStrict, Router, error, cors } from "itty-router" +import { handleThumbnailRequest } from "../_common/reusableHandlers.js" const { preflight, corsify } = cors({ allowMethods: "GET", }) -const router = Router({ +const router = Router< + IRequestStrict, + [URL, Env, Etag, EventContext>] +>({ before: [preflight], finally: [corsify], }) @@ -20,6 +25,30 @@ router async ({ params: { slug } }, { searchParams }, env, etag) => handleConfigRequest(slug, searchParams, env, etag) ) + .get( + "/grapher/:slug.png", + async ({ params: { slug } }, { searchParams }, env, etag, ctx) => + handleThumbnailRequest( + { type: "slug", id: slug }, + searchParams, + env, + etag, + ctx, + "png" + ) + ) + .get( + "/grapher/:slug.svg", + async ({ params: { slug } }, { searchParams }, env, etag, ctx) => + handleThumbnailRequest( + { type: "slug", id: slug }, + searchParams, + env, + etag, + ctx, + "svg" + ) + ) .get( "/grapher/:slug", async ({ params: { slug } }, { searchParams }, env) => @@ -39,7 +68,8 @@ export const onRequestGet: PagesFunction = async (context) => { request, url, { ...env, url }, - request.headers.get("if-none-match") + request.headers.get("if-none-match"), + context ) .catch((e) => error(500, e)) } @@ -99,10 +129,10 @@ async function handleHtmlPageRequest( // In the case of the redirect, the browser will then request the new URL which will again be handled by this worker. if (grapherPageResp.status !== 200) return grapherPageResp - const openGraphThumbnailUrl = `/grapher/thumbnail/${lowerCaseSlug}.png?imType=og${ + const openGraphThumbnailUrl = `/grapher/${lowerCaseSlug}.png?imType=og${ url.search ? "&" + url.search.slice(1) : "" }` - const twitterThumbnailUrl = `/grapher/thumbnail/${lowerCaseSlug}.png?imType=twitter${ + const twitterThumbnailUrl = `/grapher/${lowerCaseSlug}.png?imType=twitter${ url.search ? "&" + url.search.slice(1) : "" }` diff --git a/functions/grapher/by-uuid/[uuid].ts b/functions/grapher/by-uuid/[uuid].ts index d20af2b3caa..7ff1bc9a069 100644 --- a/functions/grapher/by-uuid/[uuid].ts +++ b/functions/grapher/by-uuid/[uuid].ts @@ -1,6 +1,7 @@ import { Env } from "../thumbnail/[slug].js" import { fetchGrapherConfig } from "../../_common/grapherRenderer.js" import { IRequestStrict, Router, error, cors } from "itty-router" +import { handleThumbnailRequest } from "../../_common/reusableHandlers.js" const { preflight, corsify } = cors({ allowMethods: "GET", }) @@ -15,6 +16,30 @@ router async ({ params: { uuid } }, { searchParams }, env, etag) => handleConfigRequest(uuid, searchParams, env, etag) ) + .get( + "/grapher/by-uuid/:uuid.png", + async ({ params: { uuid } }, { searchParams }, env, etag, ctx) => + handleThumbnailRequest( + { type: "uuid", id: uuid }, + searchParams, + env, + etag, + ctx, + "png" + ) + ) + .get( + "/grapher/by-uuid/:uuid.svg", + async ({ params: { uuid } }, { searchParams }, env, etag, ctx) => + handleThumbnailRequest( + { type: "uuid", id: uuid }, + searchParams, + env, + etag, + ctx, + "svg" + ) + ) .all("*", () => error(404, "Route not defined")) export const onRequestGet: PagesFunction = async (context) => { @@ -29,7 +54,8 @@ export const onRequestGet: PagesFunction = async (context) => { request, url, { ...env, url }, - request.headers.get("if-none-match") + request.headers.get("if-none-match"), + context ) .catch((e) => error(500, e)) } diff --git a/functions/grapher/thumbnail/[slug].ts b/functions/grapher/thumbnail/[slug].ts index d92306bd67d..9b66aebfdb9 100644 --- a/functions/grapher/thumbnail/[slug].ts +++ b/functions/grapher/thumbnail/[slug].ts @@ -2,22 +2,39 @@ import { Env } from "../../_common/env.js" import { fetchAndRenderGrapher } from "../../_common/grapherRenderer.js" import { IRequestStrict, Router, error } from "itty-router" +// TODO: remove the /grapher/thumbnail route two weeks or so after the change to use /grapher/:slug.png is deployed +// We keep this around for another two weeks so that cached html pages etc can still fetch the correct thumbnail const router = Router() router .get( "/grapher/thumbnail/:slug.png", async ({ params: { slug } }, { searchParams }, env) => - fetchAndRenderGrapher(slug, searchParams, "png", env) + fetchAndRenderGrapher( + { type: "slug", id: slug }, + searchParams, + "png", + env + ) ) .get( "/grapher/thumbnail/:slug.svg", async ({ params: { slug } }, { searchParams }, env) => - fetchAndRenderGrapher(slug, searchParams, "svg", env) + fetchAndRenderGrapher( + { type: "slug", id: slug }, + searchParams, + "svg", + env + ) ) .get( "/grapher/thumbnail/:slug", async ({ params: { slug } }, { searchParams }, env) => - fetchAndRenderGrapher(slug, searchParams, "svg", env) + fetchAndRenderGrapher( + { type: "slug", id: slug }, + searchParams, + "svg", + env + ) ) .all("*", () => error(404, "Route not defined"))