/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import { MainContext, MainThreadFileSystemShape } from './extHost.protocol'; import * as vscode from 'vscode'; import * as files from 'vs/platform/files/common/files'; import { FileSystemError } from 'vs/workbench/api/common/extHostTypes'; import { VSBuffer } from 'vs/base/common/buffer'; import { createDecorator } from 'vs/platform/instantiation/common/instantiation'; import { IExtHostRpcService } from 'vs/workbench/api/common/extHostRpcService'; import { IExtHostFileSystemInfo } from 'vs/workbench/api/common/extHostFileSystemInfo'; import { IDisposable, toDisposable } from 'vs/base/common/lifecycle'; export class ExtHostConsumerFileSystem { readonly _serviceBrand: undefined; readonly value: vscode.FileSystem; private readonly _proxy: MainThreadFileSystemShape; private readonly _fileSystemProvider = new Map(); constructor( @IExtHostRpcService extHostRpc: IExtHostRpcService, @IExtHostFileSystemInfo fileSystemInfo: IExtHostFileSystemInfo, ) { this._proxy = extHostRpc.getProxy(MainContext.MainThreadFileSystem); const that = this; this.value = Object.freeze({ async stat(uri: vscode.Uri): Promise { try { const provider = that._fileSystemProvider.get(uri.scheme); if (!provider) { return await that._proxy.$stat(uri); } // use shortcut await that._proxy.$ensureActivation(uri.scheme); const stat = await provider.stat(uri); return { type: stat.type, ctime: stat.ctime, mtime: stat.mtime, size: stat.size, permissions: stat.permissions }; } catch (err) { ExtHostConsumerFileSystem._handleError(err); } }, async readDirectory(uri: vscode.Uri): Promise<[string, vscode.FileType][]> { try { const provider = that._fileSystemProvider.get(uri.scheme); if (provider) { // use shortcut await that._proxy.$ensureActivation(uri.scheme); return (await provider.readDirectory(uri)).slice(); // safe-copy } else { return await that._proxy.$readdir(uri); } } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, async createDirectory(uri: vscode.Uri): Promise { try { // no shortcut: does mkdirp return await that._proxy.$mkdir(uri); } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, async readFile(uri: vscode.Uri): Promise { try { const provider = that._fileSystemProvider.get(uri.scheme); if (provider) { // use shortcut await that._proxy.$ensureActivation(uri.scheme); return (await provider.readFile(uri)).slice(); // safe-copy } else { const buff = await that._proxy.$readFile(uri); return buff.buffer; } } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, async writeFile(uri: vscode.Uri, content: Uint8Array): Promise { try { // no shortcut: does mkdirp return await that._proxy.$writeFile(uri, VSBuffer.wrap(content)); } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, async delete(uri: vscode.Uri, options?: { recursive?: boolean; useTrash?: boolean }): Promise { try { const provider = that._fileSystemProvider.get(uri.scheme); if (provider) { // use shortcut await that._proxy.$ensureActivation(uri.scheme); return await provider.delete(uri, { recursive: false, ...options }); } else { return await that._proxy.$delete(uri, { recursive: false, useTrash: false, ...options }); } } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, async rename(oldUri: vscode.Uri, newUri: vscode.Uri, options?: { overwrite?: boolean }): Promise { try { // no shortcut: potentially involves different schemes, does mkdirp return await that._proxy.$rename(oldUri, newUri, { ...{ overwrite: false }, ...options }); } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, async copy(source: vscode.Uri, destination: vscode.Uri, options?: { overwrite?: boolean }): Promise { try { // no shortcut: potentially involves different schemes, does mkdirp return await that._proxy.$copy(source, destination, { ...{ overwrite: false }, ...options }); } catch (err) { return ExtHostConsumerFileSystem._handleError(err); } }, isWritableFileSystem(scheme: string): boolean | undefined { const capabilities = fileSystemInfo.getCapabilities(scheme); if (typeof capabilities === 'number') { return !(capabilities & files.FileSystemProviderCapabilities.Readonly); } return undefined; } }); } private static _handleError(err: any): never { // desired error type if (err instanceof FileSystemError) { throw err; } // generic error if (!(err instanceof Error)) { throw new FileSystemError(String(err)); } // no provider (unknown scheme) error if (err.name === 'ENOPRO') { throw FileSystemError.Unavailable(err.message); } // file system error switch (err.name) { case files.FileSystemProviderErrorCode.FileExists: throw FileSystemError.FileExists(err.message); case files.FileSystemProviderErrorCode.FileNotFound: throw FileSystemError.FileNotFound(err.message); case files.FileSystemProviderErrorCode.FileNotADirectory: throw FileSystemError.FileNotADirectory(err.message); case files.FileSystemProviderErrorCode.FileIsADirectory: throw FileSystemError.FileIsADirectory(err.message); case files.FileSystemProviderErrorCode.NoPermissions: throw FileSystemError.NoPermissions(err.message); case files.FileSystemProviderErrorCode.Unavailable: throw FileSystemError.Unavailable(err.message); default: throw new FileSystemError(err.message, err.name as files.FileSystemProviderErrorCode); } } // --- addFileSystemProvider(scheme: string, provider: vscode.FileSystemProvider): IDisposable { this._fileSystemProvider.set(scheme, provider); return toDisposable(() => this._fileSystemProvider.delete(scheme)); } } export interface IExtHostConsumerFileSystem extends ExtHostConsumerFileSystem { } export const IExtHostConsumerFileSystem = createDecorator('IExtHostConsumerFileSystem');