Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/src/vs/workbench/api/common/extHostFileSystem.ts
3296 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
6
import { URI, UriComponents } from '../../../base/common/uri.js';
7
import { MainContext, IMainContext, ExtHostFileSystemShape, MainThreadFileSystemShape, IFileChangeDto } from './extHost.protocol.js';
8
import type * as vscode from 'vscode';
9
import * as files from '../../../platform/files/common/files.js';
10
import { IDisposable, toDisposable } from '../../../base/common/lifecycle.js';
11
import { FileChangeType } from './extHostTypes.js';
12
import * as typeConverter from './extHostTypeConverters.js';
13
import { ExtHostLanguageFeatures } from './extHostLanguageFeatures.js';
14
import { State, StateMachine, LinkComputer, Edge } from '../../../editor/common/languages/linkComputer.js';
15
import { commonPrefixLength } from '../../../base/common/strings.js';
16
import { CharCode } from '../../../base/common/charCode.js';
17
import { VSBuffer } from '../../../base/common/buffer.js';
18
import { IExtensionDescription } from '../../../platform/extensions/common/extensions.js';
19
import { checkProposedApiEnabled } from '../../services/extensions/common/extensions.js';
20
import { IMarkdownString, isMarkdownString } from '../../../base/common/htmlContent.js';
21
22
class FsLinkProvider {
23
24
private _schemes: string[] = [];
25
private _stateMachine?: StateMachine;
26
27
add(scheme: string): void {
28
this._stateMachine = undefined;
29
this._schemes.push(scheme);
30
}
31
32
delete(scheme: string): void {
33
const idx = this._schemes.indexOf(scheme);
34
if (idx >= 0) {
35
this._schemes.splice(idx, 1);
36
this._stateMachine = undefined;
37
}
38
}
39
40
private _initStateMachine(): void {
41
if (!this._stateMachine) {
42
43
// sort and compute common prefix with previous scheme
44
// then build state transitions based on the data
45
const schemes = this._schemes.sort();
46
const edges: Edge[] = [];
47
let prevScheme: string | undefined;
48
let prevState: State;
49
let lastState = State.LastKnownState;
50
let nextState = State.LastKnownState;
51
for (const scheme of schemes) {
52
53
// skip the common prefix of the prev scheme
54
// and continue with its last state
55
let pos = !prevScheme ? 0 : commonPrefixLength(prevScheme, scheme);
56
if (pos === 0) {
57
prevState = State.Start;
58
} else {
59
prevState = nextState;
60
}
61
62
for (; pos < scheme.length; pos++) {
63
// keep creating new (next) states until the
64
// end (and the BeforeColon-state) is reached
65
if (pos + 1 === scheme.length) {
66
// Save the last state here, because we need to continue for the next scheme
67
lastState = nextState;
68
nextState = State.BeforeColon;
69
} else {
70
nextState += 1;
71
}
72
edges.push([prevState, scheme.toUpperCase().charCodeAt(pos), nextState]);
73
edges.push([prevState, scheme.toLowerCase().charCodeAt(pos), nextState]);
74
prevState = nextState;
75
}
76
77
prevScheme = scheme;
78
// Restore the last state
79
nextState = lastState;
80
}
81
82
// all link must match this pattern `<scheme>:/<more>`
83
edges.push([State.BeforeColon, CharCode.Colon, State.AfterColon]);
84
edges.push([State.AfterColon, CharCode.Slash, State.End]);
85
86
this._stateMachine = new StateMachine(edges);
87
}
88
}
89
90
provideDocumentLinks(document: vscode.TextDocument): vscode.ProviderResult<vscode.DocumentLink[]> {
91
this._initStateMachine();
92
93
const result: vscode.DocumentLink[] = [];
94
const links = LinkComputer.computeLinks({
95
getLineContent(lineNumber: number): string {
96
return document.lineAt(lineNumber - 1).text;
97
},
98
getLineCount(): number {
99
return document.lineCount;
100
}
101
}, this._stateMachine);
102
103
for (const link of links) {
104
const docLink = typeConverter.DocumentLink.to(link);
105
if (docLink.target) {
106
result.push(docLink);
107
}
108
}
109
return result;
110
}
111
}
112
113
export class ExtHostFileSystem implements ExtHostFileSystemShape {
114
115
private readonly _proxy: MainThreadFileSystemShape;
116
private readonly _linkProvider = new FsLinkProvider();
117
private readonly _fsProvider = new Map<number, vscode.FileSystemProvider>();
118
private readonly _registeredSchemes = new Set<string>();
119
private readonly _watches = new Map<number, IDisposable>();
120
121
private _linkProviderRegistration?: IDisposable;
122
private _handlePool: number = 0;
123
124
constructor(mainContext: IMainContext, private _extHostLanguageFeatures: ExtHostLanguageFeatures) {
125
this._proxy = mainContext.getProxy(MainContext.MainThreadFileSystem);
126
}
127
128
dispose(): void {
129
this._linkProviderRegistration?.dispose();
130
}
131
132
registerFileSystemProvider(extension: IExtensionDescription, scheme: string, provider: vscode.FileSystemProvider, options: { isCaseSensitive?: boolean; isReadonly?: boolean | vscode.MarkdownString } = {}) {
133
134
// validate the given provider is complete
135
ExtHostFileSystem._validateFileSystemProvider(provider);
136
137
if (this._registeredSchemes.has(scheme)) {
138
throw new Error(`a provider for the scheme '${scheme}' is already registered`);
139
}
140
141
//
142
if (!this._linkProviderRegistration) {
143
this._linkProviderRegistration = this._extHostLanguageFeatures.registerDocumentLinkProvider(extension, '*', this._linkProvider);
144
}
145
146
const handle = this._handlePool++;
147
this._linkProvider.add(scheme);
148
this._registeredSchemes.add(scheme);
149
this._fsProvider.set(handle, provider);
150
151
let capabilities = files.FileSystemProviderCapabilities.FileReadWrite;
152
if (options.isCaseSensitive) {
153
capabilities += files.FileSystemProviderCapabilities.PathCaseSensitive;
154
}
155
if (options.isReadonly) {
156
capabilities += files.FileSystemProviderCapabilities.Readonly;
157
}
158
if (typeof provider.copy === 'function') {
159
capabilities += files.FileSystemProviderCapabilities.FileFolderCopy;
160
}
161
if (typeof provider.open === 'function' && typeof provider.close === 'function'
162
&& typeof provider.read === 'function' && typeof provider.write === 'function'
163
) {
164
checkProposedApiEnabled(extension, 'fsChunks');
165
capabilities += files.FileSystemProviderCapabilities.FileOpenReadWriteClose;
166
}
167
168
let readOnlyMessage: IMarkdownString | undefined;
169
if (options.isReadonly && isMarkdownString(options.isReadonly) && options.isReadonly.value !== '') {
170
readOnlyMessage = {
171
value: options.isReadonly.value,
172
isTrusted: options.isReadonly.isTrusted,
173
supportThemeIcons: options.isReadonly.supportThemeIcons,
174
supportHtml: options.isReadonly.supportHtml,
175
baseUri: options.isReadonly.baseUri,
176
uris: options.isReadonly.uris
177
};
178
}
179
180
this._proxy.$registerFileSystemProvider(handle, scheme, capabilities, readOnlyMessage).catch(err => {
181
console.error(`FAILED to register filesystem provider of ${extension.identifier.value}-extension for the scheme ${scheme}`);
182
console.error(err);
183
});
184
185
const subscription = provider.onDidChangeFile(event => {
186
const mapped: IFileChangeDto[] = [];
187
for (const e of event) {
188
const { uri: resource, type } = e;
189
if (resource.scheme !== scheme) {
190
// dropping events for wrong scheme
191
continue;
192
}
193
let newType: files.FileChangeType | undefined;
194
switch (type) {
195
case FileChangeType.Changed:
196
newType = files.FileChangeType.UPDATED;
197
break;
198
case FileChangeType.Created:
199
newType = files.FileChangeType.ADDED;
200
break;
201
case FileChangeType.Deleted:
202
newType = files.FileChangeType.DELETED;
203
break;
204
default:
205
throw new Error('Unknown FileChangeType');
206
}
207
mapped.push({ resource, type: newType });
208
}
209
this._proxy.$onFileSystemChange(handle, mapped);
210
});
211
212
return toDisposable(() => {
213
subscription.dispose();
214
this._linkProvider.delete(scheme);
215
this._registeredSchemes.delete(scheme);
216
this._fsProvider.delete(handle);
217
this._proxy.$unregisterProvider(handle);
218
});
219
}
220
221
private static _validateFileSystemProvider(provider: vscode.FileSystemProvider) {
222
if (!provider) {
223
throw new Error('MISSING provider');
224
}
225
if (typeof provider.watch !== 'function') {
226
throw new Error('Provider does NOT implement watch');
227
}
228
if (typeof provider.stat !== 'function') {
229
throw new Error('Provider does NOT implement stat');
230
}
231
if (typeof provider.readDirectory !== 'function') {
232
throw new Error('Provider does NOT implement readDirectory');
233
}
234
if (typeof provider.createDirectory !== 'function') {
235
throw new Error('Provider does NOT implement createDirectory');
236
}
237
if (typeof provider.readFile !== 'function') {
238
throw new Error('Provider does NOT implement readFile');
239
}
240
if (typeof provider.writeFile !== 'function') {
241
throw new Error('Provider does NOT implement writeFile');
242
}
243
if (typeof provider.delete !== 'function') {
244
throw new Error('Provider does NOT implement delete');
245
}
246
if (typeof provider.rename !== 'function') {
247
throw new Error('Provider does NOT implement rename');
248
}
249
}
250
251
private static _asIStat(stat: vscode.FileStat): files.IStat {
252
const { type, ctime, mtime, size, permissions } = stat;
253
return { type, ctime, mtime, size, permissions };
254
}
255
256
$stat(handle: number, resource: UriComponents): Promise<files.IStat> {
257
return Promise.resolve(this._getFsProvider(handle).stat(URI.revive(resource))).then(stat => ExtHostFileSystem._asIStat(stat));
258
}
259
260
$readdir(handle: number, resource: UriComponents): Promise<[string, files.FileType][]> {
261
return Promise.resolve(this._getFsProvider(handle).readDirectory(URI.revive(resource)));
262
}
263
264
$readFile(handle: number, resource: UriComponents): Promise<VSBuffer> {
265
return Promise.resolve(this._getFsProvider(handle).readFile(URI.revive(resource))).then(data => VSBuffer.wrap(data));
266
}
267
268
$writeFile(handle: number, resource: UriComponents, content: VSBuffer, opts: files.IFileWriteOptions): Promise<void> {
269
return Promise.resolve(this._getFsProvider(handle).writeFile(URI.revive(resource), content.buffer, opts));
270
}
271
272
$delete(handle: number, resource: UriComponents, opts: files.IFileDeleteOptions): Promise<void> {
273
return Promise.resolve(this._getFsProvider(handle).delete(URI.revive(resource), opts));
274
}
275
276
$rename(handle: number, oldUri: UriComponents, newUri: UriComponents, opts: files.IFileOverwriteOptions): Promise<void> {
277
return Promise.resolve(this._getFsProvider(handle).rename(URI.revive(oldUri), URI.revive(newUri), opts));
278
}
279
280
$copy(handle: number, oldUri: UriComponents, newUri: UriComponents, opts: files.IFileOverwriteOptions): Promise<void> {
281
const provider = this._getFsProvider(handle);
282
if (!provider.copy) {
283
throw new Error('FileSystemProvider does not implement "copy"');
284
}
285
return Promise.resolve(provider.copy(URI.revive(oldUri), URI.revive(newUri), opts));
286
}
287
288
$mkdir(handle: number, resource: UriComponents): Promise<void> {
289
return Promise.resolve(this._getFsProvider(handle).createDirectory(URI.revive(resource)));
290
}
291
292
$watch(handle: number, session: number, resource: UriComponents, opts: files.IWatchOptions): void {
293
const subscription = this._getFsProvider(handle).watch(URI.revive(resource), opts);
294
this._watches.set(session, subscription);
295
}
296
297
$unwatch(_handle: number, session: number): void {
298
const subscription = this._watches.get(session);
299
if (subscription) {
300
subscription.dispose();
301
this._watches.delete(session);
302
}
303
}
304
305
$open(handle: number, resource: UriComponents, opts: files.IFileOpenOptions): Promise<number> {
306
const provider = this._getFsProvider(handle);
307
if (!provider.open) {
308
throw new Error('FileSystemProvider does not implement "open"');
309
}
310
return Promise.resolve(provider.open(URI.revive(resource), opts));
311
}
312
313
$close(handle: number, fd: number): Promise<void> {
314
const provider = this._getFsProvider(handle);
315
if (!provider.close) {
316
throw new Error('FileSystemProvider does not implement "close"');
317
}
318
return Promise.resolve(provider.close(fd));
319
}
320
321
$read(handle: number, fd: number, pos: number, length: number): Promise<VSBuffer> {
322
const provider = this._getFsProvider(handle);
323
if (!provider.read) {
324
throw new Error('FileSystemProvider does not implement "read"');
325
}
326
const data = VSBuffer.alloc(length);
327
return Promise.resolve(provider.read(fd, pos, data.buffer, 0, length)).then(read => {
328
return data.slice(0, read); // don't send zeros
329
});
330
}
331
332
$write(handle: number, fd: number, pos: number, data: VSBuffer): Promise<number> {
333
const provider = this._getFsProvider(handle);
334
if (!provider.write) {
335
throw new Error('FileSystemProvider does not implement "write"');
336
}
337
return Promise.resolve(provider.write(fd, pos, data.buffer, 0, data.byteLength));
338
}
339
340
private _getFsProvider(handle: number): vscode.FileSystemProvider {
341
const provider = this._fsProvider.get(handle);
342
if (!provider) {
343
const err = new Error();
344
err.name = 'ENOPRO';
345
err.message = `no provider`;
346
throw err;
347
}
348
return provider;
349
}
350
}
351
352