Comlink
Comlink’s goal is to make WebWorkers enjoyable. Instead of using postMessage
to send messages back and forth, Comlink allows you to invoke functions, pass callbacks, add event listeners and create new instances of classes.
const MyClass = Comlink.proxy(new Worker('worker.js'));
const instance = await new MyClass();
await instance.logSomething();
const myValue = 42;
class MyClass {
logSomething() {
console.log(`myValue = ${myValue}`);
}
}
Comlink.expose(MyClass, self);
Browsers support & bundle size
Size: ~3.9k, ~1.6k gzip’d
Introduction
WebWorkers are a web API that allow you to run code in a separate thread. To communicate with another thread, WebWorkers offer the postMessage
API. You can send messages in form of transferable JavaScript objects using myWorker.postMessage(someObject)
, triggering a message
event inside the worker.
Comlink turns this messaged-based API into a something more developer-friendly: Values from one thread can be used within the other thread (and vice versa) just like local values.
Comlink can be used with anything that offers postMessage
like windows, iframes and ServiceWorkers.
Download
You can download Comlink from the dist folder. Alternatively, you can
install it via npm
$ npm install --save comlinkjs
or use a CDN like delivrjs:
https://cdn.jsdelivr.net/npm/comlinkjs@2.3.0/comlink.es6.min.js
Examples
There’s a collection of examples in the examples directory.
Module formats
The Comlink module is provided in 3 different formats:
-
“es6”: This package uses the native ES6 module format. Import it as follows:
import {Comlink} from '../dist/comlink.es6.js';
-
“global”: This package adds Comlink
to the global scope (i.e. self
). Useful for workers or projects without a module loader.
-
“umd”: This package uses UMD so it is compatible with AMD, CommonJS and requireJS.
API
The Comlink module exports 3 functions:
Comlink.proxy(endpoint)
Returns the value that is exposed on the other side of endpoint
.
proxy
creates an ES6 proxy and sends all operations performed on that proxy through endpoint
. endpoint
can be a Window
, a Worker
or a MessagePort
.* The other endpoint of the channel should be passed to Comlink.expose
.
If you invoke function, all parameters will be structurally cloned or transferred if they are transferable. If you want to pass a function as a parameters (e.g. callbacks), make sure to use proxyValue
(see below). Same applies to the return value of a function.
*) Technically it can be any object with postMessage
, addEventListener
and
removeEventListener
.
Comlink.expose(obj, endpoint)
Exposes obj
to endpoint
. Use Comlink.proxy
on the other end of endpoint
.
expose
is the counter-part to proxy
. It listens for RPC messages on endpoint
and applies the operations to obj
. Return values of functions will be structurally cloned or transfered if they are transferable.
Comlink.proxyValue(value)
Makes sure a parameter or return value is proxied, not copied.
By default, all parameters to a function are copied (structural clone):
const api = Comlink.proxy(new Worker('worker.js'));
const obj = {x: 0};
await api.setXto4(obj);
console.log(obj.x);
The worker receives a copy of obj
, so any mutation of obj
done by the worker won’t affect the original object. If the value should not be copied but instead be proxied, use Comlink.proxyValue
:
- await api.setXto4(obj);
+ await api.setXto4(Comlink.proxyValue(obj));
console.log(obj.x)
will now log 4.
Keep in mind that functions cannot be copied. Unless they are used in combination with Comlink.proxyValue
, they will get discarded during copy.
License Apache-2.0