tty: add ref() so process.stdin.ref() etc. work
Also squashed from: * test: move tty-wrap isrefed test to pseudo-tty/ * test: test tty-wrap handle isrefed properly * test: improve failure messages in isrefed tests PR-URL: https://github.com/nodejs/node/pull/7360 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Sam Roberts <vieuxtech@gmail.com> Reviewed-By: James M Snell <jasnell.gmail.com> Reviewed-By: Jeremiah Senkpiel <fishrock123@rocketmail.com>
This commit is contained in:
parent
d08836003c
commit
3c92200d8b
@ -34,6 +34,7 @@ void TTYWrap::Initialize(Local<Object> target,
|
|||||||
|
|
||||||
env->SetProtoMethod(t, "close", HandleWrap::Close);
|
env->SetProtoMethod(t, "close", HandleWrap::Close);
|
||||||
env->SetProtoMethod(t, "unref", HandleWrap::Unref);
|
env->SetProtoMethod(t, "unref", HandleWrap::Unref);
|
||||||
|
env->SetProtoMethod(t, "ref", HandleWrap::Ref);
|
||||||
env->SetProtoMethod(t, "hasRef", HandleWrap::HasRef);
|
env->SetProtoMethod(t, "hasRef", HandleWrap::HasRef);
|
||||||
|
|
||||||
StreamWrap::AddMethods(env, t, StreamBase::kFlagNoShutdown);
|
StreamWrap::AddMethods(env, t, StreamBase::kFlagNoShutdown);
|
||||||
|
@ -105,6 +105,11 @@ On how to run tests in this direcotry, see
|
|||||||
<td>Yes</td>
|
<td>Yes</td>
|
||||||
<td>Various tests that are able to be run in parallel.</td>
|
<td>Various tests that are able to be run in parallel.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>pseudo-tty</td>
|
||||||
|
<td>Yes</td>
|
||||||
|
<td>Tests that require stdin/stdout/stderr to be a TTY.</td>
|
||||||
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>pummel</td>
|
<td>pummel</td>
|
||||||
<td>No</td>
|
<td>No</td>
|
||||||
|
@ -1,33 +0,0 @@
|
|||||||
'use strict';
|
|
||||||
|
|
||||||
const common = require('../common');
|
|
||||||
const strictEqual = require('assert').strictEqual;
|
|
||||||
const spawn = require('child_process').spawn;
|
|
||||||
|
|
||||||
function makeAssert(message) {
|
|
||||||
return function(actual, expected) {
|
|
||||||
strictEqual(actual, expected, message);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
const assert = makeAssert('hasRef() not working on tty_wrap');
|
|
||||||
|
|
||||||
if (process.argv[2] === 'child') {
|
|
||||||
// Test tty_wrap in piped child to guarentee stdin being a TTY.
|
|
||||||
const ReadStream = require('tty').ReadStream;
|
|
||||||
const tty = new ReadStream(0);
|
|
||||||
assert(Object.getPrototypeOf(tty._handle).hasOwnProperty('hasRef'), true);
|
|
||||||
assert(tty._handle.hasRef(), true);
|
|
||||||
tty.unref();
|
|
||||||
assert(tty._handle.hasRef(), false);
|
|
||||||
tty._handle.close(
|
|
||||||
common.mustCall(() => assert(tty._handle.hasRef(), false)));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use spawn so that we can be sure that stdin has a _handle property.
|
|
||||||
// Refs: https://github.com/nodejs/node/pull/5916
|
|
||||||
const proc = spawn(process.execPath, [__filename, 'child'], { stdio: 'pipe' });
|
|
||||||
proc.stderr.pipe(process.stderr);
|
|
||||||
proc.on('exit', common.mustCall(function(exitCode) {
|
|
||||||
process.exitCode = exitCode;
|
|
||||||
}));
|
|
@ -3,99 +3,128 @@
|
|||||||
const common = require('../common');
|
const common = require('../common');
|
||||||
const strictEqual = require('assert').strictEqual;
|
const strictEqual = require('assert').strictEqual;
|
||||||
|
|
||||||
function makeAssert(message) {
|
|
||||||
return function(actual, expected) {
|
|
||||||
strictEqual(actual, expected, message);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// child_process
|
// child_process
|
||||||
{
|
{
|
||||||
const assert = makeAssert('hasRef() not working on process_wrap');
|
|
||||||
const spawn = require('child_process').spawn;
|
const spawn = require('child_process').spawn;
|
||||||
const cmd = common.isWindows ? 'rundll32' : 'ls';
|
const cmd = common.isWindows ? 'rundll32' : 'ls';
|
||||||
const cp = spawn(cmd);
|
const cp = spawn(cmd);
|
||||||
assert(Object.getPrototypeOf(cp._handle).hasOwnProperty('hasRef'), true);
|
strictEqual(Object.getPrototypeOf(cp._handle).hasOwnProperty('hasRef'),
|
||||||
assert(cp._handle.hasRef(), true);
|
true, 'process_wrap: hasRef() missing');
|
||||||
|
strictEqual(cp._handle.hasRef(),
|
||||||
|
true, 'process_wrap: not initially refed');
|
||||||
cp.unref();
|
cp.unref();
|
||||||
assert(cp._handle.hasRef(), false);
|
strictEqual(cp._handle.hasRef(),
|
||||||
|
false, 'process_wrap: unref() ineffective');
|
||||||
cp.ref();
|
cp.ref();
|
||||||
assert(cp._handle.hasRef(), true);
|
strictEqual(cp._handle.hasRef(),
|
||||||
cp._handle.close(common.mustCall(() => assert(cp._handle.hasRef(), false)));
|
true, 'process_wrap: ref() ineffective');
|
||||||
|
cp._handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(cp._handle.hasRef(),
|
||||||
|
false, 'process_wrap: not unrefed on close')));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// dgram
|
// dgram ipv4
|
||||||
{
|
{
|
||||||
const assert = makeAssert('hasRef() not working on udp_wrap');
|
|
||||||
const dgram = require('dgram');
|
const dgram = require('dgram');
|
||||||
|
|
||||||
const sock4 = dgram.createSocket('udp4');
|
const sock4 = dgram.createSocket('udp4');
|
||||||
assert(Object.getPrototypeOf(sock4._handle).hasOwnProperty('hasRef'), true);
|
strictEqual(Object.getPrototypeOf(sock4._handle).hasOwnProperty('hasRef'),
|
||||||
assert(sock4._handle.hasRef(), true);
|
true, 'udp_wrap: ipv4: hasRef() missing');
|
||||||
|
strictEqual(sock4._handle.hasRef(),
|
||||||
|
true, 'udp_wrap: ipv4: not initially refed');
|
||||||
sock4.unref();
|
sock4.unref();
|
||||||
assert(sock4._handle.hasRef(), false);
|
strictEqual(sock4._handle.hasRef(),
|
||||||
|
false, 'udp_wrap: ipv4: unref() ineffective');
|
||||||
sock4.ref();
|
sock4.ref();
|
||||||
assert(sock4._handle.hasRef(), true);
|
strictEqual(sock4._handle.hasRef(),
|
||||||
sock4._handle.close(
|
true, 'udp_wrap: ipv4: ref() ineffective');
|
||||||
common.mustCall(() => assert(sock4._handle.hasRef(), false)));
|
sock4._handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(sock4._handle.hasRef(),
|
||||||
|
false, 'udp_wrap: ipv4: not unrefed on close')));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// dgram ipv6
|
||||||
|
{
|
||||||
|
const dgram = require('dgram');
|
||||||
const sock6 = dgram.createSocket('udp6');
|
const sock6 = dgram.createSocket('udp6');
|
||||||
assert(Object.getPrototypeOf(sock6._handle).hasOwnProperty('hasRef'), true);
|
strictEqual(Object.getPrototypeOf(sock6._handle).hasOwnProperty('hasRef'),
|
||||||
assert(sock6._handle.hasRef(), true);
|
true, 'udp_wrap: ipv6: hasRef() missing');
|
||||||
|
strictEqual(sock6._handle.hasRef(),
|
||||||
|
true, 'udp_wrap: ipv6: not initially refed');
|
||||||
sock6.unref();
|
sock6.unref();
|
||||||
assert(sock6._handle.hasRef(), false);
|
strictEqual(sock6._handle.hasRef(),
|
||||||
|
false, 'udp_wrap: ipv6: unref() ineffective');
|
||||||
sock6.ref();
|
sock6.ref();
|
||||||
assert(sock6._handle.hasRef(), true);
|
strictEqual(sock6._handle.hasRef(),
|
||||||
sock6._handle.close(
|
true, 'udp_wrap: ipv6: ref() ineffective');
|
||||||
common.mustCall(() => assert(sock6._handle.hasRef(), false)));
|
sock6._handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(sock6._handle.hasRef(),
|
||||||
|
false, 'udp_wrap: ipv6: not unrefed on close')));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// pipe
|
// pipe
|
||||||
{
|
{
|
||||||
const assert = makeAssert('hasRef() not working on pipe_wrap');
|
|
||||||
const Pipe = process.binding('pipe_wrap').Pipe;
|
const Pipe = process.binding('pipe_wrap').Pipe;
|
||||||
const handle = new Pipe();
|
const handle = new Pipe();
|
||||||
assert(Object.getPrototypeOf(handle).hasOwnProperty('hasRef'), true);
|
strictEqual(Object.getPrototypeOf(handle).hasOwnProperty('hasRef'),
|
||||||
assert(handle.hasRef(), true);
|
true, 'pipe_wrap: hasRef() missing');
|
||||||
|
strictEqual(handle.hasRef(),
|
||||||
|
true, 'pipe_wrap: not initially refed');
|
||||||
handle.unref();
|
handle.unref();
|
||||||
assert(handle.hasRef(), false);
|
strictEqual(handle.hasRef(),
|
||||||
|
false, 'pipe_wrap: unref() ineffective');
|
||||||
handle.ref();
|
handle.ref();
|
||||||
assert(handle.hasRef(), true);
|
strictEqual(handle.hasRef(),
|
||||||
handle.close(common.mustCall(() => assert(handle.hasRef(), false)));
|
true, 'pipe_wrap: ref() ineffective');
|
||||||
|
handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(handle.hasRef(),
|
||||||
|
false, 'pipe_wrap: not unrefed on close')));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// tcp
|
// tcp
|
||||||
{
|
{
|
||||||
const assert = makeAssert('hasRef() not working on tcp_wrap');
|
|
||||||
const net = require('net');
|
const net = require('net');
|
||||||
const server = net.createServer(() => {}).listen(0);
|
const server = net.createServer(() => {}).listen(0);
|
||||||
assert(Object.getPrototypeOf(server._handle).hasOwnProperty('hasRef'), true);
|
strictEqual(Object.getPrototypeOf(server._handle).hasOwnProperty('hasRef'),
|
||||||
assert(server._handle.hasRef(), true);
|
true, 'tcp_wrap: hasRef() missing');
|
||||||
assert(server._unref, false);
|
strictEqual(server._handle.hasRef(),
|
||||||
|
true, 'tcp_wrap: not initially refed');
|
||||||
|
strictEqual(server._unref,
|
||||||
|
false, 'tcp_wrap: _unref initially incorrect');
|
||||||
server.unref();
|
server.unref();
|
||||||
assert(server._handle.hasRef(), false);
|
strictEqual(server._handle.hasRef(),
|
||||||
assert(server._unref, true);
|
false, 'tcp_wrap: unref() ineffective');
|
||||||
|
strictEqual(server._unref,
|
||||||
|
true, 'tcp_wrap: _unref not updated on unref()');
|
||||||
server.ref();
|
server.ref();
|
||||||
assert(server._handle.hasRef(), true);
|
strictEqual(server._handle.hasRef(),
|
||||||
assert(server._unref, false);
|
true, 'tcp_wrap: ref() ineffective');
|
||||||
server._handle.close(
|
strictEqual(server._unref,
|
||||||
common.mustCall(() => assert(server._handle.hasRef(), false)));
|
false, 'tcp_wrap: _unref not updated on ref()');
|
||||||
|
server._handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(server._handle.hasRef(),
|
||||||
|
false, 'tcp_wrap: not unrefed on close')));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// timers
|
// timers
|
||||||
{
|
{
|
||||||
const assert = makeAssert('hasRef() not working on timer_wrap');
|
|
||||||
const timer = setTimeout(() => {}, 500);
|
const timer = setTimeout(() => {}, 500);
|
||||||
timer.unref();
|
timer.unref();
|
||||||
assert(Object.getPrototypeOf(timer._handle).hasOwnProperty('hasRef'), true);
|
strictEqual(Object.getPrototypeOf(timer._handle).hasOwnProperty('hasRef'),
|
||||||
assert(timer._handle.hasRef(), false);
|
true, 'timer_wrap: hasRef() missing');
|
||||||
|
strictEqual(timer._handle.hasRef(),
|
||||||
|
false, 'timer_wrap: unref() ineffective');
|
||||||
timer.ref();
|
timer.ref();
|
||||||
assert(timer._handle.hasRef(), true);
|
strictEqual(timer._handle.hasRef(),
|
||||||
timer._handle.close(
|
true, 'timer_wrap: ref() ineffective');
|
||||||
common.mustCall(() => assert(timer._handle.hasRef(), false)));
|
timer._handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(timer._handle.hasRef(),
|
||||||
|
false, 'timer_wrap: not unrefed on close')));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// see also test/pseudo-tty/test-handle-wrap-isrefed-tty.js
|
||||||
|
27
test/pseudo-tty/ref_keeps_node_running.js
Normal file
27
test/pseudo-tty/ref_keeps_node_running.js
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
'use strict';
|
||||||
|
require('../common');
|
||||||
|
|
||||||
|
const { TTY, isTTY } = process.binding('tty_wrap');
|
||||||
|
const strictEqual = require('assert').strictEqual;
|
||||||
|
|
||||||
|
strictEqual(isTTY(0), true, 'fd 0 is not a TTY');
|
||||||
|
|
||||||
|
const handle = new TTY(0);
|
||||||
|
handle.readStart();
|
||||||
|
handle.onread = () => {};
|
||||||
|
|
||||||
|
function isHandleActive(handle) {
|
||||||
|
return process._getActiveHandles().some((active) => active === handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
strictEqual(isHandleActive(handle), true, 'TTY handle not initially active');
|
||||||
|
|
||||||
|
handle.unref();
|
||||||
|
|
||||||
|
strictEqual(isHandleActive(handle), false, 'TTY handle active after unref()');
|
||||||
|
|
||||||
|
handle.ref();
|
||||||
|
|
||||||
|
strictEqual(isHandleActive(handle), true, 'TTY handle inactive after ref()');
|
||||||
|
|
||||||
|
handle.unref();
|
0
test/pseudo-tty/ref_keeps_node_running.out
Normal file
0
test/pseudo-tty/ref_keeps_node_running.out
Normal file
23
test/pseudo-tty/test-handle-wrap-isrefed-tty.js
Normal file
23
test/pseudo-tty/test-handle-wrap-isrefed-tty.js
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
// see also test/parallel/test-handle-wrap-isrefed.js
|
||||||
|
|
||||||
|
const common = require('../common');
|
||||||
|
const strictEqual = require('assert').strictEqual;
|
||||||
|
const ReadStream = require('tty').ReadStream;
|
||||||
|
const tty = new ReadStream(0);
|
||||||
|
const isTTY = process.binding('tty_wrap').isTTY;
|
||||||
|
strictEqual(isTTY(0), true, 'tty_wrap: stdin is not a TTY');
|
||||||
|
strictEqual(Object.getPrototypeOf(tty._handle).hasOwnProperty('hasRef'),
|
||||||
|
true, 'tty_wrap: hasRef() missing');
|
||||||
|
strictEqual(tty._handle.hasRef(),
|
||||||
|
true, 'tty_wrap: not initially refed');
|
||||||
|
tty.unref();
|
||||||
|
strictEqual(tty._handle.hasRef(),
|
||||||
|
false, 'tty_wrap: unref() ineffective');
|
||||||
|
tty.ref();
|
||||||
|
strictEqual(tty._handle.hasRef(),
|
||||||
|
true, 'tty_wrap: ref() ineffective');
|
||||||
|
tty._handle.close(common.mustCall(() =>
|
||||||
|
strictEqual(tty._handle.hasRef(),
|
||||||
|
false, 'tty_wrap: not unrefed on close')));
|
0
test/pseudo-tty/test-handle-wrap-isrefed-tty.out
Normal file
0
test/pseudo-tty/test-handle-wrap-isrefed-tty.out
Normal file
Loading…
x
Reference in New Issue
Block a user