Skip to content

Commit 542ff38

Browse files
authored
do not use shell integration for Python subshell if Python setting is disabled (#24418)
Resolves: #24422 If the user has explicitly set their Python shell integration setting to false, then do not use shell integration inside the sub-shell (Python Terminal REPL in this case), regardless of upper shell integration status (Terminal shell integration setting in vscode). We should still be safe from manual installation of shell integration with the setting disabled(https://code.visualstudio.com/docs/terminal/shell-integration#_manual-installation) since we are still reading the value of terminal.shellIntegration, and looking at Python's shell integration setting first to begin with.
1 parent ceabc46 commit 542ff38

File tree

6 files changed

+102
-11
lines changed

6 files changed

+102
-11
lines changed

src/client/common/terminal/service.ts

+13-3
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ import {
2020
TerminalShellType,
2121
} from './types';
2222
import { traceVerbose } from '../../logging';
23+
import { getConfiguration } from '../vscodeApis/workspaceApis';
2324

2425
@injectable()
2526
export class TerminalService implements ITerminalService, Disposable {
@@ -64,7 +65,7 @@ export class TerminalService implements ITerminalService, Disposable {
6465
this.terminal!.show(true);
6566
}
6667

67-
await this.executeCommand(text);
68+
await this.executeCommand(text, false);
6869
}
6970
/** @deprecated */
7071
public async sendText(text: string): Promise<void> {
@@ -74,7 +75,10 @@ export class TerminalService implements ITerminalService, Disposable {
7475
}
7576
this.terminal!.sendText(text);
7677
}
77-
public async executeCommand(commandLine: string): Promise<TerminalShellExecution | undefined> {
78+
public async executeCommand(
79+
commandLine: string,
80+
isPythonShell: boolean,
81+
): Promise<TerminalShellExecution | undefined> {
7882
const terminal = this.terminal!;
7983
if (!this.options?.hideFromUser) {
8084
terminal.show(true);
@@ -98,7 +102,13 @@ export class TerminalService implements ITerminalService, Disposable {
98102
await promise;
99103
}
100104

101-
if (terminal.shellIntegration) {
105+
const config = getConfiguration('python');
106+
const pythonrcSetting = config.get<boolean>('terminal.shellIntegration.enabled');
107+
if (isPythonShell && !pythonrcSetting) {
108+
// If user has explicitly disabled SI for Python, use sendText for inside Terminal REPL.
109+
terminal.sendText(commandLine);
110+
return undefined;
111+
} else if (terminal.shellIntegration) {
102112
const execution = terminal.shellIntegration.executeCommand(commandLine);
103113
traceVerbose(`Shell Integration is enabled, executeCommand: ${commandLine}`);
104114
return execution;

src/client/common/terminal/syncTerminalService.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -145,8 +145,8 @@ export class SynchronousTerminalService implements ITerminalService, Disposable
145145
public sendText(text: string): Promise<void> {
146146
return this.terminalService.sendText(text);
147147
}
148-
public executeCommand(commandLine: string): Promise<TerminalShellExecution | undefined> {
149-
return this.terminalService.executeCommand(commandLine);
148+
public executeCommand(commandLine: string, isPythonShell: boolean): Promise<TerminalShellExecution | undefined> {
149+
return this.terminalService.executeCommand(commandLine, isPythonShell);
150150
}
151151
public show(preserveFocus?: boolean | undefined): Promise<void> {
152152
return this.terminalService.show(preserveFocus);

src/client/common/terminal/types.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ export interface ITerminalService extends IDisposable {
5454
): Promise<void>;
5555
/** @deprecated */
5656
sendText(text: string): Promise<void>;
57-
executeCommand(commandLine: string): Promise<TerminalShellExecution | undefined>;
57+
executeCommand(commandLine: string, isPythonShell: boolean): Promise<TerminalShellExecution | undefined>;
5858
show(preserveFocus?: boolean): Promise<void>;
5959
}
6060

src/client/terminals/codeExecution/terminalCodeExecution.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,7 @@ export class TerminalCodeExecutionProvider implements ICodeExecutionService {
5959
this.configurationService.updateSetting('REPL.enableREPLSmartSend', false, resource);
6060
}
6161
} else {
62-
await this.getTerminalService(resource).executeCommand(code);
62+
await this.getTerminalService(resource).executeCommand(code, true);
6363
}
6464
}
6565

src/test/common/terminals/service.unit.test.ts

+81
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33

44
import { expect } from 'chai';
55
import * as path from 'path';
6+
import * as sinon from 'sinon';
67
import * as TypeMoq from 'typemoq';
78
import {
89
Disposable,
@@ -22,6 +23,7 @@ import { IDisposableRegistry } from '../../../client/common/types';
2223
import { IServiceContainer } from '../../../client/ioc/types';
2324
import { ITerminalAutoActivation } from '../../../client/terminals/types';
2425
import { createPythonInterpreter } from '../../utils/interpreters';
26+
import * as workspaceApis from '../../../client/common/vscodeApis/workspaceApis';
2527

2628
suite('Terminal Service', () => {
2729
let service: TerminalService;
@@ -37,6 +39,9 @@ suite('Terminal Service', () => {
3739
let terminalShellIntegration: TypeMoq.IMock<TerminalShellIntegration>;
3840
let onDidEndTerminalShellExecutionEmitter: EventEmitter<TerminalShellExecutionEndEvent>;
3941
let event: TerminalShellExecutionEndEvent;
42+
let getConfigurationStub: sinon.SinonStub;
43+
let pythonConfig: TypeMoq.IMock<WorkspaceConfiguration>;
44+
let editorConfig: TypeMoq.IMock<WorkspaceConfiguration>;
4045

4146
setup(() => {
4247
terminal = TypeMoq.Mock.ofType<VSCodeTerminal>();
@@ -88,12 +93,22 @@ suite('Terminal Service', () => {
8893
mockServiceContainer.setup((c) => c.get(IWorkspaceService)).returns(() => workspaceService.object);
8994
mockServiceContainer.setup((c) => c.get(ITerminalActivator)).returns(() => terminalActivator.object);
9095
mockServiceContainer.setup((c) => c.get(ITerminalAutoActivation)).returns(() => terminalAutoActivator.object);
96+
getConfigurationStub = sinon.stub(workspaceApis, 'getConfiguration');
97+
pythonConfig = TypeMoq.Mock.ofType<WorkspaceConfiguration>();
98+
editorConfig = TypeMoq.Mock.ofType<WorkspaceConfiguration>();
99+
getConfigurationStub.callsFake((section: string) => {
100+
if (section === 'python') {
101+
return pythonConfig.object;
102+
}
103+
return editorConfig.object;
104+
});
91105
});
92106
teardown(() => {
93107
if (service) {
94108
service.dispose();
95109
}
96110
disposables.filter((item) => !!item).forEach((item) => item.dispose());
111+
sinon.restore();
97112
});
98113

99114
test('Ensure terminal is disposed', async () => {
@@ -103,13 +118,15 @@ suite('Terminal Service', () => {
103118
const os: string = 'windows';
104119
service = new TerminalService(mockServiceContainer.object);
105120
const shellPath = 'powershell.exe';
121+
// TODO: switch over legacy Terminal code to use workspace getConfiguration from workspaceApis instead of directly from vscode.workspace
106122
workspaceService
107123
.setup((w) => w.getConfiguration(TypeMoq.It.isValue('terminal.integrated.shell')))
108124
.returns(() => {
109125
const workspaceConfig = TypeMoq.Mock.ofType<WorkspaceConfiguration>();
110126
workspaceConfig.setup((c) => c.get(os)).returns(() => shellPath);
111127
return workspaceConfig.object;
112128
});
129+
pythonConfig.setup((p) => p.get('terminal.shellIntegration.enabled')).returns(() => false);
113130

114131
platformService.setup((p) => p.isWindows).returns(() => os === 'windows');
115132
platformService.setup((p) => p.isLinux).returns(() => os === 'linux');
@@ -134,6 +151,7 @@ suite('Terminal Service', () => {
134151
});
135152

136153
test('Ensure command is sent to terminal and it is shown', async () => {
154+
pythonConfig.setup((p) => p.get('terminal.shellIntegration.enabled')).returns(() => false);
137155
terminalHelper
138156
.setup((helper) => helper.getEnvironmentActivationCommands(TypeMoq.It.isAny(), TypeMoq.It.isAny()))
139157
.returns(() => Promise.resolve(undefined));
@@ -171,6 +189,69 @@ suite('Terminal Service', () => {
171189
terminal.verify((t) => t.sendText(TypeMoq.It.isValue(textToSend)), TypeMoq.Times.exactly(1));
172190
});
173191

192+
test('Ensure sendText is used when Python shell integration is disabled', async () => {
193+
pythonConfig
194+
.setup((p) => p.get('terminal.shellIntegration.enabled'))
195+
.returns(() => false)
196+
.verifiable(TypeMoq.Times.once());
197+
198+
terminalHelper
199+
.setup((helper) => helper.getEnvironmentActivationCommands(TypeMoq.It.isAny(), TypeMoq.It.isAny()))
200+
.returns(() => Promise.resolve(undefined));
201+
service = new TerminalService(mockServiceContainer.object);
202+
const textToSend = 'Some Text';
203+
terminalHelper.setup((h) => h.identifyTerminalShell(TypeMoq.It.isAny())).returns(() => TerminalShellType.bash);
204+
terminalManager.setup((t) => t.createTerminal(TypeMoq.It.isAny())).returns(() => terminal.object);
205+
206+
service.ensureTerminal();
207+
service.executeCommand(textToSend, true);
208+
209+
terminal.verify((t) => t.show(TypeMoq.It.isValue(true)), TypeMoq.Times.exactly(1));
210+
terminal.verify((t) => t.sendText(TypeMoq.It.isValue(textToSend)), TypeMoq.Times.exactly(1));
211+
});
212+
213+
test('Ensure sendText is called when terminal.shellIntegration enabled but Python shell integration disabled', async () => {
214+
pythonConfig
215+
.setup((p) => p.get('terminal.shellIntegration.enabled'))
216+
.returns(() => false)
217+
.verifiable(TypeMoq.Times.once());
218+
219+
terminalHelper
220+
.setup((helper) => helper.getEnvironmentActivationCommands(TypeMoq.It.isAny(), TypeMoq.It.isAny()))
221+
.returns(() => Promise.resolve(undefined));
222+
service = new TerminalService(mockServiceContainer.object);
223+
const textToSend = 'Some Text';
224+
terminalHelper.setup((h) => h.identifyTerminalShell(TypeMoq.It.isAny())).returns(() => TerminalShellType.bash);
225+
terminalManager.setup((t) => t.createTerminal(TypeMoq.It.isAny())).returns(() => terminal.object);
226+
227+
service.ensureTerminal();
228+
service.executeCommand(textToSend, true);
229+
230+
terminal.verify((t) => t.show(TypeMoq.It.isValue(true)), TypeMoq.Times.exactly(1));
231+
terminal.verify((t) => t.sendText(TypeMoq.It.isValue(textToSend)), TypeMoq.Times.exactly(1));
232+
});
233+
234+
test('Ensure sendText is NOT called when Python shell integration and terminal shell integration are both enabled', async () => {
235+
pythonConfig
236+
.setup((p) => p.get('terminal.shellIntegration.enabled'))
237+
.returns(() => true)
238+
.verifiable(TypeMoq.Times.once());
239+
240+
terminalHelper
241+
.setup((helper) => helper.getEnvironmentActivationCommands(TypeMoq.It.isAny(), TypeMoq.It.isAny()))
242+
.returns(() => Promise.resolve(undefined));
243+
service = new TerminalService(mockServiceContainer.object);
244+
const textToSend = 'Some Text';
245+
terminalHelper.setup((h) => h.identifyTerminalShell(TypeMoq.It.isAny())).returns(() => TerminalShellType.bash);
246+
terminalManager.setup((t) => t.createTerminal(TypeMoq.It.isAny())).returns(() => terminal.object);
247+
248+
service.ensureTerminal();
249+
service.executeCommand(textToSend, true);
250+
251+
terminal.verify((t) => t.show(TypeMoq.It.isValue(true)), TypeMoq.Times.exactly(1));
252+
terminal.verify((t) => t.sendText(TypeMoq.It.isValue(textToSend)), TypeMoq.Times.never());
253+
});
254+
174255
test('Ensure terminal is not shown if `hideFromUser` option is set to `true`', async () => {
175256
terminalHelper
176257
.setup((helper) => helper.getEnvironmentActivationCommands(TypeMoq.It.isAny(), TypeMoq.It.isAny()))

src/test/terminals/codeExecution/terminalCodeExec.unit.test.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -643,10 +643,10 @@ suite('Terminal - Code Execution', () => {
643643
terminalSettings.setup((t) => t.launchArgs).returns(() => terminalArgs);
644644

645645
await executor.execute('cmd1');
646-
terminalService.verify(async (t) => t.executeCommand('cmd1'), TypeMoq.Times.once());
646+
terminalService.verify(async (t) => t.executeCommand('cmd1', true), TypeMoq.Times.once());
647647

648648
await executor.execute('cmd2');
649-
terminalService.verify(async (t) => t.executeCommand('cmd2'), TypeMoq.Times.once());
649+
terminalService.verify(async (t) => t.executeCommand('cmd2', true), TypeMoq.Times.once());
650650
});
651651

652652
test('Ensure code is sent to the same terminal for a particular resource', async () => {
@@ -668,10 +668,10 @@ suite('Terminal - Code Execution', () => {
668668
terminalSettings.setup((t) => t.launchArgs).returns(() => terminalArgs);
669669

670670
await executor.execute('cmd1', resource);
671-
terminalService.verify(async (t) => t.executeCommand('cmd1'), TypeMoq.Times.once());
671+
terminalService.verify(async (t) => t.executeCommand('cmd1', true), TypeMoq.Times.once());
672672

673673
await executor.execute('cmd2', resource);
674-
terminalService.verify(async (t) => t.executeCommand('cmd2'), TypeMoq.Times.once());
674+
terminalService.verify(async (t) => t.executeCommand('cmd2', true), TypeMoq.Times.once());
675675
});
676676
});
677677
});

0 commit comments

Comments
 (0)