Understand How Data Flows In SvelteKit

Published May 12, 2023

Table of Contents

Introduction

Understanding how data flows through your app in SvelteKit can be confusing because SvelteKit blurs the line between your backend and frontend.

🔥 If you want to learn SvelteKit you can watch the Complete SvelteKit Course For Building Modern Web Apps on YouTube and support my work by becoming a patron.

In another post I wrote about page versus standalone endpoints in SvelteKit if you need a better understanding through examples when to use each.

By the end you should have a clear understanding how data flows through your routes but also how you can pass data around in SvelteKit.

Before I start talking about data flow it’s important you understand what each file related to routing in SvelteKit does.

SvelteKit uses file-based routing and splits routes in directories where src/routes/posts/+page.svelte creates a new /posts route.

FileResponsibility
app.htmlHTML template SvelteKit uses for routes

🐿️ app.html is useful if you need to run some code before the page component mounts, like checking the user’s theme from local storage to set a theme.

A page is just a Svelte component. The load function inside +page.ts runs on the server and client but if you need to use a secret, talk to a database, or write to a file add .server, so the load function only runs on the server.

🐿️ The +page|layout.svelte file always has a sibling +page|layout.ts, or +page|layout.server.ts which does nothing on its own because it only provides data for the page through data.

FileResponsibility
+page.server.tsData for +page.svelte and form actions
+page.tsData for +page.svelte
+page.sveltePage with export let data prop from load

Layout wraps around your page using <slot /> and you can have nested layouts.

Layouts also pass the data to child routes through export let data.

Use +layout.ts to get the data for the page but if you need to use a secret, talk to a database, or write to a file append .server to the file.

FileResponsibility
+layout.server.tsData for +layout.svelte and child routes
+layout.tsData for +layout.svelte and child routes
+layout.svelteLayout with export let data prop from load

A standalone API endpoint named +server.ts can return anything and can be used by multiple routes. You can export GET, POST, PUT, PATCH, DELETE functions that map to the same HTTP request methods.

🐿️ If you need to send data from the browser to the server it’s easier to use SvelteKit form actions.

FileResponsibility
+server.tsStandalone API endpoint

SvelteKit hooks are functions used to intercept and override the framework’s default behavior.

FileResponsibility
hooks.server.tsRuns on every request and returns response but is optional unless you want to modify that request and response using the handle hook
hooks.client.tsNot as useful as hook.server.ts but you can use it to catch unexpected errors on the frontend with the handleError hook

Honorable mention for customizing error pages.

FileResponsibility
+error.svelteCustom error page

You can see there’s actually not a lot of files in SvelteKit you have to know about.

The +page|layout.svelte files paired with +page|layout.ts or +page|layout.server.ts use load functions to get the data for the page with the expection of +page.server.ts which can also export form actions.

Files that include load functions can also export page options to change the rendering method for individual, or multiple routes.

Understanding The Order In Which Things Run

To understand the order in which things run I’ve created an example with every file related to SvelteKit routing.

I’ve included a working example (you have to enable cookies) you can play around with. You can find the project files on GitHub.

I’m using the package chalk to style the output in the terminal and browser console with some helper functions I wrote inside lib/utils/log.ts but you can use a regular console.log() method.

Here is the SvelteKit project structure.

project
src/
src/
├── routes/
│   ├── actions
│   ├── api/secret
│   ├── error
│   ├── locals
│   ├── nested/route
│   ├── redirect
│   ├── stores
│   ├── +error.svelte
│   ├── +layout.server.ts
│   ├── +layout.svelte
│   ├── +layout.ts
│   ├── +page.server.ts
│   ├── +page.svelte
│   └── +page.ts
├── app.html
├── hooks.client.ts
└── hooks.server.ts

There’s a lot of files, so I’m only going to reference the relevant code here but you can open the example, or look at the project repository.

When you enter a URL in the browser like https://joyofcode.xyz/ you’re making a GET request to the server (SvelteKit).

You can listen to every request using the optional hooks.server.ts file inside the handle function which listens to requests and sends a response based on the request.

The default behavior is to resolve the request.

src/hooks.server.ts
export async function handle({ event, resolve }) {
  return resolve(event)
}

🐿️ The event object is the default argument for load functions and functions inside +server.ts files. You can send extra data to routes through event.locals.

Everything in a SvelteKit app is going to happen in-between the request and response until the page or resource is ready.

