🔝

Polystore npm install polystore test badge gzip size

A key-value library to unify the API of many clients, like localStorage, Redis, FileSystem, etc:

import kv from "polystore";
const store1 = kv(new Map()); // in-memory
const store2 = kv(localStorage); // Persist in the browser
const store3 = kv(redisClient); // Use a Redis client for backend persistence
const store4 = kv(yourOwnStore); // Create a store based on your code
// Many more here

These are all the methods of the API (they are all async):

  • .get(key): read a single value, or null if it doesn't exist or is expired.
  • .set(key, value, options?): save a single value that is serializable.
  • .add(value, options?): same as .set(), but auto-generates the key.
  • .has(key): check whether a key exists or not.
  • .del(key): delete a single value from the store.
  • .keys(): get a list of all the available strings in the store.
  • .values(): get a list of all the values in the store.
  • .entries(): get a list of all the key-value pairs.
  • .all(): get an object with the key:values mapped.
  • .clear(): delete ALL of the data in the store, effectively resetting it.
  • .close(): (only some stores) ends the connection to the store.
  • .prefix(prefix): create a sub-store that manages the keys with that prefix.

This library has very high performance with the item methods (GET/SET/ADD/HAS/DEL). For other methods or to learn more, see the performance considerations and read the docs on your specific client.

Available clients for the KV store:

I made this library to be used as a "building block" of other libraries, so that your library can accept many cache stores effortlessly! It's isomorphic (Node.js, Bun and the Browser) and tiny (~3KB). For example, let's say you create an API library, then you can accept the stores from your client:

import MyApi from "my-api";

MyApi({ cache: new Map() }); // OR
MyApi({ cache: localStorage }); // OR
MyApi({ cache: redisClient }); // OR
MyApi({ cache: env.KV_NAMESPACE }); // OR
// ...

Getting started

First, install polystore and whatever supported client that you prefer. Let's see Redis as an example here:

npm i polystore redis

Then import both, initialize the Redis client and pass it to Polystore:

import kv from "polystore";
import { createClient } from "redis";

// Import the Redis configuration
const REDIS = process.env.REDIS_URL;

// Wrap the redis creation with Polystore (kv())
const store = kv(createClient(REDIS).connect());

Now your store is ready to use! Add, set, get, del different keys. See full API.

const key = await store.add("Hello");

console.log(await store.get(key));
// Hello

await store.del(key);

API

See how to initialize each store in the Clients list documentation. But basically for every store, it's like this:

import kv from "polystore";

// Initialize it; NO "new"; NO "await", just a plain function wrap:
const store = kv(MyClientOrStoreInstance);

// use the store

While you can keep a reference to the store and access it directly, we strongly recommend if you are going to use a store, to only access it through polystore, since we might add custom serialization and extra properties for e.g. expiration time:

const map = new Map();
const store = kv(map);

// Works as expected
await store.set("a", "b");
console.log(await store.get("a"));

// DON'T DO THIS; this will break the app since we apply more
// advanced serialization to the values stored in memory
map.set("a", "b");
console.log(await store.get("a")); // THROWS ERROR

.get()

Retrieve a single value from the store. Will return null if the value is not set in the store, or if it was set but has already expired:

const value = await store.get(key: string);

console.log(await store.get("key1"));  // "Hello World"
console.log(await store.get("key2"));  // ["my", "grocery", "list"]
console.log(await store.get("key3"));  // { name: "Francisco" }

If the value is returned, it can be a simple type like boolean, string or number, or it can be a plain Object or Array, or any combination of those.

When there's no value (either never set, or expired), null will be returned from the operation.

.set()

Create or update a value in the store. Will return a promise that resolves with the key when the value has been saved. The value needs to be serializable:

await store.set(key: string, value: any, options?: { expires: number|string });

await store.set("key1", "Hello World");
await store.set("key2", ["my", "grocery", "list"], { expires: "1h" });
await store.set("key3", { name: "Francisco" }, { expires: 60 * 60 });

The value can be a simple type like boolean, string or number, or it can be a plain Object or Array, or a combination of those. It cannot be a more complex or non-serializable values like a Date(), Infinity, undefined (casted to null), a Symbol, etc.

  • By default the keys don't expire.
  • Setting the value to null, or the expires to 0 is the equivalent of deleting the key+value.
  • Conversely, setting expires to null or undefined will make the value never to expire.

Expires

When the expires option is set, it can be a number (seconds) or a string representing some time:

// Valid "expire" values:
0 - expire immediately (AKA delete it)
0.1 - expire after 100ms*
60 * 60 - expire after 1h
3_600 - expire after 1h
"10s" - expire after 10 seconds
"2minutes" - expire after 2 minutes
"5d" - expire after 5 days

