node/test/parallel/test-abortsignal-any.mjs
Antoine du Hamel bae14b7914
test: do not set concurrency on parallelized runs
Our CI already run test files in parallel, having `node:test` spawns
child processes concurrently could lead to oversubscribing the CI
machine. This commit sets the `concurrency` depending
on the presence of `TEST_PARALLEL` in the env, so running the test
file individually still spawns child processes concurrently, and
running the whole test suite does not oversubscribe the machine.

PR-URL: https://github.com/nodejs/node/pull/52177
Reviewed-By: Vinícius Lourenço Claro Cardoso <contact@viniciusl.com.br>
Reviewed-By: Luigi Pinca <luigipinca@gmail.com>
Reviewed-By: Marco Ippolito <marcoippolito54@gmail.com>
2024-03-23 21:11:28 +00:00

105 lines
4.1 KiB
JavaScript

import * as common from '../common/index.mjs';
import { describe, it } from 'node:test';
import { once } from 'node:events';
import assert from 'node:assert';
describe('AbortSignal.any()', { concurrency: !process.env.TEST_PARALLEL }, () => {
it('should throw when not receiving an array', () => {
const expectedError = { code: 'ERR_INVALID_ARG_TYPE' };
assert.throws(() => AbortSignal.any(), expectedError);
assert.throws(() => AbortSignal.any(null), expectedError);
assert.throws(() => AbortSignal.any(undefined), expectedError);
});
it('should throw when input contains non-signal values', () => {
assert.throws(
() => AbortSignal.any([AbortSignal.abort(), undefined]),
{
code: 'ERR_INVALID_ARG_TYPE',
message: 'The "signals[1]" argument must be an instance of AbortSignal. Received undefined'
},
);
});
it('creates a non-aborted signal for an empty input', () => {
const signal = AbortSignal.any([]);
assert.strictEqual(signal.aborted, false);
signal.addEventListener('abort', common.mustNotCall());
});
it('returns a new signal', () => {
const originalSignal = new AbortController().signal;
const signalAny = AbortSignal.any([originalSignal]);
assert.notStrictEqual(originalSignal, signalAny);
});
it('returns an aborted signal if input has an aborted signal', () => {
const signal = AbortSignal.any([AbortSignal.abort('some reason')]);
assert.strictEqual(signal.aborted, true);
assert.strictEqual(signal.reason, 'some reason');
signal.addEventListener('abort', common.mustNotCall());
});
it('returns an aborted signal with the reason of first aborted signal input', () => {
const signal = AbortSignal.any([AbortSignal.abort('some reason'), AbortSignal.abort('another reason')]);
assert.strictEqual(signal.aborted, true);
assert.strictEqual(signal.reason, 'some reason');
signal.addEventListener('abort', common.mustNotCall());
});
it('returns the correct signal in the event target', async () => {
const signal = AbortSignal.any([AbortSignal.timeout(5)]);
const interval = setInterval(() => {}, 100000); // Keep event loop alive
const [{ target }] = await once(signal, 'abort');
clearInterval(interval);
assert.strictEqual(target, signal);
assert.ok(signal.aborted);
assert.strictEqual(signal.reason.name, 'TimeoutError');
});
it('aborts with reason of first aborted signal', () => {
const controllers = Array.from({ length: 3 }, () => new AbortController());
const combinedSignal = AbortSignal.any(controllers.map((c) => c.signal));
controllers[1].abort(1);
controllers[2].abort(2);
assert.ok(combinedSignal.aborted);
assert.strictEqual(combinedSignal.reason, 1);
});
it('can accept the same signal more than once', () => {
const controller = new AbortController();
const signal = AbortSignal.any([controller.signal, controller.signal]);
assert.strictEqual(signal.aborted, false);
controller.abort('reason');
assert.ok(signal.aborted);
assert.strictEqual(signal.reason, 'reason');
});
it('handles deeply aborted signals', async () => {
const controllers = Array.from({ length: 2 }, () => new AbortController());
const composedSignal1 = AbortSignal.any([controllers[0].signal]);
const composedSignal2 = AbortSignal.any([composedSignal1, controllers[1].signal]);
composedSignal2.onabort = common.mustCall();
controllers[0].abort();
assert.ok(composedSignal2.aborted);
assert.ok(composedSignal2.reason instanceof DOMException);
assert.strictEqual(composedSignal2.reason.name, 'AbortError');
});
it('executes abort handlers in correct order', () => {
const controller = new AbortController();
const signals = [];
signals.push(controller.signal);
signals.push(AbortSignal.any([controller.signal]));
signals.push(AbortSignal.any([controller.signal]));
signals.push(AbortSignal.any([signals[0]]));
signals.push(AbortSignal.any([signals[1]]));
let result = '';
signals.forEach((signal, i) => signal.addEventListener('abort', () => result += i));
controller.abort();
assert.strictEqual(result, '01234');
});
});