@theia/filesystem 1.53.0-next.5 → 1.53.0-next.55

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (102) hide show
  1. package/README.md +30 -30
  2. package/lib/browser/file-resource.d.ts +2 -0
  3. package/lib/browser/file-resource.d.ts.map +1 -1
  4. package/lib/browser/file-resource.js +11 -0
  5. package/lib/browser/file-resource.js.map +1 -1
  6. package/lib/browser/file-resource.spec.d.ts +2 -0
  7. package/lib/browser/file-resource.spec.d.ts.map +1 -0
  8. package/lib/browser/file-resource.spec.js +191 -0
  9. package/lib/browser/file-resource.spec.js.map +1 -0
  10. package/lib/browser/file-service.d.ts +1 -0
  11. package/lib/browser/file-service.d.ts.map +1 -1
  12. package/lib/browser/file-service.js +3 -0
  13. package/lib/browser/file-service.js.map +1 -1
  14. package/package.json +4 -4
  15. package/src/browser/breadcrumbs/filepath-breadcrumb.ts +43 -43
  16. package/src/browser/breadcrumbs/filepath-breadcrumbs-container.ts +65 -65
  17. package/src/browser/breadcrumbs/filepath-breadcrumbs-contribution.ts +129 -129
  18. package/src/browser/download/file-download-command-contribution.ts +83 -83
  19. package/src/browser/download/file-download-frontend-module.ts +25 -25
  20. package/src/browser/download/file-download-service.ts +179 -179
  21. package/src/browser/file-dialog/file-dialog-container.ts +67 -67
  22. package/src/browser/file-dialog/file-dialog-hidden-files-renderer.tsx +59 -59
  23. package/src/browser/file-dialog/file-dialog-model.ts +96 -96
  24. package/src/browser/file-dialog/file-dialog-module.ts +44 -44
  25. package/src/browser/file-dialog/file-dialog-service.ts +99 -99
  26. package/src/browser/file-dialog/file-dialog-tree-filters-renderer.tsx +100 -100
  27. package/src/browser/file-dialog/file-dialog-tree.ts +89 -89
  28. package/src/browser/file-dialog/file-dialog-widget.ts +75 -75
  29. package/src/browser/file-dialog/file-dialog.ts +434 -434
  30. package/src/browser/file-dialog/index.ts +20 -20
  31. package/src/browser/file-resource.spec.ts +255 -0
  32. package/src/browser/file-resource.ts +402 -390
  33. package/src/browser/file-selection.ts +44 -44
  34. package/src/browser/file-service.ts +1845 -1841
  35. package/src/browser/file-tree/file-tree-container.ts +36 -36
  36. package/src/browser/file-tree/file-tree-decorator-adapter.ts +159 -159
  37. package/src/browser/file-tree/file-tree-label-provider.ts +53 -53
  38. package/src/browser/file-tree/file-tree-model.ts +212 -212
  39. package/src/browser/file-tree/file-tree-widget.tsx +327 -327
  40. package/src/browser/file-tree/file-tree.ts +183 -183
  41. package/src/browser/file-tree/index.ts +22 -22
  42. package/src/browser/file-upload-service.ts +547 -547
  43. package/src/browser/filesystem-frontend-contribution.ts +396 -396
  44. package/src/browser/filesystem-frontend-module.ts +77 -77
  45. package/src/browser/filesystem-preferences.ts +139 -139
  46. package/src/browser/filesystem-saveable-service.ts +138 -138
  47. package/src/browser/filesystem-watcher-error-handler.ts +60 -60
  48. package/src/browser/index.ts +21 -21
  49. package/src/browser/location/index.ts +18 -18
  50. package/src/browser/location/location-renderer.tsx +406 -406
  51. package/src/browser/location/location-service.ts +22 -22
  52. package/src/browser/remote-file-service-contribution.ts +38 -38
  53. package/src/browser/style/file-dialog.css +208 -208
  54. package/src/browser/style/file-icons.css +64 -64
  55. package/src/browser/style/filepath-breadcrumbs.css +20 -20
  56. package/src/browser/style/index.css +36 -36
  57. package/src/browser-only/browser-only-filesystem-frontend-module.ts +38 -38
  58. package/src/browser-only/browser-only-filesystem-provider-server.ts +32 -32
  59. package/src/browser-only/browserfs-filesystem-initialization.ts +61 -61
  60. package/src/browser-only/browserfs-filesystem-provider.ts +462 -462
  61. package/src/common/delegating-file-system-provider.ts +226 -226
  62. package/src/common/download/README.md +30 -30
  63. package/src/common/download/file-download-data.ts +27 -27
  64. package/src/common/file-upload.ts +17 -17
  65. package/src/common/files.spec.ts +51 -51
  66. package/src/common/files.ts +997 -997
  67. package/src/common/filesystem-utils.spec.ts +411 -411
  68. package/src/common/filesystem-utils.ts +64 -64
  69. package/src/common/filesystem-watcher-protocol.ts +96 -96
  70. package/src/common/filesystem.ts +43 -43
  71. package/src/common/index.ts +18 -18
  72. package/src/common/io.ts +150 -150
  73. package/src/common/remote-file-system-provider.ts +549 -549
  74. package/src/electron-browser/file-dialog/electron-file-dialog-module.ts +24 -24
  75. package/src/electron-browser/file-dialog/electron-file-dialog-service.ts +165 -165
  76. package/src/electron-browser/preload.ts +31 -31
  77. package/src/electron-common/electron-api.ts +55 -55
  78. package/src/electron-main/electron-api-main.ts +78 -78
  79. package/src/electron-main/electron-main-module.ts +23 -23
  80. package/src/node/disk-file-system-provider.spec.ts +142 -142
  81. package/src/node/disk-file-system-provider.ts +915 -915
  82. package/src/node/download/directory-archiver.spec.ts +104 -104
  83. package/src/node/download/directory-archiver.ts +126 -126
  84. package/src/node/download/file-download-backend-module.ts +32 -32
  85. package/src/node/download/file-download-cache.ts +88 -88
  86. package/src/node/download/file-download-endpoint.ts +63 -63
  87. package/src/node/download/file-download-handler.ts +304 -304
  88. package/src/node/download/test/mock-directory-archiver.ts +30 -30
  89. package/src/node/file-change-collection.spec.ts +110 -110
  90. package/src/node/file-change-collection.ts +78 -78
  91. package/src/node/filesystem-backend-module.ts +140 -140
  92. package/src/node/filesystem-watcher-client.ts +72 -72
  93. package/src/node/filesystem-watcher-dispatcher.ts +82 -82
  94. package/src/node/node-file-upload-service.ts +80 -80
  95. package/src/node/nsfw-watcher/index.ts +45 -45
  96. package/src/node/nsfw-watcher/nsfw-filesystem-service.ts +481 -481
  97. package/src/node/nsfw-watcher/nsfw-filesystem-watcher.spec.ts +182 -182
  98. package/src/node/nsfw-watcher/nsfw-options.ts +23 -23
  99. package/src/typings/dom.webkit.d.ts +77 -77
  100. package/src/typings/mv/index.d.ts +21 -21
  101. package/src/typings/nsfw/index.d.ts +18 -18
  102. package/src/typings/trash/index.d.ts +20 -20
