From 2ceb6709c9e6da92578657daa66bcc081c7ca9b6 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Tue, 16 May 2023 15:21:16 +0300 Subject: [PATCH] lib: support FORCE_COLOR for non TTY streams --- lib/internal/console/constructor.js | 11 ++++-- lib/internal/errors.js | 18 ++++----- lib/internal/test_runner/reporter/spec.js | 4 +- lib/internal/util/colors.js | 16 +++++++- lib/internal/util/debuglog.js | 8 +++- lib/repl.js | 5 ++- test/common/assertSnapshot.js | 4 +- test/fixtures/console/force_colors.js | 5 +++ test/fixtures/console/force_colors.snapshot | 1 + test/fixtures/errors/force_colors.js | 1 + test/fixtures/errors/force_colors.snapshot | 14 +++++++ test/parallel/test-node-output-console.mjs | 5 ++- test/parallel/test-node-output-errors.mjs | 5 ++- test/parallel/test-repl-envvars.js | 42 ++++++++++++++------- 14 files changed, 99 insertions(+), 40 deletions(-) create mode 100644 test/fixtures/console/force_colors.js create mode 100644 test/fixtures/console/force_colors.snapshot create mode 100644 test/fixtures/errors/force_colors.js create mode 100644 test/fixtures/errors/force_colors.snapshot diff --git a/lib/internal/console/constructor.js b/lib/internal/console/constructor.js index fc91bc9b3851d3..44fa0f32d76ff3 100644 --- a/lib/internal/console/constructor.js +++ b/lib/internal/console/constructor.js @@ -79,6 +79,12 @@ const kMaxGroupIndentation = 1000; // Lazy loaded for startup performance. let cliTable; +let utilColors; +function lazyUtilColors() { + utilColors ??= require('internal/util/colors'); + return utilColors; +} + // Track amount of indentation required via `console.group()`. const kGroupIndent = Symbol('kGroupIndent'); const kGroupIndentationWidth = Symbol('kGroupIndentWidth'); @@ -95,7 +101,6 @@ const kUseStdout = Symbol('kUseStdout'); const kUseStderr = Symbol('kUseStderr'); const optionsMap = new SafeWeakMap(); - function Console(options /* or: stdout, stderr, ignoreErrors = true */) { // We have to test new.target here to see if this function is called // with new, because we need to define a custom instanceof to accommodate @@ -314,9 +319,7 @@ ObjectDefineProperties(Console.prototype, { value: function(stream) { let color = this[kColorMode]; if (color === 'auto') { - color = stream.isTTY && ( - typeof stream.getColorDepth === 'function' ? - stream.getColorDepth() > 2 : true); + color = lazyUtilColors().shouldColorize(stream); } const options = optionsMap.get(this); diff --git a/lib/internal/errors.js b/lib/internal/errors.js index 8a490ea3a4281a..384dd5c3b89c15 100644 --- a/lib/internal/errors.js +++ b/lib/internal/errors.js @@ -190,6 +190,12 @@ function lazyInternalUtilInspect() { return internalUtilInspect; } +let utilColors; +function lazyUtilColors() { + utilColors ??= require('internal/util/colors'); + return utilColors; +} + let buffer; function lazyBuffer() { buffer ??= require('buffer').Buffer; @@ -789,16 +795,8 @@ const fatalExceptionStackEnhancers = { useColors = false; } } - const { - inspect, - inspectDefaultOptions: { - colors: defaultColors, - }, - } = lazyInternalUtilInspect(); - const colors = useColors && - ((internalBinding('util').guessHandleType(2) === 'TTY' && - require('internal/tty').hasColors()) || - defaultColors); + const { inspect } = lazyInternalUtilInspect(); + const colors = useColors && lazyUtilColors().shouldColorize(process.stderr); try { return inspect(error, { colors, diff --git a/lib/internal/test_runner/reporter/spec.js b/lib/internal/test_runner/reporter/spec.js index 848b5cfa245d1e..c57d84fc39343f 100644 --- a/lib/internal/test_runner/reporter/spec.js +++ b/lib/internal/test_runner/reporter/spec.js @@ -14,11 +14,11 @@ const { const assert = require('assert'); const Transform = require('internal/streams/transform'); const { inspectWithNoCustomRetry } = require('internal/errors'); -const { green, blue, red, white, gray, hasColors } = require('internal/util/colors'); +const { green, blue, red, white, gray, shouldColorize } = require('internal/util/colors'); const { kSubtestsFailed } = require('internal/test_runner/test'); const { getCoverageReport } = require('internal/test_runner/utils'); -const inspectOptions = { __proto__: null, colors: hasColors, breakLength: Infinity }; +const inspectOptions = { __proto__: null, colors: shouldColorize(process.stdout), breakLength: Infinity }; const colors = { '__proto__': null, diff --git a/lib/internal/util/colors.js b/lib/internal/util/colors.js index e81a97ad8771a1..9e1ed05d88130e 100644 --- a/lib/internal/util/colors.js +++ b/lib/internal/util/colors.js @@ -1,5 +1,11 @@ 'use strict'; +let internalTTy; +function lazyInternalTTY() { + internalTTy ??= require('internal/tty'); + return internalTTy; +} + module.exports = { blue: '', green: '', @@ -8,9 +14,17 @@ module.exports = { gray: '', clear: '', hasColors: false, + shouldColorize(stream) { + if (process.env.FORCE_COLOR !== undefined) { + return lazyInternalTTY().getColorDepth() > 2; + } + return stream?.isTTY && ( + typeof stream.getColorDepth === 'function' ? + stream.getColorDepth() > 2 : true); + }, refresh() { if (process.stderr.isTTY) { - const hasColors = process.stderr.hasColors(); + const hasColors = module.exports.shouldColorize(process.stderr); module.exports.blue = hasColors ? '\u001b[34m' : ''; module.exports.green = hasColors ? '\u001b[32m' : ''; module.exports.white = hasColors ? '\u001b[39m' : ''; diff --git a/lib/internal/util/debuglog.js b/lib/internal/util/debuglog.js index 9ecfc2aac4c3ed..854441216e67e5 100644 --- a/lib/internal/util/debuglog.js +++ b/lib/internal/util/debuglog.js @@ -45,13 +45,19 @@ function emitWarningIfNeeded(set) { const noop = () => {}; +let utilColors; +function lazyUtilColors() { + utilColors ??= require('internal/util/colors'); + return utilColors; +} + function debuglogImpl(enabled, set) { if (debugImpls[set] === undefined) { if (enabled) { const pid = process.pid; emitWarningIfNeeded(set); debugImpls[set] = function debug(...args) { - const colors = process.stderr.hasColors && process.stderr.hasColors(); + const colors = lazyUtilColors().shouldColorize(process.stderr); const msg = formatWithOptions({ colors }, ...args); const coloredPID = inspect(pid, { colors }); process.stderr.write(format('%s %s: %s\n', set, coloredPID, msg)); diff --git a/lib/repl.js b/lib/repl.js index 0a5a9b44ed16b8..55cefefbb2dd02 100644 --- a/lib/repl.js +++ b/lib/repl.js @@ -121,6 +121,7 @@ const { commonPrefix, } = require('internal/readline/utils'); const { Console } = require('console'); +const { shouldColorize } = require('internal/util/colors'); const CJSModule = require('internal/modules/cjs/loader').Module; let _builtinLibs = ArrayPrototypeFilter( CJSModule.builtinModules, @@ -270,8 +271,8 @@ function REPLServer(prompt, if (options.terminal && options.useColors === undefined) { // If possible, check if stdout supports colors or not. - if (options.output.hasColors) { - options.useColors = options.output.hasColors(); + if (shouldColorize(options.output)) { + options.useColors = true; } else if (process.env.NODE_DISABLE_COLORS === undefined) { options.useColors = true; } diff --git a/test/common/assertSnapshot.js b/test/common/assertSnapshot.js index ed9db64e654dcc..c403751ac3ef5e 100644 --- a/test/common/assertSnapshot.js +++ b/test/common/assertSnapshot.js @@ -54,7 +54,7 @@ async function assertSnapshot(actual, filename = process.argv[1]) { * @param {boolean} [options.tty] - whether to spawn the process in a pseudo-tty * @returns {Promise} */ -async function spawnAndAssert(filename, transform = (x) => x, { tty = false } = {}) { +async function spawnAndAssert(filename, transform = (x) => x, { tty = false, ...options } = {}) { if (tty && common.isWindows) { test({ skip: 'Skipping pseudo-tty tests, as pseudo terminals are not available on Windows.' }); return; @@ -62,7 +62,7 @@ async function spawnAndAssert(filename, transform = (x) => x, { tty = false } = const flags = common.parseTestFlags(filename); const executable = tty ? 'tools/pseudo-tty.py' : process.execPath; const args = tty ? [process.execPath, ...flags, filename] : [...flags, filename]; - const { stdout, stderr } = await common.spawnPromisified(executable, args); + const { stdout, stderr } = await common.spawnPromisified(executable, args, options); await assertSnapshot(transform(`${stdout}${stderr}`), filename); } diff --git a/test/fixtures/console/force_colors.js b/test/fixtures/console/force_colors.js new file mode 100644 index 00000000000000..7ced4b82eee035 --- /dev/null +++ b/test/fixtures/console/force_colors.js @@ -0,0 +1,5 @@ +'use strict'; + +require('../../common'); + +console.log(123, 'foo', { bar: 'baz' }); diff --git a/test/fixtures/console/force_colors.snapshot b/test/fixtures/console/force_colors.snapshot new file mode 100644 index 00000000000000..0c754ba4c53554 --- /dev/null +++ b/test/fixtures/console/force_colors.snapshot @@ -0,0 +1 @@ +123 foo { bar: 'baz' } diff --git a/test/fixtures/errors/force_colors.js b/test/fixtures/errors/force_colors.js new file mode 100644 index 00000000000000..1502c86f6f69d2 --- /dev/null +++ b/test/fixtures/errors/force_colors.js @@ -0,0 +1 @@ +throw new Error('Should include grayed stack trace') \ No newline at end of file diff --git a/test/fixtures/errors/force_colors.snapshot b/test/fixtures/errors/force_colors.snapshot new file mode 100644 index 00000000000000..a38745e396a8d9 --- /dev/null +++ b/test/fixtures/errors/force_colors.snapshot @@ -0,0 +1,14 @@ +*force_colors.js:1 +throw new Error('Should include grayed stack trace') +^ + +Error: Should include grayed stack trace + at Object. (/test*force_colors.js:1:7) + at Module._compile (node:internal*modules*cjs*loader:1255:14) + at Module._extensions..js (node:internal*modules*cjs*loader:1309:10) + at Module.load (node:internal*modules*cjs*loader:1113:32) + at Module._load (node:internal*modules*cjs*loader:960:12) + at Function.executeUserEntryPoint [as runMain] (node:internal*modules*run_main:83:12) + at node:internal*main*run_main_module:23:47 + +Node.js * diff --git a/test/parallel/test-node-output-console.mjs b/test/parallel/test-node-output-console.mjs index 633f2db9f08bc1..4863605d27b445 100644 --- a/test/parallel/test-node-output-console.mjs +++ b/test/parallel/test-node-output-console.mjs @@ -22,12 +22,13 @@ describe('console output', { concurrency: true }, () => { transform: snapshot .transform(snapshot.replaceWindowsLineEndings, snapshot.replaceWindowsPaths, normalize) }, + { name: 'console/force_colors.js', env: { FORCE_COLOR: 1 } }, ]; const defaultTransform = snapshot .transform(snapshot.replaceWindowsLineEndings, snapshot.replaceWindowsPaths, replaceStackTrace); - for (const { name, transform } of tests) { + for (const { name, transform, env } of tests) { it(name, async () => { - await snapshot.spawnAndAssert(fixtures.path(name), transform ?? defaultTransform); + await snapshot.spawnAndAssert(fixtures.path(name), transform ?? defaultTransform, { env }); }); } }); diff --git a/test/parallel/test-node-output-errors.mjs b/test/parallel/test-node-output-errors.mjs index 1de7b52bb61a64..fb796b3f94419e 100644 --- a/test/parallel/test-node-output-errors.mjs +++ b/test/parallel/test-node-output-errors.mjs @@ -43,10 +43,11 @@ describe('errors output', { concurrency: true }, () => { { name: 'errors/throw_in_line_with_tabs.js', transform: errTransform }, { name: 'errors/throw_non_error.js', transform: errTransform }, { name: 'errors/promise_always_throw_unhandled.js', transform: promiseTransform }, + { name: 'errors/force_colors.js', env: { FORCE_COLOR: 1 } }, ]; - for (const { name, transform } of tests) { + for (const { name, transform, env } of tests) { it(name, async () => { - await snapshot.spawnAndAssert(fixtures.path(name), transform ?? defaultTransform); + await snapshot.spawnAndAssert(fixtures.path(name), transform ?? defaultTransform, { env }); }); } }); diff --git a/test/parallel/test-repl-envvars.js b/test/parallel/test-repl-envvars.js index b9216bc4aa0303..0f02cef8cbd1a8 100644 --- a/test/parallel/test-repl-envvars.js +++ b/test/parallel/test-repl-envvars.js @@ -4,6 +4,7 @@ require('../common'); const stream = require('stream'); +const { describe, test } = require('node:test'); const REPL = require('internal/repl'); const assert = require('assert'); const inspect = require('util').inspect; @@ -18,13 +19,21 @@ const tests = [ env: { NODE_DISABLE_COLORS: '1' }, expected: { terminal: true, useColors: false } }, + { + env: { NODE_DISABLE_COLORS: '1', FORCE_COLOR: '1' }, + expected: { terminal: true, useColors: true } + }, { env: { NODE_NO_READLINE: '1' }, expected: { terminal: false, useColors: false } }, { env: { TERM: 'dumb' }, - expected: { terminal: true, useColors: false } + expected: { terminal: true, useColors: true } + }, + { + env: { TERM: 'dumb', FORCE_COLOR: '1' }, + expected: { terminal: true, useColors: true } }, { env: { NODE_NO_READLINE: '1', NODE_DISABLE_COLORS: '1' }, @@ -56,20 +65,25 @@ function run(test) { Object.assign(process.env, env); - REPL.createInternalRepl(process.env, opts, function(err, repl) { - assert.ifError(err); + return new Promise((resolve) => { + REPL.createInternalRepl(process.env, opts, function(err, repl) { + assert.ifError(err); - assert.strictEqual(repl.terminal, expected.terminal, - `Expected ${inspect(expected)} with ${inspect(env)}`); - assert.strictEqual(repl.useColors, expected.useColors, - `Expected ${inspect(expected)} with ${inspect(env)}`); - assert.strictEqual(repl.replMode, expected.replMode || REPL_MODE_SLOPPY, - `Expected ${inspect(expected)} with ${inspect(env)}`); - for (const key of Object.keys(env)) { - delete process.env[key]; - } - repl.close(); + assert.strictEqual(repl.terminal, expected.terminal, + `Expected ${inspect(expected)} with ${inspect(env)}`); + assert.strictEqual(repl.useColors, expected.useColors, + `Expected ${inspect(expected)} with ${inspect(env)}`); + assert.strictEqual(repl.replMode, expected.replMode || REPL_MODE_SLOPPY, + `Expected ${inspect(expected)} with ${inspect(env)}`); + for (const key of Object.keys(env)) { + delete process.env[key]; + } + repl.close(); + resolve(); + }); }); } -tests.forEach(run); +describe('REPL environment variables', { concurrency: 1 }, () => { + tests.forEach((testCase) => test(inspect(testCase.env), () => run(testCase))); +});