# Waku ⛩️ The minimal React framework ## Introduction **Waku** _(wah-ku)_ or **わく** is the minimal React framework. It’s lightweight and designed for a fun developer experience, yet supports all the latest React 19 features like server components and actions. Built for marketing sites, headless commerce, and web apps. For large enterprise applications, you may prefer a heavier framework. > Please try Waku on non-production projects and report any issues you find. Contributors are welcome. ## Getting started Start a new Waku project with the `create` command for your preferred package manager. It will scaffold a new project with our default [Waku starter](https://github.com/wakujs/waku/tree/main/examples/01_template). ```sh npm create waku@latest ``` #### Commands - `waku dev` to start the local development server - `waku build` to generate a production build - `waku start` to serve the production build locally **Node.js version requirement:** `^24.0.0` or `^22.12.0` or `^20.19.0` ## Rendering While there’s a bit of a learning curve to modern React rendering, it introduces powerful new patterns of full-stack composability that are only possible with the advent of [server components](https://github.com/reactjs/rfcs/blob/main/text/0188-server-components.md). So please don’t be intimidated by the `'use client'` directive! Once you get the hang of it, you’ll appreciate how awesome it is to flexibly move server-client boundaries with a single line of code as your full-stack React codebase evolves over time. It’s way simpler than maintaining separate codebases for your backend and frontend. And please don’t fret about client components! Even if you only lightly optimize towards server components, your client bundle size will be smaller than traditional React frameworks, which are always 100% client components. > Future versions of Waku may provide additional opt-in APIs to abstract some of the complexity away for an improved developer experience. #### Server components Server components can be made async and can securely perform server-side logic and data fetching. Feel free to access the local file-system and import heavy dependencies since they aren’t included in the client bundle. They have no state, interactivity, or access to browser APIs since they run _exclusively_ on the server. ```tsx // server component import db from 'some-db'; import { Gallery } from '../components/gallery'; export const Store = async () => { const products = await db.query('SELECT * FROM products'); return ; }; ``` #### Client components A `'use client'` directive placed at the top of a file will create a server-client boundary when imported into a server component. All components imported below the boundary will be hydrated to run in the browser as well. They can use all traditional React features such as state, effects, and event handlers. ```tsx // client component 'use client'; import { useState } from 'react'; export const Counter = () => { const [count, setCount] = useState(0); return ( <>
Count: {count}
); }; ``` #### Shared components Simple React components that [meet all of the rules](https://github.com/reactjs/rfcs/blob/main/text/0188-server-components.md#sharing-code-between-server-and-client) of both server and client components can be imported into either server or client components without affecting the server-client boundary. ```tsx // shared component export const Headline = ({ children }) => { return

{children}

; }; ``` #### Weaving patterns Server components can import client components and doing so will create a server-client boundary. Client components cannot import server components, but they can accept server components as props such as `children`. For example, you may want to add global context providers this way. ```tsx // ./src/pages/_layout.tsx import { Providers } from '../components/providers'; export default async function RootLayout({ children }) { return (
{children}
); } export const getConfig = async () => { return { render: 'static', } as const; }; ``` ```tsx // ./src/components/providers.tsx 'use client'; import { Provider } from 'jotai'; export const Providers = ({ children }) => { return {children}; }; ``` #### Server-side rendering Waku provides static prerendering (SSG) and server-side rendering (SSR) options for both layouts and pages including all of their server _and_ client components. Note that SSR is a distinct concept from RSC. #### tl;dr: Each layout and page in Waku is composed of a React component hierarchy. It begins with a server component at the top of the tree. Then at points down the hierarchy, you’ll eventually import a component that needs client component APIs. Mark this file with a `'use client'` directive at the top. When imported into a server component, it will create a server-client boundary. Below this point, all imported components are hydrated and will run in the browser as well. Server components can be rendered below this boundary, but only via composition (e.g., `children` props). Together they form [a new “React server” layer](https://github.com/reactwg/server-components/discussions/4) that runs _before_ the traditional “React client” layer with which you’re already familiar. Client components are still server-side rendered as SSR is separate from RSC. Please see the [linked diagrams](https://github.com/reactwg/server-components/discussions/4) for a helpful visual. #### Further reading To learn more about the modern React architecture, we recommend [Making Sense of React Server Components](https://www.joshwcomeau.com/react/server-components/) and [The Two Reacts](https://overreacted.io/the-two-reacts/). ## Routing Waku provides a minimal file-based “pages router” experience built for the server components era. Its underlying [low-level API](https://github.com/wakujs/waku/blob/main/docs/create-pages.mdx) is also available for those that prefer programmatic routing. This documentation covers file-based routing since many React developers prefer it, but please feel free to try both and see which you like more! ### Overview The directory for file-based routing in Waku projects is `./src/pages`. Layouts and pages can be created by making a new file with two exports: a default function for the React component and a named `getConfig` function that returns a configuration object to specify the render method and other options. Waku currently supports two rendering options: - `'static'` for static prerendering (SSG) - `'dynamic'` for server-side rendering (SSR) Layouts, pages, and slices are all `static` by default, while api handlers default to `dynamic`. For example, you can statically prerender a global header and footer in the root layout at build time, but dynamically render the rest of a home page at request time for personalized user experiences. ```tsx // ./src/pages/_layout.tsx import '../styles.css'; import { Providers } from '../components/providers'; import { Header } from '../components/header'; import { Footer } from '../components/footer'; // Create root layout export default async function RootLayout({ children }) { return (
{children}