diff --git a/src/client/common/terminal/service.ts b/src/client/common/terminal/service.ts index 45ce9afac47e..f20ad8496bdf 100644 --- a/src/client/common/terminal/service.ts +++ b/src/client/common/terminal/service.ts @@ -2,17 +2,18 @@ // Licensed under the MIT License. import { inject, injectable } from 'inversify'; -import { CancellationToken, Disposable, Event, EventEmitter, Terminal, TerminalShellExecution } from 'vscode'; +import { CancellationToken, Disposable, Event, EventEmitter, Terminal, window } from 'vscode'; import '../../common/extensions'; import { IInterpreterService } from '../../interpreter/contracts'; import { IServiceContainer } from '../../ioc/types'; import { captureTelemetry } from '../../telemetry'; import { EventName } from '../../telemetry/constants'; -import { ITerminalAutoActivation } from '../../terminals/types'; +import { ICodeExecutionService, ITerminalAutoActivation } from '../../terminals/types'; import { ITerminalManager } from '../application/types'; import { _SCRIPTS_DIR } from '../process/internal/scripts/constants'; import { IConfigurationService, IDisposableRegistry } from '../types'; import { + IExecuteCommandResult, ITerminalActivator, ITerminalHelper, ITerminalService, @@ -57,14 +58,18 @@ export class TerminalService implements ITerminalService, Disposable { }); } } - public async sendCommand(command: string, args: string[], _?: CancellationToken): Promise { + public async sendCommand( + command: string, + args: string[], + _?: CancellationToken, + ): Promise { await this.ensureTerminal(); const text = this.terminalHelper.buildCommandForTerminal(this.terminalShellType, command, args); if (!this.options?.hideFromUser) { this.terminal!.show(true); } - await this.executeCommand(text); + return this.executeCommand(text); } /** @deprecated */ public async sendText(text: string): Promise { @@ -74,7 +79,7 @@ export class TerminalService implements ITerminalService, Disposable { } this.terminal!.sendText(text); } - public async executeCommand(commandLine: string): Promise { + public async executeCommand(commandLine: string): Promise { const terminal = this.terminal!; if (!this.options?.hideFromUser) { terminal.show(true); @@ -97,11 +102,26 @@ export class TerminalService implements ITerminalService, Disposable { }); await promise; } - + // python in a shell , exit code is undefined . startCommand event happen, we call end command event if (terminal.shellIntegration) { + // TODO: Await the python REPL execute promise here. So we know python repl launched for sure before executing other python code. + // So we would not be interrupted. + + await this.serviceContainer.get(ICodeExecutionService).replActive; + const execution = terminal.shellIntegration.executeCommand(commandLine); traceVerbose(`Shell Integration is enabled, executeCommand: ${commandLine}`); - return execution; + return { + execution, + exitCode: new Promise((resolve) => { + const disposable = window.onDidEndTerminalShellExecution((event) => { + if (event.execution === execution) { + disposable.dispose(); + resolve(event.exitCode); + } + }); + }), + }; } else { terminal.sendText(commandLine); traceVerbose(`Shell Integration is disabled, sendText: ${commandLine}`); diff --git a/src/client/common/terminal/syncTerminalService.ts b/src/client/common/terminal/syncTerminalService.ts index 60f8ed7a6847..265635cf18b6 100644 --- a/src/client/common/terminal/syncTerminalService.ts +++ b/src/client/common/terminal/syncTerminalService.ts @@ -4,7 +4,7 @@ 'use strict'; import { inject } from 'inversify'; -import { CancellationToken, Disposable, Event, TerminalShellExecution } from 'vscode'; +import { CancellationToken, Disposable, Event } from 'vscode'; import { IInterpreterService } from '../../interpreter/contracts'; import { traceVerbose } from '../../logging'; import { PythonEnvironment } from '../../pythonEnvironments/info'; @@ -14,7 +14,7 @@ import * as internalScripts from '../process/internal/scripts'; import { createDeferred, Deferred } from '../utils/async'; import { noop } from '../utils/misc'; import { TerminalService } from './service'; -import { ITerminalService } from './types'; +import { IExecuteCommandResult, ITerminalService } from './types'; enum State { notStarted = 0, @@ -124,7 +124,7 @@ export class SynchronousTerminalService implements ITerminalService, Disposable args: string[], cancel?: CancellationToken, swallowExceptions: boolean = true, - ): Promise { + ): Promise { if (!cancel) { return this.terminalService.sendCommand(command, args); } @@ -145,7 +145,7 @@ export class SynchronousTerminalService implements ITerminalService, Disposable public sendText(text: string): Promise { return this.terminalService.sendText(text); } - public executeCommand(commandLine: string): Promise { + public async executeCommand(commandLine: string): Promise { return this.terminalService.executeCommand(commandLine); } public show(preserveFocus?: boolean | undefined): Promise { diff --git a/src/client/common/terminal/types.ts b/src/client/common/terminal/types.ts index db2b7f80e4b1..0a85e3f38545 100644 --- a/src/client/common/terminal/types.ts +++ b/src/client/common/terminal/types.ts @@ -51,15 +51,20 @@ export interface ITerminalService extends IDisposable { args: string[], cancel?: CancellationToken, swallowExceptions?: boolean, - ): Promise; + ): Promise; /** @deprecated */ sendText(text: string): Promise; - executeCommand(commandLine: string): Promise; + executeCommand(commandLine: string): Promise; show(preserveFocus?: boolean): Promise; } export const ITerminalServiceFactory = Symbol('ITerminalServiceFactory'); +export interface IExecuteCommandResult { + execution: TerminalShellExecution; + exitCode: Promise; +} + export type TerminalCreationOptions = { /** * Object with environment variables that will be added to the Terminal. diff --git a/src/client/terminals/codeExecution/terminalCodeExecution.ts b/src/client/terminals/codeExecution/terminalCodeExecution.ts index 3cba6141763b..09c9669e52a6 100644 --- a/src/client/terminals/codeExecution/terminalCodeExecution.ts +++ b/src/client/terminals/codeExecution/terminalCodeExecution.ts @@ -24,7 +24,7 @@ import { sendTelemetryEvent } from '../../telemetry'; export class TerminalCodeExecutionProvider implements ICodeExecutionService { private hasRanOutsideCurrentDrive = false; protected terminalTitle!: string; - private replActive?: Promise; + replActive?: Promise; constructor( @inject(ITerminalServiceFactory) protected readonly terminalServiceFactory: ITerminalServiceFactory, @@ -73,6 +73,7 @@ export class TerminalCodeExecutionProvider implements ICodeExecutionService { this.replActive = new Promise(async (resolve) => { const replCommandArgs = await this.getExecutableInfo(resource); let listener: IDisposable; + // TODO: There's a race condition here as well; the send text from file command may happen before this timeout resolves Promise.race([ new Promise((resolve) => setTimeout(() => resolve(true), 3000)), new Promise((resolve) => { @@ -99,9 +100,17 @@ export class TerminalCodeExecutionProvider implements ICodeExecutionService { } resolve(true); }); + // python + // python shell, exit code undefined. + // ondidStartShellExecution happens, si fires ondidEndsi - await terminalService.sendCommand(replCommandArgs.command, replCommandArgs.args); + // TODO: Store this promise and make sure no further commands are executed without awaiting .exitCode + const replExecution = await terminalService.sendCommand(replCommandArgs.command, replCommandArgs.args); // need to make sure this is resolved before starting executing something in repl. + + // TODO: Should we await replyExecution.exitCode here? + await replExecution?.exitCode; }); + this.disposables.push( terminalService.onDidCloseTerminal(() => { this.replActive = undefined; diff --git a/src/client/terminals/types.ts b/src/client/terminals/types.ts index 5fd129e8fe89..705606ec722e 100644 --- a/src/client/terminals/types.ts +++ b/src/client/terminals/types.ts @@ -7,6 +7,7 @@ import { IDisposable, Resource } from '../common/types'; export const ICodeExecutionService = Symbol('ICodeExecutionService'); export interface ICodeExecutionService { + replActive?: Promise; execute(code: string, resource?: Uri): Promise; executeFile(file: Uri, options?: { newTerminalPerFile: boolean }): Promise; initializeRepl(resource?: Uri): Promise;