Hello and welcome to easy-web-worker with easy-cancelable-promise β your go-to solution for seamless Web Workers integration, now enhanced with cancelable promises! π
Web Workers are a native tool provided by JavaScript, allowing you to use them within your favorite framework like React, Angular, Vue, and others, or with pure JavaScript and TypeScript.
Check out the running example with React and TypeScript at CODEPEN; let's explore the capabilities of JavaScript's concurrent processing with Web Workers!"
[Important!] Starting from version 4.0.0, EasyWebWorker supports concurrency mode. This means you can now configure whether a single EasyWebWorker should use multiple Web Worker instances. This feature is extremely powerful for code that requires not only heavy computations occasionally but also robust concurrent processing. For more detailed information, please see the section below. concurrency mode
If you were previously using easy-web-worker with cancelable-promise-jq, please note that the cancelable-promise-jq package has been renamed/deprecated. To continue using the latest version of easy-web-worker, simply uninstall cancelable-promise-jq and replace all imports with easy-cancelable-promise.
I sincerely apologize for any inconvenience this may cause.
/**
* The callback parameter will be the body of the worker
*/
const worker = createEasyWebWorker(({ onMessage }) => {
const fibonacci = (n) => {
if (n <= 1) return n;
return fibonacci(n - 1) + fibonacci(n - 2);
};
/**
* Inside the worker we have to define an action when onMessage
*/
onMessage((message) => {
/**
* The payload includes whatever parameters are sent from the main thread
*/
const { payload } = message;
const result = fibonacci(payload.base);
message.resolve(result);
//message.reject(); // or reject
});
});
Then, for sending a message to the worker:
/**
* This returns a CancelablePromise
*/
await worker.send(40);
And that's it! You now have a worker running heavy computation in a real separate thread, with real asynchronous programming in JavaScript.
You can also create an easy web worker from a static file, or from a native worker instance, here various examples:
// on vite and typescript
import workerUrl from './worker?worker&url';
const isProduction = import.meta.env.MODE === 'production';
const workerSource = isProduction
? workerUrl
: new URL('./worker.ts', import.meta.url);
const worker = new EasyWebWorker(workerSource, {
workerOptions: {
type: 'module',
},
});
// or
const worker = createEasyWebWorker(workerSource, {
workerOptions: {
type: 'module',
},
});
// if you are using ESM - ECMAScript Modules
const worker = createEasyWebWorker([
new Worker(new URL('./worker.js', import.meta.url)),
]);
// other ways with vanilla js
const worker = createEasyWebWorker('./worker.js');
const worker = createEasyWebWorker(new Worker('./worker.js')); // Ζ Worker() { [native code] }
const worker = new EasyWebWorker('./worker.js');
const worker = new EasyWebWorker(new Worker('./worker.js')); // Ζ Worker() { [native code] }
When working with static files, which can offer substantial benefits with web workers, you simply need to create an instance of StaticEasyWebWorker.
The StaticEasyWebWorker provides an interface to continue working with easy-cancelable-promise and build more complex APIs within your worker.
From inside your worker, the message callbacks receive a message that includes multiple methods and functions. You can use these to communicate back with the main thread, or to subscribe to and react to the lifecycle of a worker.
const { onMessage } = new StaticEasyWebWorker();
/**
* For adding a default onMessage
*/
onMessage((message) => {
/** Your message receives a payload,
* which is any information sent from the main thread.*/
const { payload } = message;
const bigArrayBuffer = new new ArrayBuffer(1000000)();
/** You can resolve the message and respond to the main thread's promise that is listening.
* This promise could also send data back to the main thread and transfer data if needed */
message.resolve({ bigArrayBuffer }, [bigArrayBuffer]);
/** You can reject the message and send back a reason,
* if no transfer is necessary just avoid the second parameter */
message.reject(new Error('something happened'));
const metadata = { message: 'progress from inside the worker' };
/** You can report progress */
message.reportProgress(10, metadata, []); // all the methods allows you to send Transferable[]
/** You can cancel an operation from within the worker */
message.cancel('the operating was canceled from the worker');
/** You can subscribe to the cancellation event of the message,
* regardless of whether this cancellation is internal or external to the worker.*/
const unsubscribeCancel = message.onCancel((reason) => {});
// you can unsubscribe from the cancel event as well
unsubscribeCancel();
/**
* And there is extra listeners for the other events
* This events represent the lifecycle of the message
*/
const unsubscribeResolve = message.onResolve((data) => {});
const unsubscribeReject = message.onReject((data) => {});
const unsubscribeProgress = message.onProgress((data) => {});
const unsubscribeFinalize = message.onFinalize((data) => {});
/** You can also review the status of the message at any time*/
const status = message.getStatus(); // pending | resolved | rejected | canceled
});
/**
* For adding specific actions
*/
onMessage('readCSV', (message) => {
// do something
});
It's important to mention that the cancel method is the only one that provides two-way binding. It can travel all the way from the main thread to the worker, cancel something inside the worker, and notify the main thread upon completion.
easy-web-worker is designed to enhance the capabilities of the Worker class by integrating a pattern of cancelable promises from the easy-cancelable-promise library. For straightforward tasks, it simplifies the process by eliminating the need to configure webpack or other bundlers. And for more complex requirements, the StaticEasyWebWorker class allows the integration of easy worker and cancelable promises capabilities into your static workers.
Start enhancing your applications with robust, cancelable promises and easy web worker integration today! π
Experience it in action with a Live Example featuring text-diff π.
For a comprehensive understanding, watch our informative introduction video π₯. You can also dive deeper into the code and explore on easy-web-workers-examples π§©.
Creating a new worker is as simple as
const backgroundWorker = createEasyWebWorker<string, string>(({ onMessage }) => {
onMessage((message) => {
const { payload } = message;
message.resolve(`this is a message from the worker: ${payload}`);
});
// you could also define and send specific methods which allow you to create a better structured API
onMessage<number, number>('doSomething', (message) => {
const { payload } = message;
message.resolve(payload + 2);
});
});
// outside your worker
const messageResult = await backgroundWorker.send('hello!');
// for specific methods us the sendToMethod function
const messageResult2 = await backgroundWorker.sendToMethod('doSomething', 2);
EasyWebWorker<IPayload, IResult> has two generic parameters... They will affect the typing of the send() and response() methods.
Take into consideration that the workerBody is a template to create a worker in run time, so you'll not be able to use anything outside of the Worker-Scope.
const message = 'Hello';
await createEasyWebWorker<null, string>(({ onMessage }) => {
onMessage((message) => {
message.resolve(message); // THIS WILL PRODUCE AND ERROR!! the variable *message* will not exist in Worker-Scope.
});
}).send('hello!');
If you need to pass a primitive parameter to the body of the worker, you can use the primitiveParameters configuration. This is an array of values that will be serialized and embedded into the worker's body.
const message = 'Hello';
await createEasyWebWorker<null, string>(
({ onMessage }, context) => {
const [message] = context.primitiveParameters;
console.log(message); // "hello!" // π it works!
},
{
primitiveParameters: [message],
}
).send('hello!');
Take a look at Workers API if you don't know yet how they work: https://developer.mozilla.org/en-US/docs/Web/API/Web_Workers_API,
If you need t to send data to the worker, please define IPayload while creating a worker. new EasyWebWorker
When you defined an onMessage callback in your Worker, this will receive all messages from the send method:
easyWorker.onMessage((message) => {
// the *message* will be strongly typed with TS
// the message could resolve the *send* promise.
message.resolve();
// the message could be rejected from the worker
message.reject(new Error());
// this message could be cancelled from inside the worker
message.cancel();
// the message is also able to listen to cancelation evens
message.onCancel(() => {
// release resources
})
// you could also report progress to the principal thread if you configured a onProgress callback
message.reportProgress(25);
});
Let say you are performing some heavy process in your worker, but you still wanted to implement some kind of progress bar in the main thread... you could add an onProgress callback.
await worker.send().onProgress((progress: number) => {
// change some progress bar percentage
}).then(doSomething);
onProgress Is gonna be executed every time you call message.reportProgress inside the worker... the cool part here is that the reportProgress is not gonna finish the main promise returned by the send method.
As WorkerBody are just templates, you could reuse them on other Workers, or use them as plugins for your Workers. Let's see:
const reusableWorkerSegment: EasyWebWorkerBody = ({ onMessage, close, importScripts }, context) => {
context.doSomething = () => Promise.resolve('This is a plugin example');
};
const reusableWorkerSegment = await createEasyWebWorker([
reusableWorkerSegment,
({ onMessage }, context) => onMessage(async (message) => {
// context will have all stuff we added on other plugins
const result = await context.doSomething();
message.resolve(result);
})]).send();
In this way, you could avoid having to create more than once the same template for your worker.
Web Workers has this amazing method called importScripts, are you passed an array of strings in the EasyWorker extra configuration, all those files are gonna be imported into your worker.
// test.js
self.message = 'Hello coders!';
self.doSomething = () => console.log(self.message);
await createEasyWebWorker(({ onMessage }, context) => {
onMessage((message) => context.doSomething());
}, {
scripts: ['http://localhost:3000/test.js'],
}).send();
This is a very simple example, but you could import a whole library into your worker, as JQUERY, Bluebird for example
If you want to create a Worker with a static .js file and don't want to lose the structure of messages and promises and the onProgress callback from the library... you could use StaticEasyWebWorker
let's see how to use it:
// worker.js // This is gonna be the content of your worker // onMessage Callback is gonna receive all send method calls.
// this is gonna create the same message structure the runtime Workers
const { onMessage } = createStaticEasyWebWorker();
onMessage((message) => {
setTimeout(() => {
message.resolve(200);
}, 5000);
});
onMessage('action', (message) => {
setTimeout(() => {
message.resolve(200);
}, 5000);
});
By the way, if you're in need of a super simple static worker, just know that the first parameter of createStaticEasyWebWorker is a function which will be used as the default onmessage callback.
createStaticEasyWebWorker((message) => {
// this is the default onMessage
});
and in your main thread:
const worker = createEasyWebWorker<null,number>('./worker.js');
await worker.send();
Super easy right?
With EasyWebWorker, you can create operations that require heavy concurrency and delegate them to a web workers queue, or create workers on demand depending on the traffic and specific tasks. Let's take a look:
/**
* Notice that the structure of the worker remains the same;
* the only changes are in the configuration parameters of the worker.
* Take a look below.*/
const worker = createEasyWebWorker(({ onMessage }) => {
onMessage((message) => {
const { payload } = message;
// heavy computation like fibonacci
message.resolve(result);
});
}, {
// We will now scale up to four workers if necessary.
maxWorkers: 4
});
By default, creating an EasyWebWorker also creates a single native JavaScript worker. Without any added configuration, this Worker instance will remain active unless it is programmatically disposed. However, by modifying the maxWorkers parameter, you can control the number of native workers used, allowing the EasyWebWorker to execute multiple messages across multiple threads.
You can also control whether these additional workers should be created on demand when messages are sent to the EasyWebWorker. This can be done along with setting the terminationDelay, which indicates how long to wait before disposing of a Worker to avoid unnecessary resource consumption. Alternatively, you can choose to warmUp and keep the Workers alive from the moment the EasyWebWorker is created.
From the main thread, the consumption of the worker remains the same. Depending on the configuration, the workers will be created either statically or on-demand as needed. The EasyWebWorker will be responsible for selecting them from a pool. Additionally, the EasyWebWorker will manage the distribution of messages among the available workers.
// Since maxWorkers is configured to 4, a total of 3 native workers, will be created.
const results = await Promise.all([
worker.send(payload),
worker.send(payload1),
worker.send(payload2),
]);
In the previous example, since the warmUp parameter wasnβt included, only 3 Native workers will be created to handle the 3 messages, even though the maximum available is 4. Additionally, since the keepAlive parameter was not included, if a total of 1 second passes without new messages, the Native workers will be disposed of to save resources.
Here is an example of how you could easily create data filter into a Worker, to avoid performing loops process into the main thread that could end affecting user experience.
interface FilterSource {
filter: string,
collection: any[],
reportProgress: boolean,
}
const worker = createEasyWebWorker<FilterSource, any[]>(({ onMessage }) => {
const containsValue = (item: any, filter: string): boolean => {
const itemKeys = Object.keys(item);
return itemKeys.some((key) => {
const prop = item[key] || null;
if (typeof prop !== 'string' && Object.keys(prop).length) return containsValue(prop, filter);
if (prop.toString().replace(/(\r\n|\n|\r)/gm, '').trim().toLowerCase()
.indexOf(filter) !== -1) return true;
return false;
});
};
onMessage((message: IEasyWebWorkerMessage<FilterSource, any[]>) => {
const { payload } = message;
const { collection, filter = '', reportProgress: countProgress } = payload;
const { length: collectionLength } = collection;
const result = filter === '' ? collection : [];
const progressPerItem = collectionLength ? 100 / collectionLength : 0;
let currentProgress = 0;
if (filter) {
for (let index = 0; index < collectionLength; index += 1) {
if (countProgress) {
currentProgress += progressPerItem;
message.reportProgress(currentProgress);
}
const item = collection[index];
if (containsValue(item, filter)) result.push(item);
}
}
message.resolve(result);
});
});
And how to use this?
worker.send({
collection: [{ name: 'julio perez' }, { name: 'carol starling' }, { name: 'goku' }, { name: { firstname: 'johnny' } }],
filter: 'johnny',
reportProgress: true,
}).onProgress((progressPercentage) => console.log(progressPercentage))
.then((filtered: any[]) => console.log(filtered));
the output should be: => 25 => 50 => 75 => 100 => [{ name: { firstname: 'johnny' } }]
Of course this is a very tiny array, but is just to give you and idea, actually you also could make fetch requests into workers... give it a try.
EasyWebWorker.reboot(reason?: unknown): CancelableCancelablePromise<void>[]
This method will reboot the worker and cancel all the messages in the queue.
reason
- (optional) reason why the worker will be restarted.Returns an array of promises that are resolved with the rejection reason provided when the messages are canceled.
Example usage:
const worker = createEasyWebWorker<string, string>(({ onMessage }) => {
onMessage((message) => {
message.resolve(`Received message: ${message.payload}`);
});
});
const messagePromise = worker.send('Hello!');
worker.reboot('Worker was restarted');
// The message promise will be rejected with the reason 'Worker was restarted'
Cancel all current messages and send a new one.
Cancels all messages that are currently waiting to be processed by the worker.
reason
- (optional) The reason for the cancellation.Returns an array of promises that are resolved with the rejection reason provided when the messages are canceled.
Cancel all the messages but the current execution and add a new message
Sends a message to the worker
payload
- (optional) The message payload.options
- (optional) Additional send options.Thanks for reading, hope this help someone