From 6981eee60eb8e2e9bf2c156ecfea6b73d623dcc9 Mon Sep 17 00:00:00 2001 From: Anna Henningsen Date: Fri, 13 Dec 2024 08:41:17 -0500 Subject: [PATCH] chore(cli-repl): return a cached promise from `.close()` MONGOSH-1943 (#2297) This is mostly just making code a bit more resilient against cases in which we would unintentionally call `.close()` twice. --- packages/cli-repl/src/cli-repl.ts | 94 +++++++++++++++---------------- 1 file changed, 46 insertions(+), 48 deletions(-) diff --git a/packages/cli-repl/src/cli-repl.ts b/packages/cli-repl/src/cli-repl.ts index e69350f6d..a138552fb 100644 --- a/packages/cli-repl/src/cli-repl.ts +++ b/packages/cli-repl/src/cli-repl.ts @@ -129,7 +129,7 @@ export class CliRepl implements MongoshIOProvider { toggleableAnalytics: ToggleableAnalytics = new ToggleableAnalytics(); warnedAboutInaccessibleFiles = false; onExit: (code?: number) => Promise; - closing = false; + closingPromise?: Promise; isContainerizedEnvironment = false; hasOnDiskTelemetryId = false; proxyOptions: DevtoolsProxyOptions = { @@ -1088,56 +1088,54 @@ export class CliRepl implements MongoshIOProvider { * Close all open resources held by this REPL instance. */ async close(): Promise { - markTime(TimingCategories.REPLInstantiation, 'start closing'); - if (this.closing) { - return; - } - this.agent?.destroy(); - if (!this.output.destroyed) { - // Wait for output to be fully flushed before exiting. - if (this.output.writableEnded) { - // .end() has been called but not finished; 'close' will be emitted in that case. - // (This should not typically happen in the context of mongosh, but there's also - // no reason not to handle this case properly.) - try { - await once(this.output, 'close'); - } catch { - /* ignore */ + return (this.closingPromise ??= (async () => { + markTime(TimingCategories.REPLInstantiation, 'start closing'); + this.agent?.destroy(); + if (!this.output.destroyed) { + // Wait for output to be fully flushed before exiting. + if (this.output.writableEnded) { + // .end() has been called but not finished; 'close' will be emitted in that case. + // (This should not typically happen in the context of mongosh, but there's also + // no reason not to handle this case properly.) + try { + await once(this.output, 'close'); + } catch { + /* ignore */ + } + } else { + // .end() has not been called; write an empty chunk and wait for it to be fully written. + await new Promise((resolve) => this.output.write('', resolve)); } - } else { - // .end() has not been called; write an empty chunk and wait for it to be fully written. - await new Promise((resolve) => this.output.write('', resolve)); } - } - markTime(TimingCategories.REPLInstantiation, 'output flushed'); - this.closing = true; - const analytics = this.toggleableAnalytics; - let flushError: string | null = null; - let flushDuration: number | null = null; - if (analytics) { - const flushStart = Date.now(); - try { - await analytics.flush(); - markTime(TimingCategories.Telemetry, 'flushed analytics'); - } catch (err: any) { - flushError = err.message; - } finally { - flushDuration = Date.now() - flushStart; - } - } - this.logWriter?.info( - 'MONGOSH', - mongoLogId(1_000_000_045), - 'analytics', - 'Flushed outstanding data', - { - flushError, - flushDuration, + markTime(TimingCategories.REPLInstantiation, 'output flushed'); + const analytics = this.toggleableAnalytics; + let flushError: string | null = null; + let flushDuration: number | null = null; + if (analytics) { + const flushStart = Date.now(); + try { + await analytics.flush(); + markTime(TimingCategories.Telemetry, 'flushed analytics'); + } catch (err: any) { + flushError = err.message; + } finally { + flushDuration = Date.now() - flushStart; + } } - ); - await this.logWriter?.flush(); - markTime(TimingCategories.Logging, 'flushed log writer'); - this.bus.emit('mongosh:closed'); + this.logWriter?.info( + 'MONGOSH', + mongoLogId(1_000_000_045), + 'analytics', + 'Flushed outstanding data', + { + flushError, + flushDuration, + } + ); + await this.logWriter?.flush(); + markTime(TimingCategories.Logging, 'flushed log writer'); + this.bus.emit('mongosh:closed'); + })()); } /**