🐿️ You can learn more about SvelteKit hooks through example.

src/hooks.server.ts
import log from '$lib/utils/log'

export async function handle({ event, resolve }) {
	log.bold(`📣 NEW REQUEST IS BEING MADE FROM ${event.url.pathname}`)
	log.hooks('hooks.server.ts')

	const response = await resolve(event)

  log.bold(`🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE`)
  return response
}

Start the development server with pnpm run dev and open http://localhost:5173/.

First the server-only code runs which you can observe in the terminal.

terminal
📣 NEW REQUEST IS BEING MADE FROM /
hooks.server.ts
+layout.server.ts (load)
+page.server.ts (load)
+layout.ts (load)
+page.ts (load)
+layout.svelte
+page.svelte
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE

As expected the order is:

  • hooks.server.ts
  • +layout.server.ts’s load function
  • +page.server.ts’s load function
  • +layout.ts’s load function
  • +page.ts’s load function

The server-only files run first, after that other load functions, then the layout and pages. This is because pages are rendered twice, once on the server (SSR) and the second time on the client (hydration).

Time to start work in the browser where you can see the order if you look at the browser console inside your developer tools which you can open with Ctrl + Shift + I.

console
app.html
+layout.ts (load)
+page.ts (load)
+layout.svelte
+page.svelte
  • app.html script runs first
  • +layout.ts’s load function runs
  • +page.ts’s load function runs
  • +layout.svelte is mounted
  • +page.svelte is mounted

As you might expect +layout.ts should run before +page.ts and +layout.svelte should be mounted before +page.svelte.

If you’re curious this is an approximation of how a layout and components work in SvelteKit.

example.html
<Layout {data}>
  <!-- 🔥 if no error render child route -->
  <Page {data} />

  <!-- 💩 if error render nearest `+error.svelte` component -->
  <Error>

  <!-- 🪆 nested layouts -->
  <NestedLayout>
    <!-- ... -->
  <NestedLayout />
</Layout>

SvelteKit Does Server-Side Rendering (SSR) And Client-Side Rendering (CSR)

By default when you open a page for the first time meaning you enter it in the URL and press enter (or refresh a page) SvelteKit is going to server-side render your page meaning it has everything it needs to return the HTML.

This is awesome but if you navigate to another page it would cause a refresh using SSR if you’re not using JavaScript for client-side navigation (this is how a multi-page app or MPA like PHP behaves).

To demonstrate this I’m going to type http://localhost:5173/ in the browser and press enter which should give us what we’ve seen before.

terminal
📣 NEW REQUEST IS BEING MADE FROM /
hooks.server.ts
+layout.server.ts (load)
+page.server.ts (load)
+layout.ts (load)
+page.ts (load)
+layout.svelte
+page.svelte
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE
console
app.html
+page.ts (load)
+layout.ts (load)
+layout.svelte
+page.svelte

When the SvelteKit app loads it’s going to initialize a client-side router that uses JavaScript and client-side rendering (CSR) but only does server-side rendering (SSR) when you visit the page for the first time or refresh it.

To demonstrate this I’m going to go to any other route and from there I’m going to navigate to home inside the browser which is going to use client-side navigation.

terminal
📣 NEW REQUEST IS BEING MADE FROM /
hooks.server.ts
+page.server.ts
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE
console
+page.ts
+page.svelte

SvelteKit is smart enough to only run the load functions it has to and +layout.svelte is already mounted, so on navigation the only thing that gets destroyed (unmounted) and created (mounted) is the +page.svelte component.

🐿️ SvelteKit tucks the data in the HTML page for reuse inside a <script> tag. If you look at the page source you can find the JSON for the page data inside data. This is your page endpoint that starts with __data.json in the network tab and you open it by going to http://localhost:5173/__data.json.

You get great SEO because crawlers are going to get SSR’d pages and you can progressiely enhance the user experience with JavaScript but it’s not like a pure SPA where your site doesn’t work without JavaScript (try opening Twitter with JavaScript disabled).

I want to emphasize how the point is not that your site should work without JavaScript but before JavaScript to be more resilient and use JavaScript to enhance the user experience.

How Data Is Passed Through Routes

Most of the time you’re going to have a +page|layout.svelte file and a +page|layout.ts, or +page|layout.server.ts file for a given route (using one load function).

That being said you could have multiple load functions if you have +page|layout.svelte, +page|layout.server.ts and +page|layout.ts in which case the last load function always wins.

