WXT Storage
A simplified wrapper around the extension storage APIs.
Installation
With WXT
This module is built-in to WXT, so you don't need to install anything.
import { storage } from 'wxt/storage';
If you use auto-imports, storage
is auto-imported for you, so you don't even need to import it!
Without WXT
Install the NPM package:
npm i @wxt-dev/storage
pnpm add @wxt-dev/storage
yarn add @wxt-dev/storage
bun add @wxt-dev/storage
import { storage } from '@wxt-dev/storage';
Storage Permission
To use the wxt/storage
API, the "storage"
permission must be added to the manifest:
// wxt.config.ts
export default defineConfig({
manifest: {
permissions: ['storage'],
},
});
Basic Usage
All storage keys must be prefixed by their storage area.
// ❌ This will throw an error
await storage.getItem('installDate');
// ✅ This is good
await storage.getItem('local:installDate');
You can use local:
, session:
, sync:
, or managed:
.
If you use TypeScript, you can add a type parameter to most methods to specify the expected type of the key's value:
await storage.getItem<number>('local:installDate');
await storage.watch<number>(
'local:installDate',
(newInstallDate, oldInstallDate) => {
// ...
},
);
await storage.getMeta<{ v: number }>('local:installDate');
For a full list of methods available, see the API reference.
Watchers
To listen for storage changes, use the storage.watch
function. It lets you set up a listener for a single key:
const unwatch = storage.watch<number>('local:counter', (newCount, oldCount) => {
console.log('Count changed:', { newCount, oldCount });
});
To remove the listener, call the returned unwatch
function:
const unwatch = storage.watch(...);
// Some time later...
unwatch();
Metadata
wxt/storage
also supports setting metadata for keys, stored at key + "$"
. Metadata is a collection of properties associated with a key. It might be a version number, last modified date, etc.
Other than versioning, you are responsible for managing a field's metadata:
await Promise.all([
storage.setItem('local:preference', true),
storage.setMeta('local:preference', { lastModified: Date.now() }),
]);
When setting different properties of metadata from multiple calls, the properties are combined instead of overwritten:
await storage.setMeta('local:preference', { lastModified: Date.now() });
await storage.setMeta('local:preference', { v: 2 });
await storage.getMeta('local:preference'); // { v: 2, lastModified: 1703690746007 }
You can remove all metadata associated with a key, or just specific properties:
// Remove all properties
await storage.removeMeta('local:preference');
// Remove only the "lastModified" property
await storage.removeMeta('local:preference', 'lastModified');
// Remove multiple properties
await storage.removeMeta('local:preference', ['lastModified', 'v']);
Defining Storage Items
Writing the key and type parameter for the same key over and over again can be annoying. As an alternative, you can use storage.defineItem
to create a "storage item".
Storage items contain the same APIs as the storage
variable, but you can configure its type, default value, and more in a single place:
// utils/storage.ts
const showChangelogOnUpdate = storage.defineItem<boolean>(
'local:showChangelogOnUpdate',
{
fallback: true,
},
);
Now, instead of using the storage
variable, you can use the helper functions on the storage item you created:
await showChangelogOnUpdate.getValue();
await showChangelogOnUpdate.setValue(false);
await showChangelogOnUpdate.removeValue();
const unwatch = showChangelogOnUpdate.watch((newValue) => {
// ...
});
For a full list of properties and methods available, see the API reference.
Versioning
You can add versioning to storage items if you expect them to grow or change over time. When defining the first version of an item, start with version 1.
For example, consider a storage item that stores a list of websites that are ignored by an extension.
type IgnoredWebsiteV1 = string;
export const ignoredWebsites = storage.defineItem<IgnoredWebsiteV1[]>(
'local:ignoredWebsites',
{
fallback: [],
version: 1,
},
);
import { nanoid } from 'nanoid';
type IgnoredWebsiteV1 = string;
interface IgnoredWebsiteV2 {
id: string;
website: string;
}
export const ignoredWebsites = storage.defineItem<IgnoredWebsiteV1[]>(
export const ignoredWebsites = storage.defineItem<IgnoredWebsiteV2[]>(
'local:ignoredWebsites',
{
fallback: [],
version: 1,
version: 2,
migrations: {
// Ran when migrating from v1 to v2
2: (websites: IgnoredWebsiteV1[]): IgnoredWebsiteV2[] => {
return websites.map((website) => ({ id: nanoid(), website }));
},
},
},
);
import { nanoid } from 'nanoid';
type IgnoredWebsiteV1 = string;
interface IgnoredWebsiteV2 {
id: string;
website: string;
}
interface IgnoredWebsiteV3 {
id: string;
website: string;
enabled: boolean;
}
export const ignoredWebsites = storage.defineItem<IgnoredWebsiteV2[]>(
export const ignoredWebsites = storage.defineItem<IgnoredWebsiteV3[]>(
'local:ignoredWebsites',
{
fallback: [],
version: 2,
version: 3,
migrations: {
// Ran when migrating from v1 to v2
2: (websites: IgnoredWebsiteV1[]): IgnoredWebsiteV2[] => {
return websites.map((website) => ({ id: nanoid(), website }));
},
// Ran when migrating from v2 to v3
3: (websites: IgnoredWebsiteV2[]): IgnoredWebsiteV3[] => {
return websites.map((website) => ({ ...website, enabled: true }));
},
},
},
);
INFO
Internally, this uses a metadata property called v
to track the value's current version.
In this case, we thought that the ignored website list might change in the future, and were able to set up a versioned storage item from the start.
Realistically, you won't know an item needs versioning until you need to change its schema. Thankfully, it's simple to add versioning to an unversioned storage item.
When a previous version isn't found, WXT assumes the version was 1
. That means you just need to set version: 2
and add a migration for 2
, and it will just work!
Let's look at the same ignored websites example from before, but start with an unversioned item this time:
export const ignoredWebsites = storage.defineItem<string[]>(
'local:ignoredWebsites',
{
fallback: [],
},
);
import { nanoid } from 'nanoid';
// Retroactively add a type for the first version
type IgnoredWebsiteV1 = string;
interface IgnoredWebsiteV2 {
id: string;
website: string;
}
export const ignoredWebsites = storage.defineItem<string[]>(
export const ignoredWebsites = storage.defineItem<IgnoredWebsiteV2[]>(
'local:ignoredWebsites',
{
fallback: [],
version: 2,
migrations: {
// Ran when migrating from v1 to v2
2: (websites: IgnoredWebsiteV1[]): IgnoredWebsiteV2[] => {
return websites.map((website) => ({ id: nanoid(), website }));
},
},
},
);
Running Migrations
As soon as storage.defineItem
is called, WXT checks if migrations need to be run, and if so, runs them. Calls to get or update the storage item's value or metadata (getValue
, setValue
, removeValue
, getMeta
, etc.) will automatically wait for the migration process to finish before actually reading or writing values.
Default Values
With storage.defineItem
, there are multiple ways of defining default values:
fallback
- Return this value fromgetValue
instead ofnull
if the value is missing.This option is great for providing default values for settings:
tsconst theme = storage.defineItem('local:theme', { fallback: 'dark', }); const allowEditing = storage.defineItem('local:allow-editing', { fallback: true, });
init
- Initialize and save a value in storage if it is not already saved.This is great for values that need to be initialized or set once:
tsconst userId = storage.defineItem('local:user-id', { init: () => globalThis.crypto.randomUUID(), }); const installDate = storage.defineItem('local:install-date', { init: () => new Date().getTime(), });
The value is initialized in storage immediately.
Bulk Operations
When getting or setting multiple values in storage, you can perform bulk operations to improve performance by reducing the number of individual storage calls. The storage
API provides several methods for performing bulk operations:
getItems
- Get multiple values at once.getMetas
- Get metadata for multiple items at once.setItems
- Set multiple values at once.setMetas
- Set metadata for multiple items at once.removeItems
- Remove multiple values (and optionally metadata) at once.
All these APIs support both string keys and defined storage items:
const userId = storage.defineItem('local:userId');
await storage.setItems([
{ key: 'local:installDate', value: Date.now() },
{ item: userId, value: generateUserId() },
]);
Refer to the API Reference for types and examples of how to use all the bulk APIs.