How Craft, GraphQL, and a TypeScript frontend forces me to develop defensively

02/02/2024

The current iteration of this site still uses Craft CMS as a backend (I think that's the first time I've ever re-developed this site but kept the backend the same), and it's using GraphQL as it did previously, but it's now using Astro with TypeScript for the frontend, rather than Next.js. I like working with React for lots of things, but it annoys me so much in many ways that I didn't really want to go through that again. We've started using Astro for some headless stuff, owing to its simplicity and the fact that you can run it directly in Cloudflare, so it's totally abstracted from your backend. I still remain surprised by how quickly we can go Browser -> Cloudflare Pages -> Craft -> Cloudflare Pages -> Browser. It doesn't seem like it should be physically possible, but it seems to be happening!

I wanted to write about this because every time I learn something new like this, I feel like I have to un-learn everything that I had previously thought was bad, and change my opinions before I can really get to grips with something. I've never considered myself to be so mercurial before, but apparently you have to be to keep up with modern software trends. By the time I'm done with this post, I expect it to be outdated.

Something that really helped me to get behind this approach is the fact that this blog has three different post types: a block-based structure where I choose from text/images/YouTube videos/Github Gists, and a couple of other legacy block types to construct a post; a status-style post which is super stripped down; and a legacy style which is just a WYSIWYG everything gets dumped into. The reason this is so fundamental to everything is that these are all different types in GraphQL, so they're different types in Astro, and if you're in the context of a blog post, you have to be super defensive about which type of blog post when accessing fields in case they don't exist. In Craft's implementation of GraphQL, it seems like every field is optional, so you always have to check if things are undefined, but you also need to check the __typename of everything before even accessing those variables otherwise linters and LSPs start to complain (and rightly so - no point using TS if you're just going to ignore it).

Astro's markup structure is HTML-like with JS. This is something that I call JSX sometimes but I don't think that's technically correct. JS devs are always quick to point out when you're not being technically correct. It's really useful to be able to use JS in your templates for simple flow control and loops, but anything beyond that begins to get really horrible to read and maintain. I wanted to avoid, as much as possible {entry.__typename == 'blog_Blog_Entry' && (<div... throughout everything, and instead determine what the typename is once, then call a component specifically build to handle that one post type. I thought it'd make things easier to read. And to make matters slightly more complicated, my content block approach above is used both in posts and pages, so I wanted to adhere to Don't Repeat Yourself here. The way I've done this is with a Factory-style approach that just acts like a wrapper that decides what to do when each type is encountered.

The factory component is called from my main page - the only thing I need to do in the page is make sure that I pull in the fragment defined in the factory into my query document so that the data is there when I include the component, then pass that data to the component. This way of working seems totally logical to me, but also goes against everything I learned for a significant part of my career, which is that your view logic should be decoupled from everything else. Now my way of thinking is very much that my view logic is heavily dependent on everything else, so why wouldn't I define it together? I'm so tired of having to hop from file to file to find out how some data got from where it was called to where it was used to try and figure out why it didn't arrive. This approach, whilst obviously not infallible, reduces those kinds of issues significantly. It also means that once everything's set up, if I wanted to add a new block type, I can just do it in one file, and not have to add calls to it in multiple locations just to make sure it ends up in the implementation itself.

Rather than waffle on about this any longer, it's probably best to just show an example. This is excerpted and I've removed some stuff that isn't relevant. Typically, Github has decided that the order I defined the files isn't the order that I actually wanted to show them in, so here they all are here.

---
import Layout from "../../components/Layout.astro";
import { graphqlClient } from "../../graphql";
import { getFragmentData, graphql } from "../../gql";
import Factory, {
BlogShowFragment
} from "../../components/blog/show/Factory.astro";
import Pagination from "../../components/blog/show/Pagination.astro";
import Menu from "../../components/Menu.astro";
export interface Params {
slug: string;
}
const BlogShowDocument = graphql(`
query BlogShow($slug: [String]) {
...BlogShowFragment
...MenuFragment
}
`);
const PaginationDocument = graphql(`
query BlogShowPagination($date: String!) {
...BlogShowPaginationFragment
}
`);
const post = await graphqlClient.request({
document: BlogShowDocument,
variables: { slug: Astro.params.slug },
});
const data = getFragmentData(BlogShowFragment, post);
const { entry } = data;
if (!entry) {
return new Response("404", { status: 404, statusText: "Not Found" });
}
const pagination = await graphqlClient.request({
document: PaginationDocument,
variables: { date: entry.postDate },
});
---
<Layout
title={`Jasper is blogging ${entry.title}`}
type="article"
url={entry.url ?? "https://jasper.tandy.is"}
>
<Menu data={post} />
<section class="w-full max-w-prose grow">
<Factory data={post} />
<Pagination data={pagination} />
</section>
</Layout>
view raw [slug].astro hosted with ❤ by GitHub
---
import { type FragmentType, graphql, getFragmentData } from "../../../gql";
// my post type components
import BlogEntry from "./BlogEntry.astro";
import LegacyBlogEntry from "./LegacyBlogEntry.astro";
import StatusBlogEntry from "./StatusBlogEntry.astro";
// my fragment just wants to know the __typename and nothing else
export const BlogShowFragment = graphql(`
fragment BlogShowFragment on Query {
entry(section: "blog", slug: $slug) {
__typename
...StatusBlogShowFragment
...BlogEntryShowFragment
...LegacyBlogShowFragment
}
}
`);
// some graphql-codegen boilerplate so that the linter knows
// what I'm expecting within this template - defining this
// next to my fragments keeps everything neat by convention
export interface Props {
data: FragmentType<typeof BlogShowFragment>;
}
const { entry } = getFragmentData(BlogShowFragment, Astro.props.data);
if (!entry) {
return new Response("404", { status: 404, statusText: "Not Found" });
}
---
<>
{entry.__typename === "blog_blog_Entry" && <BlogEntry entry={entry} />}
{
entry.__typename === "blog_legacyBlog_Entry" && (
<LegacyBlogEntry entry={entry} />
)
}
{
entry.__typename === "blog_legacyStatus_Entry" && (
<StatusBlogEntry entry={entry} />
)
}
</>
view raw Factory.astro hosted with ❤ by GitHub
---
import { type FragmentType, graphql, getFragmentData } from "../../../gql";
import { formatRelative } from "date-fns";
// external shared components that this layout depends on
import Currently from "./Currently.astro";
import TagList from "./TagList.astro";
import ContentBlocks from "../../ContentBlocks.astro";
export const BlogEntryShowFragment = graphql(`
fragment BlogEntryShowFragment on blog_blog_Entry {
title
displayTitle
postDate
excerpt
...TagListFragment
...CurrentlyFragment
contentBlocks {
...ContentBlocksFragment
}
}
`);
export interface Props {
entry: FragmentType<typeof BlogEntryShowFragment>;
}
const entry = getFragmentData(BlogEntryShowFragment, Astro.props.entry);
---
<div>
{
entry.displayTitle && (
<div class="border-gray prose mx-6 mb-2 border-b pb-3 pt-5">
<h1>{entry.title}</h1>
</div>
)
}
<p class="m-6 mt-1 italic">
<small>{formatRelative(entry.postDate, new Date())}</small>
</p>
{entry.contentBlocks.map((c) => c && <ContentBlocks data={c} />)}
<Currently data={entry} />
<TagList data={entry} />
</div>
view raw BlogEntry.astro hosted with ❤ by GitHub
---
import { type FragmentType, graphql, getFragmentData } from "../gql";
import ImageBlock from "./blog/show/ImageBlock.astro";
// this fragment is called from BlogEntry.astro but doesn't actually
// do anything related to it. This approach can result in some
// redundancy (like if two fragments at the same level both use
// `title`) but that does mean that each component can be confident
// that its data is available regardless of what else might be on
// your page
export const ContentBlocksFragment = graphql(`
fragment ContentBlocksFragment on contentBlocks_MatrixField {
__typename
... on contentBlocks_text_BlockType {
text
containsSpoilers
illuminate
}
... on contentBlocks_image_BlockType {
image {
title
url @transform(handle: "f1500", immediately: true)
width @transform(handle: "f1500", immediately: true)
height @transform(handle: "f1500", immediately: true)
}
caption
captionAlign
}
... on contentBlocks_gallery_BlockType {
id
image: images {
title
url @transform(handle: "f1500", immediately: true)
width @transform(handle: "f1500", immediately: true)
height @transform(handle: "f1500", immediately: true)
originalUrl: url
... on images_Asset {
caption: excerpt
panorama
}
}
carousel
}
... on contentBlocks_hr_BlockType {
display
}
... on contentBlocks_gist_BlockType {
gist
file
}
... on contentBlocks_youtube_BlockType {
vid
padding
}
... on contentBlocks_conversation_BlockType {
conversation {
person
said
participantNumber
}
}
}
`);
export interface Props {
data: FragmentType<typeof ContentBlocksFragment>;
}
const block = getFragmentData(ContentBlocksFragment, Astro.props.data);
// Tailwind won't pull in classnames if you interpolate. I get it, but
// it's annoying sometimes! I don't even use this block-type.
const participantClassNames: { [keyof: string]: string } = {
"1": "participant-1",
"2": "participant-2",
"3": "participant-3",
"4": "participant-4",
};
---
{
block &&
((block.__typename === "contentBlocks_text_BlockType" && (
<div class="prose mx-6 pb-3" set:html={block.text} />
)) ||
(block.__typename === "contentBlocks_image_BlockType" && (
<ImageBlock image={block.image?.[0]} />
)) ||
(block.__typename === "contentBlocks_gallery_BlockType" &&
block.image.map((image) => <ImageBlock image={image} />)) ||
(block.__typename === "contentBlocks_hr_BlockType" && (
<div class="prose mx-6">
<hr />
</div>
)) ||
(block.__typename === "contentBlocks_gist_BlockType" && (
<div class="mx-6">
<script
src={`https://gist.github.com/${block.gist}.js?file=${block.file}`}
/>
</div>
)) ||
(block.__typename === "contentBlocks_youtube_BlockType" && (
<div class="aspect-h-9 aspect-w-16 mb-5">
<iframe
src={`https://www.youtube.com/embed/${block.vid}`}
title={`YouTube: ${block.vid}`}
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share"
allowfullscreen
/>
</div>
)) ||
(block.__typename === "contentBlocks_conversation_BlockType" &&
block.conversation && (
<div class="mx-6 mb-3">
{block.conversation.map(
(line) =>
line && (
<div
class:list={[
participantClassNames[line.participantNumber ?? "1"],
"p-1",
]}
>
<strong class="mr-1">{line.person}:</strong>
{line.said}
</div>
),
)}
</div>
)))
}

You can see how messy ContentBlocks.astro looks in comparison because it has the typename checks in there as well as the block implementation. Whilst there's an argument to be made for going a level deeper and making each of those their own components, I can live with this for maintenance's sake, otherwise adding another block type does become quite a lot more labour intensive.

Because Astro only lets you define one component per file, I have ended up with a lot of files defining each of my entry types. Whilst this was a drag to create (and learn - I felt like I was losing my mind when it wasn't working!), it's a lot easier to maintain. Once my [slug] file is calling my Factory fragment, which calls my post type fragments, which call the data they rely on, maintenance and debugging become so much easier. To add a content block, I only need to add it in ContentBlocks.astro, and it will appear wherever I have previously set up to use ContentBlocks because I changed the fragment they all reference. Writing it like that seems like stating the obvious, but I have been pretty opposed to this way of working for a very long time and I'm now wondering very loudly why that is!
 

Like a Sunday Morning I've added a new Media Diet area on this site. I have been bouncing the idea of doing this for a...