The data flows top to bottom:

  • +layout.server.ts
  • +layout.ts
  • +layout.svelte
  • +page.svelte

This leaves a lot of questions open I want to explore such as “can you pass props down?” and if you can “what happens if you pass a prop with the same name?“.

If you open +page.svelte at the root I’m showing the value of export let data alongside the $page.data store on the page, so you can see any changes easily instead of squinting at the console.

🐿️ When you navigate to a page that returns data from a load function the $page.data store is going to be updated, so a parent can access data from a child route.

Let’s start with +layout.server.ts.

src/routes/+layout.server.ts
export async function load() {
	return {
		a: 1,
	}
}

This data is now available to +layout.svelte and child routes through export let data.

If you return the key with the same name from +layout.ts since it’s the last load function that runs it wins and overrides the value.

src/routes/+layout.ts
export async function load() {
  return {
    a: 2
  }
}

The SvelteKit docs refer to load functions that run on the server and browser as universal (+page|layout.ts) and server (+page|layout.server.ts) for ones that always run on the server.

This is important because server load functions return the value which you can get through the data argument from universal load functions.

src/routes/+layout.ts
export async function load({ data }) {
  // `data` is the return value of `+layout.server.ts`
  console.log(data) // { a: 1 }
  // return new data
  return {
    ...data,
    b: 2
  }
}

On the other hand server load functions don’t have a data prop but receive the data from the layout.

src/routes/+page.server.ts
export async function load() {
  return {
    c: 3
  }
}

The return from the load function is going to have the data from the layout and the new prop { a: 1, b: 2, c: 3 }.

You could change everything you received so far, or combine the received data with new data.

src/routes/+page.ts
export async function load({ data }) {
  // a) change the props
  return {
    a: 10,
    b: 20,
    c: 30,
  }

  // b) return new data
  return {
    ...data,
    d: 4
  }
}

The last load function that runs wins.

The page output should be the same for +page.server data and the $page.data store.

data
{
  "a": 1,
  "b": 2,
  "c": 3,
  "d": 4
}

Accessing Parent Layout Data

You’re already familiar how +page.svelte and +layout.svelte have access to all the data from parent load functions but you might want to get data inside a load function from a parent layout load function.

To get the data from a parent layout load function you can use await parent() inside load functions.

If you use await parent() in load functions for +page|layout.server.ts you’re going to get the data from a parent +page|layout.server.ts load function.

src/routes/nested/route/+page.server.ts
export async function load({ parent }) {
	// parent `+layout.server.ts` data
	const data = await parent()
	console.log(data) // { a: 1 }
}

If you use await parent() inside load functions for +page|layout.ts you’re going to get the data from a parent +page|layout.server.ts file.

src/routes/nested/route/+page.ts
export async function load({ parent }) {
	// parent `+layout.server.ts` data
	const data = await parent()
	console.log(data) // { a: 1, b: 2 }
}

$app/stores

SvelteKit has some useful modules which are just Svelte stores you can subscribe to and be notified when someting changes.

One of those modules is $app/stores which has a getStores, navigating, page, and updates exports.

If you navigate to /error you can see the entire output for $page.

$page
{
  "params": {},
  "route": {
    "id": "/error"
  },
  "status": 500,
  "url": "http://localhost:5173/error",
  "form": null,
  "data": {
    "a": 1,
    "b": 2
  }
}

We can use this to customize the error page.

src/routes/+error.svelte.
<script lang="ts">
	import { page } from '$app/stores'
</script>

<h1>Error</h1>

<p>
	Even though you're on the <b>{$page.url.pathname}</b> route this is not the
	<code>+page.svelte</code> component but the <code>+error.svelte</code> component.
</p>

App stores use Svelte’s context API on the server, so the page you visit is unique to that request to prevent shared state on the server because a server is a long running process which isn’t a problem in the browser because you’re not sharing state with anyone else.

Adding Custom Data To A Request

Adding custom data to a request is super useful because you can use hooks.server.ts on each request to make sure the user is authenticated and pass the user data to handlers in +server.ts and server load functions by using the event.locals object.

Here is how Lucia, an authenticaion library for SvelteKit uses it.

src/hooks.ts
import { auth } from '$lib/server/lucia'

export async function handle({ event, resolve }) {
	event.locals.auth = auth.handleRequest(event)
	return await resolve(event)
}

