diff --git a/src/client/pythonEnvironments/common/environmentManagers/conda.ts b/src/client/pythonEnvironments/common/environmentManagers/conda.ts index bd4aba219416..bc60745dfeff 100644 --- a/src/client/pythonEnvironments/common/environmentManagers/conda.ts +++ b/src/client/pythonEnvironments/common/environmentManagers/conda.ts @@ -495,6 +495,15 @@ export class Conda { ); } + /** + * Retrieves list of directories where conda environments are stored. + */ + @cache(30_000, true, 10_000) + public async getEnvDirs(): Promise { + const info = await this.getInfo(); + return info.envs_dirs ?? []; + } + public async getName(prefix: string, info?: CondaInfo): Promise { info = info ?? (await this.getInfo(true)); if (info.root_prefix && arePathsSame(prefix, info.root_prefix)) { @@ -619,3 +628,8 @@ export class Conda { export function setCondaBinary(executable: string): void { Conda.setConda(executable); } + +export async function getCondaEnvDirs(): Promise { + const conda = await Conda.getConda(); + return conda?.getEnvDirs(); +} diff --git a/src/client/pythonEnvironments/nativeAPI.ts b/src/client/pythonEnvironments/nativeAPI.ts index 7e2f7aa3515b..e069a3746ab6 100644 --- a/src/client/pythonEnvironments/nativeAPI.ts +++ b/src/client/pythonEnvironments/nativeAPI.ts @@ -23,11 +23,11 @@ import { createDeferred, Deferred } from '../common/utils/async'; import { Architecture, getUserHomeDir } from '../common/utils/platform'; import { parseVersion } from './base/info/pythonVersion'; import { cache } from '../common/utils/decorators'; -import { traceError, traceLog, traceWarn } from '../logging'; +import { traceError, traceInfo, traceLog, traceWarn } from '../logging'; import { StopWatch } from '../common/utils/stopWatch'; import { FileChangeType } from '../common/platform/fileSystemWatcher'; import { categoryToKind, NativePythonEnvironmentKind } from './base/locators/common/nativePythonUtils'; -import { setCondaBinary } from './common/environmentManagers/conda'; +import { getCondaEnvDirs, setCondaBinary } from './common/environmentManagers/conda'; import { setPyEnvBinary } from './common/environmentManagers/pyenv'; import { createPythonWatcher, @@ -157,26 +157,53 @@ function getEnvType(kind: PythonEnvKind): PythonEnvType | undefined { } } -function getName(nativeEnv: NativeEnvInfo, kind: PythonEnvKind): string { +function isSubDir(pathToCheck: string | undefined, parents: string[]): boolean { + return parents.some((prefix) => { + if (pathToCheck) { + return path.normalize(pathToCheck).startsWith(path.normalize(prefix)); + } + return false; + }); +} + +function getName(nativeEnv: NativeEnvInfo, kind: PythonEnvKind, condaEnvDirs: string[]): string { if (nativeEnv.name) { return nativeEnv.name; } const envType = getEnvType(kind); - if (nativeEnv.prefix && (envType === PythonEnvType.Conda || envType === PythonEnvType.Virtual)) { + if (nativeEnv.prefix && envType === PythonEnvType.Virtual) { return path.basename(nativeEnv.prefix); } + + if (nativeEnv.prefix && envType === PythonEnvType.Conda) { + if (nativeEnv.name === 'base') { + return 'base'; + } + + const workspaces = (getWorkspaceFolders() ?? []).map((wf) => wf.uri.fsPath); + if (isSubDir(nativeEnv.prefix, workspaces)) { + traceInfo(`Conda env is --prefix environment: ${nativeEnv.prefix}`); + return ''; + } + + if (condaEnvDirs.length > 0 && isSubDir(nativeEnv.prefix, condaEnvDirs)) { + traceInfo(`Conda env is --named environment: ${nativeEnv.prefix}`); + return path.basename(nativeEnv.prefix); + } + } + return ''; } -function toPythonEnvInfo(nativeEnv: NativeEnvInfo): PythonEnvInfo | undefined { +function toPythonEnvInfo(nativeEnv: NativeEnvInfo, condaEnvDirs: string[]): PythonEnvInfo | undefined { if (!validEnv(nativeEnv)) { return undefined; } const kind = categoryToKind(nativeEnv.kind); const arch = toArch(nativeEnv.arch); const version: PythonVersion = parseVersion(nativeEnv.version ?? ''); - const name = getName(nativeEnv, kind); + const name = getName(nativeEnv, kind, condaEnvDirs); const displayName = nativeEnv.version ? getDisplayName(version, kind, arch, name) : nativeEnv.displayName ?? 'Python'; @@ -211,6 +238,9 @@ function toPythonEnvInfo(nativeEnv: NativeEnvInfo): PythonEnvInfo | undefined { } function hasChanged(old: PythonEnvInfo, newEnv: PythonEnvInfo): boolean { + if (old.name !== newEnv.name) { + return true; + } if (old.executable.filename !== newEnv.executable.filename) { return true; } @@ -247,6 +277,8 @@ class NativePythonEnvironments implements IDiscoveryAPI, Disposable { private _disposables: Disposable[] = []; + private _condaEnvDirs: string[] = []; + constructor(private readonly finder: NativePythonFinder) { this._onProgress = new EventEmitter(); this._onChanged = new EventEmitter(); @@ -381,7 +413,7 @@ class NativePythonEnvironments implements IDiscoveryAPI, Disposable { } private addEnv(native: NativeEnvInfo, searchLocation?: Uri): PythonEnvInfo | undefined { - const info = toPythonEnvInfo(native); + const info = toPythonEnvInfo(native, this._condaEnvDirs); if (info) { const old = this._envs.find((item) => item.executable.filename === info.executable.filename); if (old) { @@ -417,6 +449,9 @@ class NativePythonEnvironments implements IDiscoveryAPI, Disposable { } const native = await this.finder.resolve(envPath); if (native) { + if (native.kind === NativePythonEnvironmentKind.Conda && this._condaEnvDirs.length === 0) { + this._condaEnvDirs = (await getCondaEnvDirs()) ?? []; + } return this.addEnv(native); } return undefined;