17 April 2024
I’m tired of Next. I want to try something fresh. So I thought this was the time I look into other frameworks. Where would I go? Nuxt? Remix? Nah…
Astro! Let’s test it and see if it can handle modern project requirements.
Here is the repo, follow along!
So I guess you could say that Next.js is the reason I’m looking into Astro. 😁
I’ve been ignoring it like I do with Remix and Svelte and a bunch of other frameworks. However, I’ve learned that Astro is different, and I’m gonna talk about how it’s different later in the post.
First, I needed to define a set of criteria with which I could judge its usefulness. How do I identify whether I should choose it (or recommend it) for a new project?
So I started thinking and reflecting on all the companies and all the projects I ever I worked with. What was I doing there? What requirements they had in common? What kind of work they were assigning to me?
And so I defined a list of such things, things that I was doing on EVERY job or contract. This is what everyone wanted me to do from tiny startups to humongous enterprises.
.module.css
): no one wants to load ALL of the CSS for every page and then use 1/100th of it.@tanstack/query
), e.g. you know that sidebar with filters on a page that lists a couple of dozens of products?So I did this silly “Frankenstein” project with those requirements. Let’s talk about the results!
The main conclusion that I got out of it is that Astro.js is like putting your project on a diet. On a JavaScript diet.
The thing about Astro is that it promotes the ****“reduced JavaScript” approach. That’s a whole other kind of mentality shift than what I initially was hoping for.
So it doesn’t render JavaScript unless you EXPLICITLY tell it to.
“Explicit” is the word I’d associate with Astro. While Next hides a lot of implementation details, Astro does nothing of such a thing and leaves implementation to us, developers.
If you want some interactivity with JavaScript on the front-end - you need to write it, e.g. addListener
s in the script
tags. Otherwise, Astro ships nothing to the browser.
Before I wasn’t really thinking explicitly much about how much JavaScript is and isn’t used in the browser. Most of the time, I just let frameworks figure this out for me, and only when we were starting to have issues with performance, etc. only then I start looking “under the hood” and optimising.
All in all, I’m happy with this approach. I kinda like it. Cognitively it feels easier to work this way.
This leads to the next point.
Astro allows you to use ALL sorts of UI frameworks. Including React. BUT!
React is a UI library. And Astro is using it as such. RSC is not a thing there (yet?). So if you’re writing a .tsx
/ .jsx
file, you can be 100% sure it’s going to be rendered, live and work only in the browser.
Yes, you can certainly feed it server-fetched data, so the user doesn’t see the loader for too long. But it still remains a client only component.
For that reason, you can’t really use React ONLY or go too deep component-wise. They’re all end up on the client.
Do you want that? Is that a good thing? It depends on the project.
If you need to render things on the server first, say, for SEO reasons, and make it interactive, then I’d assume you’d need to have a conditional variable, e.g.
const hasBeenChangedByUser = false return {hasBeenChangedByUser ? <ReactComponent /> : <AstroComponent /> }
And both <AstroComponent />
and <ReactComponent />
, in this case, would look very similar. You could do a sharable code though, because Astro still allows rendering .tsx
component on the server, it just strips them of all the JavaScript, making them redundant for React purpose but great for layout, etc.
In my experience, there weren’t many cases when we’d had to create something drastic like that, rather a healthy balance of server vs client. So this “React == UI only” approach doesn’t really bother me. I can render what I need on the server inside .astro
component, and components that need to be overly interactive, they’re client in any framework anyway, because interactivity == JavaScript.
With Astro, we have two types of files: .astro
and .tsx
. Guess what .astro
components can’t have that React just LOVES?
Providers!
Here’s another mentality shift for you. If you were thinking in a providers pattern, you need to snap out of it. You can no longer store your theme in a provider. No more RadixUI or MantineUI for you.
So to have a theme and reusable CSS variables you’d need to go with how Tailwind works. Tailwind doesn’t use providers. It stores all the variables in it’s own config file.
Something like NextUI would work, i.e. a “New Gen” UI framework. It’s Tailwind based AND it works in both .astro
and .tsx
files. Amazing!
To be frank, my last 4 projects were Tailwind-based. So I don’t see a problem with that. I can live with that choice.
Guess WHAT also generally uses providers?
State managers!
We want to store state across Astro and React components. One state, two frameworks. How do you do that?
Framework agnostic state managers!
In its docs, Astro recommends nanostores, but I’ve used effector in the past. And LOVED IT. So I’ve used it for this project as well.
State managers like these are just JavaScript. They don’t care what frameworks they’re used with. They do have wrappers for them, e.g.
import { useUnit } from "effector-react"; // ... const { text, size } = useUnit({ text: $text, size: $size, });
but you can do without them.
Have you noticed that in v5 @tanstack/query
moved on from using provider pattern for their QueryClient, e.g.
import { QueryClient, QueryClientProvider } from "react-query"; const queryClient = new QueryClient(); function App() { return <QueryClientProvider client={queryClient}>...</QueryClientProvider>; }
to a more function-based approach, e.g.
import { QueryClient, useQuery } from '@tanstack/react-query' const queryClient = new QueryClient() function App() { const query = useQuery({ ... }, queryClient) //... }
There’s a reason for that, I think. It’s more flexible, more universal, doesn’t lock you into a particular framework. You can fetch data on the server in .astro
component, store it in your queryClient
and then use it in a React component on the client with useQuery
.
Switch React to Vue and you won’t need to touch the state in any way. Amazing refactoring experience, isn’t it? 🤯
Following the providers rant…
Since there are no providers, we need to somehow communicate the language we’re working with to every component when setting up our localisation. So we do it this way 👇
// In Astro component --- const lang = getLangFromUrl(Astro.url); --- return <Modal client:only="react" lang={lang} /> // Then in Modal.tsx export const Modal = ({ lang }) => { const t = useTranslations(lang); return ( <ModalHeader> {t("modal.header")} </ModalHeader> ) }
So the language won’t be coming from a provider, but rather as a prop. And we’ll be getting it from our server .astro
pages.
Which, again, to me isn’t a big deal. It’s just different. It works and it still looks nice. We can even define a “parent” interface that we’ll be using for all the prop types, e.g.
interface PropsBase { lang: string } interface ComponentProp extends PropsBase { //... }
I keep coming back to that word.
In a lot of other aspects, Astro is not that different from Next developer experience and mentality-wise. I didn’t even feel pain when switching. I already knew where things would go, where things would come from, how to structure a project, etc.
Astro has its own place for getServerSideProps
(or use server
); it has an api
folder you can create to store some serverless functions in; it runs in node and you can build a Dockerfile for it. and you can throw all your Google Analytics scripts to PARTYTOWN, just like you’d do in Next.
And if you need to set up SSR authentication, you can do that, it’s pretty straightforward. I was able to set up one with Supabase Auth.
So it’s just like Next, but better.
Just a few ‘gotchas’ though…
Astro has no runtime. This means no unit tests. This also means no Storybook for your Astro components (although, they’re working on it!)
Can you live with that?
I’d say yes. When it comes to testing, server-rendered components don’t have much interactivity going on anyway, no user clicks, no state changes, etc. So tests might be considered redundant or over-engineer.
And the layout render we can always test in e2e/integration/visual kinds of tests with Cypress or PlayWright.
As for Storybook, well, not everything needs to be in a Storybook. Surely we can find a nice balance.
At the moment ISR comes down to cache control/invalidation in SSR pages, not static page regeneration.
There is no per-page building, no ISR. If you rebuild static pages, you rebuild all of them.
The build is fast though, as it was pointed out, Astro certainly can handle 10K+ pages.
Nonetheless, the Astro team is already thinking and working on the build engine. So it will only become better.
Not even more confusing as it was in Next.js case, I hope… 😅
Astro is great. But it is slightly different from the way Next.js projects are done and not everyone would be happy to compromise. And that’s ok. We all have our own set of requirements, values, goals, the environment we work in, etc.
However, I think Astro is the future. And I don’t mean Astro framework. I think the Astro way of doing things: explicit, transparent, and minimalistic. This is the future where all the frameworks work as one, and where frameworks don’t overwrite standardised APIs and expose important bits to the developers allowing us to modify and hack it to each project’s own needs.
The future is simple, vanilla JavaScript because simplicity always wins. 🚀
For more talking and less reading, feel free to watch my stream 👇