@@ -1,78 +1,78 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2023 STMicroelectronics and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- import { injectable } from '@theia/core/shared/inversify';
18
-
19
- import { ElectronMainApplication, ElectronMainApplicationContribution } from '@theia/core/lib/electron-main/electron-main-application';
20
- import { MaybePromise } from '@theia/core';
21
- import { CHANNEL_SHOW_OPEN, CHANNEL_SHOW_SAVE, OpenDialogOptions, SaveDialogOptions } from '../electron-common/electron-api';
22
- import { ipcMain, OpenDialogOptions as ElectronOpenDialogOptions, SaveDialogOptions as ElectronSaveDialogOptions, BrowserWindow, dialog }
23
- from '@theia/core/electron-shared/electron';
24
-
25
- @injectable()
26
- export class ElectronApi implements ElectronMainApplicationContribution {
27
- onStart(application: ElectronMainApplication): MaybePromise<void> {
28
- // dialogs
29
- ipcMain.handle(CHANNEL_SHOW_OPEN, async (event, options: OpenDialogOptions) => {
30
- const properties: ElectronOpenDialogOptions['properties'] = [];
31
-
32
- // checking proper combination of file/dir opening is done on the renderer side
33
- if (options.openFiles) {
34
- properties.push('openFile');
35
- }
36
- if (options.openFolders) {
37
- properties.push('openDirectory');
38
- }
39
-
40
- if (options.selectMany === true) {
41
- properties.push('multiSelections');
42
- }
43
-
44
- const dialogOpts: ElectronOpenDialogOptions = {
45
- defaultPath: options.path,
46
- buttonLabel: options.buttonLabel,
47
- filters: options.filters,
48
- title: options.title,
49
- properties: properties
50
- };
51
-
52
- if (options.modal) {
53
- const win = BrowserWindow.fromWebContents(event.sender);
54
- if (win) {
55
- return (await dialog.showOpenDialog(win, dialogOpts)).filePaths;
56
- }
57
- }
58
- return (await dialog.showOpenDialog(dialogOpts)).filePaths;
59
- });
60
-
61
- ipcMain.handle(CHANNEL_SHOW_SAVE, async (event, options: SaveDialogOptions) => {
62
- const dialogOpts: ElectronSaveDialogOptions = {
63
- defaultPath: options.path,
64
- buttonLabel: options.buttonLabel,
65
- filters: options.filters,
66
- title: options.title
67
- };
68
- if (options.modal) {
69
- const win = BrowserWindow.fromWebContents(event.sender);
70
- if (win) {
71
- return (await dialog.showSaveDialog(win, dialogOpts)).filePath;
72
- }
73
- }
74
- return (await dialog.showSaveDialog(dialogOpts)).filePath;
75
- });
76
-
77
- }
78
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2023 STMicroelectronics and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ import { injectable } from '@theia/core/shared/inversify';
18
+
19
+ import { ElectronMainApplication, ElectronMainApplicationContribution } from '@theia/core/lib/electron-main/electron-main-application';
20
+ import { MaybePromise } from '@theia/core';
21
+ import { CHANNEL_SHOW_OPEN, CHANNEL_SHOW_SAVE, OpenDialogOptions, SaveDialogOptions } from '../electron-common/electron-api';
22
+ import { ipcMain, OpenDialogOptions as ElectronOpenDialogOptions, SaveDialogOptions as ElectronSaveDialogOptions, BrowserWindow, dialog }
23
+ from '@theia/core/electron-shared/electron';
24
+
25
+ @injectable()
26
+ export class ElectronApi implements ElectronMainApplicationContribution {
27
+ onStart(application: ElectronMainApplication): MaybePromise<void> {
28
+ // dialogs
29
+ ipcMain.handle(CHANNEL_SHOW_OPEN, async (event, options: OpenDialogOptions) => {
30
+ const properties: ElectronOpenDialogOptions['properties'] = [];
31
+
32
+ // checking proper combination of file/dir opening is done on the renderer side
33
+ if (options.openFiles) {
34
+ properties.push('openFile');
35
+ }
36
+ if (options.openFolders) {
37
+ properties.push('openDirectory');
38
+ }
39
+
40
+ if (options.selectMany === true) {
41
+ properties.push('multiSelections');
42
+ }
43
+
44
+ const dialogOpts: ElectronOpenDialogOptions = {
45
+ defaultPath: options.path,
46
+ buttonLabel: options.buttonLabel,
47
+ filters: options.filters,
48
+ title: options.title,
49
+ properties: properties
50
+ };
51
+
52
+ if (options.modal) {
53
+ const win = BrowserWindow.fromWebContents(event.sender);
54
+ if (win) {
55
+ return (await dialog.showOpenDialog(win, dialogOpts)).filePaths;
56
+ }
57
+ }
58
+ return (await dialog.showOpenDialog(dialogOpts)).filePaths;
59
+ });
60
+
61
+ ipcMain.handle(CHANNEL_SHOW_SAVE, async (event, options: SaveDialogOptions) => {
62
+ const dialogOpts: ElectronSaveDialogOptions = {
63
+ defaultPath: options.path,
64
+ buttonLabel: options.buttonLabel,
65
+ filters: options.filters,
66
+ title: options.title
67
+ };
68
+ if (options.modal) {
69
+ const win = BrowserWindow.fromWebContents(event.sender);
70
+ if (win) {
71
+ return (await dialog.showSaveDialog(win, dialogOpts)).filePath;
72
+ }
73
+ }
74
+ return (await dialog.showSaveDialog(dialogOpts)).filePath;
75
+ });
76
+
77
+ }
78
+ }
@@ -1,23 +1,23 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2023 STMicroelectronics and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
- import { ContainerModule } from '@theia/core/shared/inversify';
17
- import { ElectronMainApplicationContribution } from '@theia/core/lib/electron-main/electron-main-application';
18
- import { ElectronApi } from './electron-api-main';
19
-
20
- export default new ContainerModule(bind => {
21
- bind(ElectronApi).toSelf().inSingletonScope();
22
- bind(ElectronMainApplicationContribution).toService(ElectronApi);
23
- });
1
+ // *****************************************************************************
2
+ // Copyright (C) 2023 STMicroelectronics and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+ import { ContainerModule } from '@theia/core/shared/inversify';
17
+ import { ElectronMainApplicationContribution } from '@theia/core/lib/electron-main/electron-main-application';
18
+ import { ElectronApi } from './electron-api-main';
19
+
20
+ export default new ContainerModule(bind => {
21
+ bind(ElectronApi).toSelf().inSingletonScope();
22
+ bind(ElectronMainApplicationContribution).toService(ElectronApi);
23
+ });
@@ -1,142 +1,142 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2023 Arduino SA and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- import { Disposable, DisposableCollection } from '@theia/core/lib/common/disposable';
18
- import { EncodingService } from '@theia/core/lib/common/encoding-service';
19
- import { ILogger } from '@theia/core/lib/common/logger';
20
- import { MockLogger } from '@theia/core/lib/common/test/mock-logger';
21
- import { FileUri } from '@theia/core/lib/common/file-uri';
22
- import { IPCConnectionProvider } from '@theia/core/lib/node/messaging/ipc-connection-provider';
23
- import { Container, ContainerModule } from '@theia/core/shared/inversify';
24
- import { equal, fail } from 'assert';
25
- import { promises as fs } from 'fs';
26
- import { join } from 'path';
27
- import * as temp from 'temp';
28
- import { generateUuid } from '@theia/core/lib/common/uuid';
29
- import { FilePermission, FileSystemProviderCapabilities, FileSystemProviderError, FileSystemProviderErrorCode } from '../common/files';
30
- import { DiskFileSystemProvider } from './disk-file-system-provider';
31
- import { bindFileSystemWatcherServer } from './filesystem-backend-module';
32
-
33
- const tracked = temp.track();
34
-
35
- describe('disk-file-system-provider', () => {
36
- let toDisposeAfter: DisposableCollection;
37
- let fsProvider: DiskFileSystemProvider;
38
-
39
- before(() => {
40
- fsProvider = createContainer().get<DiskFileSystemProvider>(
41
- DiskFileSystemProvider
42
- );
43
- toDisposeAfter = new DisposableCollection(
44
- fsProvider,
45
- Disposable.create(() => tracked.cleanupSync())
46
- );
47
- });
48
-
49
- after(() => {
50
- toDisposeAfter.dispose();
51
- });
52
-
53
- describe('stat', () => {
54
- it("should omit the 'permissions' property of the stat if the file can be both read and write", async () => {
55
- const tempDirPath = tracked.mkdirSync();
56
- const tempFilePath = join(tempDirPath, `${generateUuid()}.txt`);
57
- await fs.writeFile(tempFilePath, 'some content', { encoding: 'utf8' });
58
-
59
- let content = await fs.readFile(tempFilePath, { encoding: 'utf8' });
60
- equal(content, 'some content');
61
-
62
- await fs.writeFile(tempFilePath, 'other content', { encoding: 'utf8' });
63
-
64
- content = await fs.readFile(tempFilePath, { encoding: 'utf8' });
65
- equal(content, 'other content');
66
-
67
- const stat = await fsProvider.stat(FileUri.create(tempFilePath));
68
- equal(stat.permissions, undefined);
69
- });
70
-
71
- it("should set the 'permissions' property to `Readonly` if the file can be read but not write", async () => {
72
- const tempDirPath = tracked.mkdirSync();
73
- const tempFilePath = join(tempDirPath, `${generateUuid()}.txt`);
74
- await fs.writeFile(tempFilePath, 'readonly content', {
75
- encoding: 'utf8',
76
- });
77
-
78
- await fs.chmod(tempFilePath, '444'); // read-only for owner/group/world
79
-
80
- try {
81
- await fsProvider.writeFile(FileUri.create(tempFilePath), new Uint8Array(), { create: false, overwrite: true });
82
- fail('Expected an EACCES error for readonly (chmod 444) files');
83
- } catch (err) {
84
- equal(err instanceof FileSystemProviderError, true);
85
- equal((<FileSystemProviderError>err).code, FileSystemProviderErrorCode.NoPermissions);
86
- }
87
-
88
- const content = await fs.readFile(tempFilePath, { encoding: 'utf8' });
89
- equal(content, 'readonly content');
90
-
91
- const stat = await fsProvider.stat(FileUri.create(tempFilePath));
92
- equal(stat.permissions, FilePermission.Readonly);
93
- });
94
- });
95
-
96
- describe('delete', () => {
97
- it('delete is able to delete folder', async () => {
98
- const tempDirPath = tracked.mkdirSync();
99
- const testFolder = join(tempDirPath, 'test');
100
- const folderUri = FileUri.create(testFolder);
101
- for (const recursive of [true, false]) {
102
- // Note: useTrash = true fails on Linux
103
- const useTrash = false;
104
- if ((fsProvider.capabilities & FileSystemProviderCapabilities.Access) === 0 && useTrash) {
105
- continue;
106
- }
107
- await fsProvider.mkdir(folderUri);
108
- if (recursive) {
109
- await fsProvider.writeFile(FileUri.create(join(testFolder, 'test.file')), Buffer.from('test'), { overwrite: false, create: true });
110
- await fsProvider.mkdir(FileUri.create(join(testFolder, 'subFolder')));
111
- }
112
- await fsProvider.delete(folderUri, { recursive, useTrash });
113
- }
114
- });
115
-
116
- it('delete is able to delete file', async () => {
117
- const tempDirPath = tracked.mkdirSync();
118
- const testFile = join(tempDirPath, 'test.file');
119
- const testFileUri = FileUri.create(testFile);
120
- for (const recursive of [true, false]) {
121
- for (const useTrash of [true, false]) {
122
- await fsProvider.writeFile(testFileUri, Buffer.from('test'), { overwrite: false, create: true });
123
- await fsProvider.delete(testFileUri, { recursive, useTrash });
124
- }
125
- }
126
- });
127
- });
128
-
129
- function createContainer(): Container {
130
- const container = new Container({ defaultScope: 'Singleton' });
131
- const module = new ContainerModule(bind => {
132
- bind(DiskFileSystemProvider).toSelf().inSingletonScope();
133
- bind(EncodingService).toSelf().inSingletonScope();
134
- bindFileSystemWatcherServer(bind);
135
- bind(MockLogger).toSelf().inSingletonScope();
136
- bind(ILogger).toService(MockLogger);
137
- bind(IPCConnectionProvider).toSelf().inSingletonScope();
138
- });
139
- container.load(module);
140
- return container;
141
- }
142
- });
1
+ // *****************************************************************************
2
+ // Copyright (C) 2023 Arduino SA and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ import { Disposable, DisposableCollection } from '@theia/core/lib/common/disposable';
18
+ import { EncodingService } from '@theia/core/lib/common/encoding-service';
19
+ import { ILogger } from '@theia/core/lib/common/logger';
20
+ import { MockLogger } from '@theia/core/lib/common/test/mock-logger';
21
+ import { FileUri } from '@theia/core/lib/common/file-uri';
22
+ import { IPCConnectionProvider } from '@theia/core/lib/node/messaging/ipc-connection-provider';
23
+ import { Container, ContainerModule } from '@theia/core/shared/inversify';
24
+ import { equal, fail } from 'assert';
25
+ import { promises as fs } from 'fs';
26
+ import { join } from 'path';
27
+ import * as temp from 'temp';
28
+ import { generateUuid } from '@theia/core/lib/common/uuid';
29
+ import { FilePermission, FileSystemProviderCapabilities, FileSystemProviderError, FileSystemProviderErrorCode } from '../common/files';
30
+ import { DiskFileSystemProvider } from './disk-file-system-provider';
31
+ import { bindFileSystemWatcherServer } from './filesystem-backend-module';
32
+
33
+ const tracked = temp.track();
34
+
35
+ describe('disk-file-system-provider', () => {
36
+ let toDisposeAfter: DisposableCollection;
37
+ let fsProvider: DiskFileSystemProvider;
38
+
39
+ before(() => {
40
+ fsProvider = createContainer().get<DiskFileSystemProvider>(
41
+ DiskFileSystemProvider
42
+ );
43
+ toDisposeAfter = new DisposableCollection(
44
+ fsProvider,
45
+ Disposable.create(() => tracked.cleanupSync())
46
+ );
47
+ });
48
+
49
+ after(() => {
50
+ toDisposeAfter.dispose();
51
+ });
52
+
53
+ describe('stat', () => {
54
+ it("should omit the 'permissions' property of the stat if the file can be both read and write", async () => {
55
+ const tempDirPath = tracked.mkdirSync();
56
+ const tempFilePath = join(tempDirPath, `${generateUuid()}.txt`);
57
+ await fs.writeFile(tempFilePath, 'some content', { encoding: 'utf8' });
58
+
59
+ let content = await fs.readFile(tempFilePath, { encoding: 'utf8' });
60
+ equal(content, 'some content');
61
+
62
+ await fs.writeFile(tempFilePath, 'other content', { encoding: 'utf8' });
63
+
64
+ content = await fs.readFile(tempFilePath, { encoding: 'utf8' });
65
+ equal(content, 'other content');
66
+
67
+ const stat = await fsProvider.stat(FileUri.create(tempFilePath));
68
+ equal(stat.permissions, undefined);
69
+ });
70
+
71
+ it("should set the 'permissions' property to `Readonly` if the file can be read but not write", async () => {
72
+ const tempDirPath = tracked.mkdirSync();
73
+ const tempFilePath = join(tempDirPath, `${generateUuid()}.txt`);
74
+ await fs.writeFile(tempFilePath, 'readonly content', {
75
+ encoding: 'utf8',
76
+ });
77
+
78
+ await fs.chmod(tempFilePath, '444'); // read-only for owner/group/world
79
+
80
+ try {
81
+ await fsProvider.writeFile(FileUri.create(tempFilePath), new Uint8Array(), { create: false, overwrite: true });
82
+ fail('Expected an EACCES error for readonly (chmod 444) files');
83
+ } catch (err) {
84
+ equal(err instanceof FileSystemProviderError, true);
85
+ equal((<FileSystemProviderError>err).code, FileSystemProviderErrorCode.NoPermissions);
86
+ }
87
+
88
+ const content = await fs.readFile(tempFilePath, { encoding: 'utf8' });
89
+ equal(content, 'readonly content');
90
+
91
+ const stat = await fsProvider.stat(FileUri.create(tempFilePath));
92
+ equal(stat.permissions, FilePermission.Readonly);
93
+ });
94
+ });
95
+
96
+ describe('delete', () => {
97
+ it('delete is able to delete folder', async () => {
98
+ const tempDirPath = tracked.mkdirSync();
99
+ const testFolder = join(tempDirPath, 'test');
100
+ const folderUri = FileUri.create(testFolder);
101
+ for (const recursive of [true, false]) {
102
+ // Note: useTrash = true fails on Linux
103
+ const useTrash = false;
104
+ if ((fsProvider.capabilities & FileSystemProviderCapabilities.Access) === 0 && useTrash) {
105
+ continue;
106
+ }
107
+ await fsProvider.mkdir(folderUri);
108
+ if (recursive) {
109
+ await fsProvider.writeFile(FileUri.create(join(testFolder, 'test.file')), Buffer.from('test'), { overwrite: false, create: true });
110
+ await fsProvider.mkdir(FileUri.create(join(testFolder, 'subFolder')));
111
+ }
112
+ await fsProvider.delete(folderUri, { recursive, useTrash });
113
+ }
114
+ });
115
+
116
+ it('delete is able to delete file', async () => {
117
+ const tempDirPath = tracked.mkdirSync();
118
+ const testFile = join(tempDirPath, 'test.file');
119
+ const testFileUri = FileUri.create(testFile);
120
+ for (const recursive of [true, false]) {
121
+ for (const useTrash of [true, false]) {
122
+ await fsProvider.writeFile(testFileUri, Buffer.from('test'), { overwrite: false, create: true });
123
+ await fsProvider.delete(testFileUri, { recursive, useTrash });
124
+ }
125
+ }
126
+ });
127
+ });
128
+
129
+ function createContainer(): Container {
130
+ const container = new Container({ defaultScope: 'Singleton' });
131
+ const module = new ContainerModule(bind => {
132
+ bind(DiskFileSystemProvider).toSelf().inSingletonScope();
133
+ bind(EncodingService).toSelf().inSingletonScope();
134
+ bindFileSystemWatcherServer(bind);
135
+ bind(MockLogger).toSelf().inSingletonScope();
136
+ bind(ILogger).toService(MockLogger);
137
+ bind(IPCConnectionProvider).toSelf().inSingletonScope();
138
+ });
139
+ container.load(module);
140
+ return container;
141
+ }
142
+ });