@fast-check/worker v0.5.0
@fast-check/worker
Provide built-ins to run predicates directly within dedicated workers
Why?
fast-check
alone is great but what if it led your code into an infinite and synchronous loop for such inputs? In such case, it would neither be able to shrink the issue, nor to report any for you as the single threaded philosophy at the root of JavaScript will prevent it from anything except waiting for the main thread to come back.
This package tends to provide a way to run easily your properties within dedicated workers automatically spawed by it.
Example
Here are some of the changes you will have to do:
- hoist properties so that they get declared on the root scope
- replace
fc.property
byproperty
coming frompropertyFor(<path-to-file>)
- replace
fc.assert
byassert
coming from@fast-check/worker
for automatic cleaning of the workers as the test ends - transpilation has not been addressed yet but it may probably work
- in theory, if you were only using
propertyFor
andassert
without any external framework fortest
,it
and others, the separation of the property from the assertion would be useless as the check for main thread is fully handled within@fast-check/worker
itself so no hoisting needing in such case
import { test } from '@jest/globals';
import fc from 'fast-check';
import { isMainThread } from 'node:worker_threads';
import { assert, propertyFor } from '@fast-check/worker';
const property = propertyFor(new URL(import.meta.url)); // or propertyFor(pathToFileURL(__filename)) in commonjs
const p1 = property(fc.nat(), fc.nat(), (start, end) => {
// starting a possibly infinite loop
for (let i = start; i !== end; ++i) {
// doing stuff...
}
});
if (isMainThread) {
test('should assess p1', async () => {
await assert(p1, { timeout: 1000 });
});
}
Refer to the tests defined test/main.spec.ts
for a living example of how you can use this package with a test runner such as Jest.
Extra options
The builder of properties propertyFor
accepts a second parameter to customize how the workers will behave. By default, workers will be shared across properties. In case you want a more isolation between your runs, you can use:
const property = propertyFor(new URL(import.meta.url), { isolationLevel: 'predicate' });
// Other values:
// - "file": Re-use workers cross properties (default)
// - "property": Re-use workers for each run of the predicate. Not shared across properties!
// - "predicate": One worker per run of the predicate
By default, workers will receive the generated values from their parent thread. In some cases, such sending is made impossible as the generated values include non-serializable pieces. In such cases, you can opt-in to generate the values directly within the workers by using:
const property = propertyFor(new URL(import.meta.url), { randomSource: 'worker' });
// Other values:
// - "main-thread": The main thread will be responsible to generate the random values and send them to the worker thread. It unfortunately cannot send any value that cannot be serialized between threads. (default)
// - "worker": The worker is responsible to generate its own values based on the instructions provided by the main thread. Switching to a worker mode allows to support non-serializable values, unfortunately it drops all shrinking. capabilities.
Minimal requirements
- Node ≥14.18.0(1)(2)(3)
- TypeScript ≥4.1 (optional)
(1): worker_threads
alone would only require Node ≥10.5.0, but our usage of require(node:*)
forces us to request at least Node ≥14.18.0
(2): this package targets ES2020 specification which is quite well supported (more than 94%) by any Node ≥14.5.0
(3): this package requires a version of node able to understand package.json#exports
, supported by any Node ≥12.17.0