* not all stores support sub-second expirations, notably Redis and Cookies don't, so it's safer to always use an integer or an amount larger than 1s. There will be a note in each store for this.

These are all the units available:

"ms", "millisecond", "s", "sec", "second", "m", "min", "minute", "h", "hr", "hour", "d", "day", "w", "wk", "week", "b" (month), "month", "y", "yr", "year"

.add()

Create a value in the store with a random key string. Will return a promise that resolves with the key when the value has been saved. The value needs to be serializable:

const key:string = await store.add(value: any, options?: { expires: number|string });

const key1 = await store.add("Hello World");
const key2 = await store.add(["my", "grocery", "list"], { expires: "1h" });
const key3 = await store.add({ name: "Francisco" }, { expires: 60 * 60  });

The options and details are similar to .set(), except for the lack of the first argument, since .add() will generate the key automatically.

The default key will be 24 AlphaNumeric characters (upper+lower case), however this can change if you are using a .prefix() or some clients might generate it differently (only custom clients can do that right now).

Key Generation details The default key will be 24 AlphaNumeric characters (including upper and lower case) generated with random cryptography to make sure it's unguessable, high entropy and safe to use in most contexts like URLs, queries, etc. We use [`nanoid`](https://github.com/ai/nanoid/) with a custom dictionary, so you can check the entropy [in this dictionary](https://zelark.github.io/nano-id-cc/) by removing the "\_" and "", and setting it to 24 characters.

Here is the safety: "If you generate 1 million keys/second, it will take ~14 million years in order to have a 1% probability of at least one collision."

The main reason why .add() exists is to allow it to work with the prefix seamlessly:

const session = store.prefix("session:");

// Creates a key with the prefix (returns only the key)
const key1 = await session.add("value1");
// "c4ONlvweshXPUEy76q3eFHPL"

console.log(await session.keys()); // on the "session" store
// ["c4ONlvweshXPUEy76q3eFHPL"]
console.log(await store.keys()); // on the root store
// ["session:c4ONlvweshXPUEy76q3eFHPL"]

Remember that substores with .prefix() behave as if they were an independent store, so when adding, manipulating, etc. a value you should treat the key as if it had no prefix. This is explained in detail in the .prefix() documentation.

.has()

Check whether the key:value is available in the store and not expired:

await store.has(key: string);

if (await store.has("cookie-consent")) {
  loadCookies();
}

In many cases, internally the check for .has() is the same as .get(), so if you are going to use the value straight away it's usually better to just read it:

const val = await store.get("key1");
if (val) { ... }

An example of an exception of the above is when you use it as a cache, then you can write code like this:

// First time for a given user does a network roundtrip, while
// the second time for the same user gets it from cache
async function fetchUser(id) {
  if (!(await store.has(id))) {
    const { data } = await axios.get(`/users/${id}`);
    await store.set(id, data, { expires: "1h" });
  }
  return store.get(id);
}

An example with a prefix:

const session = store.prefix("session:");

// These three perform the same operation internally
const has1 = await session.has("key1");
const has2 = await store.prefix("session:").has("key1");
const has3 = await store.has("session:key1");

.del()

Remove a single key from the store and return the key itself:

await store.del(key: string);

It will ignore the operation if the key or value don't exist already (but won't thorw). The API makes it easy to delete multiple keys at once:

const keys = ["key1", "key2"];
await Promise.all(keys.map(store.del));
console.log(done);

An example with a prefix:

const session = store.prefix("session:");

// These three perform the same operation internally
await session.del("key1");
await store.prefix("session:").del("key1");
await store.del("session:key1");

Iterator

You can iterate over the whole store with an async iterator:

for await (const [key, value] of store) {
  console.log(key, value);
}

This is very useful for performance resons since it will retrieve the data sequentially, avoiding blocking the client while retrieving it all at once. The main disadvantage is if you keep writing data while the async iterator is running.

You can also iterate on a subset of the entries with .prefix() (the prefix is stripped from the key here, see .prefix()):

const session = store.prefix("session:");
for await (const [key, value] of session) {
  console.log(key, value);
}

// Same as this (both have the prefix stripped):

for await (const [key, value] of store.prefix("session:")) {
  console.log(key, value);
}

There are also methods to retrieve all of the keys, values, or entries at once below, but those have worse performance.

.keys()

Get all of the keys in the store, optionally filtered by a prefix:

await store.keys(filter?: string);

We ensure that all of the keys returned by this method are not expired, while discarding any potentially expired key. See expiration explained for more details.

