From d3997971a42a6c8e5599d16c8c457c792ce943c6 Mon Sep 17 00:00:00 2001 From: Oky Antoro Date: Sun, 7 Jul 2019 21:33:03 +0700 Subject: [PATCH] Fake `hasColors()` in worker processes Fixes #2170. --- lib/fork.js | 1 + lib/worker/fake-tty-has-colors.js | 19 +++++++++++++++++++ lib/worker/fake-tty.js | 7 ++++++- test/helper/simulate-tty.js | 12 +++++++++--- test/integration/node-assertions.js | 3 ++- 5 files changed, 37 insertions(+), 5 deletions(-) create mode 100644 lib/worker/fake-tty-has-colors.js diff --git a/lib/fork.js b/lib/fork.js index f748e03cb..30f1cc180 100644 --- a/lib/fork.js +++ b/lib/fork.js @@ -25,6 +25,7 @@ if (process.env.NODE_PATH) { const describeTTY = tty => ({ colorDepth: tty.getColorDepth ? tty.getColorDepth() : undefined, + hasColors: typeof tty.hasColors === 'function', columns: tty.columns || 80, rows: tty.rows }); diff --git a/lib/worker/fake-tty-has-colors.js b/lib/worker/fake-tty-has-colors.js new file mode 100644 index 000000000..15ea4d121 --- /dev/null +++ b/lib/worker/fake-tty-has-colors.js @@ -0,0 +1,19 @@ +'use strict'; +const tty = require('tty'); + +// Call original method to ensure the correct errors are thrown. +const assertHasColorsArguments = count => { + tty.WriteStream.prototype.hasColors(count); +}; + +const makeHasColors = colorDepth => (count = 16, env) => { + // `count` is optional too, so make sure it's not an env object. + if (env === undefined && typeof count === 'object' && count !== null) { + count = 16; + } + + assertHasColorsArguments(count); + return count <= 2 ** colorDepth; +}; + +module.exports = makeHasColors; diff --git a/lib/worker/fake-tty.js b/lib/worker/fake-tty.js index a8292c225..1321c466c 100644 --- a/lib/worker/fake-tty.js +++ b/lib/worker/fake-tty.js @@ -2,13 +2,14 @@ const tty = require('tty'); const ansiEscapes = require('ansi-escapes'); const options = require('./options').get(); +const makeHasColors = require('./fake-tty-has-colors'); const fakeTTYs = new Set(); const {isatty} = tty; tty.isatty = fd => fakeTTYs.has(fd) || isatty(fd); -const simulateTTY = (stream, {colorDepth, columns, rows}) => { +const simulateTTY = (stream, {colorDepth, hasColors, columns, rows}) => { Object.assign(stream, {isTTY: true, columns, rows}); stream.clearLine = dir => { @@ -35,6 +36,10 @@ const simulateTTY = (stream, {colorDepth, columns, rows}) => { if (colorDepth !== undefined) { stream.getColorDepth = () => colorDepth; } + + if (hasColors) { + stream.hasColors = makeHasColors(colorDepth); + } }; if (options.tty.stderr) { diff --git a/test/helper/simulate-tty.js b/test/helper/simulate-tty.js index af3e5c947..7ebe96fac 100644 --- a/test/helper/simulate-tty.js +++ b/test/helper/simulate-tty.js @@ -1,6 +1,7 @@ 'use strict'; +const makeHasColors = require('../../lib/worker/fake-tty-has-colors'); -const simulateTTY = (stream, colorDepth) => { +const simulateTTY = (stream, colorDepth, hasColors) => { stream.isTTY = true; stream.columns = 80; stream.rows = 24; @@ -8,6 +9,10 @@ const simulateTTY = (stream, colorDepth) => { if (colorDepth) { stream.getColorDepth = () => colorDepth; } + + if (hasColors) { + stream.hasColors = makeHasColors(colorDepth); + } }; // The execCli helper spawns tests in a child process. This means that stdout is @@ -17,7 +22,8 @@ if (process.env.AVA_SIMULATE_TTY) { const colorDepth = process.env.AVA_TTY_COLOR_DEPTH ? parseInt(process.env.AVA_TTY_COLOR_DEPTH, 10) : undefined; + const hasColors = process.env.AVA_TTY_HAS_COLORS !== undefined; - simulateTTY(process.stderr, colorDepth); - simulateTTY(process.stdout, colorDepth); + simulateTTY(process.stderr, colorDepth, hasColors); + simulateTTY(process.stdout, colorDepth, hasColors); } diff --git a/test/integration/node-assertions.js b/test/integration/node-assertions.js index 964a84039..1b37e9bb9 100644 --- a/test/integration/node-assertions.js +++ b/test/integration/node-assertions.js @@ -8,7 +8,8 @@ test('node assertion failures are reported to the console when running in a term dirname: 'fixture/node-assertions', env: { AVA_SIMULATE_TTY: true, - AVA_TTY_COLOR_DEPTH: 8 + AVA_TTY_COLOR_DEPTH: 8, + AVA_TTY_HAS_COLORS: typeof process.stderr.hasColors === 'function' } };