Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/osaton/do-proxy
Durable Object Proxy
https://github.com/osaton/do-proxy
cloudflare cloudflare-workers durableobject durableobjects
Last synced: 2 months ago
JSON representation
Durable Object Proxy
- Host: GitHub
- URL: https://github.com/osaton/do-proxy
- Owner: osaton
- License: mit
- Created: 2022-11-26T21:39:42.000Z (about 2 years ago)
- Default Branch: main
- Last Pushed: 2024-06-03T07:11:20.000Z (7 months ago)
- Last Synced: 2024-10-25T03:45:45.922Z (2 months ago)
- Topics: cloudflare, cloudflare-workers, durableobject, durableobjects
- Language: TypeScript
- Homepage:
- Size: 1.54 MB
- Stars: 2
- Watchers: 1
- Forks: 0
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
README
# Durable Object Proxy (do-proxy)
[![Types included](https://badgen.net/npm/types/do-proxy?style=flat-square)](https://www.npmjs.com/package/do-proxy)
[![Bundle Size](https://badgen.net/bundlephobia/minzip/do-proxy?style=flat-square)](https://bundlephobia.com/result?p=do-proxy)
[![Bundle Size](https://badgen.net/bundlephobia/dependency-count/do-proxy?style=flat-square)](https://bundlephobia.com/result?p=do-proxy)
[![Version](https://img.shields.io/npm/v/do-proxy.svg?style=flat-square)](https://npmjs.com/package/do-proxy)> Simple interface for accessing [Cloudflare Durable Objects](https://developers.cloudflare.com/workers/learning/using-durable-objects/)' storage and class methods.
This library handles request building, fetching and responding behind the scenes via lightweight proxy object which provides interface for accessing DO instance's storage and class methods.
## Install
```text
npm install do-proxy
```## Examples
- [Todo app](https://stackblitz.com/fork/github/osaton/do-proxy/tree/main/examples/todo-storage?file=src%2Findex.ts&terminal='start-stackblitz') utilizing `storage` and `batch` methods
- [Todo app](https://stackblitz.com/fork/github/osaton/do-proxy/tree/main/examples/todo-class?file=src%2Findex.ts&terminal='start-stackblitz') extending `DOProxy` class and utilizing `class` methods## Usage briefly
Make your Durable Object class methods accessible by extending the `DOProxy`.
```ts
import { DOProxy } from 'do-proxy';
class MyDOClass extends DOProxy {
// Arguments & return values have to be JSON serialiazable
myClassMethod(param: string) {
// Do what ever you would do inside DO
}
}
```Inside your Worker's `fetch` method:
```ts
// Get `DurableObjectNamespace` wrapped inside our proxy
const MY_DO_BINDING = MyDOClass.wrap(env.MY_DO_BINDING);
// You can use the default namespace methods or shorthand methods `getByName` & `getById`
const stub = MY_DO_BINDING.getByName('name');// You can access instance's storage methods
const res1 = await stub.storage.get('my-store');
// You can also access your class's methods.
const res2 = await stub.class.myClassMethod('foo');// Or handle both with a single fetch behind the scenes using `batch` method
const [res3, res4] = await stub.batch(() => [
stub.storage.get('my-store'),
stub.class.myClassMethod('foo'),
]);
```## Usage
You can use `DOProxy` as is for Durable Object bindings. This enables you to use [`storage`](#storage-methods) methods.
Here we expect you to have DO class `Todo` bound to `TODO` inside `wrangler.toml`:
```ts
import { DOProxy } from 'do-proxy';
export { DOProxy as Todo };
export default {
async fetch(req: Request, env: any) {
const TODO = DOProxy.wrap(env.TODO);
const stub = TODO.getByName('name');
await todo.storage.put('todo:1', 'has to be done');
const list = Object.fromEntries(await todo.storage.list());
return Response.json(list);
},
};
```Or you can extend it, which enables you to call class methods via `class` property:
```ts
import { DOProxy } from 'do-proxy';class Todo extends DOProxy {
state: DurableObjectState;
constructor(state: DurableObjectState) {
super(state);
this.state = state;
}
async add(todo: string) {
const id = Math.ceil(Math.random() * 100);
this.state.storage.put(`todo:${id}`, todo);
return id;
}async get(id: number) {
return this.state.storage.get(`todo:${id}`);
}
}
export default {
async fetch(req: Request, env: any) {
const stub = Todo.wrap(env.TODO).getByName('my-todos');
const id = await stub.class.add('has to be done');
const todo = await stub.class.get(id);
return Response.json({
id,
todo,
});
},
};
export { Todo };
```You can also utilize the [`batch`](#batch) method which allows you to run multiple methods with one fetch request to DO instance:
```ts
// See previous example for `Todo` details
const [, , list] = await stub.batch(() => [
stub.class.add('my todo'),
stub.class.add('my other todo'),
stub.storage.list(),
]);return Response.json(Object.fromEntries(list as Map));
```## `storage` methods
`DOProxy` can be used as Durable Object class as is. It gives you access to Durable Object instance's [Transactional storage API](https://developers.cloudflare.com/workers/runtime-apis/durable-objects/#transactional-storage-api) methods (excluding `transaction` which can't be proxied because of JSON serialization. See [batch method](#batch)).
Available methods: `DurableObjectStubProxy.storage.get|put|delete|deleteAll|list|getAlarm|setAlarm|deleteAlarm|sync`
## `batch` method
If you need to invoke Durable Object instance's multiple times, `DurableObjectStubProxy` has a `batch` method which allows you to run multiple method calls inside one fetch request.
Method calls passed to `batch` will be run in sequence.
```ts
const COUNTER = Counter.wrap(env.Counter);
const stub = COUNTER.get(COUNTER.newUniqueId());await stub.batch(() => [
stub.class.increment(),
stub.class.increment(),
stub.storage.deleteAll(),
stub.class.increment(),
]); // => [1, 2, null, 1]
```## static `wrap(binding: DurableObjectNamespace): DurableObjectNamespaceProxy`
This method return `DurableObjectNamespace` wrapped inside proxy.
It has all the same methods that `DurableObjectNamespace`:- `newUniqueId(options?: DurableObjectNamespaceNewUniqueIdOptions | undefined): DurableObjectId`;
- `idFromName(name: string): DurableObjectId`
- `idFromString(id: string): DurableObjectId`
- `get(id: DurableObjectId): DurableObjectStubProxy`It also has some custom shorthand methods:
- `getByName(name: string): DurableObjectStubProxy`: Shorthand for `DO.get(DO.idFromName('foo'))`
- `getByString(id: string): DurableObjectStubProxy`: Shorthand for `DO.get(DO.idFromString(hexId))``get` Method returns `DurableObjectStubProxy` instead of `DurableObjectStub`.
## `DurableObjectStubProxy` properties
- `id: DurableObjectId`
- `stub: DurableObjectStub`: The actual stub if you need access to it
- `storage: Object`: Storage methods
- `batch: (callback: () => Promise[]) => unknown[]`
- `class: Object|undefined`: All the class methods if `wrap` was called on an extended class## Limitations
Remember that we are still doing fetch requests even if it is done in the background, so everything sent to `class` and `storage` methods must be JSON serializable.
## Other libraries
Not the Durable Object proxy you were looking for?
- [`do-proxy`](https://github.com/fisherdarling/do-proxy) for Rust by [@fisherdarling](https://www.github.com/fisherdarling)