2017-06-02 01:07:20 +08:00
|
|
|
// Flags: --inspect=0
|
|
|
|
'use strict';
|
|
|
|
const common = require('../common');
|
2017-07-05 11:35:50 +08:00
|
|
|
const assert = require('assert');
|
2017-06-02 01:07:20 +08:00
|
|
|
|
|
|
|
// With the current behavior of Node.js (at least as late as 8.1.0), this
|
|
|
|
// test fails with the following error:
|
|
|
|
// `AssertionError [ERR_ASSERTION]: worker 2 failed to bind port`
|
|
|
|
// Ideally, there would be a way for the user to opt out of sequential port
|
|
|
|
// assignment.
|
|
|
|
//
|
|
|
|
// Refs: https://github.com/nodejs/node/issues/13343
|
|
|
|
|
2017-07-05 11:35:50 +08:00
|
|
|
// This following check should be replaced by common.skipIfInspectorDisabled()
|
|
|
|
// if moved out of the known_issues directory.
|
2019-01-30 23:19:45 +08:00
|
|
|
if (!process.features.inspector) {
|
2017-07-05 11:35:50 +08:00
|
|
|
// When the V8 inspector is disabled, using either --without-inspector or
|
|
|
|
// --without-ssl, this test will not fail which it is expected to do.
|
|
|
|
// The following fail will allow this test to be skipped by failing it.
|
|
|
|
assert.fail('skipping as V8 inspector is disabled');
|
|
|
|
}
|
2017-06-02 01:07:20 +08:00
|
|
|
|
|
|
|
const cluster = require('cluster');
|
|
|
|
const net = require('net');
|
|
|
|
|
|
|
|
const ports = [process.debugPort];
|
|
|
|
const clashPort = process.debugPort + 2;
|
|
|
|
function serialFork() {
|
|
|
|
return new Promise((res) => {
|
|
|
|
const worker = cluster.fork();
|
|
|
|
worker.on('error', (err) => assert.fail(err));
|
2019-03-07 08:03:53 +08:00
|
|
|
// No common.mustCall since 1 out of 3 should fail.
|
2017-06-02 01:07:20 +08:00
|
|
|
worker.on('online', () => {
|
|
|
|
worker.on('message', common.mustCall((message) => {
|
|
|
|
ports.push(message.debugPort);
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
worker.on('exit', common.mustCall((code, signal) => {
|
|
|
|
assert.strictEqual(signal, null);
|
|
|
|
// worker 2 should fail because of port clash with `server`
|
|
|
|
if (code === 12) {
|
|
|
|
return assert.fail(`worker ${worker.id} failed to bind port`);
|
|
|
|
}
|
2018-10-13 00:55:49 +08:00
|
|
|
assert.strictEqual(code, 0);
|
2017-06-02 01:07:20 +08:00
|
|
|
}));
|
|
|
|
worker.on('disconnect', common.mustCall(res));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-12-11 05:53:44 +08:00
|
|
|
if (cluster.isPrimary) {
|
2017-06-02 01:07:20 +08:00
|
|
|
cluster.on('online', common.mustCall((worker) => worker.send('dbgport'), 2));
|
|
|
|
|
2019-03-07 08:03:53 +08:00
|
|
|
// Block one of the ports with a listening socket.
|
2017-06-02 01:07:20 +08:00
|
|
|
const server = net.createServer();
|
|
|
|
server.listen(clashPort, common.localhostIPv4, common.mustCall(() => {
|
2019-03-07 08:03:53 +08:00
|
|
|
// Try to fork 3 workers. No.2 should fail.
|
2017-06-02 01:07:20 +08:00
|
|
|
Promise.all([serialFork(), serialFork(), serialFork()])
|
|
|
|
.then(common.mustNotCall())
|
|
|
|
.catch((err) => console.error(err));
|
|
|
|
}));
|
|
|
|
server.unref();
|
|
|
|
} else {
|
|
|
|
const sentinel = common.mustCall();
|
|
|
|
process.on('message', (message) => {
|
|
|
|
if (message !== 'dbgport') return;
|
|
|
|
process.send({ debugPort: process.debugPort });
|
|
|
|
sentinel();
|
|
|
|
process.disconnect();
|
|
|
|
});
|
|
|
|
}
|