.values()

Get all of the values in the store, optionally filtered by a key prefix:

await store.values(filter?: string);

This is useful specially when you already have the id/key within the value as an object, then you can just get a list of all of them:

const sessions = await store.values("session:");
// A list of all the sessions

const companies = await store.values("company:");
// A list of all the companies

We ensure that all of the values returned by this method are not expired, while discarding any potentially expired key. See expiration explained for more details.

.entries()

Get all of the entries (key:value tuples) in the store, optionally filtered by a key prefix:

await store.entries(filter?: string);

It is in a format that you can easily build an object out of it:

const sessionEntries = await store.entries("session:");
const sessions = Object.fromEntries(sessionEntries);

We ensure that all of the entries returned by this method are not expired, while discarding any potentially expired key. See expiration explained for more details.

.clear()

Remove all of the data from the store and resets it to the original state:

await store.clear();

.prefix()

There's an in-depth explanation about Substores that is very informative for production usage.

Creates a new instance of the Store, with the same client as you provided, but now any key you read, write, etc. will be passed with the given prefix to the client. You only write .prefix() once and then don't need to worry about any prefix for any method anymore, it's all automatic. It's the only method that you don't need to await:

const store = kv(new Map());
const session = store.prefix("session:");

Then all of the operations will be converted internally to add the prefix when reading, writing, etc:

const session = store.prefix("session:");
const val = await session.get("key1"); // store.get('session:key1');
await session.set("key2", "some data"); // store.set('session:key2', ...);
const val = await session.has("key3"); // store.has('session:key3');
await session.del("key4"); // store.del('session:key4');
await session.keys(); // store.keys(); + filter
// ['key1', 'key2', ...]   Note no prefix here
await session.clear(); // delete only keys with the prefix

Different clients have better/worse support for substores, and in some cases some operations might be slower. This should be documented on each client's documentation (see below). As an alternative, you can always create two different stores instead of a substore:

// Two in-memory stores
const store = kv(new Map());
const session = kv(new Map());

// Two file-stores
const users = kv(new URL(`file://${import.meta.dirname}/users.json`));
const books = kv(new URL(`file://${import.meta.dirname}/books.json`));

The main reason this is not stable is because some store engines don't allow for atomic deletion of keys given a prefix. While we do still clear them internally in those cases, that is a non-atomic operation and it could have some trouble if some other thread is reading/writing the data at the same time.

Clients

A client is the library that manages the low-level store operations. For example, the Redis Client, or the browser's localStorage API. In some exceptions it's just a string and we do a bit more work on Polystore, like with "cookie" or "file:///users/me/data.json".

Polystore provides a unified API you can use Promises, expires and .prefix() even with those stores that do not support these operations natively.

Memory

An in-memory KV store, with promises and expiration time:

import kv from "polystore";

const store = kv(new Map());

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"
Why use polystore with new Map()?

Besides the other benefits already documented elsewhere, these are the ones specifically for wrapping Map() with polystore:

  • Expiration: you can now set lifetime to your values so that they are automatically evicted when the time passes. Expiration explained.
  • Substores: you can also create substores and manage partial data with ease. Details about substores.

Local Storage

The traditional localStorage that we all know and love, this time with a unified API, and promises:

import kv from "polystore";

const store = kv(localStorage);

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

Same limitations as always apply to localStorage, if you think you are going to use too much storage try instead our integration with Local Forage!

Session Storage

Same as localStorage, but now for the session only:

import kv from "polystore";

const store = kv(sessionStorage);

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

Cookies

Supports native browser cookies, including setting the expire time:

import kv from "polystore";

const store = kv("cookie"); // just a plain string

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

It is fairly limited for how powerful cookies are, but in exchange it has the same API as any other method or KV store. It works with browser-side Cookies (no http-only).

Note: the cookie expire resolution is in the seconds, so times shorter than 1 second like expires: 0.02 (20 ms) don't make sense for this storage method and won't properly save them.

Local Forage

Supports localForage (with any driver it uses) so that you have a unified API. It also adds the expires option to the setters!

import kv from "polystore";
import localForage from "localforage";

const store = kv(localForage);

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

Redis Client

Supports the official Node Redis Client. You can pass either the client or the promise:

import kv from "polystore";
import { createClient } from "redis";

const store = kv(createClient().connect());

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

You don't need to await for the connect or similar, this will process it properly.

Note: the Redis client expire resolution is in the seconds, so times shorter than 1 second like expires: 0.02 (20 ms) don't make sense for this storage method and won't properly save them.

File

Treat a JSON file in your filesystem as the source for the KV store:

