forked from microsoft/vscode-python
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathpythonExecutionFactory.ts
More file actions
158 lines (144 loc) · 7.8 KB
/
pythonExecutionFactory.ts
File metadata and controls
158 lines (144 loc) · 7.8 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
import { inject, injectable } from 'inversify';
import { IEnvironmentActivationService } from '../../interpreter/activation/types';
import { IActivatedEnvironmentLaunch, IComponentAdapter } from '../../interpreter/contracts';
import { IServiceContainer } from '../../ioc/types';
import { sendTelemetryEvent } from '../../telemetry';
import { EventName } from '../../telemetry/constants';
import { IFileSystem } from '../platform/types';
import { IConfigurationService, IDisposableRegistry, IInterpreterPathService } from '../types';
import { ProcessService } from './proc';
import { createCondaEnv, createPythonEnv, createMicrosoftStoreEnv } from './pythonEnvironment';
import { createPythonProcessService } from './pythonProcess';
import {
ExecutionFactoryCreateWithEnvironmentOptions,
ExecutionFactoryCreationOptions,
IProcessLogger,
IProcessService,
IProcessServiceFactory,
IPythonEnvironment,
IPythonExecutionFactory,
IPythonExecutionService,
} from './types';
import { IInterpreterAutoSelectionService } from '../../interpreter/autoSelection/types';
import { sleep } from '../utils/async';
import { traceError } from '../../logging';
@injectable()
export class PythonExecutionFactory implements IPythonExecutionFactory {
private readonly disposables: IDisposableRegistry;
private readonly logger: IProcessLogger;
private readonly fileSystem: IFileSystem;
constructor(
@inject(IServiceContainer) private serviceContainer: IServiceContainer,
@inject(IEnvironmentActivationService) private readonly activationHelper: IEnvironmentActivationService,
@inject(IProcessServiceFactory) private readonly processServiceFactory: IProcessServiceFactory,
@inject(IConfigurationService) private readonly configService: IConfigurationService,
@inject(IComponentAdapter) private readonly pyenvs: IComponentAdapter,
@inject(IInterpreterAutoSelectionService) private readonly autoSelection: IInterpreterAutoSelectionService,
@inject(IInterpreterPathService) private readonly interpreterPathExpHelper: IInterpreterPathService,
) {
// Acquire other objects here so that if we are called during dispose they are available.
this.disposables = this.serviceContainer.get<IDisposableRegistry>(IDisposableRegistry);
this.logger = this.serviceContainer.get<IProcessLogger>(IProcessLogger);
this.fileSystem = this.serviceContainer.get<IFileSystem>(IFileSystem);
}
public async create(options: ExecutionFactoryCreationOptions): Promise<IPythonExecutionService> {
let { pythonPath } = options;
if (!pythonPath || pythonPath === 'python') {
const activatedEnvLaunch = this.serviceContainer.get<IActivatedEnvironmentLaunch>(
IActivatedEnvironmentLaunch,
);
await activatedEnvLaunch.selectIfLaunchedViaActivatedEnv();
// If python path wasn't passed in, we need to auto select it and then read it
// from the configuration.
const interpreterPath = this.interpreterPathExpHelper.get(options.resource);
if (!interpreterPath || interpreterPath === 'python') {
// Block on autoselection if no interpreter selected.
// Note autoselection blocks on discovery, so we do not want discovery component
// to block on this code. Discovery component should 'options.pythonPath' before
// calling into this, so this scenario should not happen. But in case consumer
// makes such an error. So break the loop via timeout and log error.
const success = await Promise.race([
this.autoSelection.autoSelectInterpreter(options.resource).then(() => true),
sleep(50000).then(() => false),
]);
if (!success) {
traceError(
'Autoselection timeout out, this is likely a issue with how consumer called execution factory API. Using default python to execute.',
);
}
}
pythonPath = this.configService.getSettings(options.resource).pythonPath;
}
const processService: IProcessService = await this.processServiceFactory.create(options.resource);
const condaExecutionService = await this.createCondaExecutionService(pythonPath, processService);
if (condaExecutionService) {
return condaExecutionService;
}
const windowsStoreInterpreterCheck = this.pyenvs.isMicrosoftStoreInterpreter.bind(this.pyenvs);
const env = (await windowsStoreInterpreterCheck(pythonPath))
? createMicrosoftStoreEnv(pythonPath, processService)
: createPythonEnv(pythonPath, processService, this.fileSystem);
return createPythonService(processService, env);
}
public async createActivatedEnvironment(
options: ExecutionFactoryCreateWithEnvironmentOptions,
): Promise<IPythonExecutionService> {
const envVars = await this.activationHelper.getActivatedEnvironmentVariables(
options.resource,
options.interpreter,
options.allowEnvironmentFetchExceptions,
);
const hasEnvVars = envVars && Object.keys(envVars).length > 0;
sendTelemetryEvent(EventName.PYTHON_INTERPRETER_ACTIVATION_ENVIRONMENT_VARIABLES, undefined, { hasEnvVars });
if (!hasEnvVars) {
return this.create({
resource: options.resource,
pythonPath: options.interpreter ? options.interpreter.path : undefined,
});
}
const pythonPath = options.interpreter
? options.interpreter.path
: this.configService.getSettings(options.resource).pythonPath;
const processService: IProcessService = new ProcessService({ ...envVars });
processService.on('exec', this.logger.logProcess.bind(this.logger));
this.disposables.push(processService);
const condaExecutionService = await this.createCondaExecutionService(pythonPath, processService);
if (condaExecutionService) {
return condaExecutionService;
}
const env = createPythonEnv(pythonPath, processService, this.fileSystem);
return createPythonService(processService, env);
}
public async createCondaExecutionService(
pythonPath: string,
processService: IProcessService,
): Promise<IPythonExecutionService | undefined> {
const condaLocatorService = this.serviceContainer.get<IComponentAdapter>(IComponentAdapter);
const [condaEnvironment] = await Promise.all([condaLocatorService.getCondaEnvironment(pythonPath)]);
if (!condaEnvironment) {
return undefined;
}
const env = await createCondaEnv(condaEnvironment, processService, this.fileSystem);
if (!env) {
return undefined;
}
return createPythonService(processService, env);
}
}
function createPythonService(procService: IProcessService, env: IPythonEnvironment): IPythonExecutionService {
const procs = createPythonProcessService(procService, env);
return {
getInterpreterInformation: () => env.getInterpreterInformation(),
getExecutablePath: () => env.getExecutablePath(),
isModuleInstalled: (m) => env.isModuleInstalled(m),
getModuleVersion: (m) => env.getModuleVersion(m),
getExecutionInfo: (a) => env.getExecutionInfo(a),
execObservable: (a, o) => procs.execObservable(a, o),
execModuleObservable: (m, a, o) => procs.execModuleObservable(m, a, o),
exec: (a, o) => procs.exec(a, o),
execModule: (m, a, o) => procs.execModule(m, a, o),
execForLinter: (m, a, o) => procs.execForLinter(m, a, o),
};
}