From 67dcdd318b508fae9ae1de59e604a2279eae508e Mon Sep 17 00:00:00 2001 From: James Southern Date: Thu, 21 Mar 2024 09:38:42 +0000 Subject: [PATCH] Fix static generation & switch to importing constituency json (#82) * Fix static generation & switch to importing constituency json * The storyblok cache avoidance was breaking static serving of non-dynamic routes * The unstable_cache for constituencydata import was causing issues with locally building for prod * update comments & remove cache control variable as per asibs suggestions --- app/[slug]/page.tsx | 4 ++- app/api/constituencies/[slug]/route.ts | 21 ++---------- app/layout.tsx | 30 +++++++++++------ utils/constituencyData.ts | 46 +++++--------------------- 4 files changed, 33 insertions(+), 68 deletions(-) diff --git a/app/[slug]/page.tsx b/app/[slug]/page.tsx index 3156eea..95a49b5 100644 --- a/app/[slug]/page.tsx +++ b/app/[slug]/page.tsx @@ -24,7 +24,9 @@ export async function generateStaticParams() { const files = await readdir(directory); // Get rid of the .json extension - const slugs = files.map((file) => file.substring(0, file.indexOf("."))); + const slugs = files + .map((file) => file.substring(0, file.indexOf("."))) + .filter((slug) => slug !== ""); console.log("Top-level static slugs:", slugs); return slugs.map((c) => ({ diff --git a/app/api/constituencies/[slug]/route.ts b/app/api/constituencies/[slug]/route.ts index a78f65b..17986ce 100644 --- a/app/api/constituencies/[slug]/route.ts +++ b/app/api/constituencies/[slug]/route.ts @@ -11,7 +11,7 @@ export const revalidate = false; // Never revalidate, always use cached version // Return a list of `params` to populate the [slug] dynamic segment export async function generateStaticParams() { - const constituencySlugs = await getConstituencySlugs(); + const constituencySlugs = getConstituencySlugs(); return constituencySlugs.map((slug) => ({ slug: slug })); } @@ -21,26 +21,9 @@ export async function GET( request: NextRequest, { params }: { params: { slug: string } }, ) { - console.log(`GENERATING API RESPONSE FOR CONSTITUENCY ${params.slug}`); - - // The cached version of constituency data doesn't seem to work in this file - because - // it's an API route perhaps? - const constituencyData: ConstituencyData = await getConstituencyData( - params.slug, - false, - ); - console.log("Successfully fetched constituencies data"); + const constituencyData: ConstituencyData = getConstituencyData(params.slug); if (constituencyData) { - console.log( - `Found constituency data for ${params.slug}, - constituencyIdentifiers ${constituencyData.constituencyIdentifiers}, - partySlug ${constituencyData.recommendation.partySlug}, - partyName ${partyNameFromSlug( - constituencyData.recommendation.partySlug, - )}`, - ); - return Response.json({ constituencyIdentifiers: constituencyData.constituencyIdentifiers, recommendation: { diff --git a/app/layout.tsx b/app/layout.tsx index cc03437..b5d0354 100644 --- a/app/layout.tsx +++ b/app/layout.tsx @@ -8,16 +8,27 @@ import StoryblokProvider from "@/storyblok/components/StoryblokProvider"; import StoryblokWrapper from "@/storyblok/components/StoryblokWrapper"; import { ComponentsMap } from "@/storyblok/components/ComponentsMap"; -// Force next.js not to cache API calls by default. This means caching is OPT-IN rather than OPT-OUT. -// This avoids issues with the Storyblok js client, which has it's own built-in caching, and Next.js caching interferes with this... -export const revalidate = 0; +/* + * This line prevents caching when storyblok live editing is enabled: + * (revalidate = 0) + * Otherwise it sets the default for indefinite caching: + * (revalidate = false) + */ +export const revalidate = + process.env.ENABLE_STORYBLOK_LIVE_EDITING === "true" ? 0 : false; -storyblokInit({ - accessToken: process.env.STORYBLOK_API_TOKEN, - use: [apiPlugin], - apiOptions: { region: "eu" }, - components: ComponentsMap, -}); +if (process.env.ENABLE_STORYBLOK_LIVE_EDITING === "true") { + storyblokInit({ + accessToken: process.env.STORYBLOK_API_TOKEN, + use: [apiPlugin], + apiOptions: { region: "eu" }, + components: ComponentsMap, + }); +} else { + storyblokInit({ + components: ComponentsMap, + }); +} export const metadata: Metadata = { title: { @@ -80,7 +91,6 @@ export default function RootLayout({ {children} - ); } diff --git a/utils/constituencyData.ts b/utils/constituencyData.ts index 64f92c2..19791a0 100644 --- a/utils/constituencyData.ts +++ b/utils/constituencyData.ts @@ -1,15 +1,7 @@ -import { readFileSync } from "fs"; -import path from "path"; -import { unstable_cache } from "next/cache"; -import getConfig from "next/config"; +import constituencyJson from "@/data/constituency.json"; -const { serverRuntimeConfig } = getConfig() || {}; - -const getConstituencyData = async ( - constituencySlug: string, - cached: boolean = true, -): Promise => { - const constituenciesData = await getConstituenciesData(cached); +const getConstituencyData = (constituencySlug: string) => { + const constituenciesData = getConstituenciesData(); const constituencyData = constituenciesData.find( (c: ConstituencyData) => c.constituencyIdentifiers.slug === constituencySlug, @@ -24,38 +16,16 @@ const getConstituencyData = async ( return constituencyData; }; -const getConstituencySlugs = async ( - cached: boolean = true, -): Promise => { - const constituenciesData = await getConstituenciesData(cached); +const getConstituencySlugs = (): string[] => { + const constituenciesData = getConstituenciesData(); return constituenciesData.map( (c: ConstituencyData) => c.constituencyIdentifiers.slug, ); }; -const getConstituenciesData = async ( - cached: boolean = true, -): Promise => { - return cached - ? await getConstituenciesDataCached() - : await getConstituenciesDataUncached(); -}; - -const getConstituenciesDataCached = unstable_cache( - // Cache data function - async () => getConstituenciesDataUncached(), - // Cache key - [`data/constituency.json.${serverRuntimeConfig.appVersion}`], - // Cache options - { revalidate: false }, // Cache will never refresh -); - -const getConstituenciesDataUncached = async () => { - console.debug("constituencyData: fetching constituencies data"); - const filePath = path.join(process.cwd(), "data", "constituency.json"); - const fileContent = readFileSync(filePath, "utf8"); - console.debug("constituencyData: fetched constituencies data from file"); - return JSON.parse(fileContent); +const getConstituenciesData = (): ConstituencyData[] => { + // Have to typecast as the json load doesn't know about our enums + return constituencyJson as ConstituencyData[]; }; /**