UNPKG

21.2 kBJavaScriptView Raw
1import { labels, utils } from '@neo-one/utils-esnext-esm';
2import { concat, of as _of, timer } from 'rxjs';
3import { concatMap, shareReplay } from 'rxjs/operators';
4export class NodeAdapter {
5 constructor({ monitor, name, binary, dataPath, settings, }) {
6 this.monitor = monitor;
7 this.name = name;
8 this.binary = binary;
9 this.dataPath = dataPath;
10 this.mutableSettings = settings;
11 const { rpcAddress, tcpAddress, telemetryAddress } = this.getNodeStatus();
12 this.node$ = concat(_of({
13 name: this.name,
14 ready: false,
15 live: false,
16 rpcAddress,
17 tcpAddress,
18 telemetryAddress,
19 }), timer(0, 500).pipe(concatMap(async () => {
20 const config = this.getNodeStatus();
21 const [ready, live] = await Promise.all([this.isReady(), this.isLive()]);
22 return {
23 name: this.name,
24 ready,
25 live,
26 rpcAddress: config.rpcAddress,
27 tcpAddress: config.tcpAddress,
28 telemetryAddress: config.telemetryAddress,
29 };
30 }))).pipe(shareReplay(1));
31 }
32 getDebug() {
33 return [
34 ['Data Path', this.dataPath],
35 [
36 'Settings',
37 {
38 type: 'describe',
39 table: [
40 ['Type', this.mutableSettings.type],
41 [
42 'Is Test Net',
43 this.mutableSettings.isTestNet === undefined ? "'null'" : JSON.stringify(this.mutableSettings.isTestNet),
44 ],
45 [
46 'Seconds Per Block',
47 this.mutableSettings.secondsPerBlock === undefined
48 ? "'null'"
49 : JSON.stringify(this.mutableSettings.secondsPerBlock),
50 ],
51 [
52 'Standby Validators',
53 this.mutableSettings.standbyValidators === undefined
54 ? "'null'"
55 : JSON.stringify(this.mutableSettings.standbyValidators, undefined, 2),
56 ],
57 [
58 'Address',
59 this.mutableSettings.address === undefined ? "'null'" : JSON.stringify(this.mutableSettings.address),
60 ],
61 ['RPC Port', JSON.stringify(this.mutableSettings.rpcPort)],
62 ['Listen TCP Port', JSON.stringify(this.mutableSettings.listenTCPPort)],
63 ['Telemetry Port', JSON.stringify(this.mutableSettings.telemetryPort)],
64 ['Consensus Enabled', this.mutableSettings.consensus.enabled ? 'Yes' : 'No'],
65 ['Consensus Private Key', this.mutableSettings.consensus.options.privateKey],
66 ['Seeds', JSON.stringify(this.mutableSettings.seeds, undefined, 2)],
67 ['RPC Endpoints', JSON.stringify(this.mutableSettings.rpcEndpoints, undefined, 2)],
68 ],
69 },
70 ],
71 ];
72 }
73 async create() {
74 await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(async () => this.createInternal(), {
75 name: 'neo_node_adapter_create',
76 message: `Created node ${this.name}`,
77 error: `Failed to create node ${this.name}`,
78 });
79 }
80 async update(settings) {
81 await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(async () => {
82 await this.updateInternal(settings);
83 this.mutableSettings = settings;
84 }, {
85 name: 'neo_node_adapter_update',
86 message: `Updated node ${this.name}`,
87 error: `Failed to update node ${this.name}`,
88 });
89 }
90 async start() {
91 await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(async () => {
92 await this.startInternal();
93 }, {
94 name: 'neo_node_adapter_start',
95 message: `Started node ${this.name}`,
96 error: `Failed to start node ${this.name}`,
97 });
98 }
99 async stop() {
100 await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(async () => {
101 await this.stopInternal();
102 }, {
103 name: 'neo_node_adapter_stop',
104 message: `Stopped node ${this.name}`,
105 error: `Failed to stop node ${this.name}`,
106 });
107 }
108 async live(timeoutSeconds) {
109 const start = utils.nowSeconds();
110 // tslint:disable-next-line no-loop-statement
111 while (utils.nowSeconds() - start < timeoutSeconds) {
112 const isLive = await this.isLive();
113 if (isLive) {
114 return;
115 }
116 await new Promise((resolve) => setTimeout(resolve, 50));
117 }
118 throw new Error(`Node ${this.name} did not start.`);
119 }
120 async ready(timeoutSeconds) {
121 const start = utils.nowSeconds();
122 // tslint:disable-next-line no-loop-statement
123 while (utils.nowSeconds() - start < timeoutSeconds) {
124 const isLive = await this.isReady();
125 if (isLive) {
126 return;
127 }
128 await new Promise((resolve) => setTimeout(resolve, 50));
129 }
130 throw new Error(`Node ${this.name} is not ready.`);
131 }
132}
133
134//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["NodeAdapter.ts"],"names":[],"mappings":"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","file":"neo-one-server-plugin-network/src/node/NodeAdapter.js","sourcesContent":["import { Monitor } from '@neo-one/monitor';\nimport { Binary, DescribeTable } from '@neo-one/server-plugin';\nimport { labels, utils } from '@neo-one/utils';\nimport { concat, Observable, of as _of, timer } from 'rxjs';\nimport { concatMap, shareReplay } from 'rxjs/operators';\nimport { NodeSettings } from '../types';\n\nexport interface Node {\n  readonly name: string;\n  readonly live: boolean;\n  readonly ready: boolean;\n  readonly rpcAddress: string;\n  readonly tcpAddress: string;\n  readonly telemetryAddress: string;\n}\n\nexport interface NodeStatus {\n  readonly rpcAddress: string;\n  readonly tcpAddress: string;\n  readonly telemetryAddress: string;\n}\n\nexport abstract class NodeAdapter {\n  public readonly name: string;\n  public readonly node$: Observable<Node>;\n  protected readonly binary: Binary;\n  protected readonly dataPath: string;\n  protected readonly monitor: Monitor;\n  protected mutableSettings: NodeSettings;\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    this.monitor = monitor;\n    this.name = name;\n    this.binary = binary;\n    this.dataPath = dataPath;\n\n    this.mutableSettings = settings;\n\n    const { rpcAddress, tcpAddress, telemetryAddress } = this.getNodeStatus();\n    this.node$ = concat(\n      _of({\n        name: this.name,\n        ready: false,\n        live: false,\n        rpcAddress,\n        tcpAddress,\n        telemetryAddress,\n      }),\n      timer(0, 500).pipe(\n        concatMap(async () => {\n          const config = this.getNodeStatus();\n          const [ready, live] = await Promise.all([this.isReady(), this.isLive()]);\n\n          return {\n            name: this.name,\n            ready,\n            live,\n            rpcAddress: config.rpcAddress,\n            tcpAddress: config.tcpAddress,\n            telemetryAddress: config.telemetryAddress,\n          };\n        }),\n      ),\n    ).pipe(shareReplay(1));\n  }\n\n  public getDebug(): DescribeTable {\n    return [\n      ['Data Path', this.dataPath],\n      [\n        'Settings',\n        {\n          type: 'describe',\n          table: [\n            ['Type', this.mutableSettings.type],\n            [\n              'Is Test Net',\n              this.mutableSettings.isTestNet === undefined ? \"'null'\" : JSON.stringify(this.mutableSettings.isTestNet),\n            ],\n            [\n              'Seconds Per Block',\n              this.mutableSettings.secondsPerBlock === undefined\n                ? \"'null'\"\n                : JSON.stringify(this.mutableSettings.secondsPerBlock),\n            ],\n            [\n              'Standby Validators',\n              this.mutableSettings.standbyValidators === undefined\n                ? \"'null'\"\n                : JSON.stringify(this.mutableSettings.standbyValidators, undefined, 2),\n            ],\n            [\n              'Address',\n              this.mutableSettings.address === undefined ? \"'null'\" : JSON.stringify(this.mutableSettings.address),\n            ],\n            ['RPC Port', JSON.stringify(this.mutableSettings.rpcPort)],\n            ['Listen TCP Port', JSON.stringify(this.mutableSettings.listenTCPPort)],\n            ['Telemetry Port', JSON.stringify(this.mutableSettings.telemetryPort)],\n            ['Consensus Enabled', this.mutableSettings.consensus.enabled ? 'Yes' : 'No'],\n            ['Consensus Private Key', this.mutableSettings.consensus.options.privateKey],\n            ['Seeds', JSON.stringify(this.mutableSettings.seeds, undefined, 2)],\n            ['RPC Endpoints', JSON.stringify(this.mutableSettings.rpcEndpoints, undefined, 2)],\n          ],\n        },\n      ],\n    ];\n  }\n\n  public async create(): Promise<void> {\n    await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(async () => this.createInternal(), {\n      name: 'neo_node_adapter_create',\n      message: `Created node ${this.name}`,\n      error: `Failed to create node ${this.name}`,\n    });\n  }\n\n  public async update(settings: NodeSettings): Promise<void> {\n    await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(\n      async () => {\n        await this.updateInternal(settings);\n        this.mutableSettings = settings;\n      },\n      {\n        name: 'neo_node_adapter_update',\n        message: `Updated node ${this.name}`,\n        error: `Failed to update node ${this.name}`,\n      },\n    );\n  }\n\n  public async start(): Promise<void> {\n    await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(\n      async () => {\n        await this.startInternal();\n      },\n      {\n        name: 'neo_node_adapter_start',\n        message: `Started node ${this.name}`,\n        error: `Failed to start node ${this.name}`,\n      },\n    );\n  }\n\n  public async stop(): Promise<void> {\n    await this.monitor.withData({ [labels.NODE_NAME]: this.name }).captureLog(\n      async () => {\n        await this.stopInternal();\n      },\n      {\n        name: 'neo_node_adapter_stop',\n        message: `Stopped node ${this.name}`,\n        error: `Failed to stop node ${this.name}`,\n      },\n    );\n  }\n\n  public abstract getNodeStatus(): NodeStatus;\n\n  public async live(timeoutSeconds: number): Promise<void> {\n    const start = utils.nowSeconds();\n    // tslint:disable-next-line no-loop-statement\n    while (utils.nowSeconds() - start < timeoutSeconds) {\n      const isLive = await this.isLive();\n      if (isLive) {\n        return;\n      }\n\n      await new Promise<void>((resolve) => setTimeout(resolve, 50));\n    }\n\n    throw new Error(`Node ${this.name} did not start.`);\n  }\n\n  public async ready(timeoutSeconds: number): Promise<void> {\n    const start = utils.nowSeconds();\n    // tslint:disable-next-line no-loop-statement\n    while (utils.nowSeconds() - start < timeoutSeconds) {\n      const isLive = await this.isReady();\n      if (isLive) {\n        return;\n      }\n\n      await new Promise<void>((resolve) => setTimeout(resolve, 50));\n    }\n\n    throw new Error(`Node ${this.name} is not ready.`);\n  }\n\n  protected abstract async isLive(): Promise<boolean>;\n  protected abstract async isReady(): Promise<boolean>;\n  protected abstract async createInternal(): Promise<void>;\n  protected abstract async updateInternal(_settings: NodeSettings): Promise<void>;\n  protected abstract async startInternal(): Promise<void>;\n  protected abstract async stopInternal(): Promise<void>;\n}\n"]}