Now you can use validation methods from auth inside load functions and +server.ts handlers through locals.auth.

🚨 Careful that you never send sensitive information using event.locals and only send the information for the user.

Let’s say I have a secret route and I only want to show it to the user if the secret is banana, otherwise I’m going to redirect them.

src/hooks.ts
export async function handle({ event, resolve }) {
	event.locals.secret = '🍌'
	return await resolve(event)
}

If you’re using TypeScript update the types in app.d.ts.

src/app.d.ts
declare global {
	namespace App {
		interface Locals {
      secret: string
    }
	}
}

Inside +page.server.ts we can check locals and show data on the page or redirect the user.

src/routes/locals/+page.server.ts
import { redirect } from '@sveltejs/kit'

export async function load({ locals }) {
	if (locals.secret !== '🍌') {
		throw redirect(307, '/')
	}

	return {
		secret: locals.secret,
	}
}

If you navigate to /locals you should see the output on the page.

Standalone Endpoints

So far we didn’t talk about standalone endpoints but for a good reason because there’s nothing special about them and you can use them if you need the same data in multiple routes, or you’re making a REST API you can use inside, or outside your app.

I made a api/secret/+server.ts endpoint and I’m curious in what order things run if I navigate to it

termina
📣 NEW REQUEST IS BEING MADE FROM /api/secret
hooks.server.ts
+server.ts (GET)
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE

As you can see a standalone API endpoint always runs on the server because you need a server to make a HTTP request. If you look at the network tab it returns a text document but you can return anything you want.

If you want to learn more I wrote about page versus standalone endpoints in SvelteKit.

Form Actions

SvelteKit form actions are the easiest way to send data from the browser to the server compared to using standalone endpoints because SvelteKit handles everything for you.

By default forms work without JavaScript but you can use the enhance Svelte form action to progressively enhance them.

To use form actions you export an actions object from +page.server.ts.

src/routes/actions/+page.server.svelte
export const actions = {
	async login() {
    // ...
	},
	async register() {
    // ...
	},
}

You can use a form prop to handle errors but I’m going to keep it simple.

src/routes/actions/+page.svelte
<script lang="ts">
	import { enhance } from '$app/forms'

	export let data
</script>

<h1>Form Actions</h1>

<form method="POST" action="?/login" use:enhance>
	<label for="username">
		<p>Username</p>
		<input type="text" name="username" id="username" />
	</label>

	<label for="password">
		<p>Password</p>
		<input type="password" name="password" id="password" />
	</label>

	<div class="actions">
		<button type="submit">Log In</button>
		<button type="submit" formaction="?/register">Register</button>
	</div>
</form>

First we navigate to the /actions page where you can see the usual suspects.

terminal
📣 NEW REQUEST IS BEING MADE FROM /actions
hooks.server.ts
/forms/+page.server.ts (load)
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE
console
/forms/+page.svelte

Let’s see what happens when you submit the form.

terminal
📣 NEW REQUEST IS BEING MADE FROM /actions
hooks.server.ts
LOGIN ACTION: http://localhost:5173/actions?/login
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE
📣 NEW REQUEST IS BEING MADE FROM /actions
hooks.server.ts
/forms/+page.server.ts (load)
+layout.server.ts (load)
🔥 THE PAGE IS READY, I'M SENDING THE RESPONSE
console
+layout.ts (load)
  • a new request is made to /actions (you could log the type of request but you can only use a POST request when using form actions)
  • as you can see a form action is just a URL that invokes an action
  • you can replace “PAGE” with “RESOURCE” in this case
  • SvelteKit is going to invalidate the data which fires another request
  • the load functions for /forms/+page.server.ts and src/routes/+layout.server.ts are going to rerun to update the data
  • you don’t see +page.svelte because it’s already mounted

That’s it! 😄

By the way don’t sweat the details because this is a lot to take in at once. The important thing to take away from this is that you have a general idea how data flows in SvelteKit to help you build your next idea.

SvelteKit Data Flow Cheat Sheet

I made an awesome SvelteKit data flow cheat sheet to go over what we learned which should be helpful.

Sveltekit data flow

You can get the SVG version of the SvelteKit data flow cheat sheet as a light or dark version:

I hope this helped you understand how data flows in SvelteKit and answered any questions you might have.

There’s always more than one way of doing things, so don’t concern yourself with the “right way” of doing things and go make some happy accidents.