Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/cloudnc/observable-webworker
Simplified API for working with Web Workers with RxJS
https://github.com/cloudnc/observable-webworker
Last synced: about 2 months ago
JSON representation
Simplified API for working with Web Workers with RxJS
- Host: GitHub
- URL: https://github.com/cloudnc/observable-webworker
- Owner: cloudnc
- License: mit
- Created: 2019-06-08T22:47:58.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2024-06-17T20:10:37.000Z (7 months ago)
- Last Synced: 2024-11-08T21:40:56.852Z (2 months ago)
- Language: TypeScript
- Homepage: https://cloudnc.github.io/observable-webworker
- Size: 6.31 MB
- Stars: 226
- Watchers: 7
- Forks: 15
- Open Issues: 8
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-angular - observable-webworker - Simplified API for working with Web Workers with RxJS. (Table of contents / Third Party Components)
- fucking-awesome-angular - observable-webworker - Simplified API for working with Web Workers with RxJS. (Table of contents / Third Party Components)
README
# Observable Webworker
Simple API for using [web workers](https://developer.mozilla.org/en-US/docs/Web/API/Web_Workers_API/Using_web_workers) with [RxJS](https://rxjs-dev.firebaseapp.com/guide/overview) observables
[![Strict TypeScript Checked](https://badgen.net/badge/TS/Strict "Strict TypeScript Checked")](https://www.typescriptlang.org)
[![npm version](https://badge.fury.io/js/observable-webworker.svg)](https://www.npmjs.com/package/observable-webworker)
[![Build Status](https://github.com/cloudnc/observable-webworker/workflows/CI/badge.svg)](https://github.com/cloudnc/observable-webworker/actions)
[![codecov](https://codecov.io/gh/cloudnc/observable-webworker/branch/master/graph/badge.svg)](https://codecov.io/gh/cloudnc/observable-webworker)
[![Commitizen friendly](https://img.shields.io/badge/commitizen-friendly-brightgreen.svg)](https://commitizen.github.io/cz-cli/)
[![License](https://img.shields.io/github/license/cloudnc/observable-webworker)](https://raw.githubusercontent.com/cloudnc/observable-webworker/master/LICENSE)
![npm peer dependency version](https://img.shields.io/npm/dependency-version/observable-webworker/peer/rxjs)[![NPM](https://nodei.co/npm/observable-webworker.png?compact=true)](https://nodei.co/npm/observable-webworker/)
# Features
- Simple `fromWorker` function from main thread side
- Fully RxJS interfaces allowing both main thread and worker thread streaming
- Error handling across the thread boundaries is propagated
- Under the hood `materialize` and `dematerialize` is used as a robust transport of streaming errors
- Automatic handling of worker termination on main thread unsubscription of observable
- Framework agnostic - while the demo uses Angular, the only dependency is rxjs, so React, Vue or plain old JS is
completely compatible
- Fully compatible with [Webpack worker-plugin](https://github.com/GoogleChromeLabs/worker-plugin)
- Therefore compatible with [Angular webworker bundling](https://angular.io/guide/web-worker) which uses this
- Class interface based worker creation (should be familiar API for Angular developers)
- Unopinionated on stream switching behavior, feel free to use `mergeMap`, `switchMap` or `exhaustMap` in your worker if
the input stream contains multiple items that generate their own stream of results
- Built in interfaces for handling [`Transferable`](https://developer.mozilla.org/en-US/docs/Web/API/Transferable) parts
of message payloads so large binaries can transferred efficiently without copying - See [Transferable](#transferable)
section for usage
- Automatic destruction of worker on unsubscription of output stream, this allows for smart cancelling of computation
using `switchMap` operator, or parallelisation of computation with `mergeMap`
- [Worker Pool strategy](#worker-pool-strategy) - maximise the throughput of units of work by utilising all cores on the host machine## Demo
https://cloudnc.github.io/observable-webworker## Tutorial
https://dev.to/zakhenry/observable-webworkers-with-angular-8-4k6## Articles
* [Observable Web Workers, a deep dive into a realistic use case](https://dev.to/zakhenry/observable-web-workers-a-deep-dive-into-a-realistic-use-case-4042)
* [Parallel computation in the browser with observable webworkers](https://dev.to/zakhenry/parallel-computation-in-the-browser-with-observable-webworkers-hci)## Install
Install the [npm package](https://www.npmjs.com/package/observable-webworker): `observable-webworker`
```sh
# with npm
npm install observable-webworker
# or with yarn
yarn add observable-webworker
```## Usage
### Quickstart
#### Main Thread
💡 Take note! The webworker construction syntax differs for different version of webpack:
#### Webpack < 5 (deprecated)
```ts
// src/readme/hello-legacy-webpack.tsimport { fromWorker } from 'observable-webworker';
import { of } from 'rxjs';const input$ = of('Hello from main thread');
fromWorker(() => new Worker('./hello.worker', { type: 'module' }), input$).subscribe(message => {
console.log(message); // Outputs 'Hello from webworker'
});```
#### Webpack 5```ts
// src/readme/hello.ts#L2-L12import { fromWorker } from 'observable-webworker';
import { of } from 'rxjs';const input$ = of('Hello from main thread');
fromWorker(
() => new Worker(new URL('./hello.worker', import.meta.url), { type: 'module' }),
input$,
).subscribe(message => {
console.log(message); // Outputs 'Hello from webworker'
});
```#### Worker Thread
```ts
// src/readme/hello.worker.tsimport { DoWork, runWorker } from 'observable-webworker';
import { Observable } from 'rxjs';
import { map } from 'rxjs/operators';export class HelloWorker implements DoWork {
public work(input$: Observable): Observable {
return input$.pipe(
map(message => {
console.log(message); // outputs 'Hello from main thread'
return `Hello from webworker`;
}),
);
}
}runWorker(HelloWorker);
```
#### Decorator deprecation notice
Future versions of webpack (Webpack 5) minify webworkers overly aggressively, causing
the `@ObservableWorker()` decorator pattern to no longer function. This decorator
has now been deprecated, and will be removed in the next major version of this library.To migrate from decorators, simply remove the decorator, and invoke the `runWorker`
with your class passed as argument (see example above).Make sure you **don't forget to remove the decorator** when you add the `runWorker(...)`
function, otherwise the worker will be run twice, each acting on any message sent.## Transferable
If either your input or output (or both!) streams are passing large messages to or from the worker, it is highly
recommended to use message types that implement the [Transferable](https://developer.mozilla.org/en-US/docs/Web/API/Transferable)
interface (`ArrayBuffer`, `MessagePort`, `ImageBitmap`).Bear in mind that when transferring a message to a webworker that the main thread relinquishes ownership of the data.
Recommended reading:
- https://developer.mozilla.org/en-US/docs/Web/API/Transferable
- https://developer.mozilla.org/en-US/docs/Web/API/Worker/postMessageTo use `Transferable`s with observable-worker, a slightly more complex interface is provided for both sides of the
main/worker thread.If the main thread is transferring `Transferable`s _to the worker_, simply add a callback to the `fromWorker` function
call to select which elements of the input stream are transferable.```ts
// src/readme/transferable.main.ts#L7-L11return fromWorker(
() => new Worker(new URL('./transferable.worker', import.meta.url), { type: 'module' }),
input$,
input => [input],
);
```If the worker is transferring `Transferable`s _to the main thread_ simply implement `DoTransferableWork`, which will
require you to add an additional method `selectTransferables?(output: O): Transferable[];` which you implement to select
which elements of the output object are `Transferable`.Both strategies are compatible with each other, so if for example you're computing the hash of a large `ArrayBuffer` in
a worker, you would only need to use add the transferable selector callback in the main thread in order to mark the
`ArrayBuffer` as being transferable in the input. The library will handle the rest, and you can just use `DoWork` in the
worker thread, as the return type `string` is not `Transferable`.## Worker Pool Strategy
If you have a large amount of work that needs to be done, you can use the `fromWorkerPool` function to automatically
manage a pool of workers to allow true concurrency of work, distributed evenly across all available cores.The worker pool strategy has the following features
* Work can be provided as either `Observable`, `Array`, or `Iterable`
* Concurrency is limited to `navigation.hardwareConcurrency - 1` to keep the main core free.
* This is a configurable option if you know you already have other workers running
* Workers are only created when there is need for them (work is available)
* Workers are terminated when there is no more work, freeing up threads for other processes
* for `Observable`, work is considered remaining while the observable is not completed
* for `Array`, work remains while there are items in the array
* for `Iterable`, work remains while the iterator is not `result.done`
* Workers are kept running while work remains, preventing unnecessary downloading of the worker script
* Custom observable flattening operator can be passed, allowing for custom behaviour such as correlating the output
order with input order
* default operator is `mergeAll()`, which means the output from the webworker(s) is output as soon as available
### ExampleIn this simple example, we have a function that receives an array of files and returns an observable of the MD5 sum
hashes of those files. For simplicity, we're passing the primitives back and forth, however in reality you are likely to
want to construct your own interface to define the messages being passed to and from the worker.#### Main Thread
```ts
// src/readme/worker-pool.main.tsimport { Observable } from 'rxjs';
import { fromWorkerPool } from 'observable-webworker';export function computeHashes(files: File[]): Observable {
return fromWorkerPool(
() => new Worker(new URL('./worker-pool-hash.worker', import.meta.url), { type: 'module' }),
files,
);
}```
#### Worker Thread
```ts
// src/readme/worker-pool-hash.worker.tsimport * as md5 from 'js-md5';
import { DoWorkUnit, runWorker } from 'observable-webworker';
import { Observable } from 'rxjs';
import { map } from 'rxjs/operators';export class WorkerPoolHashWorker implements DoWorkUnit {
public workUnit(input: File): Observable {
return this.readFileAsArrayBuffer(input).pipe(map(arrayBuffer => md5(arrayBuffer)));
}private readFileAsArrayBuffer(blob: Blob): Observable {
return new Observable(observer => {
if (!(blob instanceof Blob)) {
observer.error(new Error('`blob` must be an instance of File or Blob.'));
return;
}const reader = new FileReader();
reader.onerror = err => observer.error(err);
reader.onload = () => observer.next(reader.result as ArrayBuffer);
reader.onloadend = () => observer.complete();reader.readAsArrayBuffer(blob);
return () => reader.abort();
});
}
}runWorker(WorkerPoolHashWorker);
```
Note here that the worker class `implements DoWorkUnit`. This is different to before where we implemented
`DoWork` which had the slightly more complex signature of inputting an observable and outputting one.If using the `fromWorkerPool` strategy, you must only implement `DoWorkUnit` as it relies on the completion of the
returned observable to indicate that the unit of work is finished processing, and the next unit of work can be
transferred to the worker.Commonly, a worker that implements `DoWorkUnit` only needs to return a single value, so you may instead return a `Promise`
from the `workUnit` method.```ts
// src/app/doc/async-work.worker.ts#L7-L14export class FactorizationWorker implements DoWorkUnit {
public async workUnit(input: number): Promise {
return factorize(input);
}
}runWorker(FactorizationWorker);
```
---
Are you using observable-webworker? [Tell us about it!](https://github.com/cloudnc/observable-webworker/discussions/69) We'd love to hear about the weird and wonderful ways developers are working with streaming workers.