UNPKG

48.2 kBJavaScriptView Raw
1import { createEndpoint } from '@neo-one/node-core-esnext-esm';
2import { Config, killProcess } from '@neo-one/server-plugin-esnext-esm';
3import fetch from 'cross-fetch';
4import execa from 'execa';
5import * as fs from 'fs-extra';
6import _ from 'lodash';
7import * as path from 'path';
8import { take } from 'rxjs/operators';
9import { NodeAdapter } from './NodeAdapter';
10const 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];
21const 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];
32const 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});
75export 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});
230export 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":"AAEA,OAAO,EAAE,cAAc,EAAkB,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAU,MAAM,EAAiB,WAAW,EAAE,MAAM,wBAAwB,CAAC;AACpF,OAAO,KAAK,MAAM,aAAa,CAAC;AAChC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,EAAE,MAAM,UAAU,CAAC;AAC/B,OAAO,CAAC,MAAM,QAAQ,CAAC;AACvB,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAC7B,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;AAEtC,OAAO,EAAE,WAAW,EAAc,MAAM,eAAe,CAAC;AAqBxD,MAAM,gBAAgB,GAA0B;IAC9C,2CAA2C;IAC3C,2CAA2C;IAC3C,sCAAsC;IACtC,sCAAsC;IACtC,sCAAsC;IACtC,qCAAqC;IACrC,qCAAqC;IACrC,qCAAqC;IACrC,qCAAqC;CACtC,CAAC;AAEF,MAAM,aAAa,GAAkC;IACnD,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,8BAA8B,EAAE,IAAI,EAAE,KAAK,EAAE;IAClE,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,8BAA8B,EAAE,IAAI,EAAE,KAAK,EAAE;IAClE,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,oBAAoB,EAAE,IAAI,EAAE,KAAK,EAAE;IACxD,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,oBAAoB,EAAE,IAAI,EAAE,KAAK,EAAE;IACxD,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,oBAAoB,EAAE,IAAI,EAAE,KAAK,EAAE;IACxD,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,wBAAwB,EAAE,IAAI,EAAE,KAAK,EAAE;IAC5D,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,wBAAwB,EAAE,IAAI,EAAE,KAAK,EAAE;IAC5D,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,wBAAwB,EAAE,IAAI,EAAE,KAAK,EAAE;IAC5D,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,wBAAwB,EAAE,IAAI,EAAE,KAAK,EAAE;CAC7D,CAAC;AAEF,MAAM,iBAAiB,GAAG,CAAC,QAAgB,EAAc,EAAE,CAAC,CAAC;IAC3D,GAAG,EAAE;QACH,KAAK,EAAE,MAAM;QACb,OAAO,EAAE,EAAE,GAAG,IAAI,GAAG,IAAI;QACzB,QAAQ,EAAE,CAAC;KACZ;IACD,QAAQ,EAAE;QACR,IAAI,EAAE,KAAK;KACZ;IACD,WAAW,EAAE;QACX,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,MAAM,CAAC;QACxC,GAAG,EAAE,EAAE;QACP,IAAI,EAAE,EAAE;QACR,OAAO,EAAE,EAAE;KACZ;IACD,OAAO,EAAE;QACP,IAAI,EAAE;YACJ,SAAS,EAAE;gBACT,OAAO,EAAE,KAAK;gBACd,OAAO,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,EAAE;aACtD;YACD,OAAO,EAAE,CAAC,GAAG,gBAAgB,CAAC;SAC/B;QACD,OAAO,EAAE;YACP,KAAK,EAAE,aAAa,CAAC,GAAG,CAAC,cAAc,CAAC;SACzC;QACD,GAAG,EAAE;YACH,MAAM,EAAE;gBACN,gBAAgB,EAAE,KAAK;aACxB;YACD,eAAe,EAAE;gBACf,OAAO,EAAE,gBAAgB;gBACzB,MAAM,EAAE,CAAC;gBACT,SAAS,EAAE,IAAI;aAChB;YACD,gBAAgB,EAAE;gBAChB,OAAO,EAAE,gBAAgB;gBACzB,MAAM,EAAE,CAAC;gBACT,SAAS,EAAE,IAAI;aAChB;SACF;KACF;CACF,CAAC,CAAC;AAEH,MAAM,CAAC,MAAM,gBAAgB,GAAG,CAAC,EAC/B,QAAQ,EACR,aAAa,GAAG,iBAAiB,CAAC,QAAQ,CAAC,GAI5C,EAAsB,EAAE,CACvB,IAAI,MAAM,CAAC;IACT,IAAI,EAAE,MAAM;IACZ,aAAa;IACb,MAAM,EAAE;QACN,IAAI,EAAE,QAAQ;QACd,QAAQ,EAAE,CAAC,KAAK,CAAC;QACjB,UAAU,EAAE;YACV,GAAG,EAAE;gBACH,IAAI,EAAE,QAAQ;gBACd,QAAQ,EAAE,CAAC,OAAO,EAAE,SAAS,EAAE,UAAU,CAAC;gBAC1C,UAAU,EAAE;oBACV,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oBACzB,OAAO,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oBAC3B,QAAQ,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iBAC7B;aACF;YACD,QAAQ,EAAE;gBACR,IAAI,EAAE,QAAQ;gBACd,QAAQ,EAAE,CAAC,MAAM,CAAC;gBAClB,UAAU,EAAE;oBACV,IAAI,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE;oBACzB,UAAU,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE;oBAC/B,eAAe,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oBACnC,iBAAiB,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE;iBAChE;aACF;YACD,WAAW,EAAE;gBACX,IAAI,EAAE,QAAQ;gBACd,QAAQ,EAAE,CAAC,UAAU,EAAE,KAAK,EAAE,MAAM,EAAE,SAAS,CAAC;gBAChD,UAAU,EAAE;oBACV,QAAQ,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oBAC5B,GAAG,EAAE;wBACH,IAAI,EAAE,QAAQ;wBACd,QAAQ,EAAE,EAAE;wBACZ,UAAU,EAAE;4BACV,IAAI,EAAE;gCACJ,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,MAAM,EAAE,MAAM,CAAC;gCAC1B,UAAU,EAAE;oCACV,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCACxB,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iCACzB;6BACF;4BAED,KAAK,EAAE;gCACL,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,CAAC;gCACzC,UAAU,EAAE;oCACV,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCACxB,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCACxB,GAAG,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCACvB,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iCACzB;6BACF;yBACF;qBACF;oBACD,IAAI,EAAE;wBACJ,IAAI,EAAE,QAAQ;wBACd,QAAQ,EAAE,EAAE;wBACZ,UAAU,EAAE;4BACV,YAAY,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;yBACjC;qBACF;oBACD,OAAO,EAAE;wBACP,IAAI,EAAE,QAAQ;wBACd,QAAQ,EAAE,EAAE;wBACZ,UAAU,EAAE;4BACV,SAAS,EAAE;gCACT,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,MAAM,CAAC;gCAClB,UAAU,EAAE;oCACV,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCACxB,IAAI,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iCACzB;6BACF;4BAED,iBAAiB,EAAE;gCACjB,IAAI,EAAE,OAAO;gCACb,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;6BAC1B;4BAED,mBAAmB,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;4BACvC,eAAe,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;yBACpC;qBACF;iBACF;aACF;YACD,OAAO,EAAE;gBACP,IAAI,EAAE,QAAQ;gBACd,QAAQ,EAAE,CAAC,MAAM,EAAE,SAAS,EAAE,KAAK,CAAC;gBACpC,UAAU,EAAE;oBACV,IAAI,EAAE;wBACJ,IAAI,EAAE,QAAQ;wBACd,QAAQ,EAAE,CAAC,WAAW,EAAE,SAAS,CAAC;wBAClC,UAAU,EAAE;4BACV,SAAS,EAAE;gCACT,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,SAAS,EAAE,SAAS,CAAC;gCAChC,UAAU,EAAE;oCACV,OAAO,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE;oCAC5B,OAAO,EAAE;wCACP,IAAI,EAAE,QAAQ;wCACd,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,CAAC;wCACtC,UAAU,EAAE;4CACV,UAAU,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;4CAC9B,UAAU,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE;yCAChC;qCACF;iCACF;6BACF;4BACD,OAAO,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE;yBACtD;qBACF;oBACD,OAAO,EAAE;wBACP,IAAI,EAAE,QAAQ;wBACd,QAAQ,EAAE,CAAC,OAAO,CAAC;wBACnB,UAAU,EAAE;4BACV,KAAK,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE;4BACnD,iBAAiB,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;yBACtC;qBACF;oBACD,GAAG,EAAE;wBACH,IAAI,EAAE,QAAQ;wBACd,QAAQ,EAAE,CAAC,QAAQ,EAAE,iBAAiB,EAAE,kBAAkB,CAAC;wBAC3D,UAAU,EAAE;4BACV,MAAM,EAAE;gCACN,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,kBAAkB,CAAC;gCAC9B,UAAU,EAAE;oCACV,gBAAgB,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iCACrC;6BACF;4BACD,eAAe,EAAE;gCACf,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,SAAS,EAAE,QAAQ,EAAE,WAAW,CAAC;gCAC5C,UAAU,EAAE;oCACV,OAAO,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE;oCACrD,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCAC1B,SAAS,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iCAC9B;6BACF;4BACD,gBAAgB,EAAE;gCAChB,IAAI,EAAE,QAAQ;gCACd,QAAQ,EAAE,CAAC,SAAS,EAAE,QAAQ,EAAE,WAAW,CAAC;gCAC5C,UAAU,EAAE;oCACV,OAAO,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE;oCACrD,MAAM,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;oCAC1B,SAAS,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE;iCAC9B;6BACF;yBACF;qBACF;iBACF;aACF;SACF;KACF;IACD,UAAU,EAAE,QAAQ;CACrB,CAAC,CAAC;AAEL,MAAM,OAAO,iBAAkB,SAAQ,WAAW;IAIhD,YAAmB,EACjB,OAAO,EACP,IAAI,EACJ,MAAM,EACN,QAAQ,EACR,QAAQ,GAOT;QACC,KAAK,CAAC;YACJ,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC,sBAAsB,CAAC;YAC3C,IAAI;YACJ,MAAM;YACN,QAAQ;YACR,QAAQ;SACT,CAAC,CAAC;IACL,CAAC;IAEM,QAAQ;QACb,OAAO,KAAK;aACT,QAAQ,EAAE;aACV,MAAM,CAAC;YACN,CAAC,YAAY,EAAE,IAAI,CAAC,cAAc,KAAK,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,EAAE,CAAC;YACzF,CAAC,aAAa,EAAE,IAAI,CAAC,aAAa,KAAK,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC;SAC3F,CAAC,CAAC;IACP,CAAC;IAEM,aAAa;QAClB,OAAO;YACL,UAAU,EAAE,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC;YACnC,UAAU,EAAE,aAAa,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE;YAC7D,gBAAgB,EAAE,oBAAoB,IAAI,CAAC,eAAe,CAAC,aAAa,UAAU;SACnF,CAAC;IACJ,CAAC;IAES,KAAK,CAAC,MAAM;QACpB,OAAO,IAAI,CAAC,QAAQ,CAAC,oBAAoB,CAAC,CAAC;IAC7C,CAAC;IAES,KAAK,CAAC,OAAO;QACrB,OAAO,IAAI,CAAC,QAAQ,CAAC,qBAAqB,CAAC,CAAC;IAC9C,CAAC;IAES,KAAK,CAAC,cAAc;QAC5B,MAAM,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IACjD,CAAC;IAES,KAAK,CAAC,cAAc,CAAC,QAAsB;QACnD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QACnD,IAAI,OAAO,IAAI,IAAI,CAAC,cAAc,KAAK,SAAS,EAAE;YAChD,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;YAClB,MAAM,IAAI,CAAC,KAAK,EAAE,CAAC;SACpB;IACH,CAAC;IAES,KAAK,CAAC,aAAa;QAC3B,IAAI,IAAI,CAAC,cAAc,KAAK,SAAS,EAAE;YACrC,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,EAAE;gBACnG,aAAa;gBACb,WAAW,EAAE,IAAI;gBACjB,KAAK,EAAE,QAAQ;aAChB,CAAC,CAAC;YAEH,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;YAC5B,gDAAgD;YAChD,KAAK;iBACF,IAAI,CAAC,GAAG,EAAE;gBACT,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC;oBACf,IAAI,EAAE,4BAA4B;oBAClC,OAAO,EAAE,sBAAsB;iBAChC,CAAC,CAAC;gBAEH,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;YAClC,CAAC,CAAC;iBACD,KAAK,CAAC,CAAC,KAAY,EAAE,EAAE;gBACtB,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC;oBACpB,IAAI,EAAE,6BAA6B;oBACnC,OAAO,EAAE,qCAAqC;oBAC9C,KAAK;iBACN,CAAC,CAAC;gBAEH,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;YAClC,CAAC,CAAC,CAAC;SACN;IACH,CAAC;IAES,KAAK,CAAC,YAAY;QAC1B,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC;QAClC,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;QAChC,IAAI,KAAK,KAAK,SAAS,EAAE;YACvB,MAAM,WAAW,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;SAC9B;IACH,CAAC;IAEO,KAAK,CAAC,QAAQ,CAAC,OAAe;QACpC,IAAI;YACF,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;YAEvD,OAAO,QAAQ,CAAC,EAAE,CAAC;SACpB;QAAC,OAAO,KAAK,EAAE;YACd,IAAI,KAAK,CAAC,IAAI,KAAK,cAAc,EAAE;gBACjC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC,QAAQ,CAAC;oBAC3E,IAAI,EAAE,qBAAqB;oBAC3B,OAAO,EAAE,sBAAsB;oBAC/B,KAAK;iBACN,CAAC,CAAC;aACJ;YAED,OAAO,KAAK,CAAC;SACd;IACH,CAAC;IAEO,UAAU,CAAC,OAAe;QAChC,OAAO,oBAAoB,IAAI,CAAC,eAAe,CAAC,OAAO,GAAG,OAAO,EAAE,CAAC;IACtE,CAAC;IAEO,KAAK,CAAC,aAAa,CAAC,QAAsB;QAChD,IAAI,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC;QAChC,IAAI,MAAM,KAAK,SAAS,EAAE;YACxB,MAAM,GAAG,gBAAgB,CAAC;gBACxB,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,aAAa,EAAE,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC;aAC3C,CAAC,CAAC;YAEH,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;SAC7B;QAED,MAAM,UAAU,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC;QAClE,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;QAClD,MAAM,EAAE,CAAC,SAAS,CAAC,aAAa,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QACvD,MAAM,MAAM,CAAC,MAAM,CAAC,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC,CAAC;QAE/C,OAAO,CAAC,CACN,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,QAAQ,EAAE,aAAa,CAAC,QAAQ,CAAC;YACtD,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,WAAW,EAAE,aAAa,CAAC,WAAW,CAAC,CAC7D,CAAC;IACJ,CAAC;IAEO,YAAY,CAAC,QAAsB;QACzC,OAAO;YACL,GAAG,EAAE;gBACH,KAAK,EAAE,MAAM;gBACb,OAAO,EAAE,EAAE,GAAG,IAAI,GAAG,IAAI;gBACzB,QAAQ,EAAE,CAAC;aACZ;YACD,QAAQ,EAAE;gBACR,IAAI,EAAE,QAAQ,CAAC,SAAS;gBACxB,UAAU,EAAE,QAAQ,CAAC,UAAU;gBAC/B,eAAe,EAAE,QAAQ,CAAC,eAAe;gBACzC,iBAAiB,EAAE,QAAQ,CAAC,iBAAiB;gBAC7C,OAAO,EAAE,QAAQ,CAAC,OAAO;aAC1B;YACD,WAAW,EAAE;gBACX,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;gBAC9C,GAAG,EAAE;oBACH,IAAI,EAAE;wBACJ,IAAI,EAAE,QAAQ,CAAC,OAAO;wBACtB,IAAI,EAAE,SAAS;qBAChB;iBACF;gBACD,IAAI,EAAE;oBACJ,YAAY,EAAE,QAAQ,CAAC,aAAa;iBACrC;gBACD,OAAO,EAAE;oBACP,SAAS,EAAE;wBACT,IAAI,EAAE,QAAQ,CAAC,aAAa;wBAC5B,IAAI,EAAE,SAAS;qBAChB;iBACF;gBACD,SAAS,EAAE;oBACT,IAAI,EAAE,QAAQ,CAAC,aAAa;iBAC7B;aACF;YACD,OAAO,EAAE;gBACP,IAAI,EAAE;oBACJ,SAAS,EAAE,QAAQ,CAAC,SAAS;oBAC7B,OAAO,EAAE,QAAQ,CAAC,YAAY;iBAC/B;gBACD,OAAO,EAAE;oBACP,KAAK,EAAE,QAAQ,CAAC,KAAK;iBACtB;gBACD,GAAG,EAAE;oBACH,MAAM,EAAE;wBACN,gBAAgB,EAAE,KAAK;qBACxB;oBACD,eAAe,EAAE;wBACf,OAAO,EAAE,QAAQ,CAAC,YAAY;wBAC9B,MAAM,EAAE,CAAC;wBACT,SAAS,EAAE,IAAI;qBAChB;oBACD,gBAAgB,EAAE;wBAChB,OAAO,EAAE,QAAQ,CAAC,YAAY;wBAC9B,MAAM,EAAE,CAAC;wBACT,SAAS,EAAE,IAAI;qBAChB;iBACF;aACF;SACF,CAAC;IACJ,CAAC;CACF","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"]}