import kv from "polystore";

const store = kv(new URL("file:///Users/me/project/cache.json"));

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

You can also create multiple stores:

// Paths need to be absolute, but you can use process.cwd() to make
// it relative to the current process:
const store1 = kv(new URL(`file://${process.cwd()}/cache.json`));
const store2 = kv(new URL(`file://${import.meta.dirname}/data.json`));

Cloudflare KV

Supports the official Cloudflare's KV stores. Follow the official guide, then load it like this:

import kv from "polystore";

export default {
  async fetch(request, env, ctx) {
    const store = kv(env.YOUR_KV_NAMESPACE);

    await store.set("key1", "Hello world", { expires: "1h" });
    console.log(await store.get("key1"));
    // "Hello world"

    return new Response("My response");
  },
};

Why use polystore? The Cloudflare native KV store only accepts strings and has you manually calculating timeouts, but as usual with polystore you can set/get any serializable value and set the timeout in a familiar format:

// GOOD - with polystore
await store.set("user", { name: "Francisco" }, { expires: "2days" });

// COMPLEX - With native Cloudflare KV
const serialValue = JSON.stringify({ name: "Francisco" });
const twoDaysInSeconds = 2 * 24 * 3600;
await env.YOUR_KV_NAMESPACE.put("user", serialValue, {
  expirationTtl: twoDaysInSeconds,
});

Level

Support the Level ecosystem, which is itself composed of modular methods:

import kv from "polystore";
import { Level } from "level";

const store = kv(new Level("example", { valueEncoding: "json" }));

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

Etcd

Connect to Microsoft's Etcd Key-Value store:

import kv from "polystore";
import { Etcd3 } from "etcd3";

const store = kv(new Etcd3());

await store.set("key1", "Hello world", { expires: "1h" });
console.log(await store.get("key1"));
// "Hello world"

Custom store

Please see the creating a store section for more details!

Performance

TL;DR: if you only use the item operations (add,set,get,has,del) and your client supports expiration natively, you have nothing to worry about!

While all of our stores support expires, .prefix() and group operations, the nature of those makes them to have different performance characteristics.

Expires we polyfill expiration when the underlying client library does not support it. The impact on read/write operations and on data size of each key should be minimal. However, it can have a big impact in storage size, since the expired keys are not evicted automatically. Note that when attempting to read an expired key, polystore will delete that key. However, if an expired key is never read, it would remain in the datastore and could create some old-data issues. This is especially important where sensitive data is involved! To fix this, the easiest way is calling await store.entries(); on a cron job and that should evict all of the old keys (this operation is O(n) though, so not suitable for calling it on EVERY API call, see the next point).

Group operations these are there mostly for small datasets only, for one-off scripts or for dev purposes, since by their own nature they can never be high performance in the general case. But this is normal if you think about traditional DBs, reading a single record by its ID is O(1), while reading all of the IDs in the DB into an array is going to be O(n). Same applies with polystore.

Substores when dealing with a .prefix() substore, the same applies. Item operations should see no performance degradation from .prefix(), but group operations follow the above performance considerations. Some engines might have native prefix support, so performance in those is better for group operations in a substore than the whole store. But in general you should consider .prefix() as a convenient way of classifying your keys and not as a performance fix for group operations.

Expires

Warning: if a client doesn't support expiration natively, we will hide expired keys on the API calls for a nice DX, but old data might not be evicted automatically. See the notes in Performance for details on how to work around this.

We unify all of the clients diverse expiration methods into a single, easy one with expires:

// in-memory store
const store = polystore(new Map());
await store.set("a", "b", { expires: "1s" });

// These checks of course work:
console.log(await store.keys()); // ['a']
console.log(await store.has("a")); // true
console.log(await store.get("a")); // 'b'

// Make sure the key is expired
await delay(2000); // 2s

// The group methods also ignore expired keys
console.log(await store.keys()); // []
console.log(await store.has("a")); // false
console.log(await store.get("a")); // null

This is great because with polystore we do ensure that if a key has expired, it doesn't show up in .keys(), .entries(), .values(), .has() or .get().

However, in some stores this does come with some potential performance disadvantages. For example, both the in-memory example above and localStorage don't have a native expiration/eviction process, so we have to store that information as metadata, meaning that even to check if a key exists we need to read and decode its value. For one or few keys it's not a problem, but for large sets this can become an issue.

For other stores like Redis this is not a problem, because the low-level operations already do them natively, so we don't need to worry about this for performance at the user-level. Instead, Redis and cookies have the problem that they only have expiration resolution at the second level. Meaning that 800ms is not a valid Redis expiration time, it has to be 1s, 2s, etc.

