@theia/remote 1.45.0 → 1.46.0-next.72

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 (87) hide show
  1. package/README.md +61 -61
  2. package/lib/electron-browser/remote-electron-file-dialog-service.d.ts +13 -13
  3. package/lib/electron-browser/remote-electron-file-dialog-service.js +57 -57
  4. package/lib/electron-browser/remote-frontend-contribution.d.ts +25 -25
  5. package/lib/electron-browser/remote-frontend-contribution.js +164 -164
  6. package/lib/electron-browser/remote-frontend-module.d.ts +3 -3
  7. package/lib/electron-browser/remote-frontend-module.js +42 -42
  8. package/lib/electron-browser/remote-preferences.d.ts +10 -10
  9. package/lib/electron-browser/remote-preferences.js +48 -48
  10. package/lib/electron-browser/remote-registry-contribution.d.ts +18 -18
  11. package/lib/electron-browser/remote-registry-contribution.js +74 -74
  12. package/lib/electron-browser/remote-service.d.ts +5 -5
  13. package/lib/electron-browser/remote-service.js +37 -37
  14. package/lib/electron-browser/remote-ssh-contribution.d.ts +17 -17
  15. package/lib/electron-browser/remote-ssh-contribution.js +117 -117
  16. package/lib/electron-common/remote-ssh-connection-provider.d.ts +10 -10
  17. package/lib/electron-common/remote-ssh-connection-provider.js +20 -20
  18. package/lib/electron-common/remote-status-service.d.ts +14 -14
  19. package/lib/electron-common/remote-status-service.js +20 -20
  20. package/lib/electron-node/backend-remote-service-impl.d.ts +11 -11
  21. package/lib/electron-node/backend-remote-service-impl.js +50 -50
  22. package/lib/electron-node/remote-backend-module.d.ts +4 -4
  23. package/lib/electron-node/remote-backend-module.js +74 -74
  24. package/lib/electron-node/remote-connection-service.d.ts +14 -14
  25. package/lib/electron-node/remote-connection-service.js +65 -65
  26. package/lib/electron-node/remote-connection-socket-provider.d.ts +8 -8
  27. package/lib/electron-node/remote-connection-socket-provider.js +37 -37
  28. package/lib/electron-node/remote-proxy-server-provider.d.ts +5 -5
  29. package/lib/electron-node/remote-proxy-server-provider.js +43 -43
  30. package/lib/electron-node/remote-status-service.d.ts +6 -6
  31. package/lib/electron-node/remote-status-service.js +54 -54
  32. package/lib/electron-node/remote-types.d.ts +34 -34
  33. package/lib/electron-node/remote-types.js +17 -17
  34. package/lib/electron-node/setup/app-native-dependency-contribution.d.ts +7 -7
  35. package/lib/electron-node/setup/app-native-dependency-contribution.js +57 -57
  36. package/lib/electron-node/setup/main-copy-contribution.d.ts +4 -4
  37. package/lib/electron-node/setup/main-copy-contribution.js +37 -37
  38. package/lib/electron-node/setup/remote-copy-contribution.d.ts +28 -28
  39. package/lib/electron-node/setup/remote-copy-contribution.js +78 -78
  40. package/lib/electron-node/setup/remote-copy-service.d.ts +18 -18
  41. package/lib/electron-node/setup/remote-copy-service.js +126 -126
  42. package/lib/electron-node/setup/remote-native-dependency-contribution.d.ts +34 -34
  43. package/lib/electron-node/setup/remote-native-dependency-contribution.js +34 -34
  44. package/lib/electron-node/setup/remote-native-dependency-service.d.ts +23 -23
  45. package/lib/electron-node/setup/remote-native-dependency-service.js +118 -118
  46. package/lib/electron-node/setup/remote-node-setup-service.d.ts +22 -22
  47. package/lib/electron-node/setup/remote-node-setup-service.js +132 -132
  48. package/lib/electron-node/setup/remote-setup-script-service.d.ts +40 -40
  49. package/lib/electron-node/setup/remote-setup-script-service.js +132 -132
  50. package/lib/electron-node/setup/remote-setup-service.d.ts +28 -28
  51. package/lib/electron-node/setup/remote-setup-service.js +198 -198
  52. package/lib/electron-node/ssh/remote-ssh-connection-provider.d.ts +55 -55
  53. package/lib/electron-node/ssh/remote-ssh-connection-provider.d.ts.map +1 -1
  54. package/lib/electron-node/ssh/remote-ssh-connection-provider.js +344 -342
  55. package/lib/electron-node/ssh/remote-ssh-connection-provider.js.map +1 -1
  56. package/lib/electron-node/ssh/ssh-identity-file-collector.d.ts +12 -12
  57. package/lib/electron-node/ssh/ssh-identity-file-collector.js +131 -131
  58. package/lib/package.spec.js +25 -25
  59. package/package.json +6 -7
  60. package/src/electron-browser/remote-electron-file-dialog-service.ts +47 -47
  61. package/src/electron-browser/remote-frontend-contribution.ts +145 -145
  62. package/src/electron-browser/remote-frontend-module.ts +49 -49
  63. package/src/electron-browser/remote-preferences.ts +62 -62
  64. package/src/electron-browser/remote-registry-contribution.ts +70 -70
  65. package/src/electron-browser/remote-service.ts +31 -31
  66. package/src/electron-browser/remote-ssh-contribution.ts +102 -102
  67. package/src/electron-common/remote-ssh-connection-provider.ts +29 -29
  68. package/src/electron-common/remote-status-service.ts +35 -35
  69. package/src/electron-node/backend-remote-service-impl.ts +45 -45
  70. package/src/electron-node/remote-backend-module.ts +80 -80
  71. package/src/electron-node/remote-connection-service.ts +55 -55
  72. package/src/electron-node/remote-connection-socket-provider.ts +34 -34
  73. package/src/electron-node/remote-proxy-server-provider.ts +37 -37
  74. package/src/electron-node/remote-status-service.ts +41 -41
  75. package/src/electron-node/remote-types.ts +56 -56
  76. package/src/electron-node/setup/app-native-dependency-contribution.ts +48 -48
  77. package/src/electron-node/setup/main-copy-contribution.ts +28 -28
  78. package/src/electron-node/setup/remote-copy-contribution.ts +90 -90
  79. package/src/electron-node/setup/remote-copy-service.ts +114 -114
  80. package/src/electron-node/setup/remote-native-dependency-contribution.ts +63 -63
  81. package/src/electron-node/setup/remote-native-dependency-service.ts +111 -111
  82. package/src/electron-node/setup/remote-node-setup-service.ts +123 -123
  83. package/src/electron-node/setup/remote-setup-script-service.ts +146 -146
  84. package/src/electron-node/setup/remote-setup-service.ts +197 -197
  85. package/src/electron-node/ssh/remote-ssh-connection-provider.ts +358 -356
  86. package/src/electron-node/ssh/ssh-identity-file-collector.ts +137 -137
  87. package/src/package.spec.ts +29 -29
