Use stores in SvelteKit during SSR as you are used to from Svelte* without modifying server state.
* with minimal boilerplate code
- 🔥 The Issue
- 💡 The Solution
- 💻 Installation
- 🚀 Quickstart
- ⚙️ How does it Work (Implementation Details)
- 📖 Documentation
- 👤 Author
- 📄 License
Svelte is great. But even though it is very fast to render a page in the browser with Svelte, it has some drawbacks compared to server side rendered HTML:
- Displaying pre-rendered HTML is faster
- Indexing / SEO is not going to work without executing JS
- Static pages need JS to be displayed even though they would not have to
SvelteKit solves these (and more) using Server Side Rendering (SSR). But it comes with it's own caveats:
If you create a store in a module
(i.e. exported from a .ts
or .js
file) or defined globally inside a <script context="module">
tag, this store will only be created once on the server, when
the server loads the module for the first time
If your component uses such a store during loading or during component initialization while rendering on the server, then your component depends on the state of the store on the server side.
If yor component then also writes to the store during SSR, it alters server state! This means, that for all upcoming requests, the value of the store will be changed. In the best case, this results in "flickering", when reloading the page, where the SSR version of the page has old data which is shortly displayed until the hydration replaces it with the updated value.
🔥 In the worst case, it leaks private information of one user to other users! 🔥
SvelteKit has a concept called the session. It is a serializable JavaScript object which is used to pass data from the server to the client. The session object is created on the server per request.
The Isolated Stores of this package use these session objects to tell appart different requests, and re-creates the stores for each new session.
As a positive side effect, it enables
custom store functions to use
SvelteKit's fetch
method,
which serializes the responses of requests made during SSR and sends them along
the rendered page so that the client does not need to do the same request again
during hydration.
npm install -D svelte-kit-isolated-stores
Because this package relies on SvelteKit's generated code, you have to prevent
Vite from building it in advance. To do that, add the following to your
svelte.config.js
:
// svelte.config.js
/** @type {import('@sveltejs/kit').Config} */
const config = {
// ...
kit: {
// ...
vite: {
optimizeDeps: {
exclude: ["svelte-kit-isolated-stores"], // <-- Add this line
},
ssr: {
noExternal: ["svelte-kit-isolated-stores"], // <-- Add this line
},
},
},
}
export default config
Follow the installation guide. Then add the following to your
top level __layout.svelte
and any __layout.reset.svelte
files:
<!-- `__layout.svelte` and any `__layout.reset.svelte` -->
<script lang="ts" context="module">
import { loadWithStores } from 'svelte-kit-isolated-stores'
export const load = loadWithStores()
</script>
<slot />
To learn how to use a custom load
function and how to use stores within it,
have a look at loadWithStores()
.
Define stores using defineWritable()
,
defineReadable()
and defineDerived()
.
They work almost exactly like their svelte counterparts but the initial value is
provided through a function.
Custom stores can be defined with
defineStore()
:
// src/lib/stores.ts
import { defineStore, defineWritable, defineReadable, defineDerived } from 'svelte-kit-isolated-stores'
import { writable } from 'svelte/store'
export const count = defineWritable(() => 0)
export const double = defineDerived(count, $count => $count * 2)
export const time = defineReadable(
// The initial value creator
() => new Date(),
// The start function called when the first subscriber subscribes
(set) => {
set(new Date())
const interval = setInterval(() => {
set(new Date())
}, 1000)
// Return the stop function called when the last subscriber unsubscribes
return () => {
clearInterval(interval)
}
},
)
// Get SvelteKit's `fetch` by destructuring the function argument
// `-------------------------vvvvv
export const user = defineStore(({ fetch }) => {
const { subscribe, set, update } = writable()
async function loadUser(uid: string) {
// Use `fetch` -----------vvvvv
const data = await (await fetch(`/api/user/${uid}`)).json()
set(data)
}
return {
subscribe,
set,
update,
loadUser,
}
})
On your pages, you can use the stores in the load
function using the
loadWithStores()
function. In the template, use the store as you are used to
from Svelte.
<!-- src/routes/users/[userUid].svelte -->
<script lang="ts" context="module">
import { loadWithStores } from 'svelte-kit-isolated-stores'
import { user } from '$lib/stores'
export const load = loadWithStores(
{ user },
async ({ user }, { params }) => {
await user.loadUser(params['userUid'])
return {}
}
)
</script>
<script lang="ts">
import { count, double as useDouble, time } from './_stores'
// Get the real store instance.
// (Not necessary for this example. See documentation to see where it
// is useful.)
const double = useDouble()
function increment() {
count.update((n) => n + 1)
}
function decrement() {
count.update((n) => n - 1)
}
$: square = $count * $count
$count = 10
</script>
<div>
The current date and time is {$time.toLocaleString()}
</div>
<div>
<button on:click={decrement}>➖</button>
{$count}
<button on:click={increment}>➕</button>
</div>
<div>
<div>Doubled: {$double}</div>
<div>Squared: {square}</div>
</div>
<div>
Loaded User:
<pre><code>{JSON.stringify($user)}</code></pre>
</div>
To try the above example, you may also want to create the used API endpoint at
src/routes/api/user/[userUid].ts
and paste the following:
import type { RequestHandler } from '@sveltejs/kit'
export const get: RequestHandler = ({ params }) => {
const userUid = params['userUid']
return {
body: {
uid: userUid,
name: 'M1000',
firstName: 'Moritz',
lastName: 'Zimmermann',
},
}
}
Every store defined with defineStore
or one of the helper functions, is
wrapped in a
JS Proxy
and when a property of the store (eg. the subscribe
or set
function) is
accessed, the Proxy
looks up the store for the current session and returns the
property of that store.
If the store does not exist for the current session yet, it is created and saved
to a
WeakMap
so the store can be retrieved if it is used again during the same request.
The WeakMap
is used to map from the session object to a Map
of stores.
Doing so makes sure, that the garbage collector can clean away stores of
sessions that do not exist anymore, after a request is done.
On the client side, it works the same way but of course there is always only one
session object. It would be possible to just return the store instead of the
Proxy
on the client, although this would prevent the aforementioned ability to
use SvelteKit's fetch
function in custom store functions. This is because with
the Proxy
, the store is created lazily when needed and not when the store
module is loaded, which means we can pass in fetch
from the load
function.
This package provides a defineStore()
method that takes a function, that
creates a custom store.
In Svelte, a store is defined as an object that has at least a subscribe()
method which, when executed, returns an method to unsubscribe.
It can then also have other properties or methods. Common methods are set()
and update()
which are used to set and update the value of the store.
You can also include custom methods. In the example below, the methods
increment()
, decrement()
and reset()
are also added to the store object to
make it more easy to use the counter
.
// counter.ts
import { defineStore } from 'svelte-kit-isolated-stores'
import { writable } from 'svelte/store'
export const counter = defineStore(() => {
const { subscribe, set, update } = writable(0)
function increment() {
update(val => val + 1)
}
function decrement() {
update(val => val - 1)
}
return {
subscribe, // You need to return at least `subscribe`
set,
update,
increment,
decrement,
reset: () => set(0),
}
})
You do however not need to create custom stores for every simple store you have.
defineWritable()
is a helper function, that creates a writable store with an
initial value.
The difference to Svelte's writable()
function is, that you need to provide a
function (arrow function in this example) to create the initial value. This is
necessary because the store must be re-created over and over again and if the
initial value is a reference type like an object or an array, different
instances of the same store would share this data.
// count.ts
import { defineWritable } from 'svelte-kit-isolated-stores'
// --- Initial value creator -------vvvvvvv
export const count = defineWritable(() => 0)
To define simple readable stores use the helper function defineReadable()
.
Like with defineWritable()
the initial value must be returned from a function.
The second argument is the start
function. This should be familiar from
Svelte's readable()
stores.
// time.ts
import { defineReadable } from 'svelte-kit-isolated-stores'
export const time = defineReadable(
// The initial value creator
() => new Date(),
// The start function called when the first subscriber subscribes
(set) => {
set(new Date())
const interval = setInterval(() => {
set(new Date())
}, 1000)
// Return the stop function called when the last subscriber unsubscribes
return () => {
clearInterval(interval)
}
},
)
Derived stores change their value depending on other stores. Use the helper
function defineDerived()
to create an isolated derived store.
Like with Svelte's
derived()
method, it
is possible to pass a single store or an array of one or more dependent stores
to defineDerived()
.
When deriving from a single store, the dependent store can be provided without being encapsulated in an array. The first argument of the callback function is then just the store value.
// double.ts
import { defineWritable, defineDerived } from 'svelte-kit-isolated-stores'
export const someValue = defineWritable(() => 0)
export const double = defineDerived(someValue, $someValue => $someValue * 2)
Multiple stores can be derived from by passing them as an array. The first
argument of the callback function is then an array of store values instead of
just a single store value. The array can be destructured to arbitrary names
(however it is convention to prefix the store names with $
for the values).
// rectangle.ts
import { defineWritable, defineDerived } from 'svelte-kit-isolated-stores'
export const width = defineWritable(() => 0)
export const height = defineWritable(() => 0)
export const area = defineDerived(
[width, height],
([$width, $height]) => $width * $height
)
Contrary to Svelte's derive()
it is also possible to pass an object containing
dependent stores to defineDerived()
. The first argument of the callback
function is then an object of store values where each key of the stores object
is prefixed with a $
.
This feature is added to increase DX (developer experience), because by using objects (together with TypeScript), the intellisense can help with destructuring. It can can also help to reduce bugs, as swapping the positions of the stores in the input object does not silently change the order of the store values (as they would in the array example above).
However, keep in mind that this adds a layer on top of the original
derived()
implementation and, for very frequently changing stores, this might impact performance.
// rectangle.ts
import { defineWritable, defineDerived } from 'svelte-kit-isolated-stores'
export const width = defineWritable(() => 0)
export const height = defineWritable(() => 0)
export const area = defineDerived(
// Dependent stores are passed in as object (using object property value
// shorthand)
{ width, height },
// The object can directly be destructured. Keep in mind, that every key has
// been prefixed with `$`.
({ $width, $height }) => $width * $height,
)
export const diagonal = defineDerived(
{ width, height },
// Of course, you can assign different names using destructuring syntax
({ $width: w, $height: h }) => Math.sqrt(Math.pow(w, 2) + Math.pow(h, 2)),
)
Closures is the concept of bundling togehter a function with it's surrounding state or context. It let's you access variables that are defined in the outside scope of the function.
When using stores on the server side, using closures might have some side effects:
For example the Svelte tutorial for
derived stores uses a global
variable start
which contains a Date
. This start
variable is initialized
when the module (the file) is first being loaded. In the browser this behaves as
expected: start
contains the Date
when the page has loaded.
On the server however, this would not be re-evaluated on every request and therefore would contain the date of the first request (or more precise the date the module has first been loaded on the server).
🔥 In the worst case, this can leak private information of one user to other users! 🔥
To achieve the same behaviour as in the Svelte tutorial, you can use
defineStore()
and return a derived store instead of a custom store object:
// time.ts
import { defineStore, defineReadable } from 'svelte-kit-isolated-stores'
import { derived } from 'svelte/store'
export const time = defineReadable(
() => new Date(),
(set) => {
const interval = setInterval(() => {
set(new Date())
}, 1000)
return () => { clearInterval(interval) }
},
)
export const elapsed = defineStore(() => {
const start = new Date()
// Instead of returning an object with a `subscribe()` method, you can just
// return a plain Svelte store. It is safe to use sveltes `derived()` here,
// as it will be re-created when the store defined by `defineStore()` is
// recreated.
return derived(
time,
($time) => Math.round(($time.getTime() - start.getTime()) / 1000),
)
})
You can use the store as if it was a normal Svelte store during component initialization:
<script language="ts">
import { counter } from '$lib/stores/counter'
// Use auto-subscription syntax
$counter = 10
</script>
{$counter}
During component initializion the store isolation mechanism has access to
$app/stores
and can extract the session objectfrom there. This session is crucial to be able to know which stores belong to the current request.
Just don't think about it!
<script language="ts">
import { counter } from '$lib/stores/counter'
</script>
<h1>The count is: {$counter}</h1>
<div>
<button on:click={counter.decrement}>➖</button>
<button on:click={counter.increment}>➕</button>
</div>
<div>
<label>
Count:
<input type="number" bind:value={$counter} />
</label>
</div>
Layouts and pages in SvelteKit can export a load
function, which is executed
before the component is initialized (a layout / page is also just a Svelte
component).
When not in component initialization, the isolation mechanism does not have
access to the current session object through $app/stores
. Therefore, we must
pass the session
object of the input argument of the load function to the
isolation logic.
We could do that by calling the store as if it was a function and we
pass in the input
argument of the load
function:
<script lang="ts" context="module">
import { counter } from '$lib/stores/counter'
import type { Load } from '@sveltejs/kit'
export const load: Load = (input) => {
// This is now the real store instance
const _counter = counter(input)
// For some reason we want the counter value to be `1337` on this page ¯\_(ツ)_/¯
_counter.set(1337)
// Don't forget to return an object, otherwise you'll get a 404
return {}
}
</script>
{$counter}
This might look a bit strange at first and is also a little boilerplaty. But don't worry, you will not need to do this too often thanks to another convenience function:
This function is used to wrap the actual load
function. It can be used in a
variety of ways. But all of them include exporting the result as load
in the
module
context of a layout of page component.
-
You can just call it without arguments. This makes sure that the stores get access to SvelteKit's
fetch
function (more on that later).It also does some "magic" to make things easier for code that runs exclusively in the browser (and by "magic" I mean, it stores the session in a global variable, because in the browser there is no server state that can be modified 😉).
It is generally a good idea to do this in the top level
__layout.svelte
and any__layout.reset.svelte
files.<script lang="ts" context="module"> import { loadWithStores } from 'svelte-kit-isolated-stores' export const load = loadWithStores() </script> <slot />
Calling without arguments will export a load function that effectively returns
{}
to prevent a404
error. -
You can also pass in a custom load function.
<script lang="ts" context="module"> import { loadWithStores } from 'svelte-kit-isolated-stores' export const load = loadWithStores(({ params, fetch}) => { const userId = params['id'] // Do some load logic here, maybe return `props` or `stuff` or whatever. // Don't forget to return an object, otherwise you'll get a 404 return {} }) </script>
-
You can pass in an object where each value is an isolated store as the first argument and as the second argument, a custom
load
function. This custom load function is provided with two arguments. The first is an object of actual (not isolated) stores, the second is theLoadInput
object that a normalload
function receive through the first argument.<script lang="ts" context="module"> import { loadWithStores } from 'svelte-kit-isolated-stores' import { counter } from '$lib/stores/counter' export const load = loadWithStores({ counter }, ({ counter }, { params }) => { // Inside this function, the `counter` variable contains the real store // instance, not the isolated store const userId = Number(params['userId']) // For some reason we want the counter value to be the user id on this // page ¯\_(ツ)_/¯ counter.set(userId) // Don't forget to return an object, otherwise you'll get a 404 return {} }) </script> {$counter}
The isolation mechanism needs to know what the current session is. Outside
component initialization and outside load()
there is no reliable way to access
SvelteKit's session
object in general.
Consider the following code:
<script lang="ts">
import { counter } from '$lib/stores/counter'
import { onMount } from 'svelte'
import { get } from 'svelte/store'
let counterVal
function incrementBy(n) {
// ERROR: Store used outside component initialization
counter.update((c) => {
counterVal = c + n
return counterVal
})
}
</script>
<button on:click="{() => incrementBy(1)}">+1</button>
<button on:click="{() => incrementBy(10)}">+10</button>
The incrementBy()
method is executed after component initialization. So
when it accesses the update
property of counter
, the isolation wrapper has
no idea what the current session is. Normally this would fail. But there is a
fix:
While in the browser, there is only ever one session, so it is okay to work
with global variables here. Therefore, whenever the isolation mechanism first
gets access to the session
object, it stores a global reference to it, so it
can be accessed later.
If you use any isolated store during component initialization (e.g. you set it to some value or you subscribe to it) then the above example would work. But you do not want to rely on someon else having used a store before you.
Therefore, to ensure the isolation mechanism always has access to the
session
(at least in the browser), make sure to use the loadWithStores()
function (with or without arguments) in the top level __layout.svelte
and any
__layout.reset.svelte
files:
<!-- `__layout.svelte` and any `__layout.reset.svelte` -->
<script lang="ts" context="module">
import { loadWithStores } from 'svelte-kit-isolated-stores'
export const load = loadWithStores()
</script>
Luckily most code that runs outside component initialization and outside the
load
function is only ever run in the browser.
For example any event handlers do usually not run on the server and you should
avoid async code in your component initialization (outside the load
function)
during SSR anyway, as it will not effect the rendered page HTML.
Also Svelte's onMount()
function is only run in the browser.
But there is at least one situation where code runs on the server outside component initialization:
The onDestroy()
hook is triggert during SSR (or rather when SSR is done). And
if you try to use an isolated store there, it will fail to access the session
object and throw an exception.
There may be other situation where this can happen, I just didn't encounter others yet.
To mitigate this, you can either make sure the store is only accessed when in the browser:
<script lang="ts">
import { onDestroy } from 'svelte'
import { browser } from '$app/env'
import { counter } from '$lib/stores/counter'
onDestroy(() => {
if (browser) {
counter.reset()
}
})
</script>
Or you can get yourself an instance of the real store object during component initialization and use it later:
<script lang="ts">
import { onDestroy } from 'svelte'
// Import the counter with an alias name (can be anything, but I like to
// prefix the name with `use` as a convention)
import { counter as useCounter } from '$lib/stores/counter'
// Get an instance of the real store object by calling the isolated store
const counter = useCounter()
// From here on, the store can be used *exactly* like in plain Svelte
onDestroy(() => {
counter.reset()
})
</script>
A positive side effect of the store isolation is, that you can use SvelteKit's
fetch
inside your defined stores.
SvelteKit's fetch
wrapper saves the results of requests that are executed
during SSR. To speed things up, SvelteKit then serializes the results and sends
them along with the generated page to the browser. During
hydration, the result is
re-used so the browser does not have to fetch the same data again. After
hydration, it works just like the normal fetch
function.
Another advantage of using SvelteKit's fetch
is, that you can use relative
URLs on the server side (i.e. you can use just the path to the API endpoint,
without the https://yourdomain.tld
part).
When defining a custom store, you can access this fatch
wrapper through the
function arguments:
import { defineStore } from 'svelte-kit-isolated-stores'
import { writable } from 'svelte/store'
// Get SvelteKit's `fetch` by destructuring the function argument
// `-------------------------vvvvv
export const user = defineStore(({ fetch }) => {
const { subscribe, set, update } = writable()
async function loadUser(uid: string) {
// Use `fetch` -----------vvvvv
const data = await (await fetch(`/api/user/${uid}`)).json()
set(data)
}
return {
subscribe,
set,
update,
// Export the `loadUser` function
loadUser,
}
})
Copyright © 2019 Björn Richter <git@x3ro.dev>
This project is
MIT
licensed.