Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/mylofi/client-storage
Simple key-value storage API backed by various client storage mechanisms
https://github.com/mylofi/client-storage
storage
Last synced: 4 months ago
JSON representation
Simple key-value storage API backed by various client storage mechanisms
- Host: GitHub
- URL: https://github.com/mylofi/client-storage
- Owner: mylofi
- License: mit
- Created: 2024-08-26T01:28:13.000Z (4 months ago)
- Default Branch: main
- Last Pushed: 2024-08-27T21:41:59.000Z (4 months ago)
- Last Synced: 2024-08-28T02:57:31.892Z (4 months ago)
- Topics: storage
- Language: JavaScript
- Homepage: https://mylofi.github.io/client-storage/
- Size: 23.4 KB
- Stars: 27
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE.txt
Awesome Lists containing this project
README
# Client Storage
[![npm Module](https://badge.fury.io/js/@lo-fi%2Fclient-storage.svg)](https://www.npmjs.org/package/@lo-fi/client-storage)
[![License](https://img.shields.io/badge/license-MIT-a1356a)](LICENSE.txt)**Client Storage** provides a set of adapters for easy client-side key-value storage.
```js
// for IndexedDB:
import { get, set } from "@lo-fi/client-storage/idb";await set("Hello","World!"); // true
await get("Hello"); // "World!"
```----
[Library Tests (Demo)](https://mylofi.github.io/client-storage/)
----
## Overview
The main purpose of **Client Storage** is to provide a set of adapters that normalize across various client side storage mechanisms (`localStorage` / `sessionStorage`, IndexedDB, cookies, and OPFS) with a consistent key-value API (`get()`, `set()`, etc).
## Client Side Storage Adapters
**Client Storage** ships with adapters for the following storage mechanisms:
* `idb`: [IndexedDB](https://developer.mozilla.org/en-US/docs/Web/API/IndexedDB_API)
* `local-storage`: [Web Storage `localStorage`](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage)
* `session-storage`: [Web Storage `sessionStorage`](https://developer.mozilla.org/en-US/docs/Web/API/Window/sessionStorage)
* `cookie`: [Web cookies](https://developer.mozilla.org/en-US/docs/Mozilla/Add-ons/WebExtensions/API/cookies)
* `OPFS`: [Origin Private File System](https://developer.mozilla.org/en-US/docs/Web/API/File_System_API/Origin_private_file_system), specifically [virtual origin filesystem](https://developer.mozilla.org/en-US/docs/Web/API/StorageManager/getDirectory)
Each of these client-side storage mechanisms has its own pros/cons, so choice should be made carefully.
However, IndexedDB (`idb` adapter) is the most robust and flexible option, and should generally be considered the best default.
### Storage Limitations
These client storage mechanisms have different storage limits, which in some cases may be rather small (i.e., 5MB for Local-Storage, or 4KB for cookies). Be careful with `set()` calls: look for the [`QuotaExceededError` DOM exception](https://developer.mozilla.org/en-US/docs/Web/API/DOMException#quotaexceedederror) being thrown, and determine what data can be freed up, or potentially switch to another storage mechanism with higher limits.
For example:
```js
try {
await set("session-jwt",sessionJWT);
}
catch (err) {
if (err.reason?.name == "QuotaExceededError") {
// handle storage limit failure!
}
}
```#### Web Storage (`localStorage`, `sessionStorage`)
The web storage mechanisms (`localStorage`, `sessionStorage`) are by far the most common place web applications storage client-side data. However, there are some factors to consider when using the `local-storage` / `session-storage` adapters.
Each mechanism is size-limited to 5MB, on most all browsers/devices. And they are only available from main browser threads, not in workers (Web Workers, Service Workers).
#### Cookies
The `cookie` adapter stores vault data in browser cookies. There are however some strong caveats to consider before choosing this storage mechanism.
Cookies are limited to ~4KB. Moreover, the provided data object has been JSON-serialized, encrypted and then base64 string encoded, then *that value* has been put into another object that's JSON-serialized, and that string (the actual cookie value) is URI-encoded (e.g, replacing `" "` with `%20`, etc). Taking into account all these steps that inflate your data size further towards the 4KB limit, you might only be able to squeeze ~2-3KB of original application data in, under the limit.
Also, cookies are typically sent on *every request* to a first-party origin server (images, CSS, fetch calls, etc). So that data (encrypted, of course) will be sent remotely, and will significantly weigh down all those requests.
Moreover, cookies are never "persistent" storage, and are subject to both expirations (maximum allowed is ~400 days out from the last update) and to users clearing them.
All these concerns considered, the `cookie` adapter *really should not be used* except as a last resort, for small amounts of data. For example, your app might use this storage as a temporary location if normal storage quota has been reached, and later synchronize/migrate/backup off-device, etc.
#### Origin Private File System
The [Origin Private File System](https://developer.mozilla.org/en-US/docs/Web/API/File_System_API/Origin_private_file_system) (OPFS) web feature can be used to read/write "files" in a virtual filesystem on the client's device (private to the page's origin). The `opfs` adapter provided with this library creates JSON "files" in this OPFS to store the vault data, one per vault.
Be aware: [the ability to asynchronously `write()`](https://developer.mozilla.org/en-US/docs/Web/API/FileSystemWritableFileStream/write#browser_compatibility) on the main browser thread, into OPFS, is currently only supported on desktop (not mobile!) Chromium and Firefox browsers.
However, there is [widespread browser/device support for synchronous `write()`](https://developer.mozilla.org/en-US/docs/Web/API/FileSystemFileHandle/createSyncAccessHandle#browser_compatibility) into OPFS, if done off-thread in a background worker (Web Worker, Service Worker). The `opfs` adapter does *not currently* support this approach, but it may in the future.
## Deployment / Import
```cmd
npm install @lo-fi/client-storage
```The [**@lo-fi/client-storage** npm package](https://npmjs.com/package/@lo-fi/client-storage) includes a `dist/` directory with all files you need to deploy **Client Storage** (and its dependencies) into your application/project.
**Note:** If you obtain this library via git instead of npm, you'll need to [build `dist/` manually](#re-building-dist) before deployment.
### Using a bundler
If you are using a bundler (Astro, Vite, Webpack, etc) for your web application, you should not need to manually copy any files from `dist/`.
Just `import` the adapter(s) of your choice, like so:
```js
// {WHICHEVER}: "idb", "local-storage", etc
import { get, set } from "@lo-fi/client-storage/{WHICHEVER}";
```The bundler tool should pick up and find whatever files (and dependencies) are needed.
### Without using a bundler
If you are not using a bundler (Astro, Vite, Webpack, etc) for your web application, and just deploying the contents of `dist/` as-is without changes (e.g., to `/path/to/js-assets/client-storage/`), you'll need an [Import Map](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/script/type/importmap) in your app's HTML:
```html
{
"imports": {
"client-storage/idb": "/path/to/js-assets/client-storage/adapter.idb.mjs",
"client-storage/local-storage": "/path/to/js-assets/client-storage/adapter.local-storage.mjs",
"client-storage/session-storage": "/path/to/js-assets/client-storage/adapter.session-storage.mjs",
"client-storage/cookie": "/path/to/js-assets/client-storage/adapter.cookie.mjs",
"client-storage/opfs": "/path/to/js-assets/client-storage/adapter.opfs.mjs","idb-keyval": "/path/to/js-assets/client-storage/external/idb-keyval.js"
}
}```
Now, you'll be able to `import` the library in your app in a friendly/readable way:
```js
// {WHICHEVER}: "idb", "local-storage", etc
import { get, set } from "client-storage/{WHICHEVER}";
```**Note:** If you omit the above *adapter* import-map entries, you can still `import` **Client Storage** by specifying the proper full path to whichever `adapter.*.mjs` file(s) you want to use.
However, the entry above for `idb-keyval` is more required. Alternatively, you'll have to edit the `adapter.idb.mjs` file to change its `import` specifier for `idb-keyval` to the proper path to `idb-keyval.js`.
## Client Storage API
The API provided by the **Client Storage** adapters can be accessed, for each adapter, like this:
```js
// for IndexedDB:
import { has, get, set, remove } from "@lo-fi/client-storage/idb";await has("Hello"); // false
await set("Hello","World!"); // true
await has("Hello"); // true
await get("Hello"); // "World!"
await remove("Hello"); // true
```The key-value oriented methods available on each adapter's API are:
* `has(name)`: has a value of `name` been set in this storage before?
* `get(name)`: get a value of `name` (if any) from storage
* `set(name,value)`: set a `value` at `name` into storage
`value` can be any JSON-serializable object (object, array) or any primitive value; however, bare primitive values will end up being stored (and then retrieved) as strings.
Further, any string value that is parseable as JSON *will be parsed* as JSON; for example, the string value `"[1,2,3]"` will be parsed as a JSON-serialized array, and return `[1,2,3]` instead.
* `remove(name)`: remove `name` (if any) from storage
* `keys()`: returns an array of existing keys in storage
* `entries()`: returns an array of `[ key, value ]` tuples
**NOTE:** All of these methods are async (promise-returning).
## Re-building `dist/*`
If you need to rebuild the `dist/*` files for any reason, run:
```cmd
# only needed one time
npm installnpm run build:all
```## Tests
This library only works in a browser, so its automated test suite must also be run in a browser.
Visit [`https://mylofi.github.io/client-storage/`](https://mylofi.github.io/client-storage/) and click the "run tests" button.
### Run Locally
To instead run the tests locally, first make sure you've [already run the build](#re-building-dist), then:
```cmd
npm test
```This will start a static file webserver (no server logic), serving the interactive test page from `http://localhost:8080/`; visit this page in your browser and click the "run tests" button.
By default, the `test/test.js` file imports the code from the `src/*` directly. However, to test against the `dist/*` files (as included in the npm package), you can modify `test/test.js`, updating the `/src` in its `import` statements to `/dist` (see the import-map in `test/index.html` for more details).
## License
[![License](https://img.shields.io/badge/license-MIT-a1356a)](LICENSE.txt)
All code and documentation are (c) 2024 Kyle Simpson and released under the [MIT License](http://getify.mit-license.org/). A copy of the MIT License [is also included](LICENSE.txt).