1 | import { createEndpoint } from '@neo-one/node-core-esnext-esm';
|
2 | import { Config, killProcess } from '@neo-one/server-plugin-esnext-esm';
|
3 | import fetch from 'cross-fetch';
|
4 | import execa from 'execa';
|
5 | import * as fs from 'fs-extra';
|
6 | import _ from 'lodash';
|
7 | import * as path from 'path';
|
8 | import { take } from 'rxjs/operators';
|
9 | import { NodeAdapter } from './NodeAdapter';
|
10 | const DEFAULT_RPC_URLS = [
|
11 | 'http://node1.nyc3.bridgeprotocol.io:10332',
|
12 | 'http://node2.nyc3.bridgeprotocol.io:10332',
|
13 | 'https://seed1.switcheo.network:10331',
|
14 | 'https://seed2.switcheo.network:10331',
|
15 | 'https://seed3.switcheo.network:10331',
|
16 | 'http://seed1.aphelion-neo.com:10332',
|
17 | 'http://seed2.aphelion-neo.com:10332',
|
18 | 'http://seed3.aphelion-neo.com:10332',
|
19 | 'http://seed4.aphelion-neo.com:10332',
|
20 | ];
|
21 | const DEFAULT_SEEDS = [
|
22 | { type: 'tcp', host: 'node1.nyc3.bridgeprotocol.io', port: 10333 },
|
23 | { type: 'tcp', host: 'node2.nyc3.bridgeprotocol.io', port: 10333 },
|
24 | { type: 'tcp', host: 'seed1.switcheo.com', port: 10333 },
|
25 | { type: 'tcp', host: 'seed2.switcheo.com', port: 10333 },
|
26 | { type: 'tcp', host: 'seed3.switcheo.com', port: 10333 },
|
27 | { type: 'tcp', host: 'seed1.aphelion-neo.com', port: 10333 },
|
28 | { type: 'tcp', host: 'seed2.aphelion-neo.com', port: 10333 },
|
29 | { type: 'tcp', host: 'seed3.aphelion-neo.com', port: 10333 },
|
30 | { type: 'tcp', host: 'seed4.aphelion-neo.com', port: 10333 },
|
31 | ];
|
32 | const makeDefaultConfig = (dataPath) => ({
|
33 | log: {
|
34 | level: 'info',
|
35 | maxSize: 10 * 1024 * 1024,
|
36 | maxFiles: 5,
|
37 | },
|
38 | settings: {
|
39 | test: false,
|
40 | },
|
41 | environment: {
|
42 | dataPath: path.resolve(dataPath, 'node'),
|
43 | rpc: {},
|
44 | node: {},
|
45 | network: {},
|
46 | },
|
47 | options: {
|
48 | node: {
|
49 | consensus: {
|
50 | enabled: false,
|
51 | options: { privateKey: 'default', privateNet: false },
|
52 | },
|
53 | rpcURLs: [...DEFAULT_RPC_URLS],
|
54 | },
|
55 | network: {
|
56 | seeds: DEFAULT_SEEDS.map(createEndpoint),
|
57 | },
|
58 | rpc: {
|
59 | server: {
|
60 | keepAliveTimeout: 60000,
|
61 | },
|
62 | liveHealthCheck: {
|
63 | rpcURLs: DEFAULT_RPC_URLS,
|
64 | offset: 1,
|
65 | timeoutMS: 5000,
|
66 | },
|
67 | readyHealthCheck: {
|
68 | rpcURLs: DEFAULT_RPC_URLS,
|
69 | offset: 1,
|
70 | timeoutMS: 5000,
|
71 | },
|
72 | },
|
73 | },
|
74 | });
|
75 | export const createNodeConfig = ({ dataPath, defaultConfig = makeDefaultConfig(dataPath), }) => new Config({
|
76 | name: 'node',
|
77 | defaultConfig,
|
78 | schema: {
|
79 | type: 'object',
|
80 | required: ['log'],
|
81 | properties: {
|
82 | log: {
|
83 | type: 'object',
|
84 | required: ['level', 'maxSize', 'maxFiles'],
|
85 | properties: {
|
86 | level: { type: 'string' },
|
87 | maxSize: { type: 'number' },
|
88 | maxFiles: { type: 'number' },
|
89 | },
|
90 | },
|
91 | settings: {
|
92 | type: 'object',
|
93 | required: ['test'],
|
94 | properties: {
|
95 | test: { type: 'boolean' },
|
96 | privateNet: { type: 'boolean' },
|
97 | secondsPerBlock: { type: 'number' },
|
98 | standbyValidators: { type: 'array', items: { type: 'string' } },
|
99 | },
|
100 | },
|
101 | environment: {
|
102 | type: 'object',
|
103 | required: ['dataPath', 'rpc', 'node', 'network'],
|
104 | properties: {
|
105 | dataPath: { type: 'string' },
|
106 | rpc: {
|
107 | type: 'object',
|
108 | required: [],
|
109 | properties: {
|
110 | http: {
|
111 | type: 'object',
|
112 | required: ['host', 'port'],
|
113 | properties: {
|
114 | host: { type: 'string' },
|
115 | port: { type: 'number' },
|
116 | },
|
117 | },
|
118 | https: {
|
119 | type: 'object',
|
120 | required: ['host', 'port', 'key', 'cert'],
|
121 | properties: {
|
122 | host: { type: 'string' },
|
123 | port: { type: 'number' },
|
124 | key: { type: 'string' },
|
125 | cert: { type: 'string' },
|
126 | },
|
127 | },
|
128 | },
|
129 | },
|
130 | node: {
|
131 | type: 'object',
|
132 | required: [],
|
133 | properties: {
|
134 | externalPort: { type: 'number' },
|
135 | },
|
136 | },
|
137 | network: {
|
138 | type: 'object',
|
139 | required: [],
|
140 | properties: {
|
141 | listenTCP: {
|
142 | type: 'object',
|
143 | required: ['port'],
|
144 | properties: {
|
145 | host: { type: 'string' },
|
146 | port: { type: 'number' },
|
147 | },
|
148 | },
|
149 | externalEndpoints: {
|
150 | type: 'array',
|
151 | items: { type: 'string' },
|
152 | },
|
153 | connectPeersDelayMS: { type: 'number' },
|
154 | socketTimeoutMS: { type: 'number' },
|
155 | },
|
156 | },
|
157 | },
|
158 | },
|
159 | options: {
|
160 | type: 'object',
|
161 | required: ['node', 'network', 'rpc'],
|
162 | properties: {
|
163 | node: {
|
164 | type: 'object',
|
165 | required: ['consensus', 'rpcURLs'],
|
166 | properties: {
|
167 | consensus: {
|
168 | type: 'object',
|
169 | required: ['enabled', 'options'],
|
170 | properties: {
|
171 | enabled: { type: 'boolean' },
|
172 | options: {
|
173 | type: 'object',
|
174 | required: ['privateKey', 'privateNet'],
|
175 | properties: {
|
176 | privateKey: { type: 'string' },
|
177 | privateNet: { type: 'boolean' },
|
178 | },
|
179 | },
|
180 | },
|
181 | },
|
182 | rpcURLs: { type: 'array', items: { type: 'string' } },
|
183 | },
|
184 | },
|
185 | network: {
|
186 | type: 'object',
|
187 | required: ['seeds'],
|
188 | properties: {
|
189 | seeds: { type: 'array', items: { type: 'string' } },
|
190 | maxConnectedPeers: { type: 'number' },
|
191 | },
|
192 | },
|
193 | rpc: {
|
194 | type: 'object',
|
195 | required: ['server', 'liveHealthCheck', 'readyHealthCheck'],
|
196 | properties: {
|
197 | server: {
|
198 | type: 'object',
|
199 | required: ['keepAliveTimeout'],
|
200 | properties: {
|
201 | keepAliveTimeout: { type: 'number' },
|
202 | },
|
203 | },
|
204 | liveHealthCheck: {
|
205 | type: 'object',
|
206 | required: ['rpcURLs', 'offset', 'timeoutMS'],
|
207 | properties: {
|
208 | rpcURLs: { type: 'array', items: { type: 'string' } },
|
209 | offset: { type: 'number' },
|
210 | timeoutMS: { type: 'number' },
|
211 | },
|
212 | },
|
213 | readyHealthCheck: {
|
214 | type: 'object',
|
215 | required: ['rpcURLs', 'offset', 'timeoutMS'],
|
216 | properties: {
|
217 | rpcURLs: { type: 'array', items: { type: 'string' } },
|
218 | offset: { type: 'number' },
|
219 | timeoutMS: { type: 'number' },
|
220 | },
|
221 | },
|
222 | },
|
223 | },
|
224 | },
|
225 | },
|
226 | },
|
227 | },
|
228 | configPath: dataPath,
|
229 | });
|
230 | export class NEOONENodeAdapter extends NodeAdapter {
|
231 | constructor({ monitor, name, binary, dataPath, settings, }) {
|
232 | super({
|
233 | monitor: monitor.at('neo_one_node_adapter'),
|
234 | name,
|
235 | binary,
|
236 | dataPath,
|
237 | settings,
|
238 | });
|
239 | }
|
240 | getDebug() {
|
241 | return super
|
242 | .getDebug()
|
243 | .concat([
|
244 | ['Process ID', this.mutableProcess === undefined ? 'null' : `${this.mutableProcess.pid}`],
|
245 | ['Config Path', this.mutableConfig === undefined ? 'null' : this.mutableConfig.configPath],
|
246 | ]);
|
247 | }
|
248 | getNodeStatus() {
|
249 | return {
|
250 | rpcAddress: this.getAddress('/rpc'),
|
251 | tcpAddress: `localhost:${this.mutableSettings.listenTCPPort}`,
|
252 | telemetryAddress: `http://localhost:${this.mutableSettings.telemetryPort}/metrics`,
|
253 | };
|
254 | }
|
255 | async isLive() {
|
256 | return this.checkRPC('/live_health_check');
|
257 | }
|
258 | async isReady() {
|
259 | return this.checkRPC('/ready_health_check');
|
260 | }
|
261 | async createInternal() {
|
262 | await this.writeSettings(this.mutableSettings);
|
263 | }
|
264 | async updateInternal(settings) {
|
265 | const restart = await this.writeSettings(settings);
|
266 | if (restart && this.mutableProcess !== undefined) {
|
267 | await this.stop();
|
268 | await this.start();
|
269 | }
|
270 | }
|
271 | async startInternal() {
|
272 | if (this.mutableProcess === undefined) {
|
273 | const child = execa(this.binary.cmd, this.binary.firstArgs.concat(['start', 'node', this.dataPath]), {
|
274 | // @ts-ignore
|
275 | windowsHide: true,
|
276 | stdio: 'ignore',
|
277 | });
|
278 | this.mutableProcess = child;
|
279 | // tslint:disable-next-line no-floating-promises
|
280 | child
|
281 | .then(() => {
|
282 | this.monitor.log({
|
283 | name: 'neo_node_adapter_node_exit',
|
284 | message: 'Child process exited',
|
285 | });
|
286 | this.mutableProcess = undefined;
|
287 | })
|
288 | .catch((error) => {
|
289 | this.monitor.logError({
|
290 | name: 'neo_node_adapter_node_error',
|
291 | message: 'Child process exited with an error.',
|
292 | error,
|
293 | });
|
294 | this.mutableProcess = undefined;
|
295 | });
|
296 | }
|
297 | }
|
298 | async stopInternal() {
|
299 | const child = this.mutableProcess;
|
300 | this.mutableProcess = undefined;
|
301 | if (child !== undefined) {
|
302 | await killProcess(child.pid);
|
303 | }
|
304 | }
|
305 | async checkRPC(rpcPath) {
|
306 | try {
|
307 | const response = await fetch(this.getAddress(rpcPath));
|
308 | return response.ok;
|
309 | }
|
310 | catch (error) {
|
311 | if (error.code !== 'ECONNREFUSED') {
|
312 | this.monitor.withData({ [this.monitor.labels.HTTP_PATH]: rpcPath }).logError({
|
313 | name: 'http_client_request',
|
314 | message: 'Failed to check RPC.',
|
315 | error,
|
316 | });
|
317 | }
|
318 | return false;
|
319 | }
|
320 | }
|
321 | getAddress(rpcPath) {
|
322 | return `http://localhost:${this.mutableSettings.rpcPort}${rpcPath}`;
|
323 | }
|
324 | async writeSettings(settings) {
|
325 | let config = this.mutableConfig;
|
326 | if (config === undefined) {
|
327 | config = createNodeConfig({
|
328 | dataPath: this.dataPath,
|
329 | defaultConfig: this.createConfig(settings),
|
330 | });
|
331 | this.mutableConfig = config;
|
332 | }
|
333 | const nodeConfig = await config.config$.pipe(take(1)).toPromise();
|
334 | const newNodeConfig = this.createConfig(settings);
|
335 | await fs.ensureDir(newNodeConfig.environment.dataPath);
|
336 | await config.update({ config: newNodeConfig });
|
337 | return !(_.isEqual(nodeConfig.settings, newNodeConfig.settings) &&
|
338 | _.isEqual(nodeConfig.environment, newNodeConfig.environment));
|
339 | }
|
340 | createConfig(settings) {
|
341 | return {
|
342 | log: {
|
343 | level: 'info',
|
344 | maxSize: 10 * 1024 * 1024,
|
345 | maxFiles: 5,
|
346 | },
|
347 | settings: {
|
348 | test: settings.isTestNet,
|
349 | privateNet: settings.privateNet,
|
350 | secondsPerBlock: settings.secondsPerBlock,
|
351 | standbyValidators: settings.standbyValidators,
|
352 | address: settings.address,
|
353 | },
|
354 | environment: {
|
355 | dataPath: path.resolve(this.dataPath, 'chain'),
|
356 | rpc: {
|
357 | http: {
|
358 | port: settings.rpcPort,
|
359 | host: '0.0.0.0',
|
360 | },
|
361 | },
|
362 | node: {
|
363 | externalPort: settings.listenTCPPort,
|
364 | },
|
365 | network: {
|
366 | listenTCP: {
|
367 | port: settings.listenTCPPort,
|
368 | host: '0.0.0.0',
|
369 | },
|
370 | },
|
371 | telemetry: {
|
372 | port: settings.telemetryPort,
|
373 | },
|
374 | },
|
375 | options: {
|
376 | node: {
|
377 | consensus: settings.consensus,
|
378 | rpcURLs: settings.rpcEndpoints,
|
379 | },
|
380 | network: {
|
381 | seeds: settings.seeds,
|
382 | },
|
383 | rpc: {
|
384 | server: {
|
385 | keepAliveTimeout: 60000,
|
386 | },
|
387 | liveHealthCheck: {
|
388 | rpcURLs: settings.rpcEndpoints,
|
389 | offset: 1,
|
390 | timeoutMS: 5000,
|
391 | },
|
392 | readyHealthCheck: {
|
393 | rpcURLs: settings.rpcEndpoints,
|
394 | offset: 1,
|
395 | timeoutMS: 5000,
|
396 | },
|
397 | },
|
398 | },
|
399 | };
|
400 | }
|
401 | }
|
402 |
|
403 | //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["NEOONENodeAdapter.ts"],"names":[],"mappings":"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","file":"neo-one-server-plugin-network/src/node/NEOONENodeAdapter.js","sourcesContent":["import { Monitor } from '@neo-one/monitor';\nimport { FullNodeEnvironment, FullNodeOptions } from '@neo-one/node';\nimport { createEndpoint, EndpointConfig } from '@neo-one/node-core';\nimport { Binary, Config, DescribeTable, killProcess } from '@neo-one/server-plugin';\nimport fetch from 'cross-fetch';\nimport execa from 'execa';\nimport * as fs from 'fs-extra';\nimport _ from 'lodash';\nimport * as path from 'path';\nimport { take } from 'rxjs/operators';\nimport { NodeSettings } from '../types';\nimport { NodeAdapter, NodeStatus } from './NodeAdapter';\n\nexport interface NodeConfig {\n  readonly log: {\n    readonly level: string;\n    readonly maxSize: number;\n    readonly maxFiles: number;\n  };\n\n  readonly settings: {\n    readonly test?: boolean;\n    readonly privateNet?: boolean;\n    readonly secondsPerBlock?: number;\n    readonly standbyValidators?: ReadonlyArray<string>;\n    readonly address?: string;\n  };\n\n  readonly environment: FullNodeEnvironment;\n  readonly options: FullNodeOptions;\n}\n\nconst DEFAULT_RPC_URLS: ReadonlyArray<string> = [\n  'http://node1.nyc3.bridgeprotocol.io:10332',\n  'http://node2.nyc3.bridgeprotocol.io:10332',\n  'https://seed1.switcheo.network:10331',\n  'https://seed2.switcheo.network:10331',\n  'https://seed3.switcheo.network:10331',\n  'http://seed1.aphelion-neo.com:10332',\n  'http://seed2.aphelion-neo.com:10332',\n  'http://seed3.aphelion-neo.com:10332',\n  'http://seed4.aphelion-neo.com:10332',\n];\n\nconst DEFAULT_SEEDS: ReadonlyArray<EndpointConfig> = [\n  { type: 'tcp', host: 'node1.nyc3.bridgeprotocol.io', port: 10333 },\n  { type: 'tcp', host: 'node2.nyc3.bridgeprotocol.io', port: 10333 },\n  { type: 'tcp', host: 'seed1.switcheo.com', port: 10333 },\n  { type: 'tcp', host: 'seed2.switcheo.com', port: 10333 },\n  { type: 'tcp', host: 'seed3.switcheo.com', port: 10333 },\n  { type: 'tcp', host: 'seed1.aphelion-neo.com', port: 10333 },\n  { type: 'tcp', host: 'seed2.aphelion-neo.com', port: 10333 },\n  { type: 'tcp', host: 'seed3.aphelion-neo.com', port: 10333 },\n  { type: 'tcp', host: 'seed4.aphelion-neo.com', port: 10333 },\n];\n\nconst makeDefaultConfig = (dataPath: string): NodeConfig => ({\n  log: {\n    level: 'info',\n    maxSize: 10 * 1024 * 1024,\n    maxFiles: 5,\n  },\n  settings: {\n    test: false,\n  },\n  environment: {\n    dataPath: path.resolve(dataPath, 'node'),\n    rpc: {},\n    node: {},\n    network: {},\n  },\n  options: {\n    node: {\n      consensus: {\n        enabled: false,\n        options: { privateKey: 'default', privateNet: false },\n      },\n      rpcURLs: [...DEFAULT_RPC_URLS],\n    },\n    network: {\n      seeds: DEFAULT_SEEDS.map(createEndpoint),\n    },\n    rpc: {\n      server: {\n        keepAliveTimeout: 60000,\n      },\n      liveHealthCheck: {\n        rpcURLs: DEFAULT_RPC_URLS,\n        offset: 1,\n        timeoutMS: 5000,\n      },\n      readyHealthCheck: {\n        rpcURLs: DEFAULT_RPC_URLS,\n        offset: 1,\n        timeoutMS: 5000,\n      },\n    },\n  },\n});\n\nexport const createNodeConfig = ({\n  dataPath,\n  defaultConfig = makeDefaultConfig(dataPath),\n}: {\n  readonly dataPath: string;\n  readonly defaultConfig?: NodeConfig;\n}): Config<NodeConfig> =>\n  new Config({\n    name: 'node',\n    defaultConfig,\n    schema: {\n      type: 'object',\n      required: ['log'],\n      properties: {\n        log: {\n          type: 'object',\n          required: ['level', 'maxSize', 'maxFiles'],\n          properties: {\n            level: { type: 'string' },\n            maxSize: { type: 'number' },\n            maxFiles: { type: 'number' },\n          },\n        },\n        settings: {\n          type: 'object',\n          required: ['test'],\n          properties: {\n            test: { type: 'boolean' },\n            privateNet: { type: 'boolean' },\n            secondsPerBlock: { type: 'number' },\n            standbyValidators: { type: 'array', items: { type: 'string' } },\n          },\n        },\n        environment: {\n          type: 'object',\n          required: ['dataPath', 'rpc', 'node', 'network'],\n          properties: {\n            dataPath: { type: 'string' },\n            rpc: {\n              type: 'object',\n              required: [],\n              properties: {\n                http: {\n                  type: 'object',\n                  required: ['host', 'port'],\n                  properties: {\n                    host: { type: 'string' },\n                    port: { type: 'number' },\n                  },\n                },\n\n                https: {\n                  type: 'object',\n                  required: ['host', 'port', 'key', 'cert'],\n                  properties: {\n                    host: { type: 'string' },\n                    port: { type: 'number' },\n                    key: { type: 'string' },\n                    cert: { type: 'string' },\n                  },\n                },\n              },\n            },\n            node: {\n              type: 'object',\n              required: [],\n              properties: {\n                externalPort: { type: 'number' },\n              },\n            },\n            network: {\n              type: 'object',\n              required: [],\n              properties: {\n                listenTCP: {\n                  type: 'object',\n                  required: ['port'],\n                  properties: {\n                    host: { type: 'string' },\n                    port: { type: 'number' },\n                  },\n                },\n\n                externalEndpoints: {\n                  type: 'array',\n                  items: { type: 'string' },\n                },\n\n                connectPeersDelayMS: { type: 'number' },\n                socketTimeoutMS: { type: 'number' },\n              },\n            },\n          },\n        },\n        options: {\n          type: 'object',\n          required: ['node', 'network', 'rpc'],\n          properties: {\n            node: {\n              type: 'object',\n              required: ['consensus', 'rpcURLs'],\n              properties: {\n                consensus: {\n                  type: 'object',\n                  required: ['enabled', 'options'],\n                  properties: {\n                    enabled: { type: 'boolean' },\n                    options: {\n                      type: 'object',\n                      required: ['privateKey', 'privateNet'],\n                      properties: {\n                        privateKey: { type: 'string' },\n                        privateNet: { type: 'boolean' },\n                      },\n                    },\n                  },\n                },\n                rpcURLs: { type: 'array', items: { type: 'string' } },\n              },\n            },\n            network: {\n              type: 'object',\n              required: ['seeds'],\n              properties: {\n                seeds: { type: 'array', items: { type: 'string' } },\n                maxConnectedPeers: { type: 'number' },\n              },\n            },\n            rpc: {\n              type: 'object',\n              required: ['server', 'liveHealthCheck', 'readyHealthCheck'],\n              properties: {\n                server: {\n                  type: 'object',\n                  required: ['keepAliveTimeout'],\n                  properties: {\n                    keepAliveTimeout: { type: 'number' },\n                  },\n                },\n                liveHealthCheck: {\n                  type: 'object',\n                  required: ['rpcURLs', 'offset', 'timeoutMS'],\n                  properties: {\n                    rpcURLs: { type: 'array', items: { type: 'string' } },\n                    offset: { type: 'number' },\n                    timeoutMS: { type: 'number' },\n                  },\n                },\n                readyHealthCheck: {\n                  type: 'object',\n                  required: ['rpcURLs', 'offset', 'timeoutMS'],\n                  properties: {\n                    rpcURLs: { type: 'array', items: { type: 'string' } },\n                    offset: { type: 'number' },\n                    timeoutMS: { type: 'number' },\n                  },\n                },\n              },\n            },\n          },\n        },\n      },\n    },\n    configPath: dataPath,\n  });\n\nexport class NEOONENodeAdapter extends NodeAdapter {\n  private mutableConfig: Config<NodeConfig> | undefined;\n  private mutableProcess: execa.ExecaChildProcess | undefined;\n\n  public constructor({\n    monitor,\n    name,\n    binary,\n    dataPath,\n    settings,\n  }: {\n    readonly monitor: Monitor;\n    readonly name: string;\n    readonly binary: Binary;\n    readonly dataPath: string;\n    readonly settings: NodeSettings;\n  }) {\n    super({\n      monitor: monitor.at('neo_one_node_adapter'),\n      name,\n      binary,\n      dataPath,\n      settings,\n    });\n  }\n\n  public getDebug(): DescribeTable {\n    return super\n      .getDebug()\n      .concat([\n        ['Process ID', this.mutableProcess === undefined ? 'null' : `${this.mutableProcess.pid}`],\n        ['Config Path', this.mutableConfig === undefined ? 'null' : this.mutableConfig.configPath],\n      ]);\n  }\n\n  public getNodeStatus(): NodeStatus {\n    return {\n      rpcAddress: this.getAddress('/rpc'),\n      tcpAddress: `localhost:${this.mutableSettings.listenTCPPort}`,\n      telemetryAddress: `http://localhost:${this.mutableSettings.telemetryPort}/metrics`,\n    };\n  }\n\n  protected async isLive(): Promise<boolean> {\n    return this.checkRPC('/live_health_check');\n  }\n\n  protected async isReady(): Promise<boolean> {\n    return this.checkRPC('/ready_health_check');\n  }\n\n  protected async createInternal(): Promise<void> {\n    await this.writeSettings(this.mutableSettings);\n  }\n\n  protected async updateInternal(settings: NodeSettings): Promise<void> {\n    const restart = await this.writeSettings(settings);\n    if (restart && this.mutableProcess !== undefined) {\n      await this.stop();\n      await this.start();\n    }\n  }\n\n  protected async startInternal(): Promise<void> {\n    if (this.mutableProcess === undefined) {\n      const child = execa(this.binary.cmd, this.binary.firstArgs.concat(['start', 'node', this.dataPath]), {\n        // @ts-ignore\n        windowsHide: true,\n        stdio: 'ignore',\n      });\n\n      this.mutableProcess = child;\n      // tslint:disable-next-line no-floating-promises\n      child\n        .then(() => {\n          this.monitor.log({\n            name: 'neo_node_adapter_node_exit',\n            message: 'Child process exited',\n          });\n\n          this.mutableProcess = undefined;\n        })\n        .catch((error: Error) => {\n          this.monitor.logError({\n            name: 'neo_node_adapter_node_error',\n            message: 'Child process exited with an error.',\n            error,\n          });\n\n          this.mutableProcess = undefined;\n        });\n    }\n  }\n\n  protected async stopInternal(): Promise<void> {\n    const child = this.mutableProcess;\n    this.mutableProcess = undefined;\n    if (child !== undefined) {\n      await killProcess(child.pid);\n    }\n  }\n\n  private async checkRPC(rpcPath: string): Promise<boolean> {\n    try {\n      const response = await fetch(this.getAddress(rpcPath));\n\n      return response.ok;\n    } catch (error) {\n      if (error.code !== 'ECONNREFUSED') {\n        this.monitor.withData({ [this.monitor.labels.HTTP_PATH]: rpcPath }).logError({\n          name: 'http_client_request',\n          message: 'Failed to check RPC.',\n          error,\n        });\n      }\n\n      return false;\n    }\n  }\n\n  private getAddress(rpcPath: string): string {\n    return `http://localhost:${this.mutableSettings.rpcPort}${rpcPath}`;\n  }\n\n  private async writeSettings(settings: NodeSettings): Promise<boolean> {\n    let config = this.mutableConfig;\n    if (config === undefined) {\n      config = createNodeConfig({\n        dataPath: this.dataPath,\n        defaultConfig: this.createConfig(settings),\n      });\n\n      this.mutableConfig = config;\n    }\n\n    const nodeConfig = await config.config$.pipe(take(1)).toPromise();\n    const newNodeConfig = this.createConfig(settings);\n    await fs.ensureDir(newNodeConfig.environment.dataPath);\n    await config.update({ config: newNodeConfig });\n\n    return !(\n      _.isEqual(nodeConfig.settings, newNodeConfig.settings) &&\n      _.isEqual(nodeConfig.environment, newNodeConfig.environment)\n    );\n  }\n\n  private createConfig(settings: NodeSettings): NodeConfig {\n    return {\n      log: {\n        level: 'info',\n        maxSize: 10 * 1024 * 1024,\n        maxFiles: 5,\n      },\n      settings: {\n        test: settings.isTestNet,\n        privateNet: settings.privateNet,\n        secondsPerBlock: settings.secondsPerBlock,\n        standbyValidators: settings.standbyValidators,\n        address: settings.address,\n      },\n      environment: {\n        dataPath: path.resolve(this.dataPath, 'chain'),\n        rpc: {\n          http: {\n            port: settings.rpcPort,\n            host: '0.0.0.0',\n          },\n        },\n        node: {\n          externalPort: settings.listenTCPPort,\n        },\n        network: {\n          listenTCP: {\n            port: settings.listenTCPPort,\n            host: '0.0.0.0',\n          },\n        },\n        telemetry: {\n          port: settings.telemetryPort,\n        },\n      },\n      options: {\n        node: {\n          consensus: settings.consensus,\n          rpcURLs: settings.rpcEndpoints,\n        },\n        network: {\n          seeds: settings.seeds,\n        },\n        rpc: {\n          server: {\n            keepAliveTimeout: 60000,\n          },\n          liveHealthCheck: {\n            rpcURLs: settings.rpcEndpoints,\n            offset: 1,\n            timeoutMS: 5000,\n          },\n          readyHealthCheck: {\n            rpcURLs: settings.rpcEndpoints,\n            offset: 1,\n            timeoutMS: 5000,\n          },\n        },\n      },\n    };\n  }\n}\n"]}
|