mirror of
https://github.com/nodejs/node.git
synced 2025-05-07 12:03:30 +00:00

This changes the `compact` default from `true` to `3`. That mode changes arrays to be grouped together, it alignes multiple small entries on a single line in similar to `compact` true but only for the most inner three depth levels and the closing brackets are always on the same indentation as the openeing of the object instead of at the same line as another property. Big strings will be naturally broken into multiple lines instead of having one huge line that is not well readable. The output size mainly stays the same that way while it will be smaller in case of big arrays. Increasing the `breakLength` to 80 adjusts for most terminals that support at least 80 characters in a single line and improves the general output that way. A lot of calculations use the `breakLength` to determine the concrete behavior. PR-URL: https://github.com/nodejs/node/pull/27109 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Minwoo Jung <minwoo@nodesource.com> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Matteo Collina <matteo.collina@gmail.com>
284 lines
9.3 KiB
JavaScript
284 lines
9.3 KiB
JavaScript
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
'use strict';
|
|
const common = require('../common');
|
|
const assert = require('assert');
|
|
const util = require('util');
|
|
|
|
const {
|
|
hijackStdout,
|
|
hijackStderr,
|
|
restoreStdout,
|
|
restoreStderr
|
|
} = require('../common/hijackstdio');
|
|
|
|
assert.ok(process.stdout.writable);
|
|
assert.ok(process.stderr.writable);
|
|
// Support legacy API
|
|
if (common.isMainThread) {
|
|
assert.strictEqual(typeof process.stdout.fd, 'number');
|
|
assert.strictEqual(typeof process.stderr.fd, 'number');
|
|
}
|
|
|
|
common.expectWarning(
|
|
'Warning',
|
|
[
|
|
['Count for \'noLabel\' does not exist'],
|
|
['No such label \'noLabel\' for console.timeLog()'],
|
|
['No such label \'noLabel\' for console.timeEnd()'],
|
|
['Count for \'default\' does not exist'],
|
|
['No such label \'default\' for console.timeLog()'],
|
|
['No such label \'default\' for console.timeEnd()'],
|
|
['Label \'default\' already exists for console.time()'],
|
|
['Label \'test\' already exists for console.time()']
|
|
]
|
|
);
|
|
|
|
console.countReset('noLabel');
|
|
console.timeLog('noLabel');
|
|
console.timeEnd('noLabel');
|
|
|
|
console.time('label');
|
|
console.timeEnd('label');
|
|
|
|
// Test using the default label
|
|
// on console.time(), console.countReset(), console.timeLog(), console.timeEnd()
|
|
console.countReset();
|
|
console.timeLog();
|
|
console.timeEnd();
|
|
|
|
console.time();
|
|
console.time();
|
|
console.timeLog();
|
|
console.timeEnd();
|
|
|
|
// Check that the `Error` is a `TypeError` but do not check the message as it
|
|
// will be different in different JavaScript engines.
|
|
assert.throws(() => console.time(Symbol('test')),
|
|
TypeError);
|
|
assert.throws(() => console.timeEnd(Symbol('test')),
|
|
TypeError);
|
|
|
|
|
|
// An Object with a custom inspect function.
|
|
const custom_inspect = { foo: 'bar', [util.inspect.custom]: () => 'inspect' };
|
|
|
|
const strings = [];
|
|
const errStrings = [];
|
|
process.stdout.isTTY = false;
|
|
hijackStdout(function(data) {
|
|
strings.push(data);
|
|
});
|
|
process.stderr.isTTY = false;
|
|
hijackStderr(function(data) {
|
|
errStrings.push(data);
|
|
});
|
|
|
|
// Test console.log() goes to stdout
|
|
console.log('foo');
|
|
console.log('foo', 'bar');
|
|
console.log('%s %s', 'foo', 'bar', 'hop');
|
|
console.log({ slashes: '\\\\' });
|
|
console.log(custom_inspect);
|
|
|
|
// Test console.debug() goes to stdout
|
|
console.debug('foo');
|
|
console.debug('foo', 'bar');
|
|
console.debug('%s %s', 'foo', 'bar', 'hop');
|
|
console.debug({ slashes: '\\\\' });
|
|
console.debug(custom_inspect);
|
|
|
|
// Test console.info() goes to stdout
|
|
console.info('foo');
|
|
console.info('foo', 'bar');
|
|
console.info('%s %s', 'foo', 'bar', 'hop');
|
|
console.info({ slashes: '\\\\' });
|
|
console.info(custom_inspect);
|
|
|
|
// Test console.error() goes to stderr
|
|
console.error('foo');
|
|
console.error('foo', 'bar');
|
|
console.error('%s %s', 'foo', 'bar', 'hop');
|
|
console.error({ slashes: '\\\\' });
|
|
console.error(custom_inspect);
|
|
|
|
// Test console.warn() goes to stderr
|
|
console.warn('foo');
|
|
console.warn('foo', 'bar');
|
|
console.warn('%s %s', 'foo', 'bar', 'hop');
|
|
console.warn({ slashes: '\\\\' });
|
|
console.warn(custom_inspect);
|
|
|
|
// test console.dir()
|
|
console.dir(custom_inspect);
|
|
console.dir(custom_inspect, { showHidden: false });
|
|
console.dir({ foo: { bar: { baz: true } } }, { depth: 0 });
|
|
console.dir({ foo: { bar: { baz: true } } }, { depth: 1 });
|
|
|
|
// Test console.dirxml()
|
|
console.dirxml(custom_inspect, custom_inspect);
|
|
console.dirxml(
|
|
{ foo: { bar: { baz: true } } },
|
|
{ foo: { bar: { quux: false } } },
|
|
{ foo: { bar: { quux: true } } }
|
|
);
|
|
|
|
// Test console.trace()
|
|
console.trace('This is a %j %d', { formatted: 'trace' }, 10, 'foo');
|
|
|
|
// Test console.time() and console.timeEnd() output
|
|
console.time('label');
|
|
console.timeEnd('label');
|
|
|
|
// Verify that Object.prototype properties can be used as labels
|
|
console.time('__proto__');
|
|
console.timeEnd('__proto__');
|
|
console.time('constructor');
|
|
console.timeEnd('constructor');
|
|
console.time('hasOwnProperty');
|
|
console.timeEnd('hasOwnProperty');
|
|
|
|
// Verify that values are coerced to strings.
|
|
console.time([]);
|
|
console.timeEnd([]);
|
|
console.time({});
|
|
console.timeEnd({});
|
|
// Repeat the object call to verify that everything really worked.
|
|
console.time({});
|
|
console.timeEnd({});
|
|
console.time(null);
|
|
console.timeEnd(null);
|
|
console.time(undefined);
|
|
console.timeEnd('default');
|
|
console.time('default');
|
|
console.timeEnd();
|
|
console.time(NaN);
|
|
console.timeEnd(NaN);
|
|
|
|
// Make sure calling time twice without timeEnd doesn't reset the timer.
|
|
console.time('test');
|
|
const time = console._times.get('test');
|
|
setTimeout(() => {
|
|
console.time('test');
|
|
assert.deepStrictEqual(console._times.get('test'), time);
|
|
console.timeEnd('test');
|
|
}, 1);
|
|
|
|
console.time('log1');
|
|
console.timeLog('log1');
|
|
console.timeLog('log1', 'test');
|
|
console.timeLog('log1', {}, [1, 2, 3]);
|
|
console.timeEnd('log1');
|
|
|
|
console.assert(false, '%s should', 'console.assert', 'not throw');
|
|
assert.strictEqual(errStrings[errStrings.length - 1],
|
|
'Assertion failed: console.assert should not throw\n');
|
|
|
|
console.assert(false);
|
|
assert.strictEqual(errStrings[errStrings.length - 1], 'Assertion failed\n');
|
|
|
|
console.assert(true, 'this should not throw');
|
|
|
|
console.assert(true);
|
|
|
|
assert.strictEqual(strings.length, process.stdout.writeTimes);
|
|
assert.strictEqual(errStrings.length, process.stderr.writeTimes);
|
|
restoreStdout();
|
|
restoreStderr();
|
|
|
|
// Verify that console.timeEnd() doesn't leave dead links
|
|
const timesMapSize = console._times.size;
|
|
console.time('label1');
|
|
console.time('label2');
|
|
console.time('label3');
|
|
console.timeEnd('label1');
|
|
console.timeEnd('label2');
|
|
console.timeEnd('label3');
|
|
assert.strictEqual(console._times.size, timesMapSize);
|
|
|
|
const expectedStrings = [
|
|
'foo', 'foo bar', 'foo bar hop', "{ slashes: '\\\\\\\\' }", 'inspect'
|
|
];
|
|
|
|
for (const expected of expectedStrings) {
|
|
assert.strictEqual(strings.shift(), `${expected}\n`);
|
|
assert.strictEqual(errStrings.shift(), `${expected}\n`);
|
|
}
|
|
|
|
for (const expected of expectedStrings) {
|
|
assert.strictEqual(strings.shift(), `${expected}\n`);
|
|
assert.strictEqual(errStrings.shift(), `${expected}\n`);
|
|
}
|
|
|
|
for (const expected of expectedStrings) {
|
|
assert.strictEqual(strings.shift(), `${expected}\n`);
|
|
}
|
|
|
|
assert.strictEqual(strings.shift(),
|
|
"{\n foo: 'bar',\n [Symbol(nodejs.util.inspect.custom)]:" +
|
|
' [Function: [nodejs.util.inspect.custom]]\n}\n');
|
|
assert.strictEqual(strings.shift(),
|
|
"{\n foo: 'bar',\n [Symbol(nodejs.util.inspect.custom)]:" +
|
|
' [Function: [nodejs.util.inspect.custom]]\n}\n');
|
|
assert.ok(strings.shift().includes('foo: [Object]'));
|
|
assert.strictEqual(strings.shift().includes('baz'), false);
|
|
assert.strictEqual(strings.shift(), 'inspect inspect\n');
|
|
assert.ok(strings[0].includes('foo: { bar: { baz:'));
|
|
assert.ok(strings[0].includes('quux'));
|
|
assert.ok(strings.shift().includes('quux: true'));
|
|
|
|
assert.ok(/^label: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^__proto__: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^constructor: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^hasOwnProperty: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
|
|
// Verify that console.time() coerces label values to strings as expected
|
|
assert.ok(/^: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^\[object Object\]: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^\[object Object\]: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^null: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^default: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^default: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^NaN: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
|
|
assert.ok(/^log1: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
assert.ok(/^log1: \d+\.\d{3}ms test$/.test(strings.shift().trim()));
|
|
assert.ok(/^log1: \d+\.\d{3}ms {} \[ 1, 2, 3 ]$/.test(strings.shift().trim()));
|
|
assert.ok(/^log1: \d+\.\d{3}ms$/.test(strings.shift().trim()));
|
|
|
|
// Make sure that we checked all strings
|
|
assert.strictEqual(strings.length, 0);
|
|
|
|
assert.strictEqual(errStrings.shift().split('\n').shift(),
|
|
'Trace: This is a {"formatted":"trace"} 10 foo');
|
|
|
|
// Hijack stderr to catch `process.emitWarning` which is using
|
|
// `process.nextTick`
|
|
hijackStderr(common.mustCall(function(data) {
|
|
restoreStderr();
|
|
|
|
// stderr.write will catch sync error, so use `process.nextTick` here
|
|
process.nextTick(function() {
|
|
assert.strictEqual(data.includes('noLabel'), true);
|
|
});
|
|
}));
|