Skip to content

Commit

Permalink
try support both node_debug and trace by introducing a new module
Browse files Browse the repository at this point in the history
  • Loading branch information
H4ad committed Mar 28, 2024
1 parent ac6e394 commit d7b53fb
Show file tree
Hide file tree
Showing 7 changed files with 357 additions and 119 deletions.
129 changes: 34 additions & 95 deletions lib/internal/console/constructor.js
Original file line number Diff line number Diff line change
Expand Up @@ -68,12 +68,10 @@ const {
CHAR_UPPERCASE_C: kTraceCount,
} = require('internal/constants');
const kCounts = Symbol('counts');
const { time, timeEnd, timeLog } = require('internal/util/trace_timer');

const kTraceConsoleCategory = 'node,node.console';

const kSecond = 1000;
const kMinute = 60 * kSecond;
const kHour = 60 * kMinute;
const kMaxGroupIndentation = 1000;

// Lazy loaded for startup performance.
Expand All @@ -99,6 +97,7 @@ const kBindStreamsEager = Symbol('kBindStreamsEager');
const kBindStreamsLazy = Symbol('kBindStreamsLazy');
const kUseStdout = Symbol('kUseStdout');
const kUseStderr = Symbol('kUseStderr');
const kInternalTimeLogImpl = Symbol('kInternalTimeLogImpl');

