From c24deca89be5908539bed6e80747266343d76cf1 Mon Sep 17 00:00:00 2001 From: Rich Trott Date: Mon, 2 Aug 2021 21:41:19 -0700 Subject: [PATCH] debugger: prevent simultaneous heap snapshots Fixes: https://github.com/nodejs/node/issues/39555 --- lib/internal/debugger/inspect_repl.js | 11 ++++- .../test-debugger-takeHeapSnapshot-race.js | 48 ------------------- .../sequential/test-debugger-heap-profiler.js | 4 ++ 3 files changed, 14 insertions(+), 49 deletions(-) delete mode 100644 test/known_issues/test-debugger-takeHeapSnapshot-race.js diff --git a/lib/internal/debugger/inspect_repl.js b/lib/internal/debugger/inspect_repl.js index a93a8dfecf2720..401887cbda5c74 100644 --- a/lib/internal/debugger/inspect_repl.js +++ b/lib/internal/debugger/inspect_repl.js @@ -325,6 +325,7 @@ function createRepl(inspector) { const history = { control: [], debug: [] }; const watchedExpressions = []; const knownBreakpoints = []; + let heapSnapshotPromise = null; let pauseOnExceptionState = 'none'; let lastCommand; @@ -961,7 +962,13 @@ function createRepl(inspector) { }, takeHeapSnapshot(filename = 'node.heapsnapshot') { - return new Promise((resolve, reject) => { + if (heapSnapshotPromise) { + print( + 'Cannot take heap snapshot because another snapshot is in progress.' + ); + return heapSnapshotPromise; + } + heapSnapshotPromise = new Promise((resolve, reject) => { const absoluteFile = Path.resolve(filename); const writer = FS.createWriteStream(absoluteFile); let sizeWritten = 0; @@ -983,6 +990,7 @@ function createRepl(inspector) { writer.end(() => { teardown(); print(`Wrote snapshot: ${absoluteFile}`); + heapSnapshotPromise = null; resolve(); }); } @@ -1006,6 +1014,7 @@ function createRepl(inspector) { HeapProfiler.takeHeapSnapshot({ reportProgress: true }), onResolve, onReject); }); + return heapSnapshotPromise; }, get watchers() { diff --git a/test/known_issues/test-debugger-takeHeapSnapshot-race.js b/test/known_issues/test-debugger-takeHeapSnapshot-race.js deleted file mode 100644 index c1974135046671..00000000000000 --- a/test/known_issues/test-debugger-takeHeapSnapshot-race.js +++ /dev/null @@ -1,48 +0,0 @@ -'use strict'; -const common = require('../common'); - -// Refs: https://github.com/nodejs/node/issues/39555 - -// After this issue is fixed, this can perhaps be integrated into -// test/sequential/test-debugger-heap-profiler.js as it shares almost all -// the same code. - -// These skips should be uncommented once the issue is fixed. -// common.skipIfInspectorDisabled(); - -// if (!common.isMainThread) { -// common.skip('process.chdir() is not available in workers'); -// } - -// This assert.fail() can be removed once the issue is fixed. -if (!common.hasCrypto || !process.features.inspector) { - require('assert').fail('crypto is not available'); -} - -const fixtures = require('../common/fixtures'); -const startCLI = require('../common/debugger'); -const tmpdir = require('../common/tmpdir'); - -tmpdir.refresh(); -process.chdir(tmpdir.path); - -const { readFileSync } = require('fs'); - -const filename = 'node.heapsnapshot'; - -// Check that two simultaneous snapshots don't step all over each other. -{ - const cli = startCLI([fixtures.path('debugger/empty.js')]); - - function onFatal(error) { - cli.quit(); - throw error; - } - - return cli.waitForInitialBreak() - .then(() => cli.waitForPrompt()) - .then(() => cli.command('takeHeapSnapshot(); takeHeapSnapshot()')) - .then(() => JSON.parse(readFileSync(filename, 'utf8'))) - .then(() => cli.quit()) - .then(null, onFatal); -} diff --git a/test/sequential/test-debugger-heap-profiler.js b/test/sequential/test-debugger-heap-profiler.js index 0f0fdc22fbb3b4..0c8327a64f3c85 100644 --- a/test/sequential/test-debugger-heap-profiler.js +++ b/test/sequential/test-debugger-heap-profiler.js @@ -32,6 +32,10 @@ const filename = 'node.heapsnapshot'; .then(() => cli.waitForPrompt()) .then(() => cli.command('takeHeapSnapshot()')) .then(() => JSON.parse(readFileSync(filename, 'utf8'))) + // Check that two simultaneous snapshots don't step all over each other. + // Refs: https://github.com/nodejs/node/issues/39555 + .then(() => cli.command('takeHeapSnapshot(); takeHeapSnapshot()')) + .then(() => JSON.parse(readFileSync(filename, 'utf8'))) .then(() => cli.quit()) .then(null, onFatal); }