Substores

There's some basic .prefix() API info for everyday usage, this section is the in-depth explanation.

What .prefix() does is it creates a new instance of the Store, with the same client as you provided, but now any key you read, write, etc. will be passed with the given prefix to the client. The issue is that support from the underlying clients is inconsistent.

When dealing with large or complex amounts of data in a KV store, some times it's useful to divide them by categories. Some examples might be:

  • You use KV as a cache, and have different categories of data.
  • You use KV as a session store, and want to differentiate different kinds of sessions.
  • You use KV as a primary data store, and have different types of datasets.

For these and more situations, you can use .prefix() to simplify your life further.

Creating a store

To create a store, you define a class with these properties and methods:

class MyClient {
  // If this is set to `true`, the CLIENT (you) handle the expiration, so
  // the `.set()` and `.add()` receive a `expires` that is a `null` or `number`:
  EXPIRES = false;

  // Mandatory methods
  get (key): Promise<any>;
  set (key, value, { expires: null|number }): Promise<null>;
  iterate(prefix): AyncIterator<[string, any]>

  // Optional item methods (for optimization or customization)
  add (prefix, data, { expires: null|number }): Promise<string>;
  has (key): Promise<boolean>;
  del (key): Promise<null>;

  // Optional group methods
  entries (prefix): Promise<[string, any][]>;
  keys (prefix): Promise<string[]>;
  values (prefix): Promise<any[]>;
  clear (prefix): Promise<null>;

  // Optional misc method
  close (): Promise<null>;
}

Note that this is NOT the public API, it's the internal client API. It's simpler than the public API since we do some of the heavy lifting as an intermediate layer (e.g. for the client, the expires will always be a null or number, never undefined or a string), but also it differs from polystore's public API, like .add() has a different signature, and the group methods all take a explicit prefix.

Expires: if you set the EXPIRES = true, then you are indicating that the client WILL manage the lifecycle of the data. This includes all methods, for example if an item is expired, then its key should not be returned in .keys(), it's value should not be returned in .values(), and the method .has() will return false. The good news is that you will always receive the option expires, which is either null (no expiration) or a number indicating the time when it will expire.

Prefix: we manage the prefix as an invisible layer on top, you only need to be aware of it in the .add() method, as well as in the group methods:

// What the user of polystore does:
const store = await kv(client).prefix("hello:").prefix("world:");

// User calls this, then the client is called with that:
const value = await store.get("a");
// client.get("hello:world:a");

// User calls this, then the client is called with that:
for await (const entry of store.iterate()) {
}
// client.iterate("hello:world:");

Note: all of the group methods that return keys, should return them with the prefix:

client.keys = (prefix) => {
  // Filter the keys, and return them INCLUDING the prefix!
  return Object.keys(subStore).filter((key) => key.startsWith(prefix));
};

While the signatures are different, you can check each entries on the output of Polystore API to see what is expected for the methods of the client to do, e.g. .clear() will remove all of the items that match the prefix (or everything if there's no prefix).

Example: Plain Object client

This is a good example of how simple a store can be, however do not use it literally since it behaves the same as the already-supported new Map(), only use it as the base for your own clients:

const dataSource = {};

class MyClient {
  get(key) {
    return dataSource[key];
  }

  // No need to stringify it or anything for a plain object storage
  set(key, value) {
    dataSource[key] = value;
  }

  // Filter them by the prefix, note that `prefix` will always be a string
  *iterate(prefix) {
    for (const [key, value] of Object.entries(dataSource)) {
      if (key.startsWith(prefix)) {
        yield [key, value];
      }
    }
  }
}

We don't set EXPIRES to true since plain objects do NOT support expiration natively. So by not adding the EXPIRES property, it's the same as setting it to false, and polystore will manage all the expirations as a layer on top of the data. We could be more explicit and set it to EXPIRES = false, but it's not needed in this case.

Example: custom ID generation

You might want to provide your custom key generation algorithm, which I'm going to call customId() for example purposes. The only place where polystore generates IDs is in add, so you can provide your client with a custom generator:

class MyClient {

  // Add the opt method .add() to have more control over the ID generation
  async add (prefix, data, { expires }) {
    const id = customId();
    const key = prefix + id;
    return this.set(key, data, { expires });
  }

  //
  async set (...) {
    // ...
  }
}

That way, when using the store, you can simply use .add() to generate it:

import kv from "polystore";

const store = kv(MyClient);
const id = await store.add({ hello: "world" });
// this is your own custom id
const id2 = await store.prefix("hello:").add({ hello: "world" });
// this is `hello:{your own custom id}`