Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add an API to report progress of environment discovery in two phases #19125

Merged
merged 21 commits into from
May 17, 2022
Merged
1 change: 1 addition & 0 deletions news/1 Enhancements/19103.md
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Added a proposed API to report progress of environment discovery in two phases.
7 changes: 6 additions & 1 deletion src/client/apiTypes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import { Event, Uri } from 'vscode';
import { Resource } from './common/types';
import { IDataViewerDataProvider, IJupyterUriProvider } from './jupyter/types';
import { EnvPathType, PythonEnvKind } from './pythonEnvironments/base/info';
import { GetRefreshEnvironmentsOptions, ProgressNotificationEvent } from './pythonEnvironments/base/locator';

/*
* Do not introduce any breaking changes to this API.
Expand Down Expand Up @@ -203,11 +204,15 @@ export interface IProposedExtensionAPI {
* is triggered.
*/
refreshEnvironment(options?: RefreshEnvironmentsOptions): Promise<EnvPathType[] | undefined>;
/**
* Fires with details of the current discovery progress, i.e when it starts, finishes or any other relevant stage.
*/
readonly onRefreshProgress: Event<ProgressNotificationEvent>;
/**
* Returns a promise for the ongoing refresh. Returns `undefined` if there are no active
* refreshes going on.
*/
getRefreshPromise(): Promise<void> | undefined;
getRefreshPromise(options?: GetRefreshEnvironmentsOptions): Promise<void> | undefined;
/**
* This event is triggered when the known environment list changes, like when a environment
* is found, existing environment is removed, or some details changed on an environment.
Expand Down
7 changes: 3 additions & 4 deletions src/client/interpreter/contracts.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { CodeLensProvider, ConfigurationTarget, Disposable, Event, TextDocument,
import { FileChangeType } from '../common/platform/fileSystemWatcher';
import { Resource } from '../common/types';
import { PythonEnvSource } from '../pythonEnvironments/base/info';
import { PythonLocatorQuery } from '../pythonEnvironments/base/locator';
import { ProgressNotificationEvent, PythonLocatorQuery } from '../pythonEnvironments/base/locator';
import { CondaEnvironmentInfo } from '../pythonEnvironments/common/environmentManagers/conda';
import { EnvironmentType, PythonEnvironment } from '../pythonEnvironments/info';

Expand All @@ -16,9 +16,9 @@ export type PythonEnvironmentsChangedEvent = {

export const IComponentAdapter = Symbol('IComponentAdapter');
export interface IComponentAdapter {
readonly onRefreshStart: Event<void>;
readonly onProgress: Event<ProgressNotificationEvent>;
triggerRefresh(query?: PythonLocatorQuery & { clearCache?: boolean }, trigger?: 'auto' | 'ui'): Promise<void>;
readonly refreshPromise: Promise<void> | undefined;
getRefreshPromise(): Promise<void> | undefined;
readonly onChanged: Event<PythonEnvironmentsChangedEvent>;
// VirtualEnvPrompt
onDidCreate(resource: Resource, callback: () => void): Disposable;
Expand Down Expand Up @@ -63,7 +63,6 @@ export interface ICondaService {

export const IInterpreterService = Symbol('IInterpreterService');
export interface IInterpreterService {
readonly onRefreshStart: Event<void>;
triggerRefresh(query?: PythonLocatorQuery & { clearCache?: boolean }, trigger?: 'auto' | 'ui'): Promise<void>;
readonly refreshPromise: Promise<void> | undefined;
readonly onDidChangeInterpreters: Event<PythonEnvironmentsChangedEvent>;
Expand Down
14 changes: 9 additions & 5 deletions src/client/interpreter/display/progressDisplay.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import { IDisposableRegistry } from '../../common/types';
import { createDeferred, Deferred } from '../../common/utils/async';
import { Interpreters } from '../../common/utils/localize';
import { traceDecoratorVerbose } from '../../logging';
import { ProgressReportStage } from '../../pythonEnvironments/base/locator';
import { IComponentAdapter } from '../contracts';

// The parts of IComponentAdapter used here.
Expand All @@ -30,11 +31,14 @@ export class InterpreterLocatorProgressStatubarHandler implements IExtensionSing
) {}

public async activate(): Promise<void> {
this.pyenvs.onRefreshStart(
() => {
this.showProgress();
if (this.pyenvs.refreshPromise) {
this.pyenvs.refreshPromise.then(() => this.hideProgress());
this.pyenvs.onProgress(
(event) => {
if (event.stage === ProgressReportStage.discoveryStarted) {
this.showProgress();
const refreshPromise = this.pyenvs.getRefreshPromise();
if (refreshPromise) {
refreshPromise.then(() => this.hideProgress());
}
}
},
this,
Expand Down
6 changes: 1 addition & 5 deletions src/client/interpreter/interpreterService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -40,10 +40,6 @@ export class InterpreterService implements Disposable, IInterpreterService {
return this.pyenvs.hasInterpreters(filter);
}

public get onRefreshStart(): Event<void> {
return this.pyenvs.onRefreshStart;
}

public triggerRefresh(
query?: PythonLocatorQuery & { clearCache?: boolean },
trigger?: 'auto' | 'ui',
Expand All @@ -52,7 +48,7 @@ export class InterpreterService implements Disposable, IInterpreterService {
}

public get refreshPromise(): Promise<void> | undefined {
return this.pyenvs.refreshPromise;
return this.pyenvs.getRefreshPromise();
}

public get onDidChangeInterpreter(): Event<void> {
Expand Down
7 changes: 4 additions & 3 deletions src/client/proposedApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import { IInterpreterService } from './interpreter/contracts';
import { IServiceContainer } from './ioc/types';
import { PythonEnvInfo } from './pythonEnvironments/base/info';
import { getEnvPath } from './pythonEnvironments/base/info/env';
import { IDiscoveryAPI } from './pythonEnvironments/base/locator';
import { GetRefreshEnvironmentsOptions, IDiscoveryAPI } from './pythonEnvironments/base/locator';

const onDidInterpretersChangedEvent = new EventEmitter<EnvironmentsChangedParams[]>();
export function reportInterpretersChanged(e: EnvironmentsChangedParams[]): void {
Expand Down Expand Up @@ -106,12 +106,13 @@ export function buildProposedApi(
const paths = discoveryApi.getEnvs().map((e) => getEnvPath(e.executable.filename, e.location));
return Promise.resolve(paths);
},
getRefreshPromise(): Promise<void> | undefined {
return discoveryApi.refreshPromise;
getRefreshPromise(options?: GetRefreshEnvironmentsOptions): Promise<void> | undefined {
return discoveryApi.getRefreshPromise(options);
},
onDidChangeExecutionDetails: interpreterService.onDidChangeInterpreterConfiguration,
onDidEnvironmentsChanged: onDidInterpretersChangedEvent.event,
onDidActiveEnvironmentChanged: onDidActiveInterpreterChangedEvent.event,
onRefreshProgress: discoveryApi.onProgress,
},
};
return proposed;
Expand Down
16 changes: 11 additions & 5 deletions src/client/pythonEnvironments/api.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,17 @@
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

import { Event } from 'vscode';
import { StopWatch } from '../common/utils/stopWatch';
import { sendTelemetryEvent } from '../telemetry';
import { EventName } from '../telemetry/constants';
import { getEnvPath } from './base/info/env';
import { IDiscoveryAPI, PythonLocatorQuery } from './base/locator';
import {
GetRefreshEnvironmentsOptions,
IDiscoveryAPI,
ProgressNotificationEvent,
PythonLocatorQuery,
} from './base/locator';

export type GetLocatorFunc = () => Promise<IDiscoveryAPI>;

Expand All @@ -26,12 +32,12 @@ class PythonEnvironments implements IDiscoveryAPI {
this.locator = await this.getLocator();
}

public get onRefreshStart() {
return this.locator.onRefreshStart;
public get onProgress(): Event<ProgressNotificationEvent> {
return this.locator.onProgress;
}

public get refreshPromise() {
return this.locator.refreshPromise;
public getRefreshPromise(options?: GetRefreshEnvironmentsOptions) {
return this.locator.getRefreshPromise(options);
}

public get onChanged() {
Expand Down
28 changes: 24 additions & 4 deletions src/client/pythonEnvironments/base/locator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,23 @@ export interface IPythonEnvsIterator<I = PythonEnvInfo> extends IAsyncIterableIt
* If this property is not provided then it means the iterator does
* not support updates.
*/
onUpdated?: Event<PythonEnvUpdatedEvent<I> | null>;
onUpdated?: Event<PythonEnvUpdatedEvent<I> | ProgressNotificationEvent>;
}

