@theia/playwright 1.34.3 → 1.34.4

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 (124) hide show
  1. package/LICENSE +641 -641
  2. package/README.md +54 -54
  3. package/lib/index.d.ts +25 -25
  4. package/lib/index.js +52 -52
  5. package/lib/tests/fixtures/theia-fixture.d.ts +3 -3
  6. package/lib/tests/fixtures/theia-fixture.js +23 -23
  7. package/lib/tests/theia-app.test.d.ts +1 -1
  8. package/lib/tests/theia-app.test.js +29 -29
  9. package/lib/tests/theia-explorer-view.test.d.ts +1 -1
  10. package/lib/tests/theia-explorer-view.test.js +109 -109
  11. package/lib/tests/theia-main-menu.test.d.ts +1 -1
  12. package/lib/tests/theia-main-menu.test.js +73 -73
  13. package/lib/tests/theia-preference-view.test.d.ts +1 -1
  14. package/lib/tests/theia-preference-view.test.js +99 -99
  15. package/lib/tests/theia-problems-view.test.d.ts +1 -1
  16. package/lib/tests/theia-problems-view.test.js +52 -52
  17. package/lib/tests/theia-quick-command.test.d.ts +1 -1
  18. package/lib/tests/theia-quick-command.test.js +53 -53
  19. package/lib/tests/theia-sample-app.test.d.ts +1 -1
  20. package/lib/tests/theia-sample-app.test.js +80 -80
  21. package/lib/tests/theia-status-bar.test.d.ts +1 -1
  22. package/lib/tests/theia-status-bar.test.js +41 -41
  23. package/lib/tests/theia-text-editor.test.d.ts +1 -1
  24. package/lib/tests/theia-text-editor.test.js +152 -152
  25. package/lib/tests/theia-workspace.test.d.ts +1 -1
  26. package/lib/tests/theia-workspace.test.js +47 -47
  27. package/lib/theia-about-dialog.d.ts +4 -4
  28. package/lib/theia-about-dialog.js +26 -26
  29. package/lib/theia-app.d.ts +43 -43
  30. package/lib/theia-app.js +135 -135
  31. package/lib/theia-context-menu.d.ts +8 -8
  32. package/lib/theia-context-menu.js +37 -37
  33. package/lib/theia-dialog.d.ts +27 -27
  34. package/lib/theia-dialog.js +98 -98
  35. package/lib/theia-editor.d.ts +9 -9
  36. package/lib/theia-editor.js +68 -68
  37. package/lib/theia-explorer-view.d.ts +38 -38
  38. package/lib/theia-explorer-view.js +209 -209
  39. package/lib/theia-main-menu.d.ts +12 -12
  40. package/lib/theia-main-menu.js +53 -53
  41. package/lib/theia-menu-item.d.ts +14 -14
  42. package/lib/theia-menu-item.js +66 -66
  43. package/lib/theia-menu.d.ts +16 -16
  44. package/lib/theia-menu.js +86 -86
  45. package/lib/theia-notification-indicator.d.ts +7 -7
  46. package/lib/theia-notification-indicator.js +44 -44
  47. package/lib/theia-notification-overlay.d.ts +22 -22
  48. package/lib/theia-notification-overlay.js +79 -79
  49. package/lib/theia-page-object.d.ts +7 -7
  50. package/lib/theia-page-object.js +27 -27
  51. package/lib/theia-preference-view.d.ts +76 -76
  52. package/lib/theia-preference-view.js +201 -201
  53. package/lib/theia-problem-indicator.d.ts +8 -8
  54. package/lib/theia-problem-indicator.js +38 -38
  55. package/lib/theia-problem-view.d.ts +5 -5
  56. package/lib/theia-problem-view.js +30 -30
  57. package/lib/theia-quick-command-palette.d.ts +11 -11
  58. package/lib/theia-quick-command-palette.js +73 -73
  59. package/lib/theia-rename-dialog.d.ts +5 -5
  60. package/lib/theia-rename-dialog.js +35 -35
  61. package/lib/theia-status-bar.d.ts +13 -13
  62. package/lib/theia-status-bar.js +39 -39
  63. package/lib/theia-status-indicator.d.ts +10 -10
  64. package/lib/theia-status-indicator.js +48 -48
  65. package/lib/theia-text-editor.d.ts +27 -27
  66. package/lib/theia-text-editor.js +149 -149
  67. package/lib/theia-toggle-bottom-indicator.d.ts +4 -4
  68. package/lib/theia-toggle-bottom-indicator.js +26 -26
  69. package/lib/theia-tree-node.d.ts +16 -16
  70. package/lib/theia-tree-node.js +53 -53
  71. package/lib/theia-view.d.ts +32 -32
  72. package/lib/theia-view.js +151 -151
  73. package/lib/theia-workspace.d.ts +18 -18
  74. package/lib/theia-workspace.js +69 -69
  75. package/lib/util.d.ts +19 -19
  76. package/lib/util.js +93 -93
  77. package/package.json +2 -2
  78. package/src/index.ts +41 -41
  79. package/src/tests/fixtures/theia-fixture.ts +25 -25
  80. package/src/tests/resources/sample-files1/sample.txt +4 -4
  81. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-1/sampleFile1-1-1.txt +1 -1
  82. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-1/sampleFile1-1-2.txt +1 -1
  83. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-2/sampleFile1-2-1.txt +1 -1
  84. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder1/sampleFolder1-2/sampleFile1-2-2.txt +1 -1
  85. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-1/sampleFile2-1-1.txt +1 -1
  86. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-1/sampleFile2-1-2.txt +1 -1
  87. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-2/sampleFile2-2-1.txt +1 -1
  88. package/src/tests/resources/sample-files1/sampleFolder/sampleFolder2/sampleFolder2-2/sampleFile2-2-2.txt +1 -1
  89. package/src/tests/resources/sample-files2/another-sample.txt +1 -1
  90. package/src/tests/theia-app.test.ts +34 -34
  91. package/src/tests/theia-explorer-view.test.ts +124 -124
  92. package/src/tests/theia-main-menu.test.ts +89 -89
  93. package/src/tests/theia-preference-view.test.ts +118 -118
  94. package/src/tests/theia-problems-view.test.ts +60 -60
  95. package/src/tests/theia-quick-command.test.ts +61 -61
  96. package/src/tests/theia-sample-app.test.ts +92 -92
  97. package/src/tests/theia-status-bar.test.ts +47 -47
  98. package/src/tests/theia-text-editor.test.ts +182 -182
  99. package/src/tests/theia-workspace.test.ts +51 -51
  100. package/src/theia-about-dialog.ts +26 -26
  101. package/src/theia-app.ts +154 -154
  102. package/src/theia-context-menu.ts +42 -42
  103. package/src/theia-dialog.ts +113 -113
  104. package/src/theia-editor.ts +73 -73
  105. package/src/theia-explorer-view.ts +234 -234
  106. package/src/theia-main-menu.ts +54 -54
  107. package/src/theia-menu-item.ts +75 -75
  108. package/src/theia-menu.ts +96 -96
  109. package/src/theia-notification-indicator.ts +44 -44
  110. package/src/theia-notification-overlay.ts +94 -94
  111. package/src/theia-page-object.ts +29 -29
  112. package/src/theia-preference-view.ts +232 -232
  113. package/src/theia-problem-indicator.ts +37 -37
  114. package/src/theia-problem-view.ts +30 -30
  115. package/src/theia-quick-command-palette.ts +75 -75
  116. package/src/theia-rename-dialog.ts +36 -36
  117. package/src/theia-status-bar.ts +44 -44
  118. package/src/theia-status-indicator.ts +50 -50
  119. package/src/theia-text-editor.ts +172 -172
  120. package/src/theia-toggle-bottom-indicator.ts +21 -21
  121. package/src/theia-tree-node.ts +60 -60
  122. package/src/theia-view.ts +178 -178
  123. package/src/theia-workspace.ts +76 -76
  124. package/src/util.ts +91 -91
