-
Notifications
You must be signed in to change notification settings - Fork 6
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Feature/clientside refactor #215
Merged
Merged
Changes from all commits
Commits
Show all changes
11 commits
Select commit
Hold shift + click to select a range
4b77f3e
metadata
nlkluth e94846a
update
nlkluth b38f223
about
nlkluth 133cc62
categories
nlkluth dd27683
components page
nlkluth e4fa142
products page
nlkluth f02c96a
home
nlkluth 369571b
details page
nlkluth c87c88d
moving files
nlkluth 050ea1b
moving
nlkluth 87eed83
fix
nlkluth File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,294 @@ | ||
import AboutPage from "app/migration/about"; | ||
import { Metadata } from "next"; | ||
|
||
import Link from "next/link"; | ||
import { BreadIcon } from "../ui/shared-ui"; | ||
import { Breadcrumbs } from "app/components/Breadcrumbs"; | ||
import LocalImage from "app/components/LocalImage"; | ||
import bigPictureImage from "../../../../docs/img/big-picture.png"; | ||
import cacheDiagramImage from "../../../../docs/img/caching-diagram.png"; | ||
|
||
const DIAGRAM_WIDTH = 700.0; | ||
const DIAGRAM_HEIGHT = DIAGRAM_WIDTH / 1.86; | ||
const FASTLY_WIDTH = 700.0; | ||
const FASTLY_HEIGHT = FASTLY_WIDTH / 2.93; | ||
|
||
export const metadata: Metadata = { | ||
title: "About", | ||
description: "About the Formidable Boulangerie project.", | ||
}; | ||
|
||
export default async function Page() { | ||
return <AboutPage />; | ||
return ( | ||
<> | ||
<div className="container my-4"> | ||
<Breadcrumbs /> | ||
</div> | ||
<div className="container py-9 grid grid-cols-1 lg:grid-cols-2 gap-9 text-primary"> | ||
<div className="order-1 flex flex-col gap-9 justify-between"> | ||
<h1 className="text-h4 md:text-h2">Learn more about the Formidable Ecommerce demo site</h1> | ||
<p> | ||
We don’t really sell bread. The goal of the project is to provide a realistic demonstration of running a | ||
highly performant and available e-commerce site with data sourced from Sanity’s headless CMS. The app is | ||
powered by | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://nextjs.org/" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Next.js | ||
</a> | ||
, | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://www.sanity.io/" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Sanity CMS | ||
</a> | ||
, and | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://www.fastly.com/" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Fastly | ||
</a> | ||
. | ||
</p> | ||
</div> | ||
<div className="order-2"> | ||
<LocalImage | ||
alt="big picture diagram" | ||
src={bigPictureImage} | ||
width={DIAGRAM_WIDTH} | ||
height={DIAGRAM_HEIGHT} | ||
className="rounded-2xl max-w-full w-auto object-contain object-center" | ||
priority | ||
/> | ||
</div> | ||
<div className="order-3 lg:col-span-2 flex flex-col gap-9 justify-between"> | ||
<div> | ||
<h4 className="text-h4 mb-4">Headless CMS-driven architecture.</h4> | ||
<p className="mb-4"> | ||
The e-commerce data is stored in a headless CMS (powered by Sanity). The project uses Next.js (deployed on | ||
Vercel) to render the site, and Fastly is placed in front of Vercel to cache server-rendered webpages for | ||
speed and availability. | ||
</p> | ||
<ul> | ||
<li className="mb-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<h1 className="font-medium">Sanity CMS</h1> | ||
Sanity is used for storing information about our e-commerce products. The data from Sanity is fetched | ||
using | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://www.sanity.io/docs/groq" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
GROQ | ||
</a> | ||
– a query language, used for fetching data. Formidable built | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://formidable.com/open-source/groqd/" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Groqd | ||
</a> | ||
– a schema-unaware, runtime and type-safe query builder for GROQ. | ||
</div> | ||
</li> | ||
<li className="mb-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<h1 className="font-medium">Sanity Studio</h1> | ||
Sanity Studio is a web interface for Sanity’s headless CMS. It is used for creating and editing the | ||
data on the site. The models for Sanity are created in code and tracked in source control. Sanity | ||
Studio is integrated into the NextJS application and deployed alongside{" "} | ||
<Link href="studio" className="hover:underline font-bold"> | ||
as a route | ||
</Link> | ||
. | ||
</div> | ||
</li> | ||
<li className="mb-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<h1 className="font-medium">NextJS app</h1> | ||
To show the CMS data to end-users we created a Next.js web app that server-renders some common | ||
e-commerce pages, including a landing page, a Product Listing Page (PLP) with sorting and filtering, | ||
and a Product Details Page (PDP). The Next.js app is deployed to{" "} | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://vercel.com" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Vercel | ||
</a> | ||
via their git pipeline. In a real-world e-commerce app, we expect to experience some heavy loads | ||
on pages whose data doesn’t change much between visits, and therefore we can deploy caching strategies | ||
to reduce the load on our source server. | ||
</div> | ||
</li> | ||
</ul> | ||
</div> | ||
<div> | ||
<h4 className="text-h4 mb-4">The caching story.</h4> | ||
<ul> | ||
<li className="mb-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<h1 className="font-medium">Fastly CDN and Caching</h1> | ||
<p className="mb-4"> | ||
In order to enhance the speed of the app, we are utilizing Fastly’s CDN with a high cache-lifetime | ||
for server-rendered pages. We are using Fastly to both cache and host the subdomain used for this | ||
showcase app. The data flow involved in caching is illustrated below: | ||
</p> | ||
<div className="grid grid-cols-1 md:grid-cols-2 gap-9 text-primary"> | ||
<LocalImage | ||
alt="cache diagram" | ||
src={cacheDiagramImage} | ||
width={FASTLY_WIDTH} | ||
height={FASTLY_HEIGHT} | ||
className="rounded-2xl max-w-full h-auto object-contain object-center" | ||
/> | ||
<div> | ||
<div> | ||
To cache our server-rendered pages at the Fastly layer, we use response headers to indicate | ||
what/how we want Fastly to cache our responses from the source server. We need to a couple key | ||
ingredients: | ||
<ul> | ||
<li className="my-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<code>Surrogate-Control</code> response header needs to be added to pages where caching is | ||
desired. | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://docs.fastly.com/en/guides/working-with-surrogate-keys" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Learn more. | ||
</a> | ||
</div> | ||
</li> | ||
<li className="my-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<code>Surrogate-Key</code> response header needs to be added to enable appropriate cache | ||
invalidation. | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://developer.fastly.com/reference/api/purging" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
Learn more. | ||
</a> | ||
</div> | ||
</li> | ||
</ul> | ||
</div> | ||
</div> | ||
</div> | ||
<div className="mt-3"> | ||
On the Next.js side we’ll need to include a few primary response headers to then control caching (in | ||
our case, we’re setting these headers from <code>middleware</code> on server-rendered pages that | ||
we’d like to cache). | ||
<ul> | ||
<li className="my-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<code>surrogate-control</code> Fastly-specific header used to set the cache policies. | ||
</div> | ||
</li> | ||
<li className="my-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<code>surrogate-key</code> Fastly-specific header that allows purging by key. Note: this | ||
header is removed by Fastly before sending the response to the client. To see the value of | ||
this header, you must include the | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://developer.fastly.com/reference/http/http-headers/Fastly-Debug/" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
<code>Fastly-Debug</code> | ||
</a> | ||
header in your request. | ||
</div> | ||
</li> | ||
<li className="my-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<code>cache-control</code> used to indicate to browsers and Vercel to not cache so that we can | ||
handle caching solely at the Fastly layer. | ||
</div> | ||
</li> | ||
</ul> | ||
</div> | ||
<p className="mt-4"> | ||
With these response headers implemented, Fastly will start caching our responses and give us a path | ||
to invalidate our cache when necessary. In our case, we use data items’ <code>slug</code>s as part | ||
of our <code>surrogate-key</code> header to indicate what items’ data are used to render a | ||
page so that we can invalidate accordingly when any of those items’ data changes. | ||
</p> | ||
</div> | ||
</li> | ||
<li className="mb-3 last-of-type:mb-0 flex items-baseline gap-2"> | ||
<div className="top-1 relative"> | ||
<BreadIcon /> | ||
</div> | ||
<div> | ||
<h1 className="font-medium">Cache Invalidation and Purging</h1> | ||
When CMS data changes, a Sanity webhook is triggered and makes a request to an API endpoint in our | ||
Next.js app. The endpoint does some validation on the request (to make sure it’s coming from a trusted | ||
Sanity webhook), and then makes a request to Fastly’s API to invalidate/purge our cache accordingly. | ||
The Sanity webhook payload contains information (in our case, an item’s | ||
<a | ||
className="hover:underline font-bold" | ||
href="https://www.sanity.io/docs/slug-type" | ||
target="_blank" | ||
rel="noreferrer noopener" | ||
> | ||
<code>slug</code> | ||
</a> | ||
about what data changes, and our API endpoint uses that <code>slug</code> to tell Fastly which | ||
cache data to invalidate (based on the <code>surrogate-key</code> set in the original response | ||
header). | ||
</div> | ||
</li> | ||
</ul> | ||
</div> | ||
</div> | ||
</div> | ||
</> | ||
); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
"use client"; | ||
|
||
import { AnimatePresence } from "framer-motion"; | ||
import { useSearchParams } from "next/navigation"; | ||
import React from "react"; | ||
import { FadeInOut } from "shared-ui"; | ||
|
||
const AnimatedProductDetail = ({ children }: React.PropsWithChildren) => { | ||
const query = useSearchParams(); | ||
const variant = query?.get("variant"); | ||
|
||
return ( | ||
<AnimatePresence initial={false} mode="wait"> | ||
<React.Fragment key={`${query?.get("slug")}:${variant}`}> | ||
<FadeInOut>{children}</FadeInOut> | ||
</React.Fragment> | ||
</AnimatePresence> | ||
); | ||
}; | ||
|
||
export default AnimatedProductDetail; |
2 changes: 2 additions & 0 deletions
2
packages/nextjs/components/Breadcrumbs.tsx → ...ges/nextjs/app/components/Breadcrumbs.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
File renamed without changes.
2 changes: 1 addition & 1 deletion
2
packages/nextjs/components/CategoryList.tsx → ...es/nextjs/app/components/CategoryList.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
2 changes: 1 addition & 1 deletion
2
packages/nextjs/components/FeaturedList.tsx → ...es/nextjs/app/components/FeaturedList.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A few of these added to follow this pattern https://nextjs.org/docs/app/building-your-application/rendering/composition-patterns#supported-pattern-passing-server-components-to-client-components-as-props