From 9f16b7f3326c5eb4819128944ae72fb5bff80d52 Mon Sep 17 00:00:00 2001 From: Brian White Date: Sat, 14 Mar 2020 07:55:44 -0400 Subject: [PATCH] lib: improve debuglog() performance PR-URL: https://github.com/nodejs/node/pull/32260 Reviewed-By: James M Snell --- lib/_http_agent.js | 4 ++- lib/_http_common.js | 4 ++- lib/_stream_readable.js | 4 ++- lib/_tls_wrap.js | 4 ++- lib/child_process.js | 7 +++++- lib/https.js | 4 ++- lib/internal/http2/core.js | 4 ++- lib/internal/js_stream_socket.js | 7 +++++- lib/internal/main/worker_thread.js | 4 ++- lib/internal/modules/cjs/helpers.js | 4 ++- lib/internal/modules/cjs/loader.js | 4 ++- .../modules/esm/create_dynamic_module.js | 4 ++- lib/internal/modules/esm/module_map.js | 4 ++- lib/internal/modules/esm/translators.js | 6 ++--- lib/internal/policy/manifest.js | 4 ++- lib/internal/repl/history.js | 4 ++- lib/internal/repl/utils.js | 4 ++- .../source_map/prepare_stack_trace.js | 4 ++- lib/internal/source_map/source_map_cache.js | 4 ++- lib/internal/stream_base_commons.js | 4 ++- lib/internal/timers.js | 4 ++- lib/internal/util/debuglog.js | 25 +++++++++++-------- lib/internal/worker.js | 4 ++- lib/internal/worker/io.js | 4 ++- lib/net.js | 4 ++- lib/repl.js | 4 ++- lib/timers.js | 4 ++- test/sequential/test-util-debug.js | 4 ++- 28 files changed, 101 insertions(+), 40 deletions(-) diff --git a/lib/_http_agent.js b/lib/_http_agent.js index 7fb4a4929c35f5..d75d9f341347c6 100644 --- a/lib/_http_agent.js +++ b/lib/_http_agent.js @@ -30,7 +30,9 @@ const { const net = require('net'); const EventEmitter = require('events'); -const debug = require('internal/util/debuglog').debuglog('http'); +let debug = require('internal/util/debuglog').debuglog('http', (fn) => { + debug = fn; +}); const { async_id_symbol } = require('internal/async_hooks').symbols; const kOnKeylog = Symbol('onkeylog'); // New Agent code. diff --git a/lib/_http_common.js b/lib/_http_common.js index cb8eced4a85c0a..8e2387e4536769 100644 --- a/lib/_http_common.js +++ b/lib/_http_common.js @@ -42,7 +42,9 @@ const { readStop } = incoming; -const debug = require('internal/util/debuglog').debuglog('http'); +let debug = require('internal/util/debuglog').debuglog('http', (fn) => { + debug = fn; +}); const kIncomingMessage = Symbol('IncomingMessage'); const kOnHeaders = HTTPParser.kOnHeaders | 0; diff --git a/lib/_stream_readable.js b/lib/_stream_readable.js index 17f433f03d939d..dcf0b95a311e45 100644 --- a/lib/_stream_readable.js +++ b/lib/_stream_readable.js @@ -39,7 +39,9 @@ const EE = require('events'); const Stream = require('stream'); const { Buffer } = require('buffer'); -const debug = require('internal/util/debuglog').debuglog('stream'); +let debug = require('internal/util/debuglog').debuglog('stream', (fn) => { + debug = fn; +}); const BufferList = require('internal/streams/buffer_list'); const destroyImpl = require('internal/streams/destroy'); const { diff --git a/lib/_tls_wrap.js b/lib/_tls_wrap.js index 45d8c3722f44b2..154d6a0a1fd9ea 100644 --- a/lib/_tls_wrap.js +++ b/lib/_tls_wrap.js @@ -45,7 +45,9 @@ const tls = require('tls'); const common = require('_tls_common'); const JSStreamSocket = require('internal/js_stream_socket'); const { Buffer } = require('buffer'); -const debug = require('internal/util/debuglog').debuglog('tls'); +let debug = require('internal/util/debuglog').debuglog('tls', (fn) => { + debug = fn; +}); const { TCP, constants: TCPConstants } = internalBinding('tcp_wrap'); const tls_wrap = internalBinding('tls_wrap'); const { Pipe, constants: PipeConstants } = internalBinding('pipe_wrap'); diff --git a/lib/child_process.js b/lib/child_process.js index 77bce9c386bb3f..6897ad47b03d5b 100644 --- a/lib/child_process.js +++ b/lib/child_process.js @@ -37,7 +37,12 @@ const { getSystemErrorName } = require('internal/util'); const { isArrayBufferView } = require('internal/util/types'); -const debug = require('internal/util/debuglog').debuglog('child_process'); +let debug = require('internal/util/debuglog').debuglog( + 'child_process', + (fn) => { + debug = fn; + } +); const { Buffer } = require('buffer'); const { Pipe, constants: PipeConstants } = internalBinding('pipe_wrap'); const { diff --git a/lib/https.js b/lib/https.js index 2db1ee5453b356..47d59d1984ebbb 100644 --- a/lib/https.js +++ b/lib/https.js @@ -37,7 +37,9 @@ const { kServerResponse } = require('_http_server'); const { ClientRequest } = require('_http_client'); -const debug = require('internal/util/debuglog').debuglog('https'); +let debug = require('internal/util/debuglog').debuglog('https', (fn) => { + debug = fn; +}); const { URL, urlToOptions, searchParamsSymbol } = require('internal/url'); const { IncomingMessage, ServerResponse } = require('http'); const { kIncomingMessage } = require('_http_common'); diff --git a/lib/internal/http2/core.js b/lib/internal/http2/core.js index 37c06db8f4141e..7515227fa82be4 100644 --- a/lib/internal/http2/core.js +++ b/lib/internal/http2/core.js @@ -152,7 +152,9 @@ const { UV_EOF } = internalBinding('uv'); const { StreamPipe } = internalBinding('stream_pipe'); const { _connectionListener: httpConnectionListener } = http; -const debug = require('internal/util/debuglog').debuglog('http2'); +let debug = require('internal/util/debuglog').debuglog('http2', (fn) => { + debug = fn; +}); const { getOptionValue } = require('internal/options'); // TODO(addaleax): See if this can be made more efficient by figuring out diff --git a/lib/internal/js_stream_socket.js b/lib/internal/js_stream_socket.js index 1a805ae830bdb2..faad988e820ffa 100644 --- a/lib/internal/js_stream_socket.js +++ b/lib/internal/js_stream_socket.js @@ -9,7 +9,12 @@ const assert = require('internal/assert'); const { Socket } = require('net'); const { JSStream } = internalBinding('js_stream'); const uv = internalBinding('uv'); -const debug = require('internal/util/debuglog').debuglog('stream_socket'); +let debug = require('internal/util/debuglog').debuglog( + 'stream_socket', + (fn) => { + debug = fn; + } +); const { owner_symbol } = require('internal/async_hooks').symbols; const { ERR_STREAM_WRAP } = require('internal/errors').codes; diff --git a/lib/internal/main/worker_thread.js b/lib/internal/main/worker_thread.js index 0c68c5eab21cf8..d2557ebc4b30d7 100644 --- a/lib/internal/main/worker_thread.js +++ b/lib/internal/main/worker_thread.js @@ -49,7 +49,9 @@ const { } = require('internal/process/execution'); const publicWorker = require('worker_threads'); -const debug = require('internal/util/debuglog').debuglog('worker'); +let debug = require('internal/util/debuglog').debuglog('worker', (fn) => { + debug = fn; +}); const assert = require('internal/assert'); diff --git a/lib/internal/modules/cjs/helpers.js b/lib/internal/modules/cjs/helpers.js index 2f62c8623e27be..cc9e40f4401e61 100644 --- a/lib/internal/modules/cjs/helpers.js +++ b/lib/internal/modules/cjs/helpers.js @@ -15,7 +15,9 @@ const path = require('path'); const { pathToFileURL, fileURLToPath } = require('internal/url'); const { URL } = require('url'); -const debug = require('internal/util/debuglog').debuglog('module'); +let debug = require('internal/util/debuglog').debuglog('module', (fn) => { + debug = fn; +}); function loadNativeModule(filename, request) { const mod = NativeModule.map.get(filename); diff --git a/lib/internal/modules/cjs/loader.js b/lib/internal/modules/cjs/loader.js index c5d6c4221db076..7a004eef59d29d 100644 --- a/lib/internal/modules/cjs/loader.js +++ b/lib/internal/modules/cjs/loader.js @@ -224,7 +224,9 @@ ObjectDefineProperty(Module, 'wrapper', { } }); -const debug = require('internal/util/debuglog').debuglog('module'); +let debug = require('internal/util/debuglog').debuglog('module', (fn) => { + debug = fn; +}); Module._debug = deprecate(debug, 'Module._debug is deprecated.', 'DEP0077'); // Given a module name, and a list of paths to test, returns the first diff --git a/lib/internal/modules/esm/create_dynamic_module.js b/lib/internal/modules/esm/create_dynamic_module.js index 38344f69640e6c..e831db8daaf476 100644 --- a/lib/internal/modules/esm/create_dynamic_module.js +++ b/lib/internal/modules/esm/create_dynamic_module.js @@ -8,7 +8,9 @@ const { Set, } = primordials; -const debug = require('internal/util/debuglog').debuglog('esm'); +let debug = require('internal/util/debuglog').debuglog('esm', (fn) => { + debug = fn; +}); function createImport(impt, index) { const imptPath = JSONStringify(impt); diff --git a/lib/internal/modules/esm/module_map.js b/lib/internal/modules/esm/module_map.js index 9b7d86581e4159..9c51c7c48779f3 100644 --- a/lib/internal/modules/esm/module_map.js +++ b/lib/internal/modules/esm/module_map.js @@ -4,7 +4,9 @@ const ModuleJob = require('internal/modules/esm/module_job'); const { SafeMap, } = primordials; -const debug = require('internal/util/debuglog').debuglog('esm'); +let debug = require('internal/util/debuglog').debuglog('esm', (fn) => { + debug = fn; +}); const { ERR_INVALID_ARG_TYPE } = require('internal/errors').codes; const { validateString } = require('internal/validators'); diff --git a/lib/internal/modules/esm/translators.js b/lib/internal/modules/esm/translators.js index f314ba96b3476c..bb3528eddde964 100644 --- a/lib/internal/modules/esm/translators.js +++ b/lib/internal/modules/esm/translators.js @@ -30,7 +30,9 @@ const { defaultTransformSource } = require( const createDynamicModule = require( 'internal/modules/esm/create_dynamic_module'); const { fileURLToPath, URL } = require('url'); -const { debuglog } = require('internal/util/debuglog'); +let debug = require('internal/util/debuglog').debuglog('esm', (fn) => { + debug = fn; +}); const { emitExperimentalWarning } = require('internal/util'); const { ERR_UNKNOWN_BUILTIN_MODULE, @@ -43,8 +45,6 @@ const { getOptionValue } = require('internal/options'); const experimentalImportMetaResolve = getOptionValue('--experimental-import-meta-resolve'); -const debug = debuglog('esm'); - const translators = new SafeMap(); exports.translators = translators; diff --git a/lib/internal/policy/manifest.js b/lib/internal/policy/manifest.js index 7283606478d383..0531f198f21971 100644 --- a/lib/internal/policy/manifest.js +++ b/lib/internal/policy/manifest.js @@ -22,7 +22,9 @@ const { ERR_MANIFEST_INVALID_RESOURCE_FIELD, ERR_MANIFEST_UNKNOWN_ONERROR, } = require('internal/errors').codes; -const debug = require('internal/util/debuglog').debuglog('policy'); +let debug = require('internal/util/debuglog').debuglog('policy', (fn) => { + debug = fn; +}); const SRI = require('internal/policy/sri'); const crypto = require('crypto'); const { Buffer } = require('buffer'); diff --git a/lib/internal/repl/history.js b/lib/internal/repl/history.js index 845d3fdebd96db..eb0394cc3b110e 100644 --- a/lib/internal/repl/history.js +++ b/lib/internal/repl/history.js @@ -8,7 +8,9 @@ const { Interface } = require('readline'); const path = require('path'); const fs = require('fs'); const os = require('os'); -const debug = require('internal/util/debuglog').debuglog('repl'); +let debug = require('internal/util/debuglog').debuglog('repl', (fn) => { + debug = fn; +}); const { clearTimeout, setTimeout } = require('timers'); // XXX(chrisdickinson): The 15ms debounce value is somewhat arbitrary. diff --git a/lib/internal/repl/utils.js b/lib/internal/repl/utils.js index 476760a08d5934..5c77cf9aa7aa02 100644 --- a/lib/internal/repl/utils.js +++ b/lib/internal/repl/utils.js @@ -40,7 +40,9 @@ const { inspect, } = require('internal/util/inspect'); -const debug = require('internal/util/debuglog').debuglog('repl'); +let debug = require('internal/util/debuglog').debuglog('repl', (fn) => { + debug = fn; +}); const previewOptions = { colors: false, diff --git a/lib/internal/source_map/prepare_stack_trace.js b/lib/internal/source_map/prepare_stack_trace.js index 037a8dc53e0855..8a10470d077fed 100644 --- a/lib/internal/source_map/prepare_stack_trace.js +++ b/lib/internal/source_map/prepare_stack_trace.js @@ -4,7 +4,9 @@ const { Error, } = primordials; -const debug = require('internal/util/debuglog').debuglog('source_map'); +let debug = require('internal/util/debuglog').debuglog('source_map', (fn) => { + debug = fn; +}); const { findSourceMap } = require('internal/source_map/source_map_cache'); const { kNoOverride, diff --git a/lib/internal/source_map/source_map_cache.js b/lib/internal/source_map/source_map_cache.js index 06b1a2a5f52289..8952940920b88f 100644 --- a/lib/internal/source_map/source_map_cache.js +++ b/lib/internal/source_map/source_map_cache.js @@ -22,7 +22,9 @@ function ObjectGetValueSafe(obj, key) { // See https://sourcemaps.info/spec.html for SourceMap V3 specification. const { Buffer } = require('buffer'); -const debug = require('internal/util/debuglog').debuglog('source_map'); +let debug = require('internal/util/debuglog').debuglog('source_map', (fn) => { + debug = fn; +}); const { dirname, resolve } = require('path'); const fs = require('fs'); const { getOptionValue } = require('internal/options'); diff --git a/lib/internal/stream_base_commons.js b/lib/internal/stream_base_commons.js index 9ea73dd15735e3..233741285d2c72 100644 --- a/lib/internal/stream_base_commons.js +++ b/lib/internal/stream_base_commons.js @@ -37,7 +37,9 @@ const kAfterAsyncWrite = Symbol('kAfterAsyncWrite'); const kHandle = Symbol('kHandle'); const kSession = Symbol('kSession'); -const debug = require('internal/util/debuglog').debuglog('stream'); +let debug = require('internal/util/debuglog').debuglog('stream', (fn) => { + debug = fn; +}); const kBuffer = Symbol('kBuffer'); const kBufferGen = Symbol('kBufferGen'); const kBufferCb = Symbol('kBufferCb'); diff --git a/lib/internal/timers.js b/lib/internal/timers.js index 6ebbdc5b5f3026..c478a43de9b7d8 100644 --- a/lib/internal/timers.js +++ b/lib/internal/timers.js @@ -115,7 +115,9 @@ const L = require('internal/linkedlist'); const PriorityQueue = require('internal/priority_queue'); const { inspect } = require('internal/util/inspect'); -const debug = require('internal/util/debuglog').debuglog('timer'); +let debug = require('internal/util/debuglog').debuglog('timer', (fn) => { + debug = fn; +}); // *Must* match Environment::ImmediateInfo::Fields in src/env.h. const kCount = 0; diff --git a/lib/internal/util/debuglog.js b/lib/internal/util/debuglog.js index aab3aa9aaf678d..35c1d918aedc5c 100644 --- a/lib/internal/util/debuglog.js +++ b/lib/internal/util/debuglog.js @@ -33,6 +33,8 @@ function emitWarningIfNeeded(set) { } } +function noop() {} + function debuglogImpl(set) { set = set.toUpperCase(); if (debugs[set] === undefined) { @@ -46,7 +48,7 @@ function debuglogImpl(set) { process.stderr.write(format('%s %s: %s\n', set, coloredPID, msg)); }; } else { - debugs[set] = null; + debugs[set] = noop; } } return debugs[set]; @@ -56,16 +58,17 @@ function debuglogImpl(set) { // so it needs to be called lazily in top scopes of internal modules // that may be loaded before these run time states are allowed to // be accessed. -function debuglog(set) { - let debug; - return function(...args) { - if (debug === undefined) { - // Only invokes debuglogImpl() when the debug function is - // called for the first time. - debug = debuglogImpl(set); - } - if (debug !== null) - debug(...args); +function debuglog(set, cb) { + let debug = (...args) => { + // Only invokes debuglogImpl() when the debug function is + // called for the first time. + debug = debuglogImpl(set); + if (typeof cb === 'function') + cb(debug); + debug(...args); + }; + return (...args) => { + debug(...args); }; } diff --git a/lib/internal/worker.js b/lib/internal/worker.js index 6c6b462205dfaa..abdaacf205385e 100644 --- a/lib/internal/worker.js +++ b/lib/internal/worker.js @@ -68,7 +68,9 @@ const kOnErrorMessage = Symbol('kOnErrorMessage'); const kParentSideStdio = Symbol('kParentSideStdio'); const SHARE_ENV = SymbolFor('nodejs.worker_threads.SHARE_ENV'); -const debug = require('internal/util/debuglog').debuglog('worker'); +let debug = require('internal/util/debuglog').debuglog('worker', (fn) => { + debug = fn; +}); let cwdCounter; diff --git a/lib/internal/worker/io.js b/lib/internal/worker/io.js index 0a97f17595faa8..60dd8cd67d6186 100644 --- a/lib/internal/worker/io.js +++ b/lib/internal/worker/io.js @@ -31,7 +31,9 @@ const { const { Readable, Writable } = require('stream'); const EventEmitter = require('events'); const { inspect } = require('internal/util/inspect'); -const debug = require('internal/util/debuglog').debuglog('worker'); +let debug = require('internal/util/debuglog').debuglog('worker', (fn) => { + debug = fn; +}); const kIncrementsPortRef = Symbol('kIncrementsPortRef'); const kName = Symbol('kName'); diff --git a/lib/net.js b/lib/net.js index d0d7cfcb92addc..06e08361515745 100644 --- a/lib/net.js +++ b/lib/net.js @@ -35,7 +35,9 @@ const { const EventEmitter = require('events'); const stream = require('stream'); const { inspect } = require('internal/util/inspect'); -const debug = require('internal/util/debuglog').debuglog('net'); +let debug = require('internal/util/debuglog').debuglog('net', (fn) => { + debug = fn; +}); const { deprecate } = require('internal/util'); const { isIP, diff --git a/lib/repl.js b/lib/repl.js index 29a061d2f9e5ac..2c0dcc7757a70e 100644 --- a/lib/repl.js +++ b/lib/repl.js @@ -88,7 +88,9 @@ const { Console } = require('console'); const cjsLoader = require('internal/modules/cjs/loader'); const { Module: CJSModule } = cjsLoader; const domain = require('domain'); -const debug = require('internal/util/debuglog').debuglog('repl'); +let debug = require('internal/util/debuglog').debuglog('repl', (fn) => { + debug = fn; +}); const { codes: { ERR_CANNOT_WATCH_SIGINT, diff --git a/lib/timers.js b/lib/timers.js index b0dfbda6c52d5e..de50e56a11909a 100644 --- a/lib/timers.js +++ b/lib/timers.js @@ -56,7 +56,9 @@ const { promisify: { custom: customPromisify }, deprecate } = require('internal/util'); -const debug = require('internal/util/debuglog').debuglog('timer'); +let debug = require('internal/util/debuglog').debuglog('timer', (fn) => { + debug = fn; +}); const { validateCallback } = require('internal/validators'); const { diff --git a/test/sequential/test-util-debug.js b/test/sequential/test-util-debug.js index 6cad46f101c0af..8bf5ae5e5abbb0 100644 --- a/test/sequential/test-util-debug.js +++ b/test/sequential/test-util-debug.js @@ -118,7 +118,9 @@ function child(section) { value: tty.WriteStream.prototype.hasColors }); // eslint-disable-next-line no-restricted-syntax - const debug = util.debuglog(section); + const debug = util.debuglog(section, common.mustCall((cb) => { + assert.strictEqual(typeof cb, 'function'); + })); debug('this', { is: 'a' }, /debugging/); debug('num=%d str=%s obj=%j', 1, 'a', { foo: 'bar' }); console.log('ok');