Comparing version 1.1.5 to 1.2.0
@@ -1,3 +0,3 @@ | ||
import { D as Driver, S as Storage, a as StorageValue } from './types-4336cb3d.js'; | ||
export { c as StorageMeta, U as Unwatch, b as WatchCallback, W as WatchEvent } from './types-4336cb3d.js'; | ||
import { D as Driver, S as Storage, a as StorageValue } from './types-bacc64fa.js'; | ||
export { c as StorageMeta, U as Unwatch, b as WatchCallback, W as WatchEvent } from './types-bacc64fa.js'; | ||
@@ -21,2 +21,5 @@ interface CreateStorageOptions { | ||
declare const builtinDrivers: { | ||
azureStorageTable: string; | ||
azureCosmos: string; | ||
azureStorageBlob: string; | ||
cloudflareKVHTTP: string; | ||
@@ -33,5 +36,7 @@ cloudflareKVBinding: string; | ||
memory: string; | ||
mongodb: string; | ||
overlay: string; | ||
planetscale: string; | ||
redis: string; | ||
azureKeyVault: string; | ||
}; | ||
@@ -38,0 +43,0 @@ type BuiltinDriverName = keyof typeof builtinDrivers; |
import { RequestListener } from 'node:http'; | ||
import { S as Storage } from './types-4336cb3d.js'; | ||
import { EventHandler } from 'h3'; | ||
import { S as Storage } from './types-bacc64fa.js'; | ||
interface StorageServerOptions { | ||
} | ||
interface StorageServer { | ||
declare function createH3StorageHandler(storage: Storage, _options?: StorageServerOptions): EventHandler; | ||
declare function createStorageServer(storage: Storage, options?: StorageServerOptions): { | ||
handle: RequestListener; | ||
} | ||
declare function createStorageServer(storage: Storage, _options?: StorageServerOptions): StorageServer; | ||
}; | ||
export { StorageServer, StorageServerOptions, createStorageServer }; | ||
export { StorageServerOptions, createH3StorageHandler, createStorageServer }; |
@@ -1,7 +0,25 @@ | ||
import { RedisOptions as _RedisOptions } from "ioredis"; | ||
import { ClusterNode, ClusterOptions, RedisOptions as _RedisOptions } from "ioredis"; | ||
export interface RedisOptions extends _RedisOptions { | ||
base: string; | ||
url: string; | ||
/** | ||
* Optional prefix to use for all keys. Can be used for namespacing. | ||
*/ | ||
base?: string; | ||
/** | ||
* Url to use for connecting to redis. Takes precedence over `host` option. Has the format `redis://<REDIS_USER>:<REDIS_PASSWORD>@<REDIS_HOST>:<REDIS_PORT>` | ||
*/ | ||
url?: string; | ||
/** | ||
* List of redis nodes to use for cluster mode. Takes precedence over `url` and `host` options. | ||
*/ | ||
cluster?: ClusterNode[]; | ||
/** | ||
* Options to use for cluster mode. | ||
*/ | ||
clusterOptions?: ClusterOptions; | ||
/** | ||
* Default TTL for all items in seconds. | ||
*/ | ||
ttl?: number; | ||
} | ||
declare const _default: (opts?: RedisOptions | undefined) => import("../types").Driver; | ||
export default _default; |
{ | ||
"name": "unstorage", | ||
"version": "1.1.5", | ||
"version": "1.2.0", | ||
"description": "Universal Storage Layer", | ||
@@ -38,24 +38,26 @@ "repository": "unjs/unstorage", | ||
"ioredis": "^5.3.1", | ||
"listhen": "^1.0.2", | ||
"lru-cache": "^7.16.0", | ||
"mkdir": "^0.0.2", | ||
"listhen": "^1.0.3", | ||
"lru-cache": "^7.17.0", | ||
"mri": "^1.2.0", | ||
"node-fetch-native": "^1.0.2", | ||
"ofetch": "^1.0.1", | ||
"ufo": "^1.1.0", | ||
"ws": "^8.12.1" | ||
"ufo": "^1.1.0" | ||
}, | ||
"optionalDependencies": { | ||
"@planetscale/database": "^1.5.0" | ||
}, | ||
"devDependencies": { | ||
"@azure/app-configuration": "^1.3.1", | ||
"@azure/cosmos": "^3.17.2", | ||
"@azure/data-tables": "^13.2.1", | ||
"@azure/identity": "^3.1.3", | ||
"@azure/keyvault-secrets": "^4.6.0", | ||
"@azure/storage-blob": "^12.12.0", | ||
"@cloudflare/workers-types": "^4.20230214.0", | ||
"@planetscale/database": "^1.5.0", | ||
"@types/ioredis-mock": "^8", | ||
"@types/jsdom": "^21.1.0", | ||
"@types/mri": "^1.1.1", | ||
"@types/node": "^18.13.0", | ||
"@types/ws": "^8.5.4", | ||
"@types/node": "^18.14.1", | ||
"@vitejs/plugin-vue": "^4.0.0", | ||
"@vitest/coverage-c8": "^0.28.5", | ||
"@vitest/coverage-c8": "^0.29.1", | ||
"@vue/compiler-sfc": "^3.2.47", | ||
"azurite": "^3.21.0", | ||
"c8": "^7.13.0", | ||
@@ -65,15 +67,27 @@ "changelogen": "^0.4.1", | ||
"eslint-config-unjs": "^0.1.0", | ||
"jiti": "^1.17.0", | ||
"ioredis-mock": "^8.2.6", | ||
"jiti": "^1.17.1", | ||
"jsdom": "^21.1.0", | ||
"monaco-editor": "^0.35.0", | ||
"msw": "^1.0.1", | ||
"monaco-editor": "^0.36.0", | ||
"mongodb": "^5.0.1", | ||
"mongodb-memory-server": "^8.11.4", | ||
"msw": "^1.1.0", | ||
"prettier": "^2.8.4", | ||
"types-cloudflare-worker": "^1.2.0", | ||
"typescript": "^4.9.5", | ||
"unbuild": "^1.1.1", | ||
"vite": "^4.1.1", | ||
"vitest": "^0.28.5", | ||
"unbuild": "^1.1.2", | ||
"vite": "^4.1.4", | ||
"vitest": "^0.29.1", | ||
"vue": "^3.2.47" | ||
}, | ||
"packageManager": "pnpm@7.27.0", | ||
"optionalDependencies": { | ||
"@azure/app-configuration": "^1.3.1", | ||
"@azure/cosmos": "^3.17.2", | ||
"@azure/data-tables": "^13.2.1", | ||
"@azure/identity": "^3.1.3", | ||
"@azure/keyvault-secrets": "^4.6.0", | ||
"@azure/storage-blob": "^12.12.0", | ||
"@planetscale/database": "^1.5.0" | ||
}, | ||
"packageManager": "pnpm@7.28.0", | ||
"scripts": { | ||
@@ -83,3 +97,4 @@ "build": "unbuild", | ||
"dev": "vitest", | ||
"lint": "eslint --ext .ts . && prettier -c src test demo", | ||
"lint": "eslint --ext .ts . && prettier -c src test demo docs/content", | ||
"lint:fix": "eslint --ext .ts . --fix && prettier -w src test demo docs/content", | ||
"release": "pnpm test && changelogen --release && git push --follow-tags && pnpm publish", | ||
@@ -86,0 +101,0 @@ "test": "pnpm lint && vitest run --coverage", |
643
README.md
@@ -9,30 +9,20 @@ # unstorage | ||
> 💾 Universal Storage Layer | ||
💾 Unstorage provides an async Key-Value storage API with conventional features like multi driver mounting, watching and working with metadata, dozens of built-in drivers and a [tiny core](https://bundlephobia.com/package/unstorage). | ||
**Why ❓** | ||
👉 [documentation](https://unstorage.unjs.io) | ||
Typically, we choose one or more data storages based on our use-cases like a filesystem, a database like Redis, Mongo, or LocalStorage for browsers but it will soon start to be lots of trouble for supporting and combining more than one or switching between them. For javascript library authors, this usually means they have to decide how many platforms they support and implement storage for each. | ||
## Features | ||
💡 Unstorage solution is a unified and powerful Key-Value (KV) interface that allows combining drivers that are either built-in or can be implemented via a super simple interface and adding conventional features like mounting, watching, and working with metadata. | ||
- Designed for all environments: Browser, NodeJS, and Workers | ||
- Lots of Built-in drivers | ||
- Asynchronous API | ||
- Unix-style driver mounting to combine storages | ||
- Default [in-memory](/drivers/memory) storage | ||
- Tree-shakable utils and tiny core | ||
- Auto JSON value serialization and deserialization | ||
- Banary and raw value support | ||
- State [snapshots](/utils#snapshots) and hydration | ||
- Storage watcher | ||
- HTTP Storage with [built-in server](/server) | ||
Comparing to similar solutions like [localforage](https://localforage.github.io/localForage/), unstorage core is almost 6x smaller (28.9 kB vs 4.7 kB), using modern ESM/Typescript/Async syntax and many more features to be used universally. | ||
<br> | ||
✅ Designed to work in all environments (Browser, NodeJS, and Workers) <br> | ||
✅ Multiple built-in drivers (Memory, FS, LocalStorage, HTTP, Redis) <br> | ||
✅ Asynchronous API <br> | ||
✅ Unix-style driver mounting to combine storages<br> | ||
✅ Default in-memory storage <br> | ||
✅ Tree-shakable utils and tiny core <br> | ||
✅ Driver native and user provided metadata <br> | ||
✅ Native aware value serialization and deserialization <br> | ||
✅ Restore initial state (hydration) <br> | ||
✅ State snapshot <br> | ||
✅ Driver agnostic watcher <br> | ||
✅ HTTP Storage server (cli and programmatic) <br> | ||
✅ Namespaced storage <br> | ||
✅ Overlay storage (copy-on-write) <br> | ||
✅ Binary and raw operations support (experimental) <br> | ||
<br> | ||
## Usage | ||
@@ -45,7 +35,8 @@ | ||
yarn add unstorage | ||
# npm | ||
npm install unstorage | ||
# pnpm | ||
pnpm add unstorage | ||
``` | ||
@@ -61,604 +52,4 @@ | ||
**Options:** | ||
👉 Check out the [the documentation](https://unstorage.unjs.io) for usage information. | ||
- `driver`: Default driver (using memory if not provided) | ||
## Storage Interface | ||
### `storage.hasItem(key)` | ||
Checks if storage contains a key. Resolves to either `true` or `false`. | ||
```js | ||
await storage.hasItem("foo:bar"); | ||
``` | ||
### `storage.getItem(key)` | ||
Gets the value of a key in storage. Resolves to either a javascript primitive value or `undefined`. | ||
```js | ||
await storage.getItem("foo:bar"); | ||
``` | ||
### `storage.getItemRaw(key)` | ||
**Note:** This is an experimental feature. Please check [unjs/unstorage#142](https://github.com/unjs/unstorage/issues/142) for more information. | ||
Gets the value of a key in storage in raw format. | ||
```js | ||
// Value can be a Buffer, Array or Driver's raw format | ||
const value = await storage.getItemRaw("foo:bar.bin"); | ||
``` | ||
### `storage.setItem(key, value)` | ||
Add/Update a value to the storage. | ||
If the value is not a string, it will be stringified. | ||
If value is `undefined`, it is same as calling `removeItem(key)`. | ||
```js | ||
await storage.setItem("foo:bar", "baz"); | ||
``` | ||
### `storage.setItemRaw(key, value)` | ||
**Note:** This is an experimental feature. Please check [unjs/unstorage#142](https://github.com/unjs/unstorage/issues/142) for more information. | ||
Add/Update a value to the storage in raw format. | ||
If value is `undefined`, it is same as calling `removeItem(key)`. | ||
```js | ||
await storage.setItemRaw("data/test.bin", new Uint8Array([1, 2, 3])); | ||
``` | ||
### `storage.removeItem(key, removeMeta = true)` | ||
Remove a value (and it's meta) from storage. | ||
```js | ||
await storage.removeItem("foo:bar"); | ||
``` | ||
### `storage.getMeta(key, nativeOnly?)` | ||
Get metadata object for a specific key. | ||
This data is fetched from two sources: | ||
- Driver native meta (like file creation time) | ||
- Custom meta set by `storage.setMeta` (overrides driver native meta) | ||
```js | ||
await storage.getMeta("foo:bar"); // For fs driver returns an object like { mtime, atime, size } | ||
``` | ||
### `storage.setMeta(key)` | ||
Set custom meta for a specific key by adding a `$` suffix. | ||
```js | ||
await storage.setMeta("foo:bar", { flag: 1 }); | ||
// Same as storage.setItem('foo:bar$', { flag: 1 }) | ||
``` | ||
### `storage.removeMeta(key)` | ||
Remove meta for a specific key by adding a `$` suffix. | ||
```js | ||
await storage.removeMeta("foo:bar"); | ||
// Same as storage.removeItem('foo:bar$') | ||
``` | ||
### `storage.getKeys(base?)` | ||
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 also only mounts starting with base will be queried. Keys still have a full path. | ||
```js | ||
await storage.getKeys(); | ||
``` | ||
### `storage.clear(base?)` | ||
Removes all stored key/values. If a base is provided, only mounts matching base will be cleared. | ||
```js | ||
await storage.clear(); | ||
``` | ||
### `storage.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. | ||
```js | ||
await storage.dispose(); | ||
``` | ||
### `storage.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. | ||
```js | ||
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"); | ||
``` | ||
### `storage.unmount(mountpoint, dispose = true)` | ||
Unregisters a mountpoint. Has no effect if mountpoint is not found or is root. | ||
```js | ||
await storage.unmount("/output"); | ||
``` | ||
### `storage.watch(callback)` | ||
Starts watching on all mountpoints. If driver does not supports watching, only emits even when `storage.*` methods are called. | ||
```js | ||
const unwatch = await storage.watch((event, key) => {}); | ||
// to stop this watcher | ||
await unwatch(); | ||
``` | ||
### `storage.unwatch()` | ||
Stop all watchers on all mountpoints. | ||
```js | ||
await storage.unwatch(); | ||
``` | ||
## Utils | ||
### `snapshot(storage, base?)` | ||
Snapshot from all keys in specified base into a plain javascript object (string: string). Base is removed from keys. | ||
```js | ||
import { snapshot } from "unstorage"; | ||
const data = await snapshot(storage, "/etc"); | ||
``` | ||
### `restoreSnapshot(storage, data, base?)` | ||
Restore snapshot created by `snapshot()`. | ||
```js | ||
await restoreSnapshot(storage, { "foo:bar": "baz" }, "/etc2"); | ||
``` | ||
### `prefixStorage(storage, data, base?)` | ||
Create a namespaced instance of main storage. | ||
All operations are virtually prefixed. Useful to create shorcuts and limit access. | ||
```js | ||
import { createStorage, prefixStorage } from "unstorage"; | ||
const storage = createStorage(); | ||
const assetsStorage = prefixStorage(storage, "assets"); | ||
// Same as storage.setItem('assets:x', 'hello!') | ||
await assetsStorage.setItem("x", "hello!"); | ||
``` | ||
## Storage Server | ||
We can easily expose unstorage instance to an http server to allow remote connections. | ||
Request url is mapped to key and method/body mapped to function. See below for supported http methods. | ||
**🛡️ Security Note:** Server is unprotected by default. You need to add your own authentication/security middleware like basic authentication. | ||
Also consider that even with authentication, unstorage should not be exposed to untrusted users since it has no protection for abuse (DDOS, Filesystem escalation, etc) | ||
**Programmatic usage:** | ||
```js | ||
import { listen } from "listhen"; | ||
import { createStorage } from "unstorage"; | ||
import { createStorageServer } from "unstorage/server"; | ||
const storage = createStorage(); | ||
const storageServer = createStorageServer(storage); | ||
// Alternatively we can use `storageServer.handle` as a middleware | ||
await listen(storageServer.handle); | ||
``` | ||
**Using CLI:** | ||
```sh | ||
npx unstorage . | ||
``` | ||
**Supported HTTP Methods:** | ||
- `GET`: Maps to `storage.getItem`. Returns list of keys on path if value not found. | ||
- `HEAD`: Maps to `storage.hasItem`. Returns 404 if not found. | ||
- `PUT`: Maps to `storage.setItem`. Value is read from body and returns `OK` if operation succeeded. | ||
- `DELETE`: Maps to `storage.removeItem`. Returns `OK` if operation succeeded. | ||
## Drivers | ||
### `fs` (node) | ||
Maps data to the real filesystem using directory structure for nested keys. Supports watching using [chokidar](https://github.com/paulmillr/chokidar). | ||
This driver implements meta for each key including `mtime` (last modified time), `atime` (last access time), and `size` (file size) using `fs.stat`. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import fsDriver from "unstorage/drivers/fs"; | ||
const storage = createStorage({ | ||
driver: fsDriver({ base: "./tmp" }), | ||
}); | ||
``` | ||
**Options:** | ||
- `base`: Base directory to isolate operations on this directory | ||
- `ignore`: Ignore patterns for watch <!-- and key listing --> | ||
- `watchOptions`: Additional [chokidar](https://github.com/paulmillr/chokidar) options. | ||
### `localStorage` (browser) | ||
Store data in [localStorage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage). | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import localStorageDriver from "unstorage/drivers/localstorage"; | ||
const storage = createStorage({ | ||
driver: localStorageDriver({ base: "app:" }), | ||
}); | ||
``` | ||
**Options:** | ||
- `base`: Add `${base}:` to all keys to avoid collision | ||
- `localStorage`: Optionally provide `localStorage` object | ||
- `window`: Optionally provide `window` object | ||
### `memory` (universal) | ||
Keeps data in memory using [Set](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Set). | ||
By default it is mounted to top level so it is unlikely you need to mount it again. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import memoryDriver from "unstorage/drivers/memory"; | ||
const storage = createStorage({ | ||
driver: memoryDriver(), | ||
}); | ||
``` | ||
### `lru-cache` (universal) | ||
Keeps cached data in memory using [LRU Cache](https://www.npmjs.com/package/lru-cache). | ||
See [`lru-cache`](https://www.npmjs.com/package/lru-cache) for supported options. By default `{ maxSize: 500 }` is used. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import lruCacheDriver from "unstorage/drivers/lru-cache"; | ||
const storage = createStorage({ | ||
driver: lruCacheDriver(), | ||
}); | ||
``` | ||
### `overlay` (universal) | ||
This is a special driver that creates a multi-layer overlay driver. | ||
All write operations happen on the top level layer while values are read from all layers. | ||
When removing a key, a special value `__OVERLAY_REMOVED__` will be set on the top level layer internally. | ||
In the example below, we create an in-memory overlay on top of fs. No changes will be actually written to the disk. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import overlay from "unstorage/drivers/overlay"; | ||
import memory from "unstorage/drivers/memory"; | ||
import fs from "unstorage/drivers/fs"; | ||
const storage = createStorage({ | ||
driver: overlay({ | ||
layers: [memory(), fs({ base: "./data" })], | ||
}), | ||
}); | ||
``` | ||
### `http` (universal) | ||
Use a remote HTTP/HTTPS endpoint as data storage. Supports built-in [http server](#storage-server) methods. | ||
This driver implements meta for each key including `mtime` (last modified time) and `status` from HTTP headers by making a `HEAD` request. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import httpDriver from "unstorage/drivers/http"; | ||
const storage = createStorage({ | ||
driver: httpDriver({ base: "http://cdn.com" }), | ||
}); | ||
``` | ||
**Options:** | ||
- `base`: Base URL for urls | ||
**Supported HTTP Methods:** | ||
- `getItem`: Maps to http `GET`. Returns deserialized value if response is ok | ||
- `hasItem`: Maps to http `HEAD`. Returns `true` if response is ok (200) | ||
- `setItem`: Maps to http `PUT`. Sends serialized value using body | ||
- `removeItem`: Maps to `DELETE` | ||
- `clear`: Not supported | ||
### `redis` | ||
Store data in a redis storage using [ioredis](https://github.com/luin/ioredis). | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import redisDriver from "unstorage/drivers/redis"; | ||
const storage = createStorage({ | ||
driver: redisDriver({ | ||
base: "storage:", | ||
}), | ||
}); | ||
``` | ||
**Options:** | ||
- `base`: Prefix all keys with base | ||
- `url`: (optional) connection string | ||
See [ioredis](https://github.com/luin/ioredis/blob/master/API.md#new-redisport-host-options) for all available options. | ||
`lazyConnect` option is enabled by default so that connection happens on first redis operation. | ||
### `cloudflare-kv-http` | ||
Store data in [Cloudflare KV](https://developers.cloudflare.com/workers/learning/how-kv-works/) using the [Cloudflare API v4](https://api.cloudflare.com/). | ||
You need to create a KV namespace. See [KV Bindings](https://developers.cloudflare.com/workers/runtime-apis/kv#kv-bindings) for more information. | ||
**Note:** This driver uses native fetch and works universally! For using directly in a cloudflare worker environemnt, please use `cloudflare-kv-binding` driver for best performance! | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import cloudflareKVHTTPDriver from "unstorage/drivers/cloudflare-kv-http"; | ||
// Using `apiToken` | ||
const storage = createStorage({ | ||
driver: cloudflareKVHTTPDriver({ | ||
accountId: "my-account-id", | ||
namespaceId: "my-kv-namespace-id", | ||
apiToken: "supersecret-api-token", | ||
}), | ||
}); | ||
// Using `email` and `apiKey` | ||
const storage = createStorage({ | ||
driver: cloudflareKVHTTPDriver({ | ||
accountId: "my-account-id", | ||
namespaceId: "my-kv-namespace-id", | ||
email: "me@example.com", | ||
apiKey: "my-api-key", | ||
}), | ||
}); | ||
// Using `userServiceKey` | ||
const storage = createStorage({ | ||
driver: cloudflareKVHTTPDriver({ | ||
accountId: "my-account-id", | ||
namespaceId: "my-kv-namespace-id", | ||
userServiceKey: "v1.0-my-service-key", | ||
}), | ||
}); | ||
``` | ||
**Options:** | ||
- `accountId`: Cloudflare account ID. | ||
- `namespaceId`: The ID of the KV namespace to target. **Note:** be sure to use the namespace's ID, and not the name or binding used in a worker environment. | ||
- `apiToken`: API Token generated from the [User Profile 'API Tokens' page](https://dash.cloudflare.com/profile/api-tokens). | ||
- `email`: Email address associated with your account. May be used along with `apiKey` to authenticate in place of `apiToken`. | ||
- `apiKey`: API key generated on the "My Account" page of the Cloudflare console. May be used along with `email` to authenticate in place of `apiToken`. | ||
- `userServiceKey`: A special Cloudflare API key good for a restricted set of endpoints. Always begins with "v1.0-", may vary in length. May be used to authenticate in place of `apiToken` or `apiKey` and `email`. | ||
- `apiURL`: Custom API URL. Default is `https://api.cloudflare.com`. | ||
**Supported methods:** | ||
- `getItem`: Maps to [Read key-value pair](https://api.cloudflare.com/#workers-kv-namespace-read-key-value-pair) `GET accounts/:account_identifier/storage/kv/namespaces/:namespace_identifier/values/:key_name` | ||
- `hasItem`: Maps to [Read key-value pair](https://api.cloudflare.com/#workers-kv-namespace-read-key-value-pair) `GET accounts/:account_identifier/storage/kv/namespaces/:namespace_identifier/values/:key_name`. Returns `true` if `<parsed response body>.success` is `true`. | ||
- `setItem`: Maps to [Write key-value pair](https://api.cloudflare.com/#workers-kv-namespace-write-key-value-pair) `PUT accounts/:account_identifier/storage/kv/namespaces/:namespace_identifier/values/:key_name` | ||
- `removeItem`: Maps to [Delete key-value pair](https://api.cloudflare.com/#workers-kv-namespace-delete-key-value-pair) `DELETE accounts/:account_identifier/storage/kv/namespaces/:namespace_identifier/values/:key_name` | ||
- `getKeys`: Maps to [List a Namespace's Keys](https://api.cloudflare.com/#workers-kv-namespace-list-a-namespace-s-keys) `GET accounts/:account_identifier/storage/kv/namespaces/:namespace_identifier/keys` | ||
- `clear`: Maps to [Delete key-value pair](https://api.cloudflare.com/#workers-kv-namespace-delete-multiple-key-value-pairs) `DELETE accounts/:account_identifier/storage/kv/namespaces/:namespace_identifier/bulk` | ||
### `cloudflare-kv-binding` | ||
Store data in [Cloudflare KV](https://developers.cloudflare.com/workers/runtime-apis/kv) and access from worker bindings. | ||
**Note:** This driver only works in a cloudflare worker environment! Use `cloudflare-kv-http` for other environments. | ||
You need to create and assign a KV. See [KV Bindings](https://developers.cloudflare.com/workers/runtime-apis/kv#kv-bindings) for more information. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import cloudflareKVBindingDriver from "unstorage/drivers/cloudflare-kv-binding"; | ||
// Using binding name to be picked from globalThis | ||
const storage = createStorage({ | ||
driver: cloudflareKVBindingDriver({ binding: "STORAGE" }), | ||
}); | ||
// Directly setting binding | ||
const storage = createStorage({ | ||
driver: cloudflareKVBindingDriver({ binding: globalThis.STORAGE }), | ||
}); | ||
// Using from Durable Objects and Workers using Modules Syntax | ||
const storage = createStorage({ | ||
driver: cloudflareKVBindingDriver({ binding: this.env.STORAGE }), | ||
}); | ||
// Using outside of Cloudflare Workers (like Node.js) | ||
// Use cloudflare-kv-http! | ||
``` | ||
**Options:** | ||
- `binding`: KV binding or name of namespace. Default is `STORAGE`. | ||
### `github` | ||
Map files from a remote github repository. (readonly) | ||
This driver fetches all possible keys once and keep it in cache for 10 minutes. Because of github rate limit, it is highly recommanded to provide a token. It only applies to fetching keys. | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import githubDriver from "unstorage/drivers/github"; | ||
const storage = createStorage({ | ||
driver: githubDriver({ | ||
repo: "nuxt/framework", | ||
branch: "main", | ||
dir: "/docs/content", | ||
}), | ||
}); | ||
``` | ||
**Options:** | ||
- **`repo`**: Github repository. Format is `username/repo` or `org/repo`. (Required!) | ||
- **`token`**: Github API token. (Recommended!) | ||
- `branch`: Target branch. Default is `main` | ||
- `dir`: Use a directory as driver root. | ||
- `ttl`: Filenames cache revalidate time. Default is `600` seconds (10 minutes) | ||
- `apiURL`: Github API domain. Default is `https://api.github.com` | ||
- `cdnURL`: Github RAW CDN Url. Default is `https://raw.githubusercontent.com` | ||
### planetscale | ||
Stores data in [Planetscale](https://planetscale.com/) | ||
This driver stores KV information in a Planetscale DB with columns of `id`, `value`, `created_at` and `updated_at`. | ||
To use, you will need to install `@planetscale/database` in your project: | ||
```json | ||
{ | ||
"dependencies": { | ||
"@planetscale/database": "^1.5.0" | ||
} | ||
} | ||
``` | ||
Then you can create a table to store your data by running the following query in your Planetscale database, where <storage> is the name of the table you want to use: | ||
```sql | ||
create table <storage> ( | ||
id varchar(255) not null primary key, | ||
value longtext, | ||
created_at timestamp default current_timestamp, | ||
updated_at timestamp default current_timestamp on update current_timestamp | ||
); | ||
``` | ||
You can then configure the driver like this: | ||
```js | ||
import { createStorage } from "unstorage"; | ||
import planetscaleDriver from "unstorage/drivers/planetscale"; | ||
const storage = createStorage({ | ||
driver: planetscaleDriver({ | ||
// This should certainly not be inlined in your code but loaded via runtime config | ||
// or environment variables depending on your framework/project. | ||
url: "mysql://xxxxxxxxx:************@xxxxxxxxxx.us-east-3.psdb.cloud/my-database?sslaccept=strict", | ||
// table: 'storage' | ||
}), | ||
}); | ||
``` | ||
**Options:** | ||
- **`url`** (required): You can find your URL in the [Planetscale dashboard](https://planetscale.com/docs/tutorials/connect-nodejs-app). | ||
- `storage`: The name of the table to read from. It defaults to `storage`. | ||
- `boostCache`: Whether to enable cached queries: see [docs](https://planetscale.com/docs/concepts/query-caching-with-planetscale-boost#using-cached-queries-in-your-application). | ||
## Making custom drivers | ||
It is possible to extend unstorage by creating custom drives. | ||
- Keys are always normalized in `foo:bar` convention | ||
- Mount base is removed | ||
- Returning promise or direct value is optional | ||
- You should cleanup any open watcher and handlers in `dispose` | ||
- Value returned by `getItem` can be a serializable object or string | ||
- Having `watch` method, disables default handler for mountpoint. You are responsible to emit event on `getItem`, `setItem` and `removeItem`. | ||
See [src/drivers](./src/drivers) to inspire how to implement them. Methods can | ||
**Example:** | ||
```js | ||
import { createStorage, defineDriver } from "unstorage"; | ||
const myStorageDriver = defineDriver((_opts) => { | ||
return { | ||
async hasItem(key) {}, | ||
async getItem(key) {}, | ||
async setItem(key, value) {}, | ||
async removeItem(key) {}, | ||
async getKeys() {}, | ||
async clear() {}, | ||
async dispose() {}, | ||
// async watch(callback) {} | ||
}; | ||
}); | ||
const storage = createStorage({ | ||
driver: myStorageDriver(), | ||
}); | ||
``` | ||
## Contribution | ||
@@ -665,0 +56,0 @@ |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
150702
70
4820
18
34
75
+ Added@azure/abort-controller@1.1.02.1.2(transitive)
+ Added@azure/app-configuration@1.8.0(transitive)
+ Added@azure/core-auth@1.9.0(transitive)
+ Added@azure/core-client@1.9.2(transitive)
+ Added@azure/core-http-compat@2.1.2(transitive)
+ Added@azure/core-lro@2.7.2(transitive)
+ Added@azure/core-paging@1.6.2(transitive)
+ Added@azure/core-rest-pipeline@1.18.0(transitive)
+ Added@azure/core-tracing@1.2.0(transitive)
+ Added@azure/core-util@1.11.0(transitive)
+ Added@azure/core-xml@1.4.4(transitive)
+ Added@azure/cosmos@3.17.3(transitive)
+ Added@azure/data-tables@13.2.2(transitive)
+ Added@azure/identity@3.4.2(transitive)
+ Added@azure/keyvault-common@2.0.0(transitive)
+ Added@azure/keyvault-secrets@4.9.0(transitive)
+ Added@azure/logger@1.1.4(transitive)
+ Added@azure/msal-browser@3.27.0(transitive)
+ Added@azure/msal-common@14.16.0(transitive)
+ Added@azure/msal-node@2.16.1(transitive)
+ Added@azure/storage-blob@12.25.0(transitive)
+ Addedagent-base@7.1.1(transitive)
+ Addedbuffer-equal-constant-time@1.0.1(transitive)
+ Addeddefine-lazy-prop@2.0.0(transitive)
+ Addedecdsa-sig-formatter@1.0.11(transitive)
+ Addedevents@3.3.0(transitive)
+ Addedfast-json-stable-stringify@2.1.0(transitive)
+ Addedfast-xml-parser@4.5.0(transitive)
+ Addedhttp-proxy-agent@7.0.2(transitive)
+ Addedhttps-proxy-agent@7.0.5(transitive)
+ Addedis-docker@2.2.1(transitive)
+ Addedis-wsl@2.2.0(transitive)
+ Addedjsbi@3.2.5(transitive)
+ Addedjsonwebtoken@9.0.2(transitive)
+ Addedjwa@1.4.12.0.0(transitive)
+ Addedjws@3.2.24.0.0(transitive)
+ Addedlodash.includes@4.3.0(transitive)
+ Addedlodash.isboolean@3.0.3(transitive)
+ Addedlodash.isinteger@4.0.4(transitive)
+ Addedlodash.isnumber@3.0.3(transitive)
+ Addedlodash.isplainobject@4.0.6(transitive)
+ Addedlodash.isstring@4.0.1(transitive)
+ Addedlodash.once@4.1.1(transitive)
+ Addednode-abort-controller@3.1.1(transitive)
+ Addedopen@8.4.2(transitive)
+ Addedpriorityqueuejs@1.0.0(transitive)
+ Addedsafe-buffer@5.2.1(transitive)
+ Addedsemaphore@1.1.0(transitive)
+ Addedsemver@7.6.3(transitive)
+ Addedstoppable@1.1.0(transitive)
+ Addedstrnum@1.0.5(transitive)
+ Addedtslib@2.8.1(transitive)
+ Addeduniversal-user-agent@6.0.1(transitive)
+ Addeduuid@8.3.2(transitive)
- Removedmkdir@^0.0.2
- Removedws@^8.12.1
- Removedmkdir@0.0.2(transitive)
- Removedws@8.18.0(transitive)
Updatedlisthen@^1.0.3
Updatedlru-cache@^7.17.0