const optionsMap = new SafeWeakMap();
function Console(options /* or: stdout, stderr, ignoreErrors = true */) {
Expand Down Expand Up @@ -138,14 +137,14 @@ function Console(options /* or: stdout, stderr, ignoreErrors = true */) {

if (groupIndentation !== undefined) {
validateInteger(groupIndentation, 'groupIndentation',
0, kMaxGroupIndentation);
0, kMaxGroupIndentation);

Check failure on line 140 in lib/internal/console/constructor.js

View workflow job for this annotation

GitHub Actions / lint-js-and-md

Expected indentation of 20 spaces but found 6
}

if (inspectOptions !== undefined) {
validateObject(inspectOptions, 'options.inspectOptions');

if (inspectOptions.colors !== undefined &&
options.colorMode !== undefined) {
options.colorMode !== undefined) {
throw new ERR_INCOMPATIBLE_OPTION_PAIR(
'options.inspectOptions.color', 'colorMode');
}
Expand Down Expand Up @@ -190,7 +189,7 @@ ObjectDefineProperties(Console.prototype, {
__proto__: null,
...consolePropAttributes,
// Eager version for the Console constructor
value: function(stdout, stderr) {
value: function (stdout, stderr) {

Check failure on line 192 in lib/internal/console/constructor.js

View workflow job for this annotation

GitHub Actions / lint-js-and-md

Unexpected space before function parentheses
ObjectDefineProperties(this, {
'_stdout': { __proto__: null, ...consolePropAttributes, value: stdout },
'_stderr': { __proto__: null, ...consolePropAttributes, value: stderr },
Expand All @@ -202,7 +201,7 @@ ObjectDefineProperties(Console.prototype, {
...consolePropAttributes,
// Lazily load the stdout and stderr from an object so we don't
// create the stdio streams when they are not even accessed
value: function(object) {
value: function (object) {
let stdout;
let stderr;
ObjectDefineProperties(this, {
Expand Down Expand Up @@ -232,7 +231,7 @@ ObjectDefineProperties(Console.prototype, {
[kBindProperties]: {
__proto__: null,
...consolePropAttributes,
value: function(ignoreErrors, colorMode, groupIndentation = 2) {
value: function (ignoreErrors, colorMode, groupIndentation = 2) {
ObjectDefineProperties(this, {
'_stdoutErrorHandler': {
__proto__: null,
Expand Down Expand Up @@ -273,7 +272,7 @@ ObjectDefineProperties(Console.prototype, {
[kWriteToConsole]: {
__proto__: null,
...consolePropAttributes,
value: function(streamSymbol, string) {
value: function (streamSymbol, string) {
const ignoreErrors = this._ignoreErrors;
const groupIndent = this[kGroupIndent];

Expand Down Expand Up @@ -316,7 +315,7 @@ ObjectDefineProperties(Console.prototype, {
[kGetInspectOptions]: {
__proto__: null,
...consolePropAttributes,
value: function(stream) {
value: function (stream) {
let color = this[kColorMode];
if (color === 'auto') {
color = lazyUtilColors().shouldColorize(stream);
Expand All @@ -336,7 +335,7 @@ ObjectDefineProperties(Console.prototype, {
[kFormatForStdout]: {
__proto__: null,
...consolePropAttributes,
value: function(args) {
value: function (args) {
const opts = this[kGetInspectOptions](this._stdout);
ArrayPrototypeUnshift(args, opts);
return ReflectApply(formatWithOptions, null, args);
Expand All @@ -345,7 +344,7 @@ ObjectDefineProperties(Console.prototype, {
[kFormatForStderr]: {
__proto__: null,
...consolePropAttributes,
value: function(args) {
value: function (args) {
const opts = this[kGetInspectOptions](this._stderr);
ArrayPrototypeUnshift(args, opts);
return ReflectApply(formatWithOptions, null, args);
Expand Down Expand Up @@ -374,6 +373,14 @@ function createWriteErrorHandler(instance, streamSymbol) {
};
}

function timeLogImpl(label, formatted, args) {
if (args === undefined) {
this.log('%s: %s', label, formatted);
} else {
this.log('%s: %s', label, formatted, ...new SafeArrayIterator(args));
}
}

const consoleMethods = {
log(...args) {
this[kWriteToConsole](kUseStdout, this[kFormatForStdout](args));
Expand All @@ -394,31 +401,21 @@ const consoleMethods = {
},

time(label = 'default') {
// Coerces everything other than Symbol to a string
label = `${label}`;
if (this._times.has(label)) {
process.emitWarning(`Label '${label}' already exists for console.time()`);
return;
}
trace(kTraceBegin, kTraceConsoleCategory, `time::${label}`, 0);
this._times.set(label, process.hrtime());
time(this._times, kTraceConsoleCategory, 'console.time()', label);
},

timeEnd(label = 'default') {
// Coerces everything other than Symbol to a string
label = `${label}`;
const found = timeLogImpl(this, 'timeEnd', label);
trace(kTraceEnd, kTraceConsoleCategory, `time::${label}`, 0);
if (found) {
this._times.delete(label);
}
if (this[kInternalTimeLogImpl] === undefined)
this[kInternalTimeLogImpl] = FunctionPrototypeBind(timeLogImpl, this)

timeEnd(this._times, kTraceConsoleCategory, 'console.timeEnd()', this[kInternalTimeLogImpl], label);
},

timeLog(label = 'default', ...data) {
// Coerces everything other than Symbol to a string
label = `${label}`;
timeLogImpl(this, 'timeLog', label, data);
trace(kTraceInstant, kTraceConsoleCategory, `time::${label}`, 0);
if (this[kInternalTimeLogImpl] === undefined)
this[kInternalTimeLogImpl] = FunctionPrototypeBind(timeLogImpl, this)

timeLog(this._times, kTraceConsoleCategory, 'console.timeLog()', this[kInternalTimeLogImpl], label, data);
},

trace: function trace(...args) {
Expand Down Expand Up @@ -515,9 +512,9 @@ const consoleMethods = {

const _inspect = (v) => {
const depth = v !== null &&
typeof v === 'object' &&
!isArray(v) &&
ObjectKeys(v).length > 2 ? -1 : 0;
typeof v === 'object' &&
!isArray(v) &&
ObjectKeys(v).length > 2 ? -1 : 0;
const opt = {
depth,
maxArrayLength: 3,
Expand Down Expand Up @@ -587,7 +584,7 @@ const consoleMethods = {
for (; i < indexKeyArray.length; i++) {
const item = tabularData[indexKeyArray[i]];
const primitive = item === null ||
(typeof item !== 'function' && typeof item !== 'object');
(typeof item !== 'function' && typeof item !== 'object');
if (properties === undefined && primitive) {
hasPrimitives = true;
valuesKeyArray[i] = _inspect(item);
Expand All @@ -596,7 +593,7 @@ const consoleMethods = {
for (const key of keys) {
map[key] ??= [];
if ((primitive && properties) ||
!ObjectPrototypeHasOwnProperty(item, key))
!ObjectPrototypeHasOwnProperty(item, key))
map[key][i] = '';
else
map[key][i] = _inspect(item[key]);
Expand All @@ -617,71 +614,14 @@ const consoleMethods = {
},
};

// Returns true if label was found
function timeLogImpl(self, name, label, data) {
const time = self._times.get(label);
if (time === undefined) {
process.emitWarning(`No such label '${label}' for console.${name}()`);
return false;
}
const duration = process.hrtime(time);
const ms = duration[0] * 1000 + duration[1] / 1e6;

const formatted = formatTime(ms);

if (data === undefined) {
self.log('%s: %s', label, formatted);
} else {
self.log('%s: %s', label, formatted, ...new SafeArrayIterator(data));
}
return true;
}

function pad(value) {
return StringPrototypePadStart(`${value}`, 2, '0');
}

function formatTime(ms) {
let hours = 0;
let minutes = 0;
let seconds = 0;

if (ms >= kSecond) {
if (ms >= kMinute) {
if (ms >= kHour) {
hours = MathFloor(ms / kHour);
ms = ms % kHour;
}
minutes = MathFloor(ms / kMinute);
ms = ms % kMinute;
}
seconds = ms / kSecond;
}

if (hours !== 0 || minutes !== 0) {
({ 0: seconds, 1: ms } = StringPrototypeSplit(
NumberPrototypeToFixed(seconds, 3),
'.',
));
const res = hours !== 0 ? `${hours}:${pad(minutes)}` : minutes;
return `${res}:${pad(seconds)}.${ms} (${hours !== 0 ? 'h:m' : ''}m:ss.mmm)`;
}

if (seconds !== 0) {
return `${NumberPrototypeToFixed(seconds, 3)}s`;
}

return `${Number(NumberPrototypeToFixed(ms, 3))}ms`;
}

const keyKey = 'Key';
const valuesKey = 'Values';
const indexKey = '(index)';
const iterKey = '(iteration index)';

const isArray = (v) => ArrayIsArray(v) || isTypedArray(v) || isBuffer(v);

function noop() {}
function noop() { }

for (const method of ReflectOwnKeys(consoleMethods))
Console.prototype[method] = consoleMethods[method];
Expand Down Expand Up @@ -728,5 +668,4 @@ module.exports = {
kBindStreamsLazy,
kBindProperties,
initializeGlobalConsole,
formatTime, // exported for tests
};
29 changes: 14 additions & 15 deletions lib/internal/modules/cjs/loader.js
Original file line number Diff line number Diff line change
Expand Up @@ -360,7 +360,7 @@ function setModuleParent(value) {
moduleParentCache.set(this, value);
}

const { debuglog, isDebugEnabled } = require('internal/util/debuglog');
const { debuglog, debugWithTimer } = require('internal/util/debuglog');

let debug = debuglog('module', (fn) => {
debug = fn;
Expand All @@ -372,8 +372,9 @@ let hrtimeBigIntTimingFn = () => {
return hrtimeBigIntTimingFn();
};

let debugTiming = debuglog('timing_module_cjs', (fn) => {
debugTiming = fn;
let { startTimer, endTimer } = debugWithTimer('module_cjs', (start, end) => {
startTimer = start;
endTimer = end;
});

ObjectDefineProperty(Module.prototype, 'parent', {
Expand Down Expand Up @@ -971,10 +972,6 @@ function getExportsForCircularRequire(module) {
return module.exports;
}

function logTiming(request, parent, start) {
debugTiming('[%s] [%s]: %d ms', parent?.id || '', request, Number((hrtimeBigIntTimingFn() - start)) / 1e6);
}

/**
* Load a module from cache if it exists, otherwise create a new module instance.
* 1. If a module already exists in the cache: return its exports object.
Expand All @@ -987,7 +984,9 @@ function logTiming(request, parent, start) {
* @param {boolean} isMain Whether the module is the main entry point
*/
Module._load = function(request, parent, isMain) {
const start = hrtimeBigIntTimingFn();
const label = `[${parent?.id || ''}] [${request}]`;

startTimer(label);

let relResolveCacheIdentifier;
if (parent) {
Expand All @@ -1005,12 +1004,12 @@ Module._load = function(request, parent, isMain) {
if (!cachedModule.loaded) {
const result = getExportsForCircularRequire(cachedModule);

logTiming(request, parent, start);
endTimer(label);

return result;
}

logTiming(request, parent, start);
endTimer(label);
return cachedModule.exports;
}
delete relativeResolveCache[relResolveCacheIdentifier];
Expand All @@ -1027,7 +1026,7 @@ Module._load = function(request, parent, isMain) {

const module = loadBuiltinModule(id, request);

logTiming(request, parent, start);
endTimer(label);
return module.exports;
}

Expand All @@ -1040,21 +1039,21 @@ Module._load = function(request, parent, isMain) {
if (!parseCachedModule || parseCachedModule.loaded) {
const result = getExportsForCircularRequire(cachedModule);

logTiming(request, parent, start);
endTimer(label);

return result;
}
parseCachedModule.loaded = true;
} else {
logTiming(request, parent, start);
endTimer(label);
return cachedModule.exports;
}
}

if (BuiltinModule.canBeRequiredWithoutScheme(filename)) {
const mod = loadBuiltinModule(filename, request);

logTiming(request, parent, start);
endTimer(label);

return mod.exports;
}
Expand Down Expand Up @@ -1103,7 +1102,7 @@ Module._load = function(request, parent, isMain) {
}
}

logTiming(request, parent, start);
endTimer(label);

return module.exports;
};
Expand Down
Loading

0 comments on commit d7b53fb

Please sign in to comment.