Skip to content

Commit

Permalink
plugin: add workspace file api
Browse files Browse the repository at this point in the history
Add `on(will|did)(create|rename|delete)Files` to the plugin's workspace
api. This implementation does not handle the `WorkspaceEdit` apis.

Signed-off-by: Paul Maréchal <paul.marechal@ericsson.com>
  • Loading branch information
paul-marechal committed May 7, 2020
1 parent fa7fb7d commit f2e1c5c
Show file tree
Hide file tree
Showing 18 changed files with 724 additions and 96 deletions.
40 changes: 21 additions & 19 deletions packages/core/src/common/event.ts
Original file line number Diff line number Diff line change
Expand Up @@ -199,9 +199,8 @@ export class Emitter<T = any> {

return result;
}, {
maxListeners: Emitter.LEAK_WARNING_THRESHHOLD
}
);
maxListeners: Emitter.LEAK_WARNING_THRESHHOLD
});
}
return this._event;
}
Expand Down Expand Up @@ -295,34 +294,37 @@ export class Emitter<T = any> {
}
}

export interface WaitUntilEvent {
/* eslint-disable @typescript-eslint/no-explicit-any */
export interface WaitUntilEvent<T = any> {
/**
* Allows to pause the event loop until the provided thenable resolved.
*
* *Note:* It can only be called during event dispatch and not in an asynchronous manner
*
* @param thenable A thenable that delays execution.
*/
waitUntil(thenable: Promise<any>): void;
/* eslint-enable @typescript-eslint/no-explicit-any */
waitUntil(thenable: Promise<T>): void;
}
export namespace WaitUntilEvent {
export async function fire<T extends WaitUntilEvent>(
emitter: Emitter<T>,
event: Pick<T, Exclude<keyof T, 'waitUntil'>>,
timeout: number | undefined = undefined
): Promise<void> {
const waitables: Promise<void>[] = [];
/**
* Fires an event with a `waitUntil` field and handles its semantics on your behalf.
*
* @param emitter
* @param event
* @param timeout
* @returns returned values of promises passed to `waitUntil`, `undefined` on timeout.
*/
export async function fire<E extends WaitUntilEvent<V>, V = any>(emitter: Emitter<E>, event: Omit<E, 'waitUntil'>): Promise<V[]>;
export async function fire<E extends WaitUntilEvent<V>, V = any>(emitter: Emitter<E>, event: Omit<E, 'waitUntil'>, timeout: number): Promise<V[] | undefined>;
export async function fire<E extends WaitUntilEvent<V>, V = any>(emitter: Emitter<E>, event: Omit<E, 'waitUntil'>, timeout?: number): Promise<V[] | undefined> {
const waitables: Promise<V>[] = [];
const asyncEvent = Object.assign(event, {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
waitUntil: (thenable: Promise<any>) => {
waitUntil: (thenable: Promise<V>) => {
if (Object.isFrozen(waitables)) {
throw new Error('waitUntil cannot be called asynchronously.');
}
waitables.push(thenable);
}
}) as T;
}) as E;
try {
emitter.fire(asyncEvent);
// Asynchronous calls to `waitUntil` should fail.
Expand All @@ -331,12 +333,12 @@ export namespace WaitUntilEvent {
delete asyncEvent['waitUntil'];
}
if (!waitables.length) {
return;
return [];
}
if (timeout !== undefined) {
await Promise.race([Promise.all(waitables), new Promise(resolve => setTimeout(resolve, timeout))]);
return Promise.race([Promise.all(waitables), new Promise<undefined>(resolve => setTimeout(resolve, timeout, undefined))]);
} else {
await Promise.all(waitables);
return Promise.all(waitables);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -73,8 +73,8 @@ export class FileSystemFrontendContribution implements FrontendApplicationContri
initialize(): void {
this.fileSystemWatcher.onFilesChanged(event => this.run(() => this.updateWidgets(event)));
this.fileSystemWatcher.onWillMove(event => event.waitUntil(this.runEach((uri, widget) => this.pushMove(uri, widget, event))));
this.fileSystemWatcher.onDidFailMove(event => event.waitUntil(this.runEach((uri, widget) => this.revertMove(uri, widget, event))));
this.fileSystemWatcher.onDidMove(event => event.waitUntil(this.runEach((uri, widget) => this.applyMove(uri, widget, event))));
this.fileSystemWatcher.onDidFailMove(event => this.runEach((uri, widget) => this.revertMove(uri, widget, event)));
this.fileSystemWatcher.onDidMove(event => this.runEach((uri, widget) => this.applyMove(uri, widget, event)));
}

onStart?(app: FrontendApplication): MaybePromise<void> {
Expand Down
43 changes: 30 additions & 13 deletions packages/filesystem/src/browser/filesystem-watcher.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ export namespace FileChangeEvent {
}
}

export interface FileMoveEvent extends WaitUntilEvent {
export interface FileMoveEvent {
sourceUri: URI
targetUri: URI
}
Expand All @@ -77,13 +77,14 @@ export namespace FileMoveEvent {
}
}

export interface FileEvent extends WaitUntilEvent {
export interface FileEvent {
uri: URI
}

export class FileOperationEmitter<E extends WaitUntilEvent> implements Disposable {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export class FileOperationEmitter<E extends object, V = any> implements Disposable {

protected readonly onWillEmitter = new Emitter<E>();
protected readonly onWillEmitter = new Emitter<E & WaitUntilEvent<V>>();
readonly onWill = this.onWillEmitter.event;

protected readonly onDidFailEmitter = new Emitter<E>();
Expand All @@ -102,17 +103,25 @@ export class FileOperationEmitter<E extends WaitUntilEvent> implements Disposabl
this.toDispose.dispose();
}

async fireWill(event: Pick<E, Exclude<keyof E, 'waitUntil'>>): Promise<void> {
await WaitUntilEvent.fire(this.onWillEmitter, event);
async fireWill(event: Omit<E, 'waitUntil'>): Promise<V[]> {
return WaitUntilEvent.fire(this.onWillEmitter, event);
}

async fireDid(failed: boolean, event: Pick<E, Exclude<keyof E, 'waitUntil'>>): Promise<void> {
fireDid(failed: boolean, event: E): void {
const onDidEmitter = failed ? this.onDidFailEmitter : this.onDidEmitter;
await WaitUntilEvent.fire(onDidEmitter, event);
onDidEmitter.fire(event);
}

}

/**
* React to file system events, including calls originating from the
* application or event coming from the system's filesystem directly
* (actual file watching).
*
* `on(will|did)(create|rename|delete)` events solely come from application
* usage, not from actual filesystem.
*/
@injectable()
export class FileSystemWatcher implements Disposable {

Expand All @@ -122,6 +131,11 @@ export class FileSystemWatcher implements Disposable {
protected readonly onFileChangedEmitter = new Emitter<FileChangeEvent>();
readonly onFilesChanged = this.onFileChangedEmitter.event;

protected readonly fileCreateEmitter = new FileOperationEmitter<FileEvent>();
readonly onWillCreate = this.fileCreateEmitter.onWill;
readonly onDidFailCreate = this.fileCreateEmitter.onDidFail;
readonly onDidCreate = this.fileCreateEmitter.onDid;

protected readonly fileDeleteEmitter = new FileOperationEmitter<FileEvent>();
readonly onWillDelete = this.fileDeleteEmitter.onWill;
readonly onDidFailDelete = this.fileDeleteEmitter.onDidFail;
Expand Down Expand Up @@ -164,11 +178,15 @@ export class FileSystemWatcher implements Disposable {
}));

this.filesystem.setClient({
/* eslint-disable no-void */
shouldOverwrite: this.shouldOverwrite.bind(this),
willDelete: uri => this.fileDeleteEmitter.fireWill({ uri: new URI(uri) }),
didDelete: (uri, failed) => this.fileDeleteEmitter.fireDid(failed, { uri: new URI(uri) }),
willMove: (source, target) => this.fileMoveEmitter.fireWill({ sourceUri: new URI(source), targetUri: new URI(target) }),
didMove: (source, target, failed) => this.fileMoveEmitter.fireDid(failed, { sourceUri: new URI(source), targetUri: new URI(target) })
willCreate: async uri => void await this.fileCreateEmitter.fireWill({ uri: new URI(uri) }),
didCreate: async (uri, failed) => void this.fileCreateEmitter.fireDid(failed, { uri: new URI(uri) }),
willDelete: async uri => void await this.fileDeleteEmitter.fireWill({ uri: new URI(uri) }),
didDelete: async (uri, failed) => void this.fileDeleteEmitter.fireDid(failed, { uri: new URI(uri) }),
willMove: async (sourceUri, targetUri) => void await this.fileMoveEmitter.fireWill({ sourceUri: new URI(sourceUri), targetUri: new URI(targetUri) }),
didMove: async (sourceUri, targetUri, failed) => this.fileMoveEmitter.fireDid(false, { sourceUri: new URI(sourceUri), targetUri: new URI(targetUri) }),
/* eslint-enable no-void */
});
}

Expand Down Expand Up @@ -228,4 +246,3 @@ export class FileSystemWatcher implements Disposable {
}

}

23 changes: 18 additions & 5 deletions packages/filesystem/src/common/filesystem.ts
Original file line number Diff line number Diff line change
Expand Up @@ -212,13 +212,18 @@ export interface FileSystemClient {
*/
shouldOverwrite: FileShouldOverwrite;

willCreate(uri: string): Promise<void>;

didCreate(uri: string, failed: boolean): Promise<void>;

willDelete(uri: string): Promise<void>;

didDelete(uri: string, failed: boolean): Promise<void>;

willMove(sourceUri: string, targetUri: string): Promise<void>;

didMove(sourceUri: string, targetUri: string, failed: boolean): Promise<void>;

}

@injectable()
Expand All @@ -227,25 +232,33 @@ export class DispatchingFileSystemClient implements FileSystemClient {
readonly clients = new Set<FileSystemClient>();

shouldOverwrite(originalStat: FileStat, currentStat: FileStat): Promise<boolean> {
return Promise.race([...this.clients].map(client =>
return Promise.race(Array.from(this.clients, client =>
client.shouldOverwrite(originalStat, currentStat))
);
}

async willCreate(uri: string): Promise<void> {
await Promise.all(Array.from(this.clients, client => client.willCreate(uri)));
}

async didCreate(uri: string, failed: boolean): Promise<void> {
await Promise.all(Array.from(this.clients, client => client.didCreate(uri, failed)));
}

async willDelete(uri: string): Promise<void> {
await Promise.all([...this.clients].map(client => client.willDelete(uri)));
await Promise.all(Array.from(this.clients, client => client.willDelete(uri)));
}

async didDelete(uri: string, failed: boolean): Promise<void> {
await Promise.all([...this.clients].map(client => client.didDelete(uri, failed)));
await Promise.all(Array.from(this.clients, client => client.didDelete(uri, failed)));
}

async willMove(sourceUri: string, targetUri: string): Promise<void> {
await Promise.all([...this.clients].map(client => client.willMove(sourceUri, targetUri)));
await Promise.all(Array.from(this.clients, client => client.willMove(sourceUri, targetUri)));
}

async didMove(sourceUri: string, targetUri: string, failed: boolean): Promise<void> {
await Promise.all([...this.clients].map(client => client.didMove(sourceUri, targetUri, failed)));
await Promise.all(Array.from(this.clients, client => client.didMove(sourceUri, targetUri, failed)));
}

}
Expand Down
38 changes: 38 additions & 0 deletions packages/filesystem/src/node/node-filesystem.ts
Original file line number Diff line number Diff line change
Expand Up @@ -263,6 +263,25 @@ export class FileSystemNode implements FileSystem {
}

async createFile(uri: string, options?: { content?: string, encoding?: string }): Promise<FileStat> {
if (this.client) {
await this.client.willCreate(uri);
}
let result: FileStat;
let failed = false;
try {
result = await this.doCreateFile(uri, options);
} catch (e) {
failed = true;
throw e;
} finally {
if (this.client) {
await this.client.didCreate(uri, failed);
}
}
return result;
}

protected async doCreateFile(uri: string, options?: { content?: string, encoding?: string }): Promise<FileStat> {
const _uri = new URI(uri);
const parentUri = _uri.parent;
const [stat, parentStat] = await Promise.all([this.doGetStat(_uri, 0), this.doGetStat(parentUri, 0)]);
Expand All @@ -284,6 +303,25 @@ export class FileSystemNode implements FileSystem {
}

async createFolder(uri: string): Promise<FileStat> {
if (this.client) {
await this.client.willCreate(uri);
}
let result: FileStat;
let failed = false;
try {
result = await this.doCreateFolder(uri);
} catch (e) {
failed = true;
throw e;
} finally {
if (this.client) {
await this.client.didCreate(uri, failed);
}
}
return result;
}

async doCreateFolder(uri: string): Promise<FileStat> {
const _uri = new URI(uri);
const stat = await this.doGetStat(_uri, 0);
if (stat) {
Expand Down
12 changes: 12 additions & 0 deletions packages/plugin-ext/src/common/plugin-api-rpc-model.ts
Original file line number Diff line number Diff line change
Expand Up @@ -547,3 +547,15 @@ export interface CallHierarchyOutgoingCall {
to: CallHierarchyItem;
fromRanges: Range[];
}

export interface CreateFilesEventDTO {
files: UriComponents[]
}

export interface RenameFilesEventDTO {
files: { oldUri: UriComponents, newUri: UriComponents }[]
}

export interface DeleteFilesEventDTO {
files: UriComponents[]
}
12 changes: 11 additions & 1 deletion packages/plugin-ext/src/common/plugin-api-rpc.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,10 @@ import {
FoldingRange,
SelectionRange,
CallHierarchyDefinition,
CallHierarchyReference
CallHierarchyReference,
CreateFilesEventDTO,
RenameFilesEventDTO,
DeleteFilesEventDTO,
} from './plugin-api-rpc-model';
import { ExtPluginApi } from './plugin-ext-api-contribution';
import { KeysToAnyValues, KeysToKeysToAnyValue } from './types';
Expand Down Expand Up @@ -511,6 +514,13 @@ export interface WorkspaceExt {
$fileChanged(event: FileChangeEvent): void;
$onFileRename(event: FileMoveEvent): void;
$onWillRename(event: FileWillMoveEvent): Promise<any>;

$onWillCreateFiles(event: CreateFilesEventDTO): Promise<any[]>;
$onDidCreateFiles(event: CreateFilesEventDTO): void;
$onWillRenameFiles(event: RenameFilesEventDTO): Promise<any[]>;
$onDidRenameFiles(event: RenameFilesEventDTO): void;
$onWillDeleteFiles(event: DeleteFilesEventDTO): Promise<any[]>;
$onDidDeleteFiles(event: DeleteFilesEventDTO): void;
}

export interface DialogsMain {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -96,7 +96,7 @@ export class InPluginFileSystemWatcherManager {
}

private uriMatches(subscriber: FileWatcherSubscriber, fileChange: FileChange): boolean {
return subscriber.mather(fileChange.uri.path.toString());
return subscriber.matcher(fileChange.uri.path.toString());
}

/**
Expand All @@ -118,7 +118,7 @@ export class InPluginFileSystemWatcherManager {

const subscriber: FileWatcherSubscriber = {
id: subscriberId,
mather: globPatternMatcher,
matcher: globPatternMatcher,
ignoreCreateEvents: options.ignoreCreateEvents === true,
ignoreChangeEvents: options.ignoreChangeEvents === true,
ignoreDeleteEvents: options.ignoreDeleteEvents === true,
Expand All @@ -141,7 +141,7 @@ export class InPluginFileSystemWatcherManager {

interface FileWatcherSubscriber {
id: string;
mather: ParsedPattern;
matcher: ParsedPattern;
ignoreCreateEvents: boolean;
ignoreChangeEvents: boolean;
ignoreDeleteEvents: boolean;
Expand Down
Loading

0 comments on commit f2e1c5c

Please sign in to comment.