@@ -1,197 +1,197 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2023 TypeFox 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 { inject, injectable } from '@theia/core/shared/inversify';
18
- import { RemoteConnection, RemoteExecResult, RemotePlatform, RemoteStatusReport } from '../remote-types';
19
- import { ApplicationPackage } from '@theia/core/shared/@theia/application-package';
20
- import { RemoteCopyService } from './remote-copy-service';
21
- import { RemoteNativeDependencyService } from './remote-native-dependency-service';
22
- import { OS, THEIA_VERSION } from '@theia/core';
23
- import { RemoteNodeSetupService } from './remote-node-setup-service';
24
- import { RemoteSetupScriptService } from './remote-setup-script-service';
25
-
26
- export interface RemoteSetupOptions {
27
- connection: RemoteConnection;
28
- report: RemoteStatusReport;
29
- nodeDownloadTemplate?: string;
30
- }
31
-
32
- @injectable()
33
- export class RemoteSetupService {
34
-
35
- @inject(RemoteCopyService)
36
- protected readonly copyService: RemoteCopyService;
37
-
38
- @inject(RemoteNativeDependencyService)
39
- protected readonly nativeDependencyService: RemoteNativeDependencyService;
40
-
41
- @inject(RemoteNodeSetupService)
42
- protected readonly nodeSetupService: RemoteNodeSetupService;
43
-
44
- @inject(RemoteSetupScriptService)
45
- protected readonly scriptService: RemoteSetupScriptService;
46
-
47
- @inject(ApplicationPackage)
48
- protected readonly applicationPackage: ApplicationPackage;
49
-
50
- async setup(options: RemoteSetupOptions): Promise<void> {
51
- const {
52
- connection,
53
- report,
54
- nodeDownloadTemplate
55
- } = options;
56
- report('Identifying remote system...');
57
- // 1. Identify remote platform
58
- const platform = await this.detectRemotePlatform(connection);
59
- // 2. Setup home directory
60
- const remoteHome = await this.getRemoteHomeDirectory(connection, platform);
61
- const applicationDirectory = this.scriptService.joinPath(platform, remoteHome, `.${this.getRemoteAppName()}`);
62
- await this.mkdirRemote(connection, platform, applicationDirectory);
63
- // 3. Download+copy node for that platform
64
- const nodeFileName = this.nodeSetupService.getNodeFileName(platform);
65
- const nodeDirName = this.nodeSetupService.getNodeDirectoryName(platform);
66
- const remoteNodeDirectory = this.scriptService.joinPath(platform, applicationDirectory, nodeDirName);
67
- const nodeDirExists = await this.dirExistsRemote(connection, remoteNodeDirectory);
68
- if (!nodeDirExists) {
69
- report('Downloading and installing Node.js on remote...');
70
- // Download the binaries locally and move it via SSH
71
- const nodeArchive = await this.nodeSetupService.downloadNode(platform, nodeDownloadTemplate);
72
- const remoteNodeZip = this.scriptService.joinPath(platform, applicationDirectory, nodeFileName);
73
- await connection.copy(nodeArchive, remoteNodeZip);
74
- await this.unzipRemote(connection, platform, remoteNodeZip, applicationDirectory);
75
- }
76
- // 4. Copy backend to remote system
77
- const libDir = this.scriptService.joinPath(platform, applicationDirectory, 'lib');
78
- const libDirExists = await this.dirExistsRemote(connection, libDir);
79
- if (!libDirExists) {
80
- report('Installing application on remote...');
81
- const applicationZipFile = this.scriptService.joinPath(platform, applicationDirectory, `${this.getRemoteAppName()}.tar`);
82
- await this.copyService.copyToRemote(connection, platform, applicationZipFile);
83
- await this.unzipRemote(connection, platform, applicationZipFile, applicationDirectory);
84
- }
85
- // 5. start remote backend
86
- report('Starting application on remote...');
87
- const port = await this.startApplication(connection, platform, applicationDirectory, remoteNodeDirectory);
88
- connection.remotePort = port;
89
- }
90
-
91
- protected async startApplication(connection: RemoteConnection, platform: RemotePlatform, remotePath: string, nodeDir: string): Promise<number> {
92
- const nodeExecutable = this.scriptService.joinPath(platform, nodeDir, ...(platform.os === OS.Type.Windows ? ['node.exe'] : ['bin', 'node']));
93
- const mainJsFile = this.scriptService.joinPath(platform, remotePath, 'lib', 'backend', 'main.js');
94
- const localAddressRegex = /listening on http:\/\/127.0.0.1:(\d+)/;
95
- let prefix = '';
96
- if (platform.os === OS.Type.Windows) {
97
- // We might to switch to PowerShell beforehand on Windows
98
- prefix = this.scriptService.exec(platform) + ' ';
99
- }
100
- // Change to the remote application path and start a node process with the copied main.js file
101
- // This way, our current working directory is set as expected
102
- const result = await connection.execPartial(`${prefix}cd "${remotePath}";${nodeExecutable}`,
103
- stdout => localAddressRegex.test(stdout),
104
- [mainJsFile, '--hostname=127.0.0.1', '--port=0', '--remote']);
105
-
106
- const match = localAddressRegex.exec(result.stdout);
107
- if (!match) {
108
- throw new Error('Could not start remote system: ' + result.stdout);
109
- } else {
110
- return Number(match[1]);
111
- }
112
- }
113
-
114
- protected async detectRemotePlatform(connection: RemoteConnection): Promise<RemotePlatform> {
115
- const osResult = await connection.exec('uname -s');
116
-
117
- let os: OS.Type | undefined;
118
- if (osResult.stderr) {
119
- // Only Windows systems return an error stream here
120
- os = OS.Type.Windows;
121
- } else if (osResult.stdout) {
122
- if (osResult.stdout.includes('windows32') || osResult.stdout.includes('MINGW64')) {
123
- os = OS.Type.Windows;
124
- } else if (osResult.stdout.includes('Linux')) {
125
- os = OS.Type.Linux;
126
- } else if (osResult.stdout.includes('Darwin')) {
127
- os = OS.Type.OSX;
128
- }
129
- }
130
- if (!os) {
131
- throw new Error('Failed to identify remote system: ' + osResult.stdout + '\n' + osResult.stderr);
132
- }
133
- let arch: string | undefined;
134
- if (os === OS.Type.Windows) {
135
- const wmicResult = await connection.exec('wmic OS get OSArchitecture');
136
- if (wmicResult.stdout.includes('64-bit')) {
137
- arch = 'x64';
138
- } else if (wmicResult.stdout.includes('32-bit')) {
139
- arch = 'x86';
140
- }
141
- } else {
142
- const archResult = (await connection.exec('uname -m')).stdout;
143
- if (archResult.includes('x86_64')) {
144
- arch = 'x64';
145
- } else if (archResult.match(/i\d83/)) { // i386, i483, i683
146
- arch = 'x86';
147
- } else {
148
- arch = archResult.trim();
149
- }
150
- }
151
- if (!arch) {
152
- throw new Error('Could not identify remote system architecture');
153
- }
154
- return {
155
- os,
156
- arch
157
- };
158
- }
159
-
160
- protected async getRemoteHomeDirectory(connection: RemoteConnection, platform: RemotePlatform): Promise<string> {
161
- const result = await connection.exec(this.scriptService.home(platform));
162
- return result.stdout.trim();
163
- }
164
-
165
- protected getRemoteAppName(): string {
166
- const appName = this.applicationPackage.pck.name || 'theia';
167
- const appVersion = this.applicationPackage.pck.version || THEIA_VERSION;
168
- return `${this.cleanupDirectoryName(`${appName}-${appVersion}`)}-remote`;
169
- }
170
-
171
- protected cleanupDirectoryName(name: string): string {
172
- return name.replace(/[@<>:"\\|?*]/g, '').replace(/\//g, '-');
173
- }
174
-
175
- protected async mkdirRemote(connection: RemoteConnection, platform: RemotePlatform, remotePath: string): Promise<void> {
176
- const result = await connection.exec(this.scriptService.mkdir(platform, remotePath));
177
- if (result.stderr) {
178
- throw new Error('Failed to create directory: ' + result.stderr);
179
- }
180
- }
181
-
182
- protected async dirExistsRemote(connection: RemoteConnection, remotePath: string): Promise<boolean> {
183
- const cdResult = await connection.exec(`cd "${remotePath}"`);
184
- return !Boolean(cdResult.stderr);
185
- }
186
-
187
- protected async unzipRemote(connection: RemoteConnection, platform: RemotePlatform, remoteFile: string, remoteDirectory: string): Promise<void> {
188
- const result = await connection.exec(this.scriptService.unzip(platform, remoteFile, remoteDirectory));
189
- if (result.stderr) {
190
- throw new Error('Failed to unzip: ' + result.stderr);
191
- }
192
- }
193
-
194
- protected async executeScriptRemote(connection: RemoteConnection, platform: RemotePlatform, script: string): Promise<RemoteExecResult> {
195
- return connection.exec(this.scriptService.exec(platform), [script]);
196
- }
197
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2023 TypeFox 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 { inject, injectable } from '@theia/core/shared/inversify';
18
+ import { RemoteConnection, RemoteExecResult, RemotePlatform, RemoteStatusReport } from '../remote-types';
19
+ import { ApplicationPackage } from '@theia/core/shared/@theia/application-package';
20
+ import { RemoteCopyService } from './remote-copy-service';
21
+ import { RemoteNativeDependencyService } from './remote-native-dependency-service';
22
+ import { OS, THEIA_VERSION } from '@theia/core';
23
+ import { RemoteNodeSetupService } from './remote-node-setup-service';
24
+ import { RemoteSetupScriptService } from './remote-setup-script-service';
25
+
26
+ export interface RemoteSetupOptions {
27
+ connection: RemoteConnection;
28
+ report: RemoteStatusReport;
29
+ nodeDownloadTemplate?: string;
30
+ }
31
+
32
+ @injectable()
33
+ export class RemoteSetupService {
34
+
35
+ @inject(RemoteCopyService)
36
+ protected readonly copyService: RemoteCopyService;
37
+
38
+ @inject(RemoteNativeDependencyService)
39
+ protected readonly nativeDependencyService: RemoteNativeDependencyService;
40
+
41
+ @inject(RemoteNodeSetupService)
42
+ protected readonly nodeSetupService: RemoteNodeSetupService;
43
+
44
+ @inject(RemoteSetupScriptService)
45
+ protected readonly scriptService: RemoteSetupScriptService;
46
+
47
+ @inject(ApplicationPackage)
48
+ protected readonly applicationPackage: ApplicationPackage;
49
+
50
+ async setup(options: RemoteSetupOptions): Promise<void> {
51
+ const {
52
+ connection,
53
+ report,
54
+ nodeDownloadTemplate
55
+ } = options;
56
+ report('Identifying remote system...');
57
+ // 1. Identify remote platform
58
+ const platform = await this.detectRemotePlatform(connection);
59
+ // 2. Setup home directory
60
+ const remoteHome = await this.getRemoteHomeDirectory(connection, platform);
61
+ const applicationDirectory = this.scriptService.joinPath(platform, remoteHome, `.${this.getRemoteAppName()}`);
62
+ await this.mkdirRemote(connection, platform, applicationDirectory);
63
+ // 3. Download+copy node for that platform
64
+ const nodeFileName = this.nodeSetupService.getNodeFileName(platform);
65
+ const nodeDirName = this.nodeSetupService.getNodeDirectoryName(platform);
66
+ const remoteNodeDirectory = this.scriptService.joinPath(platform, applicationDirectory, nodeDirName);
67
+ const nodeDirExists = await this.dirExistsRemote(connection, remoteNodeDirectory);
68
+ if (!nodeDirExists) {
69
+ report('Downloading and installing Node.js on remote...');
70
+ // Download the binaries locally and move it via SSH
71
+ const nodeArchive = await this.nodeSetupService.downloadNode(platform, nodeDownloadTemplate);
72
+ const remoteNodeZip = this.scriptService.joinPath(platform, applicationDirectory, nodeFileName);
73
+ await connection.copy(nodeArchive, remoteNodeZip);
74
+ await this.unzipRemote(connection, platform, remoteNodeZip, applicationDirectory);
75
+ }
76
+ // 4. Copy backend to remote system
77
+ const libDir = this.scriptService.joinPath(platform, applicationDirectory, 'lib');
78
+ const libDirExists = await this.dirExistsRemote(connection, libDir);
79
+ if (!libDirExists) {
80
+ report('Installing application on remote...');
81
+ const applicationZipFile = this.scriptService.joinPath(platform, applicationDirectory, `${this.getRemoteAppName()}.tar`);
82
+ await this.copyService.copyToRemote(connection, platform, applicationZipFile);
83
+ await this.unzipRemote(connection, platform, applicationZipFile, applicationDirectory);
84
+ }
85
+ // 5. start remote backend
86
+ report('Starting application on remote...');
87
+ const port = await this.startApplication(connection, platform, applicationDirectory, remoteNodeDirectory);
88
+ connection.remotePort = port;
89
+ }
90
+
91
+ protected async startApplication(connection: RemoteConnection, platform: RemotePlatform, remotePath: string, nodeDir: string): Promise<number> {
92
+ const nodeExecutable = this.scriptService.joinPath(platform, nodeDir, ...(platform.os === OS.Type.Windows ? ['node.exe'] : ['bin', 'node']));
93
+ const mainJsFile = this.scriptService.joinPath(platform, remotePath, 'lib', 'backend', 'main.js');
94
+ const localAddressRegex = /listening on http:\/\/127.0.0.1:(\d+)/;
95
+ let prefix = '';
96
+ if (platform.os === OS.Type.Windows) {
97
+ // We might to switch to PowerShell beforehand on Windows
98
+ prefix = this.scriptService.exec(platform) + ' ';
99
+ }
100
+ // Change to the remote application path and start a node process with the copied main.js file
101
+ // This way, our current working directory is set as expected
102
+ const result = await connection.execPartial(`${prefix}cd "${remotePath}";${nodeExecutable}`,
103
+ stdout => localAddressRegex.test(stdout),
104
+ [mainJsFile, '--hostname=127.0.0.1', '--port=0', '--remote']);
105
+
106
+ const match = localAddressRegex.exec(result.stdout);
107
+ if (!match) {
108
+ throw new Error('Could not start remote system: ' + result.stdout);
109
+ } else {
110
+ return Number(match[1]);
111
+ }
112
+ }
113
+
114
+ protected async detectRemotePlatform(connection: RemoteConnection): Promise<RemotePlatform> {
115
+ const osResult = await connection.exec('uname -s');
116
+
117
+ let os: OS.Type | undefined;
118
+ if (osResult.stderr) {
119
+ // Only Windows systems return an error stream here
120
+ os = OS.Type.Windows;
121
+ } else if (osResult.stdout) {
122
+ if (osResult.stdout.includes('windows32') || osResult.stdout.includes('MINGW64')) {
123
+ os = OS.Type.Windows;
124
+ } else if (osResult.stdout.includes('Linux')) {
125
+ os = OS.Type.Linux;
126
+ } else if (osResult.stdout.includes('Darwin')) {
127
+ os = OS.Type.OSX;
128
+ }
129
+ }
130
+ if (!os) {
131
+ throw new Error('Failed to identify remote system: ' + osResult.stdout + '\n' + osResult.stderr);
132
+ }
133
+ let arch: string | undefined;
134
+ if (os === OS.Type.Windows) {
135
+ const wmicResult = await connection.exec('wmic OS get OSArchitecture');
136
+ if (wmicResult.stdout.includes('64-bit')) {
137
+ arch = 'x64';
138
+ } else if (wmicResult.stdout.includes('32-bit')) {
139
+ arch = 'x86';
140
+ }
141
+ } else {
142
+ const archResult = (await connection.exec('uname -m')).stdout;
143
+ if (archResult.includes('x86_64')) {
144
+ arch = 'x64';
145
+ } else if (archResult.match(/i\d83/)) { // i386, i483, i683
146
+ arch = 'x86';
147
+ } else {
148
+ arch = archResult.trim();
149
+ }
150
+ }
151
+ if (!arch) {
152
+ throw new Error('Could not identify remote system architecture');
153
+ }
154
+ return {
155
+ os,
156
+ arch
157
+ };
158
+ }
159
+
160
+ protected async getRemoteHomeDirectory(connection: RemoteConnection, platform: RemotePlatform): Promise<string> {
161
+ const result = await connection.exec(this.scriptService.home(platform));
162
+ return result.stdout.trim();
163
+ }
164
+
165
+ protected getRemoteAppName(): string {
166
+ const appName = this.applicationPackage.pck.name || 'theia';
167
+ const appVersion = this.applicationPackage.pck.version || THEIA_VERSION;
168
+ return `${this.cleanupDirectoryName(`${appName}-${appVersion}`)}-remote`;
169
+ }
170
+
171
+ protected cleanupDirectoryName(name: string): string {
172
+ return name.replace(/[@<>:"\\|?*]/g, '').replace(/\//g, '-');
173
+ }
174
+
175
+ protected async mkdirRemote(connection: RemoteConnection, platform: RemotePlatform, remotePath: string): Promise<void> {
176
+ const result = await connection.exec(this.scriptService.mkdir(platform, remotePath));
177
+ if (result.stderr) {
178
+ throw new Error('Failed to create directory: ' + result.stderr);
179
+ }
180
+ }
181
+
182
+ protected async dirExistsRemote(connection: RemoteConnection, remotePath: string): Promise<boolean> {
183
+ const cdResult = await connection.exec(`cd "${remotePath}"`);
184
+ return !Boolean(cdResult.stderr);
185
+ }
186
+
187
+ protected async unzipRemote(connection: RemoteConnection, platform: RemotePlatform, remoteFile: string, remoteDirectory: string): Promise<void> {
188
+ const result = await connection.exec(this.scriptService.unzip(platform, remoteFile, remoteDirectory));
189
+ if (result.stderr) {
190
+ throw new Error('Failed to unzip: ' + result.stderr);
191
+ }
192
+ }
193
+
194
+ protected async executeScriptRemote(connection: RemoteConnection, platform: RemotePlatform, script: string): Promise<RemoteExecResult> {
195
+ return connection.exec(this.scriptService.exec(platform), [script]);
196
+ }
197
+ }