1 | "use strict";
|
2 |
|
3 |
|
4 |
|
5 |
|
6 |
|
7 |
|
8 |
|
9 |
|
10 |
|
11 |
|
12 |
|
13 |
|
14 |
|
15 |
|
16 |
|
17 | var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
18 | var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
19 | if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
20 | else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
21 | return c > 3 && r && Object.defineProperty(target, key, r), r;
|
22 | };
|
23 | var __metadata = (this && this.__metadata) || function (k, v) {
|
24 | if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
25 | };
|
26 | Object.defineProperty(exports, "__esModule", { value: true });
|
27 | exports.IPCConnectionProvider = void 0;
|
28 | const cp = require("child_process");
|
29 | const inversify_1 = require("inversify");
|
30 | const path = require("path");
|
31 | const readline_1 = require("readline");
|
32 | const common_1 = require("../../common");
|
33 | const ipc_channel_1 = require("./ipc-channel");
|
34 | const ipc_protocol_1 = require("./ipc-protocol");
|
35 | let IPCConnectionProvider = class IPCConnectionProvider {
|
36 | listen(options, acceptor) {
|
37 | return this.doListen(Object.assign({ logger: this.logger, args: [] }, options), acceptor);
|
38 | }
|
39 | doListen(options, acceptor) {
|
40 | const childProcess = this.fork(options);
|
41 | const channel = new ipc_channel_1.IPCChannel(childProcess);
|
42 | const toStop = new common_1.DisposableCollection();
|
43 | const toCancelStop = toStop.push(common_1.Disposable.create(() => childProcess.kill()));
|
44 | const errorHandler = options.errorHandler;
|
45 | if (errorHandler) {
|
46 | let errorCount = 0;
|
47 | channel.onError((err) => {
|
48 | errorCount++;
|
49 | if (errorHandler.shouldStop(err, errorCount)) {
|
50 | toStop.dispose();
|
51 | }
|
52 | });
|
53 | channel.onClose(() => {
|
54 | if (toStop.disposed) {
|
55 | return;
|
56 | }
|
57 | if (errorHandler.shouldRestart()) {
|
58 | toCancelStop.dispose();
|
59 | toStop.push(this.doListen(options, acceptor));
|
60 | }
|
61 | });
|
62 | }
|
63 | acceptor(channel);
|
64 | return toStop;
|
65 | }
|
66 | fork(options) {
|
67 | const forkOptions = {
|
68 | env: (0, ipc_protocol_1.createIpcEnv)(options),
|
69 | execArgv: [],
|
70 |
|
71 |
|
72 |
|
73 |
|
74 | stdio: ['pipe', 'pipe', 'pipe', 'ipc', 'overlapped']
|
75 | };
|
76 | const inspectArgPrefix = `--${options.serverName}-inspect`;
|
77 | const inspectArg = process.argv.find(v => v.startsWith(inspectArgPrefix));
|
78 | if (inspectArg !== undefined) {
|
79 | forkOptions.execArgv = ['--nolazy', `--inspect${inspectArg.substr(inspectArgPrefix.length)}`];
|
80 | }
|
81 | const childProcess = cp.fork(path.join(__dirname, 'ipc-bootstrap'), options.args, forkOptions);
|
82 | (0, readline_1.createInterface)(childProcess.stdout).on('line', line => this.logger.info(`[${options.serverName}: ${childProcess.pid}] ${line}`));
|
83 | (0, readline_1.createInterface)(childProcess.stderr).on('line', line => this.logger.error(`[${options.serverName}: ${childProcess.pid}] ${line}`));
|
84 | this.logger.debug(`[${options.serverName}: ${childProcess.pid}] IPC started`);
|
85 | childProcess.once('exit', () => this.logger.debug(`[${options.serverName}: ${childProcess.pid}] IPC exited`));
|
86 | return childProcess;
|
87 | }
|
88 | };
|
89 | __decorate([
|
90 | (0, inversify_1.inject)(common_1.ILogger),
|
91 | __metadata("design:type", Object)
|
92 | ], IPCConnectionProvider.prototype, "logger", void 0);
|
93 | IPCConnectionProvider = __decorate([
|
94 | (0, inversify_1.injectable)()
|
95 | ], IPCConnectionProvider);
|
96 | exports.IPCConnectionProvider = IPCConnectionProvider;
|
97 |
|
\ | No newline at end of file |