Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/reearth/quickjs-emscripten-sync

Provides a way to sync objects between browser and QuickJS
https://github.com/reearth/quickjs-emscripten-sync

Last synced: 3 months ago
JSON representation

Provides a way to sync objects between browser and QuickJS

Awesome Lists containing this project

README

        

# quickjs-emscripten-sync

[![CI](https://github.com/reearth/quickjs-emscripten-sync/actions/workflows/ci.yml/badge.svg)](https://github.com/reearth/quickjs-emscripten-sync/actions/workflows/main.yml) [![codecov](https://codecov.io/gh/reearth/quickjs-emscripten-sync/branch/main/graph/badge.svg)](https://codecov.io/gh/reearth/quickjs-emscripten-sync)

Build a secure plugin system in web browsers

This library wraps [quickjs-emscripten](https://github.com/justjake/quickjs-emscripten) and provides a way to sync object state between the browser and sandboxed QuickJS.

- Exchange and sync values between the browser (host) and QuickJS seamlessly
- Primitives (number, boolean, string, symbol)
- Arrays
- Functions
- Classes and instances
- Objects with prototypes and any property descriptors
- Promises
- Expose objects as a global object in QuickJS
- Marshaling limitation for specific objects
- Register a pair of objects that will be considered the same between the browser and QuickJS

```
npm install quickjs-emscripten quickjs-emscripten-sync
```

```js
import { getQuickJS } from "quickjs-emscripten";
import { Arena } from "quickjs-emscripten-sync";

class Cls {
field = 0;

method() {
return ++this.field;
}
}

const ctx = (await getQuickJS()).newContext();
const arena = new Arena(ctx, { isMarshalable: true });

// We can pass objects to the context and run code safely
const exposed = {
Cls,
cls: new Cls(),
syncedCls: arena.sync(new Cls()),
};
arena.expose(exposed);

arena.evalCode(`cls instanceof Cls`); // returns true
arena.evalCode(`cls.field`); // returns 0
arena.evalCode(`cls.method()`); // returns 1
arena.evalCode(`cls.field`); // returns 1

arena.evalCode(`syncedCls.field`); // returns 0
exposed.syncedCls.method(); // returns 1
arena.evalCode(`syncedCls.field`); // returns 1

arena.dispose();
ctx.dispose();
```

[Example code](src/index.test.ts) is available as the unit test code.

## Operating Environment

- Web browsers that support WebAssembly
- Node.js

If you want to run quickjs-emscripten and quickjs-emscripten-sync in a web browser, they have to be bundled with a bundler tool such as webpack, because quickjs-emscripten is now written in CommonJS format and web browsers cannot load it directly.

## Usage

```js
import { getQuickJS } from "quickjs-emscripten";
import { Arena } from "quickjs-emscripten-sync";

(async function() {
const ctx = (await getQuickJS()).newContext();

// init Arena
// ⚠️ Marshaling is opt-in for security reasons.
// ⚠️ Be careful when activating marshalling.
const arena = new Arena(ctx, { isMarshalable: true });

// expose objects as global objects in QuickJS context
arena.expose({
console: {
log: console.log
}
});
arena.evalCode(`console.log("hello, world");`); // run console.log
arena.evalCode(`1 + 1`); // 2

// expose objects but also enable sync
const data = arena.sync({ hoge: "foo" });
arena.expose({ data });

arena.evalCode(`data.hoge = "bar"`);
// eval code and operations to exposed objects are automatically synced
console.log(data.hoge); // "bar"
data.hoge = "changed!";
console.log(arena.evalCode(`data.hoge`)); // "changed!"

// Don't forget calling arena.dispose() before disposing QuickJS context!
arena.dispose();
ctx.dispose();
})();
```

## Marshaling Limitations

Objects are automatically converted when they cross between the host and the QuickJS context. The conversion of a host's object to a handle is called marshaling, and the conversion of a handle to a host's object is called unmarshaling.

And for marshalling, it is possible to control whether the conversion is performed or not.

For example, exposing the host's global object to QuickJS is very heavy and dangerous. This exposure can be limited and controlled with the `isMarshalable` option. If `false` is returned, just `undefined` is passed to QuickJS.

```js
import { Arena, complexity } from "quickjs-emscripten-sync";

const arena = new Arena(ctx, {
isMarshalable: (target: any) => {
// prevent passing globalThis to QuickJS
if (target === window) return false;
// complexity is a helper function to detect whether the object is heavy
if (complexity(target, 30) >= 30) return false;
return true; // other objects are OK
}
});

arena.evalCode(`a => a === undefined`)({}); // false
arena.evalCode(`a => a === undefined`)(window); // true
arena.evalCode(`a => a === undefined`)(document); // true
```

The `complexity` function is useful to detect whether the object is too heavy to be passed to QuickJS.

## Security Warning

QuickJS has an environment isolated from the browser, so any code can be executed safely, but there are edge cases where some exposed objects by quickjs-emscripten-sync may break security.

quickjs-emscripten-sync cannot prevent such dangerous case, so **PLEASE be very careful and deliberate about what you expose to QuickJS!**

### Case 1: Prototype pollution

```js
import { set } from "lodash-es";

arena.expose({
danger: (keys, value) => {
// This function may cause prototype pollution in the browser by QuickJS
set({}, keys, value)
}
});

arena.evalCode(`danger("__proto__.a", () => { /* injected */ })`);
```

### Case 2: Unintended HTTP request

It is very dangerous to expose or use directly or indirectly the `window` object, `localStorage`, `fetch`, `XMLHttpRequest` ...

This is because it enables the execution of unintended code such as XSS attacks, such as reading local storage, sending unintended HTTP requests, and manipulating DOM objects.

```js
arena.expose({
// This function may cause unintended HTTP requests
danger: (url, body) => {
fetch(url, {
method: "POST",
headers: {
"Content-Type": "application/json"
},
body: JSON.stringify(body)
});
}
});

arena.evalCode(`danger("/api", { dangerous: true })`);
```

By default, quickjs-emscripten-sync doesn't prevent any marshaling, even in such cases. And there are many built-in objects in the host, so please note that it's hard to prevent all dangerous cases with the `isMarshalable` option alone.

## API

### `Arena`

The Arena class manages all generated handles at once by quickjs-emscripten and automatically converts objects between the host and the QuickJS context.

#### `new Arena(ctx: QuickJSContext, options?: Options)`

Constructs a new Arena instance. It requires a quickjs-emscripten context initialized with `quickjs.newContext()`.

Options accepted:

```ts
type Options = {
/** A callback that returns a boolean value that determines whether an object is marshalled or not. If false, no marshaling will be done and undefined will be passed to the QuickJS VM, otherwise marshaling will be done. By default, all objects will be marshalled. */
isMarshalable?: boolean | "json" | ((target: any) => boolean | "json");
/** Pre-registered pairs of objects that will be considered the same between the host and the QuickJS VM. This will be used automatically during the conversion. By default, it will be registered automatically with `defaultRegisteredObjects`.
*
* Instead of a string, you can also pass a QuickJSHandle directly. In that case, however, you have to dispose of them manually when destroying the VM.
*/
registeredObjects?: Iterable<[any, QuickJSHandle | string]>;
/** Register functions to convert an object to a QuickJS handle. */
customMarshaller?: Iterable<
(target: unknown, ctx: QuickJSContext) => QuickJSHandle | undefined
>;
/** Register functions to convert a QuickJS handle to an object. */
customUnmarshaller?: Iterable<
(target: QuickJSHandle, ctx: QuickJSContext) => any
>;
/** A callback that returns a boolean value that determines whether an object is wrappable by proxies. If returns false, note that the object cannot be synchronized between the host and the QuickJS even if arena.sync is used. */
isWrappable?: (target: any) => boolean;
/** A callback that returns a boolean value that determines whether an QuickJS handle is wrappable by proxies. If returns false, note that the handle cannot be synchronized between the host and the QuickJS even if arena.sync is used. */
isHandleWrappable?: (handle: QuickJSHandle, ctx: QuickJSContext) => boolean;
/** Compatibility with quickjs-emscripten prior to v0.15. Inject code for compatibility into context at Arena class initialization time. */
compat?: boolean;
}
```

Notes:

**`isMarshalable`**: Determines how marshalling will be done when sending objects from the host to the context. **Make sure to set the marshalling to be the minimum necessary as it may reduce the security of your application.** [Please read the section on security above.](#security-warning)

- `"json"` (**default**, safety): Target object will be serialized as JSON in host and then parsed in context. Functions and classes will be lost in the process.
- `false` (safety): Target object will not be always marshalled as `undefined`.
- `(target: any) => boolean | "json"` (recoomended): You can control marshalling mode for each objects. If you want to do marshalling, usually use this method. Allow partial marshalling by returning `true` only for some objects.
- `true` (**risky and not recommended**): Target object will be always marshaled. This setting may reduce security.

**`registeredObjects`**: You can pre-register a pair of objects that will be considered the same between the host and the QuickJS context. This will be used automatically during the conversion. By default, it will be registered automatically with [`defaultRegisteredObjects`](src/default.ts). If you want to add a new pair to this, please do the following:

```js
import { defaultRegisteredObjects } from "quickjs-emscripten-sync";

const arena = new Arena(ctx, {
registeredObjects: [
...defaultRegisteredObjects,
[Math, "Math"]
]
});
```

Instead of a string, you can also pass a QuickJSHandle directly. In that case, however, you have to dispose of them manually when destroying the context.

#### `dispose()`

Dispose of the arena and managed handles. This method won't dispose the context itself, so the context has to be disposed of manually.

#### `evalCode(code: string): T | undefined`

Evaluate JS code in the context and get the result as an object on the host side. It also converts and re-throws error objects when an error is thrown during evaluation.

#### `executePendingJobs(): number`

Almost same as `ctx.runtime.executePendingJobs()`, but it converts and re-throws error objects when an error is thrown during evaluation.

#### `expose(obj: { [k: string]: any })`

Expose objects as global objects in the context.

By default, exposed objects are not synchronized between the host and the context.
If you want to sync an objects, first wrap the object with `sync` method, and then expose the wrapped object.

#### `sync(target: T): T`

Enables sync for the object between the host and the context and returns objects wrapped with proxies.

The return value is necessary in order to reflect changes to the object from the host to the context. Please note that setting a value in the field or deleting a field in the original object will not synchronize it.

#### `register(target: any, code: string | QuickJSHandle)`

Register a pair of objects that will be considered the same between the host and the QuickJS context.

#### `unregisterAll(targets: Iterable<[any, string | QuickJSHandle]>)`

Execute `register` methods for each pair.

#### `unregister(target: any)`

Unregister a pair of objects that were registered with `registeredObjects` option and `register` method.

#### `unregisterAll(targets: Iterable)`

Execute `unregister` methods for each target.

### `defaultRegisteredObjects: [any, string][]`

Default value of registeredObjects option of the Arena class constructor.

### `complexity(target: any, max?: number): number`

Measure the complexity of an object as you traverse the field and prototype chain. If max is specified, when the complexity reaches max, the traversal is terminated and it returns the max. In this function, one object and function are counted as a complexity of 1, and primitives are not counted as a complexity.

## Advanced

### How to work

quickjs-emscripten can execute JS code safely, but it requires to deal with a lot of handles and lifetimes. Also, when destroying the context, any un-destroyed handle will result in an error.

quickjs-emscripten-sync will automatically manage all handles once generated by QuickJS context in an Arena class.
And it will automatically "marshal" objects as handles and "unmarshal" handles as objects to enable seamless data exchange between the browser and QuickJS. It recursively traverses the object properties and prototype chain to transform objects. A function is called after its arguments and this arg are automatically converted for the environment in which the function is defined. The return value will be automatically converted to match the environment of the caller.
Most objects are wrapped by proxies during conversion, allowing "set" and "delete" operations on objects to be synchronized between the browser and QuickJS.

### Limitations

#### Class constructor

When initializing a new instance, it is not possible to fully proxy this arg (a.k.a. `new.target`) inside the class constructor. Therefore, after the constructor call, the fields set for this are re-set to this on the context side. Therefore, there may be some edge cases where the constructor may not work properly.

```js
class Cls {
constructor() {
this.hoge = "foo";
}
}

arena.expose({ Cls });
arena.evalCode(`new Cls()`); // Cls { hoge: "foo" }
```

#### Operation synchronization

For now, only the `set` and `deleteProperty` operations on objects are subject to synchronization. The result of `Object.defineProperty` on a proxied object will not be synchronized to the other side.

## License

[MIT License](LICENSE)