export enum ProgressReportStage {
discoveryStarted = 'discoveryStarted',
allPathsDiscovered = 'allPathsDiscovered',
discoveryFinished = 'discoveryFinished',
}

export type ProgressNotificationEvent = {
stage: ProgressReportStage;
};

export function isProgressEvent<I = PythonEnvInfo>(
event: PythonEnvUpdatedEvent<I> | ProgressNotificationEvent,
): event is ProgressNotificationEvent {
return 'stage' in event;
}

/**
Expand Down Expand Up @@ -170,11 +186,15 @@ interface IResolver {

export interface IResolvingLocator<I = PythonEnvInfo> extends IResolver, ILocator<I> {}

export interface GetRefreshEnvironmentsOptions {
stage?: ProgressReportStage;
}

export interface IDiscoveryAPI {
/**
* Fires when the known list of environments starts refreshing, i.e when discovery starts or restarts.
* Fires with details of the current discovery progress, i.e when it starts, finishes or any other relevant stage.
*/
readonly onRefreshStart: Event<void>;
readonly onProgress: Event<ProgressNotificationEvent>;
/**
* Fires with details if the known list changes.
*/
Expand All @@ -183,7 +203,7 @@ export interface IDiscoveryAPI {
* Resolves once environment list has finished refreshing, i.e all environments are
* discovered. Carries `undefined` if there is no refresh currently going on.
*/
readonly refreshPromise: Promise<void> | undefined;
getRefreshPromise(options?: GetRefreshEnvironmentsOptions): Promise<void> | undefined;
/**
* Triggers a new refresh for query if there isn't any already running.
*/
Expand Down
16 changes: 13 additions & 3 deletions src/client/pythonEnvironments/base/locatorUtils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,14 @@ import { createDeferred } from '../../common/utils/async';
import { getURIFilter } from '../../common/utils/misc';
import { traceVerbose } from '../../logging';
import { PythonEnvInfo } from './info';
import { IPythonEnvsIterator, PythonEnvUpdatedEvent, PythonLocatorQuery } from './locator';
import {
IPythonEnvsIterator,
isProgressEvent,
ProgressNotificationEvent,
ProgressReportStage,
PythonEnvUpdatedEvent,
PythonLocatorQuery,
} from './locator';

/**
* Create a filter function to match the given query.
Expand Down Expand Up @@ -71,8 +78,11 @@ export async function getEnvs<I = PythonEnvInfo>(iterator: IPythonEnvsIterator<I
if (iterator.onUpdated === undefined) {
updatesDone.resolve();
} else {
const listener = iterator.onUpdated((event: PythonEnvUpdatedEvent<I> | null) => {
if (event === null) {
const listener = iterator.onUpdated((event: PythonEnvUpdatedEvent<I> | ProgressNotificationEvent) => {
if (isProgressEvent(event)) {
if (event.stage !== ProgressReportStage.discoveryFinished) {
return;
}
updatesDone.resolve();
listener.dispose();
} else {
Expand Down
28 changes: 20 additions & 8 deletions src/client/pythonEnvironments/base/locators.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,15 @@
import { chain } from '../../common/utils/async';
import { Disposables } from '../../common/utils/resourceLifecycle';
import { PythonEnvInfo } from './info';
import { ILocator, IPythonEnvsIterator, PythonEnvUpdatedEvent, PythonLocatorQuery } from './locator';
import {
ILocator,
IPythonEnvsIterator,
isProgressEvent,
ProgressNotificationEvent,
ProgressReportStage,
PythonEnvUpdatedEvent,
PythonLocatorQuery,
} from './locator';
import { PythonEnvsWatchers } from './watchers';

/**
Expand All @@ -19,17 +27,21 @@ export function combineIterators<I>(iterators: IPythonEnvsIterator<I>[]): IPytho
}

// eslint-disable-next-line @typescript-eslint/no-explicit-any
result.onUpdated = (handleEvent: (e: PythonEnvUpdatedEvent<I> | null) => any) => {
result.onUpdated = (handleEvent: (e: PythonEnvUpdatedEvent<I> | ProgressNotificationEvent) => any) => {
const disposables = new Disposables();
let numActive = events.length;
events.forEach((event) => {
const disposable = event!((e: PythonEnvUpdatedEvent<I> | null) => {
const disposable = event!((e: PythonEnvUpdatedEvent<I> | ProgressNotificationEvent) => {
// NOSONAR
if (e === null) {
numActive -= 1;
if (numActive === 0) {
// All the sub-events are done so we're done.
handleEvent(null);
if (isProgressEvent(e)) {
if (e.stage === ProgressReportStage.discoveryFinished) {
numActive -= 1;
if (numActive === 0) {
// All the sub-events are done so we're done.
handleEvent({ stage: ProgressReportStage.discoveryFinished });
}
} else {
handleEvent({ stage: e.stage });
}
} else {
handleEvent(e);
Expand Down
Loading