Getting Started

Learn how to setup and use unstorage in your project.

Introduction

We usually choose one or more storage backends based on our use cases, such as the filesystem, a database, or LocalStorage for browsers. It soon starts to create troubles when supporting and combining multiple options or switching between them. For JavaScript library authors, this usually means that they have to decide how many platforms they are going to support and implement storage for each of them.

Installation

Install unstorage npm package:

npm i unstorage

Usage

my-storage.js
import { createStorage } from "unstorage";

const storage = createStorage(/* opts */);

await storage.getItem("foo:bar"); // or storage.getItem('/foo/bar')

Options:

  • driver: Default driver, using memory if not provided

Interface

hasItem(key, opts?)

Checks if storage contains a key. Resolves to either true or false.

getItem(key, opts?)

Gets the value of a key in storage. Resolves to either a JavaScript primitive value or undefined.

getItems(items, opts)

(Experimental) Gets the value of multiple keys in storage in parallel.

Each item in the array can either be a string or an object with { key, options? } format.

Returned value is a Promise resolving to an array of objects with { key, value } format.

getItemRaw(key, opts?)

Note: This is an experimental feature. Please check unjs/unstorage#142 for more information.

Gets the value of a key in storage in raw format.

setItem(key, value, opts?)

Add/Update a value to the storage.

If the value is not a string, it will be stringified.

If the value is undefined, it is same as calling removeItem(key).

setItems(items, opts)

(Experimental) Add/Update items in parallel to the storage.

Each item in items array should be in { key, value, options? } format.

Returned value is a Promise resolving to an array of objects with { key, value } format.

setItemRaw(key, value, opts?)

Note: This is an experimental feature. Please check unjs/unstorage#142 for more information.

Add/Update a value to the storage in raw format.

If value is undefined, it is the same as calling removeItem(key).

removeItem(key, opts = { removeMeta = false })

Remove a value (and it's meta) from storage.

getMeta(key, opts = { nativeOnly? })

Get metadata object for a specific key.

This data is fetched from two sources:

setMeta(key, opts?)

Set custom meta for a specific key by adding a $ suffix.

removeMeta(key, opts?)

Remove meta for a specific key by adding a $ suffix.

getKeys(base?, opts?)

Get all keys. Returns an array of strings.

Meta keys (ending with $) will be filtered.

If a base is provided, only keys starting with the base will be returned and only mounts starting with base will be queried. Keys still have a full path.

clear(base?, opts?)

Removes all stored key/values. If a base is provided, only mounts matching base will be cleared.

dispose()

Disposes all mounted storages to ensure there are no open-handles left. Call it before exiting process.

Note: Dispose also clears in-memory data.

mount(mountpoint, driver)

By default, everything is stored in memory. We can mount additional storage space in a Unix-like fashion.

When operating with a key that starts with mountpoint, instead of default storage, mounted driver will be called.

In addition to base, you can set readOnly and noClear to disable write and clear operations.

unmount(mountpoint, dispose = true)

Unregisters a mountpoint. Has no effect if mountpoint is not found or is root.

watch(callback)

Starts watching on all mountpoints. If driver does not support watching, only emits even when storage.* methods are called.

unwatch()

Stop all watchers on all mountpoints.

getMount(key)

Gets the mount point (driver and base) for a specific key in storage.

getMounts(base?, { parents: boolean }?)

Gets the mount points on a specific base.

await storage.hasItem("foo:bar");

You can also use the has alias:

await storage.has("foo:bar");
await storage.getItem("foo:bar");

You can also use the get alias:

await storage.get("foo:bar");
// Value can be a Buffer, Array or Driver's raw format
const value = await storage.getItemRaw("foo:bar.bin");
await storage.setItem("foo:bar", "baz");

You can also use the set alias:

await storage.set("foo:bar", "baz");
await storage.setItemRaw("data/test.bin", new Uint8Array([1, 2, 3]));
await storage.removeItem("foo:bar", { removeMeta: true });
// same as await storage.removeItem("foo:bar", true);

You can also use the del or remove aliases:

await storage.remove("foo:bar");
await storage.del("foo:bar");
  • Driver native meta (like file creation time)
  • Custom meta set by storage.setMeta (overrides driver native meta)
await storage.getMeta("foo:bar"); // For fs driver returns an object like { mtime, atime, size }
await storage.setMeta("foo:bar", { flag: 1 });
// Same as storage.setItem('foo:bar$', { flag: 1 })
await storage.removeMeta("foo:bar");
// Same as storage.removeItem('foo:bar$')
await storage.getKeys();

You can also use the keys alias:

await storage.keys();
await storage.clear();
await storage.dispose();
import { createStorage } from "unstorage";
import fsDriver from "unstorage/drivers/fs";

// Create a storage container with default memory storage
const storage = createStorage({});

storage.mount("/output", fsDriver({ base: "./output" }));

//  Writes to ./output/test file
await storage.setItem("/output/test", "works");

// Adds value to in-memory storage
await storage.setItem("/foo", "bar");
await storage.unmount("/output");
const unwatch = await storage.watch((event, key) => {});
// to stop this watcher
await unwatch();
await storage.unwatch();
storage.mount("cache" /* ... */);
storage.mount("cache:routes" /* ... */);

storage.getMount("cache:routes:foo:bar");
// => { base: "cache:routes:", driver: "..." }
storage.mount("cache" /* ... */);
storage.mount("cache:sub" /* ... */);

storage.getMounts("cache:sub");
// => [{ base: "cache:sub", driver }]

storage.getMounts("cache:");
// => [{ base: "cache:sub", driver }, { base: "cache:", driver }]

storage.getMounts("");
storage.getMounts("cache:sub", { parents: true });
// => [{ base: "cache:sub", driver }, { base: "cache:", driver }, { base: "", driver }]

Generic types

Type getItem return value:

await storage.getItem<string>("k"); // => <string>

await storage.getItemRaw<Buffer>("k"); // => <Buffer>

Type check setItem parameters:

storage.setItem<string>("k", "val"); // check ok
storage.setItemRaw<string>("k", "val"); // check ok

storage.setItem<string>("k", 123); // ts error
storage.setItemRaw<string>("k", 123); // ts error

Typed storage instance:

const storage = createStorage<string>();

await storage.getItem("k"); // => <string>

storage.setItem("k", "val"); // Check ok
storage.setItem("k", 123); // TS error
Forward references use inheritance instead of overriding types.
const storage = createStorage<string>();

storage.setItem<number>("k", 123); // TS error: <number> is not compatible with <string>

Typing a sub set using prefixStorage:

const storage = createStorage();

const htmlStorage = prefixStorage<string>(storage, "assets:html");

await htmlStorage.getItem("foo.html"); // => <string>

type Post = {
  title: string;
  content: string;
};

const postStorage = prefixStorage<Post>(storage, "assets:posts");

await postStorage.getItem("foo.json"); // => <Post>

In strict mode, it will also return the undefined type to help you handle the case when getItem is missing.

"use strict";

await storage.getItem<string>("k"); // => <string | null>