@theia/process 1.47.1 → 1.48.1
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.
- package/README.md +30 -30
- package/lib/common/process-common-module.d.ts +3 -3
- package/lib/common/process-common-module.js +22 -22
- package/lib/common/process-manager-types.d.ts +35 -35
- package/lib/common/process-manager-types.js +23 -23
- package/lib/common/shell-command-builder.d.ts +49 -49
- package/lib/common/shell-command-builder.js +169 -169
- package/lib/common/shell-command-builder.slow-spec.d.ts +9 -9
- package/lib/common/shell-command-builder.slow-spec.js +404 -404
- package/lib/common/shell-quoting.d.ts +91 -91
- package/lib/common/shell-quoting.js +145 -145
- package/lib/common/shell-quoting.spec.d.ts +1 -1
- package/lib/common/shell-quoting.spec.js +170 -170
- package/lib/node/dev-null-stream.d.ts +17 -17
- package/lib/node/dev-null-stream.js +41 -41
- package/lib/node/index.d.ts +6 -6
- package/lib/node/index.js +24 -24
- package/lib/node/multi-ring-buffer.d.ts +68 -68
- package/lib/node/multi-ring-buffer.js +299 -299
- package/lib/node/multi-ring-buffer.spec.d.ts +1 -1
- package/lib/node/multi-ring-buffer.spec.js +422 -422
- package/lib/node/process-backend-module.d.ts +3 -3
- package/lib/node/process-backend-module.js +56 -56
- package/lib/node/process-manager.d.ts +33 -33
- package/lib/node/process-manager.js +102 -102
- package/lib/node/process.d.ts +95 -95
- package/lib/node/process.js +142 -142
- package/lib/node/pseudo-pty.d.ts +22 -22
- package/lib/node/pseudo-pty.js +38 -38
- package/lib/node/raw-process.d.ts +45 -45
- package/lib/node/raw-process.js +104 -104
- package/lib/node/raw-process.spec.d.ts +1 -1
- package/lib/node/raw-process.spec.js +164 -164
- package/lib/node/task-terminal-process.d.ts +10 -10
- package/lib/node/task-terminal-process.js +42 -42
- package/lib/node/terminal-process.d.ts +60 -60
- package/lib/node/terminal-process.js +248 -248
- package/lib/node/terminal-process.spec.d.ts +1 -1
- package/lib/node/terminal-process.spec.js +103 -103
- package/lib/node/test/process-test-container.d.ts +2 -2
- package/lib/node/test/process-test-container.js +28 -28
- package/lib/node/utils.d.ts +16 -16
- package/lib/node/utils.js +77 -77
- package/package.json +4 -4
- package/src/common/process-common-module.ts +22 -22
- package/src/common/process-manager-types.ts +58 -58
- package/src/common/shell-command-builder.slow-spec.ts +486 -486
- package/src/common/shell-command-builder.ts +187 -187
- package/src/common/shell-quoting.spec.ts +176 -176
- package/src/common/shell-quoting.ts +236 -236
- package/src/common/tests/$weird(),file=name.js +1 -1
- package/src/common/tests/white space.js +1 -1
- package/src/node/dev-null-stream.ts +47 -47
- package/src/node/index.ts +22 -22
- package/src/node/multi-ring-buffer.spec.ts +486 -486
- package/src/node/multi-ring-buffer.ts +348 -348
- package/src/node/process-backend-module.ts +67 -67
- package/src/node/process-manager.ts +107 -107
- package/src/node/process.ts +207 -207
- package/src/node/pseudo-pty.ts +54 -54
- package/src/node/raw-process.spec.ts +199 -199
- package/src/node/raw-process.ts +156 -156
- package/src/node/string-argv.d.ts +21 -21
- package/src/node/task-terminal-process.ts +41 -41
- package/src/node/terminal-process.spec.ts +121 -121
- package/src/node/terminal-process.ts +290 -290
- package/src/node/test/process-fork-test.js +22 -22
- package/src/node/test/process-test-container.ts +27 -27
- package/src/node/utils.ts +79 -79
package/lib/node/raw-process.js
CHANGED
|
@@ -1,105 +1,105 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
// *****************************************************************************
|
|
3
|
-
// Copyright (C) 2017 Ericsson and others.
|
|
4
|
-
//
|
|
5
|
-
// This program and the accompanying materials are made available under the
|
|
6
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
7
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
8
|
-
//
|
|
9
|
-
// This Source Code may also be made available under the following Secondary
|
|
10
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
11
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
12
|
-
// with the GNU Classpath Exception which is available at
|
|
13
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
14
|
-
//
|
|
15
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
16
|
-
// *****************************************************************************
|
|
17
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
18
|
-
exports.RawProcess = exports.RawProcessFactory = exports.RawProcessOptions = exports.DevNullStream = void 0;
|
|
19
|
-
const tslib_1 = require("tslib");
|
|
20
|
-
const inversify_1 = require("@theia/core/shared/inversify");
|
|
21
|
-
const process_manager_1 = require("./process-manager");
|
|
22
|
-
const common_1 = require("@theia/core/lib/common");
|
|
23
|
-
const process_1 = require("./process");
|
|
24
|
-
const child_process_1 = require("child_process");
|
|
25
|
-
// The class was here before, exporting to not break anything.
|
|
26
|
-
var dev_null_stream_1 = require("./dev-null-stream");
|
|
27
|
-
Object.defineProperty(exports, "DevNullStream", { enumerable: true, get: function () { return dev_null_stream_1.DevNullStream; } });
|
|
28
|
-
const dev_null_stream_2 = require("./dev-null-stream");
|
|
29
|
-
exports.RawProcessOptions = Symbol('RawProcessOptions');
|
|
30
|
-
exports.RawProcessFactory = Symbol('RawProcessFactory');
|
|
31
|
-
let RawProcess = class RawProcess extends process_1.Process {
|
|
32
|
-
constructor(// eslint-disable-next-line @typescript-eslint/indent
|
|
33
|
-
options, processManager, logger) {
|
|
34
|
-
super(processManager, logger, process_1.ProcessType.Raw, options);
|
|
35
|
-
const executable = this.isForkOptions(options) ? options.modulePath : options.command;
|
|
36
|
-
this.logger.debug(`Starting raw process: ${executable},`
|
|
37
|
-
+ ` with args: ${options.args ? options.args.join(' ') : ''}, `
|
|
38
|
-
+ ` with options: ${JSON.stringify(options.options)}`);
|
|
39
|
-
// About catching errors: spawn will sometimes throw directly
|
|
40
|
-
// (EACCES on Linux), sometimes return a Process object with the pid
|
|
41
|
-
// property undefined (ENOENT on Linux) and then emit an 'error' event.
|
|
42
|
-
// For now, we try to normalize that into always emitting an 'error'
|
|
43
|
-
// event.
|
|
44
|
-
try {
|
|
45
|
-
if (this.isForkOptions(options)) {
|
|
46
|
-
this.process = (0, child_process_1.fork)(options.modulePath, options.args || [], options.options || {});
|
|
47
|
-
}
|
|
48
|
-
else {
|
|
49
|
-
this.process = (0, child_process_1.spawn)(options.command, options.args || [], options.options || {});
|
|
50
|
-
}
|
|
51
|
-
this.process.on('error', (error) => {
|
|
52
|
-
error.code = error.code || 'Unknown error';
|
|
53
|
-
this.emitOnError(error);
|
|
54
|
-
});
|
|
55
|
-
// When no stdio option is passed, it is null by default.
|
|
56
|
-
this.outputStream = this.process.stdout || new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
57
|
-
this.inputStream = this.process.stdin || new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
58
|
-
this.errorStream = this.process.stderr || new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
59
|
-
this.process.on('exit', (exitCode, signal) => {
|
|
60
|
-
// node's child_process exit sets the unused parameter to null,
|
|
61
|
-
// but we want it to be undefined instead.
|
|
62
|
-
this.emitOnExit(typeof exitCode === 'number' ? exitCode : undefined, typeof signal === 'string' ? signal : undefined);
|
|
63
|
-
this.processManager.unregister(this);
|
|
64
|
-
});
|
|
65
|
-
this.process.on('close', (exitCode, signal) => {
|
|
66
|
-
// node's child_process exit sets the unused parameter to null,
|
|
67
|
-
// but we want it to be undefined instead.
|
|
68
|
-
this.emitOnClose(typeof exitCode === 'number' ? exitCode : undefined, typeof signal === 'string' ? signal : undefined);
|
|
69
|
-
});
|
|
70
|
-
if (this.process.pid !== undefined) {
|
|
71
|
-
process.nextTick(this.emitOnStarted.bind(this));
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
catch (error) {
|
|
75
|
-
/* When an error is thrown, set up some fake streams, so the client
|
|
76
|
-
code doesn't break because these field are undefined. */
|
|
77
|
-
this.outputStream = new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
78
|
-
this.inputStream = new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
79
|
-
this.errorStream = new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
80
|
-
/* Call the client error handler, but first give them a chance to register it. */
|
|
81
|
-
this.emitOnErrorAsync(error);
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
get pid() {
|
|
85
|
-
if (!this.process || !this.process.pid) {
|
|
86
|
-
throw new Error('process did not start correctly');
|
|
87
|
-
}
|
|
88
|
-
return this.process.pid;
|
|
89
|
-
}
|
|
90
|
-
kill(signal) {
|
|
91
|
-
if (this.process && this.killed === false) {
|
|
92
|
-
this.process.kill(signal);
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
};
|
|
96
|
-
RawProcess = (0, tslib_1.__decorate)([
|
|
97
|
-
(0, inversify_1.injectable)(),
|
|
98
|
-
(0, tslib_1.__param)(0, (0, inversify_1.inject)(exports.RawProcessOptions)),
|
|
99
|
-
(0, tslib_1.__param)(1, (0, inversify_1.inject)(process_manager_1.ProcessManager)),
|
|
100
|
-
(0, tslib_1.__param)(2, (0, inversify_1.inject)(common_1.ILogger)),
|
|
101
|
-
(0, tslib_1.__param)(2, (0, inversify_1.named)('process')),
|
|
102
|
-
(0, tslib_1.__metadata)("design:paramtypes", [Object, process_manager_1.ProcessManager, Object])
|
|
103
|
-
], RawProcess);
|
|
104
|
-
exports.RawProcess = RawProcess;
|
|
1
|
+
"use strict";
|
|
2
|
+
// *****************************************************************************
|
|
3
|
+
// Copyright (C) 2017 Ericsson and others.
|
|
4
|
+
//
|
|
5
|
+
// This program and the accompanying materials are made available under the
|
|
6
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
7
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
8
|
+
//
|
|
9
|
+
// This Source Code may also be made available under the following Secondary
|
|
10
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
11
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
12
|
+
// with the GNU Classpath Exception which is available at
|
|
13
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
14
|
+
//
|
|
15
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
16
|
+
// *****************************************************************************
|
|
17
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
18
|
+
exports.RawProcess = exports.RawProcessFactory = exports.RawProcessOptions = exports.DevNullStream = void 0;
|
|
19
|
+
const tslib_1 = require("tslib");
|
|
20
|
+
const inversify_1 = require("@theia/core/shared/inversify");
|
|
21
|
+
const process_manager_1 = require("./process-manager");
|
|
22
|
+
const common_1 = require("@theia/core/lib/common");
|
|
23
|
+
const process_1 = require("./process");
|
|
24
|
+
const child_process_1 = require("child_process");
|
|
25
|
+
// The class was here before, exporting to not break anything.
|
|
26
|
+
var dev_null_stream_1 = require("./dev-null-stream");
|
|
27
|
+
Object.defineProperty(exports, "DevNullStream", { enumerable: true, get: function () { return dev_null_stream_1.DevNullStream; } });
|
|
28
|
+
const dev_null_stream_2 = require("./dev-null-stream");
|
|
29
|
+
exports.RawProcessOptions = Symbol('RawProcessOptions');
|
|
30
|
+
exports.RawProcessFactory = Symbol('RawProcessFactory');
|
|
31
|
+
let RawProcess = class RawProcess extends process_1.Process {
|
|
32
|
+
constructor(// eslint-disable-next-line @typescript-eslint/indent
|
|
33
|
+
options, processManager, logger) {
|
|
34
|
+
super(processManager, logger, process_1.ProcessType.Raw, options);
|
|
35
|
+
const executable = this.isForkOptions(options) ? options.modulePath : options.command;
|
|
36
|
+
this.logger.debug(`Starting raw process: ${executable},`
|
|
37
|
+
+ ` with args: ${options.args ? options.args.join(' ') : ''}, `
|
|
38
|
+
+ ` with options: ${JSON.stringify(options.options)}`);
|
|
39
|
+
// About catching errors: spawn will sometimes throw directly
|
|
40
|
+
// (EACCES on Linux), sometimes return a Process object with the pid
|
|
41
|
+
// property undefined (ENOENT on Linux) and then emit an 'error' event.
|
|
42
|
+
// For now, we try to normalize that into always emitting an 'error'
|
|
43
|
+
// event.
|
|
44
|
+
try {
|
|
45
|
+
if (this.isForkOptions(options)) {
|
|
46
|
+
this.process = (0, child_process_1.fork)(options.modulePath, options.args || [], options.options || {});
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
this.process = (0, child_process_1.spawn)(options.command, options.args || [], options.options || {});
|
|
50
|
+
}
|
|
51
|
+
this.process.on('error', (error) => {
|
|
52
|
+
error.code = error.code || 'Unknown error';
|
|
53
|
+
this.emitOnError(error);
|
|
54
|
+
});
|
|
55
|
+
// When no stdio option is passed, it is null by default.
|
|
56
|
+
this.outputStream = this.process.stdout || new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
57
|
+
this.inputStream = this.process.stdin || new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
58
|
+
this.errorStream = this.process.stderr || new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
59
|
+
this.process.on('exit', (exitCode, signal) => {
|
|
60
|
+
// node's child_process exit sets the unused parameter to null,
|
|
61
|
+
// but we want it to be undefined instead.
|
|
62
|
+
this.emitOnExit(typeof exitCode === 'number' ? exitCode : undefined, typeof signal === 'string' ? signal : undefined);
|
|
63
|
+
this.processManager.unregister(this);
|
|
64
|
+
});
|
|
65
|
+
this.process.on('close', (exitCode, signal) => {
|
|
66
|
+
// node's child_process exit sets the unused parameter to null,
|
|
67
|
+
// but we want it to be undefined instead.
|
|
68
|
+
this.emitOnClose(typeof exitCode === 'number' ? exitCode : undefined, typeof signal === 'string' ? signal : undefined);
|
|
69
|
+
});
|
|
70
|
+
if (this.process.pid !== undefined) {
|
|
71
|
+
process.nextTick(this.emitOnStarted.bind(this));
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
catch (error) {
|
|
75
|
+
/* When an error is thrown, set up some fake streams, so the client
|
|
76
|
+
code doesn't break because these field are undefined. */
|
|
77
|
+
this.outputStream = new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
78
|
+
this.inputStream = new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
79
|
+
this.errorStream = new dev_null_stream_2.DevNullStream({ autoDestroy: true });
|
|
80
|
+
/* Call the client error handler, but first give them a chance to register it. */
|
|
81
|
+
this.emitOnErrorAsync(error);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
get pid() {
|
|
85
|
+
if (!this.process || !this.process.pid) {
|
|
86
|
+
throw new Error('process did not start correctly');
|
|
87
|
+
}
|
|
88
|
+
return this.process.pid;
|
|
89
|
+
}
|
|
90
|
+
kill(signal) {
|
|
91
|
+
if (this.process && this.killed === false) {
|
|
92
|
+
this.process.kill(signal);
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
};
|
|
96
|
+
RawProcess = (0, tslib_1.__decorate)([
|
|
97
|
+
(0, inversify_1.injectable)(),
|
|
98
|
+
(0, tslib_1.__param)(0, (0, inversify_1.inject)(exports.RawProcessOptions)),
|
|
99
|
+
(0, tslib_1.__param)(1, (0, inversify_1.inject)(process_manager_1.ProcessManager)),
|
|
100
|
+
(0, tslib_1.__param)(2, (0, inversify_1.inject)(common_1.ILogger)),
|
|
101
|
+
(0, tslib_1.__param)(2, (0, inversify_1.named)('process')),
|
|
102
|
+
(0, tslib_1.__metadata)("design:paramtypes", [Object, process_manager_1.ProcessManager, Object])
|
|
103
|
+
], RawProcess);
|
|
104
|
+
exports.RawProcess = RawProcess;
|
|
105
105
|
//# sourceMappingURL=raw-process.js.map
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export {};
|
|
1
|
+
export {};
|
|
2
2
|
//# sourceMappingURL=raw-process.spec.d.ts.map
|
|
@@ -1,165 +1,165 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
// *****************************************************************************
|
|
4
|
-
// Copyright (C) 2017 Ericsson and others.
|
|
5
|
-
//
|
|
6
|
-
// This program and the accompanying materials are made available under the
|
|
7
|
-
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
8
|
-
// http://www.eclipse.org/legal/epl-2.0.
|
|
9
|
-
//
|
|
10
|
-
// This Source Code may also be made available under the following Secondary
|
|
11
|
-
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
12
|
-
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
13
|
-
// with the GNU Classpath Exception which is available at
|
|
14
|
-
// https://www.gnu.org/software/classpath/license.html.
|
|
15
|
-
//
|
|
16
|
-
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
17
|
-
// *****************************************************************************
|
|
18
|
-
const chai = require("chai");
|
|
19
|
-
const process = require("process");
|
|
20
|
-
const stream = require("stream");
|
|
21
|
-
const process_test_container_1 = require("./test/process-test-container");
|
|
22
|
-
const raw_process_1 = require("./raw-process");
|
|
23
|
-
const temp = require("temp");
|
|
24
|
-
const fs = require("fs");
|
|
25
|
-
const path = require("path");
|
|
26
|
-
const core_1 = require("@theia/core");
|
|
27
|
-
/* Allow to create temporary files, but delete them when we're done. */
|
|
28
|
-
const track = temp.track();
|
|
29
|
-
/**
|
|
30
|
-
* Globals
|
|
31
|
-
*/
|
|
32
|
-
const expect = chai.expect;
|
|
33
|
-
const FORK_TEST_FILE = path.join(__dirname, '../../src/node/test/process-fork-test.js');
|
|
34
|
-
describe('RawProcess', function () {
|
|
35
|
-
this.timeout(20000);
|
|
36
|
-
let rawProcessFactory;
|
|
37
|
-
beforeEach(() => {
|
|
38
|
-
rawProcessFactory = (0, process_test_container_1.createProcessTestContainer)().get(raw_process_1.RawProcessFactory);
|
|
39
|
-
});
|
|
40
|
-
after(() => {
|
|
41
|
-
track.cleanupSync();
|
|
42
|
-
});
|
|
43
|
-
it('test error on non-existent path', async function () {
|
|
44
|
-
const error = await new Promise((resolve, reject) => {
|
|
45
|
-
const proc = rawProcessFactory({ command: '/non-existent' });
|
|
46
|
-
proc.onStart(reject);
|
|
47
|
-
proc.onError(resolve);
|
|
48
|
-
proc.onExit(reject);
|
|
49
|
-
});
|
|
50
|
-
expect(error.code).eq('ENOENT');
|
|
51
|
-
});
|
|
52
|
-
it('test error on non-executable path', async function () {
|
|
53
|
-
// Create a non-executable file.
|
|
54
|
-
const f = track.openSync('non-executable');
|
|
55
|
-
fs.writeSync(f.fd, 'echo bob');
|
|
56
|
-
// Make really sure it's non-executable.
|
|
57
|
-
let mode = fs.fstatSync(f.fd).mode;
|
|
58
|
-
mode &= ~fs.constants.S_IXUSR;
|
|
59
|
-
mode &= ~fs.constants.S_IXGRP;
|
|
60
|
-
mode &= ~fs.constants.S_IXOTH;
|
|
61
|
-
fs.fchmodSync(f.fd, mode);
|
|
62
|
-
fs.closeSync(f.fd);
|
|
63
|
-
const error = await new Promise((resolve, reject) => {
|
|
64
|
-
const proc = rawProcessFactory({ command: f.path });
|
|
65
|
-
proc.onStart(reject);
|
|
66
|
-
proc.onError(resolve);
|
|
67
|
-
proc.onExit(reject);
|
|
68
|
-
});
|
|
69
|
-
// On Windows, we get 'UNKNOWN'.
|
|
70
|
-
const expectedCode = core_1.isWindows ? 'UNKNOWN' : 'EACCES';
|
|
71
|
-
expect(error.code).eq(expectedCode);
|
|
72
|
-
});
|
|
73
|
-
it('test start event', function () {
|
|
74
|
-
return new Promise(async (resolve, reject) => {
|
|
75
|
-
const args = ['-e', 'process.exit(3)'];
|
|
76
|
-
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
77
|
-
rawProcess.onStart(resolve);
|
|
78
|
-
rawProcess.onError(reject);
|
|
79
|
-
rawProcess.onExit(reject);
|
|
80
|
-
});
|
|
81
|
-
});
|
|
82
|
-
it('test exit', async function () {
|
|
83
|
-
const args = ['--version'];
|
|
84
|
-
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
85
|
-
const p = new Promise((resolve, reject) => {
|
|
86
|
-
rawProcess.onError(reject);
|
|
87
|
-
rawProcess.onExit(event => {
|
|
88
|
-
if (event.code === undefined) {
|
|
89
|
-
reject(new Error('event.code is undefined'));
|
|
90
|
-
}
|
|
91
|
-
else {
|
|
92
|
-
resolve(event.code);
|
|
93
|
-
}
|
|
94
|
-
});
|
|
95
|
-
});
|
|
96
|
-
const exitCode = await p;
|
|
97
|
-
expect(exitCode).equal(0);
|
|
98
|
-
});
|
|
99
|
-
it('test pipe stdout stream', async function () {
|
|
100
|
-
const output = await new Promise(async (resolve, reject) => {
|
|
101
|
-
const args = ['-e', 'console.log("text to stdout")'];
|
|
102
|
-
const outStream = new stream.PassThrough();
|
|
103
|
-
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
104
|
-
rawProcess.onError(reject);
|
|
105
|
-
rawProcess.outputStream.pipe(outStream);
|
|
106
|
-
let buf = '';
|
|
107
|
-
outStream.on('data', data => {
|
|
108
|
-
buf += data.toString();
|
|
109
|
-
});
|
|
110
|
-
outStream.on('end', () => {
|
|
111
|
-
resolve(buf.trim());
|
|
112
|
-
});
|
|
113
|
-
});
|
|
114
|
-
expect(output).to.be.equal('text to stdout');
|
|
115
|
-
});
|
|
116
|
-
it('test pipe stderr stream', async function () {
|
|
117
|
-
const output = await new Promise(async (resolve, reject) => {
|
|
118
|
-
const args = ['-e', 'console.error("text to stderr")'];
|
|
119
|
-
const outStream = new stream.PassThrough();
|
|
120
|
-
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
121
|
-
rawProcess.onError(reject);
|
|
122
|
-
rawProcess.errorStream.pipe(outStream);
|
|
123
|
-
let buf = '';
|
|
124
|
-
outStream.on('data', data => {
|
|
125
|
-
buf += data.toString();
|
|
126
|
-
});
|
|
127
|
-
outStream.on('end', () => {
|
|
128
|
-
resolve(buf.trim());
|
|
129
|
-
});
|
|
130
|
-
});
|
|
131
|
-
expect(output).to.be.equal('text to stderr');
|
|
132
|
-
});
|
|
133
|
-
it('test forked pipe stdout stream', async function () {
|
|
134
|
-
const args = ['version'];
|
|
135
|
-
const rawProcess = rawProcessFactory({ modulePath: FORK_TEST_FILE, args, options: { stdio: 'pipe' } });
|
|
136
|
-
const outStream = new stream.PassThrough();
|
|
137
|
-
const p = new Promise((resolve, reject) => {
|
|
138
|
-
let version = '';
|
|
139
|
-
outStream.on('data', data => {
|
|
140
|
-
version += data.toString();
|
|
141
|
-
});
|
|
142
|
-
outStream.on('end', () => {
|
|
143
|
-
resolve(version.trim());
|
|
144
|
-
});
|
|
145
|
-
});
|
|
146
|
-
rawProcess.outputStream.pipe(outStream);
|
|
147
|
-
expect(await p).to.be.equal('1.0.0');
|
|
148
|
-
});
|
|
149
|
-
it('test forked pipe stderr stream', async function () {
|
|
150
|
-
const rawProcess = rawProcessFactory({ modulePath: FORK_TEST_FILE, args: [], options: { stdio: 'pipe' } });
|
|
151
|
-
const outStream = new stream.PassThrough();
|
|
152
|
-
const p = new Promise((resolve, reject) => {
|
|
153
|
-
let version = '';
|
|
154
|
-
outStream.on('data', data => {
|
|
155
|
-
version += data.toString();
|
|
156
|
-
});
|
|
157
|
-
outStream.on('end', () => {
|
|
158
|
-
resolve(version.trim());
|
|
159
|
-
});
|
|
160
|
-
});
|
|
161
|
-
rawProcess.errorStream.pipe(outStream);
|
|
162
|
-
expect(await p).to.have.string('Error');
|
|
163
|
-
});
|
|
164
|
-
});
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
// *****************************************************************************
|
|
4
|
+
// Copyright (C) 2017 Ericsson and others.
|
|
5
|
+
//
|
|
6
|
+
// This program and the accompanying materials are made available under the
|
|
7
|
+
// terms of the Eclipse Public License v. 2.0 which is available at
|
|
8
|
+
// http://www.eclipse.org/legal/epl-2.0.
|
|
9
|
+
//
|
|
10
|
+
// This Source Code may also be made available under the following Secondary
|
|
11
|
+
// Licenses when the conditions for such availability set forth in the Eclipse
|
|
12
|
+
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
|
|
13
|
+
// with the GNU Classpath Exception which is available at
|
|
14
|
+
// https://www.gnu.org/software/classpath/license.html.
|
|
15
|
+
//
|
|
16
|
+
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
|
|
17
|
+
// *****************************************************************************
|
|
18
|
+
const chai = require("chai");
|
|
19
|
+
const process = require("process");
|
|
20
|
+
const stream = require("stream");
|
|
21
|
+
const process_test_container_1 = require("./test/process-test-container");
|
|
22
|
+
const raw_process_1 = require("./raw-process");
|
|
23
|
+
const temp = require("temp");
|
|
24
|
+
const fs = require("fs");
|
|
25
|
+
const path = require("path");
|
|
26
|
+
const core_1 = require("@theia/core");
|
|
27
|
+
/* Allow to create temporary files, but delete them when we're done. */
|
|
28
|
+
const track = temp.track();
|
|
29
|
+
/**
|
|
30
|
+
* Globals
|
|
31
|
+
*/
|
|
32
|
+
const expect = chai.expect;
|
|
33
|
+
const FORK_TEST_FILE = path.join(__dirname, '../../src/node/test/process-fork-test.js');
|
|
34
|
+
describe('RawProcess', function () {
|
|
35
|
+
this.timeout(20000);
|
|
36
|
+
let rawProcessFactory;
|
|
37
|
+
beforeEach(() => {
|
|
38
|
+
rawProcessFactory = (0, process_test_container_1.createProcessTestContainer)().get(raw_process_1.RawProcessFactory);
|
|
39
|
+
});
|
|
40
|
+
after(() => {
|
|
41
|
+
track.cleanupSync();
|
|
42
|
+
});
|
|
43
|
+
it('test error on non-existent path', async function () {
|
|
44
|
+
const error = await new Promise((resolve, reject) => {
|
|
45
|
+
const proc = rawProcessFactory({ command: '/non-existent' });
|
|
46
|
+
proc.onStart(reject);
|
|
47
|
+
proc.onError(resolve);
|
|
48
|
+
proc.onExit(reject);
|
|
49
|
+
});
|
|
50
|
+
expect(error.code).eq('ENOENT');
|
|
51
|
+
});
|
|
52
|
+
it('test error on non-executable path', async function () {
|
|
53
|
+
// Create a non-executable file.
|
|
54
|
+
const f = track.openSync('non-executable');
|
|
55
|
+
fs.writeSync(f.fd, 'echo bob');
|
|
56
|
+
// Make really sure it's non-executable.
|
|
57
|
+
let mode = fs.fstatSync(f.fd).mode;
|
|
58
|
+
mode &= ~fs.constants.S_IXUSR;
|
|
59
|
+
mode &= ~fs.constants.S_IXGRP;
|
|
60
|
+
mode &= ~fs.constants.S_IXOTH;
|
|
61
|
+
fs.fchmodSync(f.fd, mode);
|
|
62
|
+
fs.closeSync(f.fd);
|
|
63
|
+
const error = await new Promise((resolve, reject) => {
|
|
64
|
+
const proc = rawProcessFactory({ command: f.path });
|
|
65
|
+
proc.onStart(reject);
|
|
66
|
+
proc.onError(resolve);
|
|
67
|
+
proc.onExit(reject);
|
|
68
|
+
});
|
|
69
|
+
// On Windows, we get 'UNKNOWN'.
|
|
70
|
+
const expectedCode = core_1.isWindows ? 'UNKNOWN' : 'EACCES';
|
|
71
|
+
expect(error.code).eq(expectedCode);
|
|
72
|
+
});
|
|
73
|
+
it('test start event', function () {
|
|
74
|
+
return new Promise(async (resolve, reject) => {
|
|
75
|
+
const args = ['-e', 'process.exit(3)'];
|
|
76
|
+
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
77
|
+
rawProcess.onStart(resolve);
|
|
78
|
+
rawProcess.onError(reject);
|
|
79
|
+
rawProcess.onExit(reject);
|
|
80
|
+
});
|
|
81
|
+
});
|
|
82
|
+
it('test exit', async function () {
|
|
83
|
+
const args = ['--version'];
|
|
84
|
+
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
85
|
+
const p = new Promise((resolve, reject) => {
|
|
86
|
+
rawProcess.onError(reject);
|
|
87
|
+
rawProcess.onExit(event => {
|
|
88
|
+
if (event.code === undefined) {
|
|
89
|
+
reject(new Error('event.code is undefined'));
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
resolve(event.code);
|
|
93
|
+
}
|
|
94
|
+
});
|
|
95
|
+
});
|
|
96
|
+
const exitCode = await p;
|
|
97
|
+
expect(exitCode).equal(0);
|
|
98
|
+
});
|
|
99
|
+
it('test pipe stdout stream', async function () {
|
|
100
|
+
const output = await new Promise(async (resolve, reject) => {
|
|
101
|
+
const args = ['-e', 'console.log("text to stdout")'];
|
|
102
|
+
const outStream = new stream.PassThrough();
|
|
103
|
+
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
104
|
+
rawProcess.onError(reject);
|
|
105
|
+
rawProcess.outputStream.pipe(outStream);
|
|
106
|
+
let buf = '';
|
|
107
|
+
outStream.on('data', data => {
|
|
108
|
+
buf += data.toString();
|
|
109
|
+
});
|
|
110
|
+
outStream.on('end', () => {
|
|
111
|
+
resolve(buf.trim());
|
|
112
|
+
});
|
|
113
|
+
});
|
|
114
|
+
expect(output).to.be.equal('text to stdout');
|
|
115
|
+
});
|
|
116
|
+
it('test pipe stderr stream', async function () {
|
|
117
|
+
const output = await new Promise(async (resolve, reject) => {
|
|
118
|
+
const args = ['-e', 'console.error("text to stderr")'];
|
|
119
|
+
const outStream = new stream.PassThrough();
|
|
120
|
+
const rawProcess = rawProcessFactory({ command: process.execPath, 'args': args });
|
|
121
|
+
rawProcess.onError(reject);
|
|
122
|
+
rawProcess.errorStream.pipe(outStream);
|
|
123
|
+
let buf = '';
|
|
124
|
+
outStream.on('data', data => {
|
|
125
|
+
buf += data.toString();
|
|
126
|
+
});
|
|
127
|
+
outStream.on('end', () => {
|
|
128
|
+
resolve(buf.trim());
|
|
129
|
+
});
|
|
130
|
+
});
|
|
131
|
+
expect(output).to.be.equal('text to stderr');
|
|
132
|
+
});
|
|
133
|
+
it('test forked pipe stdout stream', async function () {
|
|
134
|
+
const args = ['version'];
|
|
135
|
+
const rawProcess = rawProcessFactory({ modulePath: FORK_TEST_FILE, args, options: { stdio: 'pipe' } });
|
|
136
|
+
const outStream = new stream.PassThrough();
|
|
137
|
+
const p = new Promise((resolve, reject) => {
|
|
138
|
+
let version = '';
|
|
139
|
+
outStream.on('data', data => {
|
|
140
|
+
version += data.toString();
|
|
141
|
+
});
|
|
142
|
+
outStream.on('end', () => {
|
|
143
|
+
resolve(version.trim());
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
rawProcess.outputStream.pipe(outStream);
|
|
147
|
+
expect(await p).to.be.equal('1.0.0');
|
|
148
|
+
});
|
|
149
|
+
it('test forked pipe stderr stream', async function () {
|
|
150
|
+
const rawProcess = rawProcessFactory({ modulePath: FORK_TEST_FILE, args: [], options: { stdio: 'pipe' } });
|
|
151
|
+
const outStream = new stream.PassThrough();
|
|
152
|
+
const p = new Promise((resolve, reject) => {
|
|
153
|
+
let version = '';
|
|
154
|
+
outStream.on('data', data => {
|
|
155
|
+
version += data.toString();
|
|
156
|
+
});
|
|
157
|
+
outStream.on('end', () => {
|
|
158
|
+
resolve(version.trim());
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
rawProcess.errorStream.pipe(outStream);
|
|
162
|
+
expect(await p).to.have.string('Error');
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
165
|
//# sourceMappingURL=raw-process.spec.js.map
|
|
@@ -1,11 +1,11 @@
|
|
|
1
|
-
import { TerminalProcess, TerminalProcessOptions } from './terminal-process';
|
|
2
|
-
export declare const TaskTerminalProcessFactory: unique symbol;
|
|
3
|
-
export interface TaskTerminalProcessFactory {
|
|
4
|
-
(options: TerminalProcessOptions): TaskTerminalProcess;
|
|
5
|
-
}
|
|
6
|
-
export declare class TaskTerminalProcess extends TerminalProcess {
|
|
7
|
-
exited: boolean;
|
|
8
|
-
attachmentAttempted: boolean;
|
|
9
|
-
protected onTerminalExit(code: number | undefined, signal: string | undefined): void;
|
|
10
|
-
}
|
|
1
|
+
import { TerminalProcess, TerminalProcessOptions } from './terminal-process';
|
|
2
|
+
export declare const TaskTerminalProcessFactory: unique symbol;
|
|
3
|
+
export interface TaskTerminalProcessFactory {
|
|
4
|
+
(options: TerminalProcessOptions): TaskTerminalProcess;
|
|
5
|
+
}
|
|
6
|
+
export declare class TaskTerminalProcess extends TerminalProcess {
|
|
7
|
+
exited: boolean;
|
|
8
|
+
attachmentAttempted: boolean;
|
|
9
|
+
protected onTerminalExit(code: number | undefined, signal: string | undefined): void;
|
|
10
|
+
}
|
|
11
11
|
//# sourceMappingURL=task-terminal-process.d.ts.map
|