package/src/theia-view.ts CHANGED
@@ -1,178 +1,178 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2021 logi.cals GmbH, EclipseSource 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 WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- import { ElementHandle } from '@playwright/test';
18
-
19
- import { TheiaApp } from './theia-app';
20
- import { TheiaContextMenu } from './theia-context-menu';
21
- import { TheiaMenu } from './theia-menu';
22
- import { TheiaPageObject } from './theia-page-object';
23
- import { containsClass, isElementVisible, textContent } from './util';
24
-
25
- export interface TheiaViewData {
26
- tabSelector: string;
27
- viewSelector: string;
28
- viewName?: string;
29
- }
30
-
31
- export class TheiaView extends TheiaPageObject {
32
-
33
- constructor(protected readonly data: TheiaViewData, app: TheiaApp) {
34
- super(app);
35
- }
36
-
37
- get tabSelector(): string {
38
- return this.data.tabSelector;
39
- }
40
-
41
- get viewSelector(): string {
42
- return this.data.viewSelector;
43
- }
44
-
45
- get name(): string | undefined {
46
- return this.data.viewName;
47
- }
48
-
49
- async open(): Promise<TheiaView> {
50
- if (!this.data.viewName) {
51
- throw new Error('View name must be specified to open via command palette');
52
- }
53
- await this.app.quickCommandPalette.type('View: Open View');
54
- await this.app.quickCommandPalette.trigger('View: Open View...', this.data.viewName);
55
- await this.waitForVisible();
56
- return this;
57
- }
58
-
59
- async focus(): Promise<void> {
60
- await this.activate();
61
- const view = await this.viewElement();
62
- await view?.click();
63
- }
64
-
65
- async activate(): Promise<void> {
66
- await this.page.waitForSelector(this.tabSelector, { state: 'visible' });
67
- if (!await this.isActive()) {
68
- const tab = await this.tabElement();
69
- await tab?.click();
70
- }
71
- return this.waitForVisible();
72
- }
73
-
74
- async waitForVisible(): Promise<void> {
75
- await this.page.waitForSelector(this.viewSelector, { state: 'visible' });
76
- }
77
-
78
- async isTabVisible(): Promise<boolean> {
79
- return isElementVisible(this.tabElement());
80
- }
81
-
82
- async isDisplayed(): Promise<boolean> {
83
- return isElementVisible(this.viewElement());
84
- }
85
-
86
- async isActive(): Promise<boolean> {
87
- return await this.isTabVisible() && containsClass(this.tabElement(), 'p-mod-current');
88
- }
89
-
90
- async isClosable(): Promise<boolean> {
91
- return await this.isTabVisible() && containsClass(this.tabElement(), 'p-mod-closable');
92
- }
93
-
94
- async close(waitForClosed = true): Promise<void> {
95
- if (!(await this.isTabVisible())) {
96
- return;
97
- }
98
- if (!(await this.isClosable())) {
99
- throw Error(`View ${this.tabSelector} is not closable`);
100
- }
101
- const tab = await this.tabElement();
102
- const side = await this.side();
103
- if (side === 'main' || side === 'bottom') {
104
- const closeIcon = await tab?.waitForSelector('div.p-TabBar-tabCloseIcon');
105
- await closeIcon?.click();
106
- } else {
107
- const menu = await this.openContextMenuOnTab();
108
- const closeItem = await menu.menuItemByName('Close');
109
- await closeItem?.click();
110
- }
111
- if (waitForClosed) {
112
- await this.waitUntilClosed();
113
- }
114
- }
115
-
116
- protected async waitUntilClosed(): Promise<void> {
117
- await this.page.waitForSelector(this.tabSelector, { state: 'detached' });
118
- }
119
-
120
- async title(): Promise<string | undefined> {
121
- if ((await this.isInSidePanel()) && !(await this.isActive())) {
122
- // we can only determine the label of a side-panel view, if it is active
123
- await this.activate();
124
- }
125
- switch (await this.side()) {
126
- case 'left':
127
- return textContent(this.page.waitForSelector('div.theia-left-side-panel > div.theia-sidepanel-title'));
128
- case 'right':
129
- return textContent(this.page.waitForSelector('div.theia-right-side-panel > div.theia-sidepanel-title'));
130
- }
131
- const tab = await this.tabElement();
132
- if (tab) {
133
- return textContent(tab.waitForSelector('div.theia-tab-icon-label > div.p-TabBar-tabLabel'));
134
- }
135
- return undefined;
136
- }
137
-
138
- async isInSidePanel(): Promise<boolean> {
139
- return (await this.side() === 'left') || (await this.side() === 'right');
140
- }
141
-
142
- async side(): Promise<'left' | 'right' | 'bottom' | 'main'> {
143
- if (!await this.isTabVisible()) {
144
- throw Error(`Unable to determine side of invisible view tab '${this.tabSelector}'`);
145
- }
146
- const tab = await this.tabElement();
147
- let appAreaElement = tab?.$('xpath=../..');
148
- if (await containsClass(appAreaElement, 'theia-app-left')) {
149
- return 'left';
150
- }
151
- if (await containsClass(appAreaElement, 'theia-app-right')) {
152
- return 'right';
153
- }
154
-
155
- appAreaElement = (await appAreaElement)?.$('xpath=../..');
156
- if (await containsClass(appAreaElement, 'theia-app-bottom')) {
157
- return 'bottom';
158
- }
159
- if (await containsClass(appAreaElement, 'theia-app-main')) {
160
- return 'main';
161
- }
162
- throw Error(`Unable to determine side of view tab '${this.tabSelector}'`);
163
- }
164
-
165
- async openContextMenuOnTab(): Promise<TheiaMenu> {
166
- await this.activate();
167
- return TheiaContextMenu.open(this.app, () => this.page.waitForSelector(this.tabSelector));
168
- }
169
-
170
- protected viewElement(): Promise<ElementHandle<SVGElement | HTMLElement> | null> {
171
- return this.page.$(this.viewSelector);
172
- }
173
-
174
- protected tabElement(): Promise<ElementHandle<SVGElement | HTMLElement> | null> {
175
- return this.page.$(this.tabSelector);
176
- }
177
-
178
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2021 logi.cals GmbH, EclipseSource 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 WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ import { ElementHandle } from '@playwright/test';
18
+
19
+ import { TheiaApp } from './theia-app';
20
+ import { TheiaContextMenu } from './theia-context-menu';
21
+ import { TheiaMenu } from './theia-menu';
22
+ import { TheiaPageObject } from './theia-page-object';
23
+ import { containsClass, isElementVisible, textContent } from './util';
24
+
25
+ export interface TheiaViewData {
26
+ tabSelector: string;
27
+ viewSelector: string;
28
+ viewName?: string;
29
+ }
30
+
31
+ export class TheiaView extends TheiaPageObject {
32
+
33
+ constructor(protected readonly data: TheiaViewData, app: TheiaApp) {
34
+ super(app);
35
+ }
36
+
37
+ get tabSelector(): string {
38
+ return this.data.tabSelector;
39
+ }
40
+
41
+ get viewSelector(): string {
42
+ return this.data.viewSelector;
43
+ }
44
+
45
+ get name(): string | undefined {
46
+ return this.data.viewName;
47
+ }
48
+
49
+ async open(): Promise<TheiaView> {
50
+ if (!this.data.viewName) {
51
+ throw new Error('View name must be specified to open via command palette');
52
+ }
53
+ await this.app.quickCommandPalette.type('View: Open View');
54
+ await this.app.quickCommandPalette.trigger('View: Open View...', this.data.viewName);
55
+ await this.waitForVisible();
56
+ return this;
57
+ }
58
+
59
+ async focus(): Promise<void> {
60
+ await this.activate();
61
+ const view = await this.viewElement();
62
+ await view?.click();
63
+ }
64
+
65
+ async activate(): Promise<void> {
66
+ await this.page.waitForSelector(this.tabSelector, { state: 'visible' });
67
+ if (!await this.isActive()) {
68
+ const tab = await this.tabElement();
69
+ await tab?.click();
70
+ }
71
+ return this.waitForVisible();
72
+ }
73
+
74
+ async waitForVisible(): Promise<void> {
75
+ await this.page.waitForSelector(this.viewSelector, { state: 'visible' });
76
+ }
77
+
78
+ async isTabVisible(): Promise<boolean> {
79
+ return isElementVisible(this.tabElement());
80
+ }
81
+
82
+ async isDisplayed(): Promise<boolean> {
83
+ return isElementVisible(this.viewElement());
84
+ }
85
+
86
+ async isActive(): Promise<boolean> {
87
+ return await this.isTabVisible() && containsClass(this.tabElement(), 'p-mod-current');
88
+ }
89
+
90
+ async isClosable(): Promise<boolean> {
91
+ return await this.isTabVisible() && containsClass(this.tabElement(), 'p-mod-closable');
92
+ }
93
+
94
+ async close(waitForClosed = true): Promise<void> {
95
+ if (!(await this.isTabVisible())) {
96
+ return;
97
+ }
98
+ if (!(await this.isClosable())) {
99
+ throw Error(`View ${this.tabSelector} is not closable`);
100
+ }
101
+ const tab = await this.tabElement();
102
+ const side = await this.side();
103
+ if (side === 'main' || side === 'bottom') {
104
+ const closeIcon = await tab?.waitForSelector('div.p-TabBar-tabCloseIcon');
105
+ await closeIcon?.click();
106
+ } else {
107
+ const menu = await this.openContextMenuOnTab();
108
+ const closeItem = await menu.menuItemByName('Close');
109
+ await closeItem?.click();
110
+ }
111
+ if (waitForClosed) {
112
+ await this.waitUntilClosed();
113
+ }
114
+ }
115
+
116
+ protected async waitUntilClosed(): Promise<void> {
117
+ await this.page.waitForSelector(this.tabSelector, { state: 'detached' });
118
+ }
119
+
120
+ async title(): Promise<string | undefined> {
121
+ if ((await this.isInSidePanel()) && !(await this.isActive())) {
122
+ // we can only determine the label of a side-panel view, if it is active
123
+ await this.activate();
124
+ }
125
+ switch (await this.side()) {
126
+ case 'left':
127
+ return textContent(this.page.waitForSelector('div.theia-left-side-panel > div.theia-sidepanel-title'));
128
+ case 'right':
129
+ return textContent(this.page.waitForSelector('div.theia-right-side-panel > div.theia-sidepanel-title'));
130
+ }
131
+ const tab = await this.tabElement();
132
+ if (tab) {
133
+ return textContent(tab.waitForSelector('div.theia-tab-icon-label > div.p-TabBar-tabLabel'));
134
+ }
135
+ return undefined;
136
+ }
137
+
138
+ async isInSidePanel(): Promise<boolean> {
139
+ return (await this.side() === 'left') || (await this.side() === 'right');
140
+ }
141
+
142
+ async side(): Promise<'left' | 'right' | 'bottom' | 'main'> {
143
+ if (!await this.isTabVisible()) {
144
+ throw Error(`Unable to determine side of invisible view tab '${this.tabSelector}'`);
145
+ }
146
+ const tab = await this.tabElement();
147
+ let appAreaElement = tab?.$('xpath=../..');
148
+ if (await containsClass(appAreaElement, 'theia-app-left')) {
149
+ return 'left';
150
+ }
151
+ if (await containsClass(appAreaElement, 'theia-app-right')) {
152
+ return 'right';
153
+ }
154
+
155
+ appAreaElement = (await appAreaElement)?.$('xpath=../..');
156
+ if (await containsClass(appAreaElement, 'theia-app-bottom')) {
157
+ return 'bottom';
158
+ }
159
+ if (await containsClass(appAreaElement, 'theia-app-main')) {
160
+ return 'main';
161
+ }
162
+ throw Error(`Unable to determine side of view tab '${this.tabSelector}'`);
163
+ }
164
+
165
+ async openContextMenuOnTab(): Promise<TheiaMenu> {
166
+ await this.activate();
167
+ return TheiaContextMenu.open(this.app, () => this.page.waitForSelector(this.tabSelector));
168
+ }
169
+
170
+ protected viewElement(): Promise<ElementHandle<SVGElement | HTMLElement> | null> {
171
+ return this.page.$(this.viewSelector);
172
+ }
173
+
174
+ protected tabElement(): Promise<ElementHandle<SVGElement | HTMLElement> | null> {
175
+ return this.page.$(this.tabSelector);
176
+ }
177
+
178
+ }
@@ -1,76 +1,76 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2021 logi.cals GmbH, EclipseSource 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 WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
-
17
- import * as fs from 'fs-extra';
18
- import { resolve } from 'path';
19
- import { OSUtil, urlEncodePath } from './util';
20
-
21
- export class TheiaWorkspace {
22
-
23
- protected workspacePath: string;
24
-
25
- /**
26
- * Creates a Theia workspace location with the specified path to files that shall be copied to this workspace.
27
- * The `pathOfFilesToInitialize` must be relative to cwd of the node process.
28
- *
29
- * @param {string[]} pathOfFilesToInitialize Path to files or folders that shall be copied to the workspace
30
- */
31
- constructor(protected pathOfFilesToInitialize?: string[]) {
32
- this.workspacePath = fs.mkdtempSync(`${OSUtil.tmpDir}${OSUtil.fileSeparator}cloud-ws-`);
33
- }
34
-
35
- /** Performs the file system operations preparing the workspace location synchronously. */
36
- initialize(): void {
37
- if (this.pathOfFilesToInitialize) {
38
- for (const initPath of this.pathOfFilesToInitialize) {
39
- const absoluteInitPath = resolve(process.cwd(), initPath);
40
- if (!fs.pathExistsSync(absoluteInitPath)) {
41
- throw Error('Workspace does not exist at ' + absoluteInitPath);
42
- }
43
- fs.copySync(absoluteInitPath, this.workspacePath);
44
- }
45
- }
46
- }
47
-
48
- get path(): string {
49
- let workspacePath = this.workspacePath;
50
- if (!OSUtil.osStartsWithFileSeparator(this.workspacePath)) {
51
- workspacePath = `${OSUtil.fileSeparator}${workspacePath}`;
52
- }
53
- if (OSUtil.isWindows) {
54
- // Drive letters in windows paths have to be lower case
55
- workspacePath = workspacePath.replace(/.:/, matchedChar => matchedChar.toLowerCase());
56
- }
57
- return workspacePath;
58
- }
59
-
60
- get urlEncodedPath(): string {
61
- return urlEncodePath(this.path);
62
- }
63
-
64
- get escapedPath(): string {
65
- return this.path.replace(/:/g, '%3A');
66
- }
67
-
68
- clear(): void {
69
- fs.emptyDirSync(this.workspacePath);
70
- }
71
-
72
- remove(): void {
73
- fs.removeSync(this.workspacePath);
74
- }
75
-
76
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2021 logi.cals GmbH, EclipseSource 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 WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+
17
+ import * as fs from 'fs-extra';
18
+ import { resolve } from 'path';
19
+ import { OSUtil, urlEncodePath } from './util';
20
+
21
+ export class TheiaWorkspace {
22
+
23
+ protected workspacePath: string;
24
+
25
+ /**
26
+ * Creates a Theia workspace location with the specified path to files that shall be copied to this workspace.
27
+ * The `pathOfFilesToInitialize` must be relative to cwd of the node process.
28
+ *
29
+ * @param {string[]} pathOfFilesToInitialize Path to files or folders that shall be copied to the workspace
30
+ */
31
+ constructor(protected pathOfFilesToInitialize?: string[]) {
32
+ this.workspacePath = fs.mkdtempSync(`${OSUtil.tmpDir}${OSUtil.fileSeparator}cloud-ws-`);
33
+ }
34
+
35
+ /** Performs the file system operations preparing the workspace location synchronously. */
36
+ initialize(): void {
37
+ if (this.pathOfFilesToInitialize) {
38
+ for (const initPath of this.pathOfFilesToInitialize) {
39
+ const absoluteInitPath = resolve(process.cwd(), initPath);
40
+ if (!fs.pathExistsSync(absoluteInitPath)) {
41
+ throw Error('Workspace does not exist at ' + absoluteInitPath);
42
+ }
43
+ fs.copySync(absoluteInitPath, this.workspacePath);
44
+ }
45
+ }
46
+ }
47
+
48
+ get path(): string {
49
+ let workspacePath = this.workspacePath;
50
+ if (!OSUtil.osStartsWithFileSeparator(this.workspacePath)) {
51
+ workspacePath = `${OSUtil.fileSeparator}${workspacePath}`;
52
+ }
53
+ if (OSUtil.isWindows) {
54
+ // Drive letters in windows paths have to be lower case
55
+ workspacePath = workspacePath.replace(/.:/, matchedChar => matchedChar.toLowerCase());
56
+ }
57
+ return workspacePath;
58
+ }
59
+
60
+ get urlEncodedPath(): string {
61
+ return urlEncodePath(this.path);
62
+ }
63
+
64
+ get escapedPath(): string {
65
+ return this.path.replace(/:/g, '%3A');
66
+ }
67
+
68
+ clear(): void {
69
+ fs.emptyDirSync(this.workspacePath);
70
+ }
71
+
72
+ remove(): void {
73
+ fs.removeSync(this.workspacePath);
74
+ }
75
+
76
+ }