mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-21 17:22:55 -05:00
231 lines
8.3 KiB
TypeScript
231 lines
8.3 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import { AsyncEmitter, Emitter, Event, IWaitUntil } from 'vs/base/common/event';
|
|
import { IRelativePattern, parse } from 'vs/base/common/glob';
|
|
import { URI, UriComponents } from 'vs/base/common/uri';
|
|
import { ExtHostDocumentsAndEditors } from 'vs/workbench/api/common/extHostDocumentsAndEditors';
|
|
import type * as vscode from 'vscode';
|
|
import { ExtHostFileSystemEventServiceShape, FileSystemEvents, IMainContext, MainContext, MainThreadTextEditorsShape, IWorkspaceFileEditDto, IWorkspaceTextEditDto } from './extHost.protocol';
|
|
import * as typeConverter from './extHostTypeConverters';
|
|
import { Disposable, WorkspaceEdit } from './extHostTypes';
|
|
import { IExtensionDescription } from 'vs/platform/extensions/common/extensions';
|
|
import { FileOperation } from 'vs/platform/files/common/files';
|
|
import { flatten } from 'vs/base/common/arrays';
|
|
import { CancellationToken } from 'vs/base/common/cancellation';
|
|
import { ILogService } from 'vs/platform/log/common/log';
|
|
|
|
class FileSystemWatcher implements vscode.FileSystemWatcher {
|
|
|
|
private readonly _onDidCreate = new Emitter<vscode.Uri>();
|
|
private readonly _onDidChange = new Emitter<vscode.Uri>();
|
|
private readonly _onDidDelete = new Emitter<vscode.Uri>();
|
|
private _disposable: Disposable;
|
|
private _config: number;
|
|
|
|
get ignoreCreateEvents(): boolean {
|
|
return Boolean(this._config & 0b001);
|
|
}
|
|
|
|
get ignoreChangeEvents(): boolean {
|
|
return Boolean(this._config & 0b010);
|
|
}
|
|
|
|
get ignoreDeleteEvents(): boolean {
|
|
return Boolean(this._config & 0b100);
|
|
}
|
|
|
|
constructor(dispatcher: Event<FileSystemEvents>, globPattern: string | IRelativePattern, ignoreCreateEvents?: boolean, ignoreChangeEvents?: boolean, ignoreDeleteEvents?: boolean) {
|
|
|
|
this._config = 0;
|
|
if (ignoreCreateEvents) {
|
|
this._config += 0b001;
|
|
}
|
|
if (ignoreChangeEvents) {
|
|
this._config += 0b010;
|
|
}
|
|
if (ignoreDeleteEvents) {
|
|
this._config += 0b100;
|
|
}
|
|
|
|
const parsedPattern = parse(globPattern);
|
|
|
|
const subscription = dispatcher(events => {
|
|
if (!ignoreCreateEvents) {
|
|
for (let created of events.created) {
|
|
const uri = URI.revive(created);
|
|
if (parsedPattern(uri.fsPath)) {
|
|
this._onDidCreate.fire(uri);
|
|
}
|
|
}
|
|
}
|
|
if (!ignoreChangeEvents) {
|
|
for (let changed of events.changed) {
|
|
const uri = URI.revive(changed);
|
|
if (parsedPattern(uri.fsPath)) {
|
|
this._onDidChange.fire(uri);
|
|
}
|
|
}
|
|
}
|
|
if (!ignoreDeleteEvents) {
|
|
for (let deleted of events.deleted) {
|
|
const uri = URI.revive(deleted);
|
|
if (parsedPattern(uri.fsPath)) {
|
|
this._onDidDelete.fire(uri);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
this._disposable = Disposable.from(this._onDidCreate, this._onDidChange, this._onDidDelete, subscription);
|
|
}
|
|
|
|
dispose() {
|
|
this._disposable.dispose();
|
|
}
|
|
|
|
get onDidCreate(): Event<vscode.Uri> {
|
|
return this._onDidCreate.event;
|
|
}
|
|
|
|
get onDidChange(): Event<vscode.Uri> {
|
|
return this._onDidChange.event;
|
|
}
|
|
|
|
get onDidDelete(): Event<vscode.Uri> {
|
|
return this._onDidDelete.event;
|
|
}
|
|
}
|
|
|
|
interface IExtensionListener<E> {
|
|
extension: IExtensionDescription;
|
|
(e: E): any;
|
|
}
|
|
|
|
export class ExtHostFileSystemEventService implements ExtHostFileSystemEventServiceShape {
|
|
|
|
private readonly _onFileSystemEvent = new Emitter<FileSystemEvents>();
|
|
|
|
private readonly _onDidRenameFile = new Emitter<vscode.FileRenameEvent>();
|
|
private readonly _onDidCreateFile = new Emitter<vscode.FileCreateEvent>();
|
|
private readonly _onDidDeleteFile = new Emitter<vscode.FileDeleteEvent>();
|
|
private readonly _onWillRenameFile = new AsyncEmitter<vscode.FileWillRenameEvent>();
|
|
private readonly _onWillCreateFile = new AsyncEmitter<vscode.FileWillCreateEvent>();
|
|
private readonly _onWillDeleteFile = new AsyncEmitter<vscode.FileWillDeleteEvent>();
|
|
|
|
readonly onDidRenameFile: Event<vscode.FileRenameEvent> = this._onDidRenameFile.event;
|
|
readonly onDidCreateFile: Event<vscode.FileCreateEvent> = this._onDidCreateFile.event;
|
|
readonly onDidDeleteFile: Event<vscode.FileDeleteEvent> = this._onDidDeleteFile.event;
|
|
|
|
|
|
constructor(
|
|
mainContext: IMainContext,
|
|
private readonly _logService: ILogService,
|
|
private readonly _extHostDocumentsAndEditors: ExtHostDocumentsAndEditors,
|
|
private readonly _mainThreadTextEditors: MainThreadTextEditorsShape = mainContext.getProxy(MainContext.MainThreadTextEditors)
|
|
) {
|
|
//
|
|
}
|
|
|
|
//--- file events
|
|
|
|
createFileSystemWatcher(globPattern: string | IRelativePattern, ignoreCreateEvents?: boolean, ignoreChangeEvents?: boolean, ignoreDeleteEvents?: boolean): vscode.FileSystemWatcher {
|
|
return new FileSystemWatcher(this._onFileSystemEvent.event, globPattern, ignoreCreateEvents, ignoreChangeEvents, ignoreDeleteEvents);
|
|
}
|
|
|
|
$onFileEvent(events: FileSystemEvents) {
|
|
this._onFileSystemEvent.fire(events);
|
|
}
|
|
|
|
|
|
//--- file operations
|
|
|
|
$onDidRunFileOperation(operation: FileOperation, target: UriComponents, source: UriComponents | undefined): void {
|
|
switch (operation) {
|
|
case FileOperation.MOVE:
|
|
this._onDidRenameFile.fire(Object.freeze({ files: [{ oldUri: URI.revive(source!), newUri: URI.revive(target) }] }));
|
|
break;
|
|
case FileOperation.DELETE:
|
|
this._onDidDeleteFile.fire(Object.freeze({ files: [URI.revive(target)] }));
|
|
break;
|
|
case FileOperation.CREATE:
|
|
this._onDidCreateFile.fire(Object.freeze({ files: [URI.revive(target)] }));
|
|
break;
|
|
default:
|
|
//ignore, dont send
|
|
}
|
|
}
|
|
|
|
|
|
getOnWillRenameFileEvent(extension: IExtensionDescription): Event<vscode.FileWillRenameEvent> {
|
|
return this._createWillExecuteEvent(extension, this._onWillRenameFile);
|
|
}
|
|
|
|
getOnWillCreateFileEvent(extension: IExtensionDescription): Event<vscode.FileWillCreateEvent> {
|
|
return this._createWillExecuteEvent(extension, this._onWillCreateFile);
|
|
}
|
|
|
|
getOnWillDeleteFileEvent(extension: IExtensionDescription): Event<vscode.FileWillDeleteEvent> {
|
|
return this._createWillExecuteEvent(extension, this._onWillDeleteFile);
|
|
}
|
|
|
|
private _createWillExecuteEvent<E extends IWaitUntil>(extension: IExtensionDescription, emitter: AsyncEmitter<E>): Event<E> {
|
|
return (listener, thisArg, disposables) => {
|
|
const wrappedListener: IExtensionListener<E> = function wrapped(e: E) { listener.call(thisArg, e); };
|
|
wrappedListener.extension = extension;
|
|
return emitter.event(wrappedListener, undefined, disposables);
|
|
};
|
|
}
|
|
|
|
async $onWillRunFileOperation(operation: FileOperation, target: UriComponents, source: UriComponents | undefined, timeout: number, token: CancellationToken): Promise<any> {
|
|
switch (operation) {
|
|
case FileOperation.MOVE:
|
|
await this._fireWillEvent(this._onWillRenameFile, { files: [{ oldUri: URI.revive(source!), newUri: URI.revive(target) }] }, timeout, token);
|
|
break;
|
|
case FileOperation.DELETE:
|
|
await this._fireWillEvent(this._onWillDeleteFile, { files: [URI.revive(target)] }, timeout, token);
|
|
break;
|
|
case FileOperation.CREATE:
|
|
await this._fireWillEvent(this._onWillCreateFile, { files: [URI.revive(target)] }, timeout, token);
|
|
break;
|
|
default:
|
|
//ignore, dont send
|
|
}
|
|
}
|
|
|
|
private async _fireWillEvent<E extends IWaitUntil>(emitter: AsyncEmitter<E>, data: Omit<E, 'waitUntil'>, timeout: number, token: CancellationToken): Promise<any> {
|
|
|
|
const edits: WorkspaceEdit[] = [];
|
|
|
|
await emitter.fireAsync(data, token, async (thenable, listener) => {
|
|
// ignore all results except for WorkspaceEdits. Those are stored in an array.
|
|
const now = Date.now();
|
|
const result = await Promise.resolve(thenable);
|
|
if (result instanceof WorkspaceEdit) {
|
|
edits.push(result);
|
|
}
|
|
|
|
if (Date.now() - now > timeout) {
|
|
this._logService.warn('SLOW file-participant', (<IExtensionListener<E>>listener).extension?.identifier);
|
|
}
|
|
});
|
|
|
|
if (token.isCancellationRequested) {
|
|
return;
|
|
}
|
|
|
|
if (edits.length > 0) {
|
|
// flatten all WorkspaceEdits collected via waitUntil-call
|
|
// and apply them in one go.
|
|
const allEdits = new Array<Array<IWorkspaceFileEditDto | IWorkspaceTextEditDto>>();
|
|
for (let edit of edits) {
|
|
let { edits } = typeConverter.WorkspaceEdit.from(edit, this._extHostDocumentsAndEditors);
|
|
allEdits.push(edits);
|
|
}
|
|
return this._mainThreadTextEditors.$tryApplyWorkspaceEdit({ edits: flatten(allEdits) });
|
|
}
|
|
}
|
|
}
|