UNPKG

22.5 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const tslib_1 = require("tslib");
4const client_common_1 = require("@neo-one/client-common");
5const node_core_1 = require("@neo-one/node-core");
6const utils_1 = require("@neo-one/utils");
7const bignumber_js_1 = tslib_1.__importDefault(require("bignumber.js"));
8const lodash_1 = tslib_1.__importDefault(require("lodash"));
9const common_1 = require("./common");
10const context_1 = require("./context");
11const createMinerTransaction = async ({ node, feeAddress, transactions, nonce, }) => {
12 const networkFee = await node_core_1.Block.calculateNetworkFee(node.blockchain.feeContext, transactions);
13 const outputs = networkFee.isZero()
14 ? []
15 : [
16 new node_core_1.Output({
17 asset: node.blockchain.settings.utilityToken.hash,
18 value: networkFee,
19 address: feeAddress,
20 }),
21 ];
22 return new node_core_1.MinerTransaction({
23 nonce: nonce.mod(node_core_1.utils.UINT_MAX.addn(1)).toNumber(),
24 outputs,
25 });
26};
27const requestChangeView = ({ context: contextIn, node, privateKey, consensusContext, }) => {
28 let context = contextIn;
29 context = context.cloneExpectedView({
30 expectedView: common_1.incrementExpectedView(context),
31 });
32 common_1.signAndRelayChangeView({ context, node, privateKey });
33 const viewNumber = context.expectedView[context.myIndex];
34 if (common_1.checkExpectedView({ context, viewNumber })) {
35 return common_1.initializeConsensusInitial({
36 blockchain: node.blockchain,
37 context,
38 viewNumber,
39 consensusContext,
40 });
41 }
42 const { secondsPerBlock } = node.blockchain.settings;
43 return {
44 context,
45 timerSeconds: secondsPerBlock << (viewNumber + 1),
46 };
47};
48exports.runConsensus = async ({ context, node, options: { privateKey, feeAddress, privateNet }, consensusContext, }) => {
49 if (context.type === 'primary' && !(context instanceof context_1.RequestSentContext)) {
50 let requestSentContext;
51 if (context instanceof context_1.SignatureSentContext) {
52 requestSentContext = context.cloneRequestSent();
53 }
54 else {
55 const nonce = node_core_1.utils.randomUInt64();
56 let mutableTransactions = Object.values(node.memPool);
57 const minerTransaction = await createMinerTransaction({
58 node,
59 feeAddress,
60 transactions: mutableTransactions,
61 nonce,
62 });
63 if (mutableTransactions.length >= node.blockchain.settings.maxTransactionsPerBlock) {
64 const mutableNetworkFees = await Promise.all(mutableTransactions.map(async (transaction) => {
65 const networkFee = await transaction.getNetworkFee(node.blockchain.feeContext);
66 return [transaction, new bignumber_js_1.default(networkFee.toString(10))];
67 }));
68 mutableNetworkFees.sort(([first, a], [second, b]) => b.div(second.size).comparedTo(a.div(first.size)));
69 mutableTransactions = lodash_1.default.take(mutableNetworkFees, node.blockchain.settings.maxTransactionsPerBlock - 1)
70 .map(([transaction, _unused]) => transaction);
71 }
72 mutableTransactions.unshift(minerTransaction);
73 const [previousHeader, validators] = await Promise.all([
74 node.blockchain.header.get({ hashOrIndex: context.previousHash }),
75 node.blockchain.getValidators(mutableTransactions),
76 ]);
77 const newContext = new context_1.RequestSentContext({
78 viewNumber: context.viewNumber,
79 myIndex: context.myIndex,
80 primaryIndex: context.primaryIndex,
81 expectedView: context.expectedView,
82 validators: context.validators,
83 blockReceivedTimeSeconds: context.blockReceivedTimeSeconds,
84 transactions: mutableTransactions.reduce((acc, transaction) => (Object.assign(Object.assign({}, acc), { [transaction.hashHex]: transaction })), {}),
85 signatures: [],
86 header: {
87 type: 'new',
88 previousHash: context.previousHash,
89 transactionHashes: mutableTransactions.map((transaction) => transaction.hashHex),
90 blockIndex: context.blockIndex,
91 nonce,
92 timestamp: Math.max(consensusContext.nowSeconds(), previousHeader.timestamp + 1),
93 nextConsensus: client_common_1.crypto.getConsensusAddress(validators),
94 },
95 });
96 const mutableSignatures = [];
97 mutableSignatures[newContext.myIndex] = client_common_1.crypto.sign({
98 message: newContext.header.message,
99 privateKey,
100 });
101 requestSentContext = newContext.cloneSignatures({ signatures: mutableSignatures });
102 }
103 if (privateNet) {
104 return common_1.checkSignatures({ node, context: requestSentContext });
105 }
106 common_1.signAndRelay({
107 context: requestSentContext,
108 node,
109 privateKey,
110 consensusMessage: new node_core_1.PrepareRequestConsensusMessage({
111 viewNumber: requestSentContext.viewNumber,
112 nonce: requestSentContext.header.consensusData,
113 nextConsensus: requestSentContext.header.nextConsensus,
114 transactionHashes: requestSentContext.transactionHashes.map((hash) => client_common_1.common.hexToUInt256(hash)),
115 minerTransaction: utils_1.utils.nullthrows(requestSentContext.transactions[requestSentContext.transactionHashes[0]]),
116 signature: utils_1.utils.nullthrows(requestSentContext.signatures[requestSentContext.myIndex]),
117 }),
118 });
119 const { secondsPerBlock } = node.blockchain.settings;
120 return {
121 context: requestSentContext,
122 timerSeconds: secondsPerBlock << (requestSentContext.viewNumber + 1),
123 };
124 }
125 if (context instanceof context_1.RequestSentContext || context.type === 'backup') {
126 return requestChangeView({
127 context,
128 node,
129 privateKey,
130 consensusContext,
131 });
132 }
133 return { context };
134};
135
136//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["runConsensus.ts"],"names":[],"mappings":"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","file":"neo-one-node-consensus/src/runConsensus.js","sourcesContent":["import { common, crypto, PrivateKey, UInt160 } from '@neo-one/client-common';\nimport {\n  Block,\n  MinerTransaction,\n  Node,\n  Output,\n  PrepareRequestConsensusMessage,\n  Transaction,\n  utils,\n} from '@neo-one/node-core';\nimport { utils as commonUtils } from '@neo-one/utils';\nimport BigNumber from 'bignumber.js';\nimport { BN } from 'bn.js';\nimport _ from 'lodash';\nimport {\n  checkExpectedView,\n  checkSignatures,\n  incrementExpectedView,\n  initializeConsensusInitial,\n  signAndRelay,\n  signAndRelayChangeView,\n} from './common';\nimport { InternalOptions } from './Consensus';\nimport { ConsensusContext } from './ConsensusContext';\nimport { Context, RequestSentContext, SignatureSentContext } from './context';\nimport { Result } from './types';\n\nconst createMinerTransaction = async ({\n  node,\n  feeAddress,\n  transactions,\n  nonce,\n}: {\n  readonly node: Node;\n  readonly feeAddress: UInt160;\n  readonly transactions: readonly Transaction[];\n  readonly nonce: BN;\n}) => {\n  const networkFee = await Block.calculateNetworkFee(node.blockchain.feeContext, transactions);\n\n  const outputs = networkFee.isZero()\n    ? []\n    : [\n        new Output({\n          asset: node.blockchain.settings.utilityToken.hash,\n          value: networkFee,\n          address: feeAddress,\n        }),\n      ];\n\n  return new MinerTransaction({\n    nonce: nonce.mod(utils.UINT_MAX.addn(1)).toNumber(),\n    outputs,\n  });\n};\n\nconst requestChangeView = ({\n  context: contextIn,\n  node,\n  privateKey,\n  consensusContext,\n}: {\n  readonly context: Context;\n  readonly node: Node;\n  readonly privateKey: PrivateKey;\n  readonly consensusContext: ConsensusContext;\n}): Result<Context> => {\n  let context = contextIn;\n\n  context = context.cloneExpectedView({\n    expectedView: incrementExpectedView(context),\n  });\n\n  signAndRelayChangeView({ context, node, privateKey });\n\n  const viewNumber = context.expectedView[context.myIndex];\n  if (checkExpectedView({ context, viewNumber })) {\n    return initializeConsensusInitial({\n      blockchain: node.blockchain,\n      context,\n      viewNumber,\n      consensusContext,\n    });\n  }\n\n  const { secondsPerBlock } = node.blockchain.settings;\n\n  return {\n    context,\n    // tslint:disable-next-line no-bitwise\n    timerSeconds: secondsPerBlock << (viewNumber + 1),\n  };\n};\n\nexport const runConsensus = async ({\n  context,\n  node,\n  options: { privateKey, feeAddress, privateNet },\n  consensusContext,\n}: {\n  readonly context: Context;\n  readonly node: Node;\n  readonly options: InternalOptions;\n  readonly consensusContext: ConsensusContext;\n}): Promise<Result<Context>> => {\n  if (context.type === 'primary' && !(context instanceof RequestSentContext)) {\n    let requestSentContext: RequestSentContext;\n    if (context instanceof SignatureSentContext) {\n      requestSentContext = context.cloneRequestSent();\n    } else {\n      const nonce = utils.randomUInt64();\n      let mutableTransactions = Object.values(node.memPool);\n      const minerTransaction = await createMinerTransaction({\n        node,\n        feeAddress,\n        transactions: mutableTransactions,\n        nonce,\n      });\n\n      if (mutableTransactions.length >= node.blockchain.settings.maxTransactionsPerBlock) {\n        const mutableNetworkFees = await Promise.all(\n          mutableTransactions.map<Promise<[Transaction, BigNumber]>>(async (transaction) => {\n            const networkFee = await transaction.getNetworkFee(node.blockchain.feeContext);\n\n            return [transaction, new BigNumber(networkFee.toString(10))];\n          }),\n        );\n\n        mutableNetworkFees.sort(([first, a], [second, b]) => b.div(second.size).comparedTo(a.div(first.size)));\n        mutableTransactions = _.take(mutableNetworkFees, node.blockchain.settings.maxTransactionsPerBlock - 1)\n          // tslint:disable-next-line no-unused\n          .map(([transaction, _unused]) => transaction);\n      }\n      mutableTransactions.unshift(minerTransaction);\n      const [previousHeader, validators] = await Promise.all([\n        node.blockchain.header.get({ hashOrIndex: context.previousHash }),\n        node.blockchain.getValidators(mutableTransactions),\n      ]);\n\n      const newContext = new RequestSentContext({\n        viewNumber: context.viewNumber,\n        myIndex: context.myIndex,\n        primaryIndex: context.primaryIndex,\n        expectedView: context.expectedView,\n        validators: context.validators,\n        blockReceivedTimeSeconds: context.blockReceivedTimeSeconds,\n        transactions: mutableTransactions.reduce<{ [key: string]: Transaction }>(\n          (acc, transaction) => ({\n            ...acc,\n            [transaction.hashHex]: transaction,\n          }),\n          {},\n        ),\n        signatures: [],\n        header: {\n          type: 'new',\n          previousHash: context.previousHash,\n          transactionHashes: mutableTransactions.map((transaction) => transaction.hashHex),\n\n          blockIndex: context.blockIndex,\n          nonce,\n          timestamp: Math.max(consensusContext.nowSeconds(), previousHeader.timestamp + 1),\n\n          nextConsensus: crypto.getConsensusAddress(validators),\n        },\n      });\n\n      const mutableSignatures = [];\n      mutableSignatures[newContext.myIndex] = crypto.sign({\n        message: newContext.header.message,\n        privateKey,\n      });\n\n      requestSentContext = newContext.cloneSignatures({ signatures: mutableSignatures });\n    }\n\n    if (privateNet) {\n      return checkSignatures({ node, context: requestSentContext });\n    }\n\n    signAndRelay({\n      context: requestSentContext,\n      node,\n      privateKey,\n      consensusMessage: new PrepareRequestConsensusMessage({\n        viewNumber: requestSentContext.viewNumber,\n        nonce: requestSentContext.header.consensusData,\n        nextConsensus: requestSentContext.header.nextConsensus,\n        transactionHashes: requestSentContext.transactionHashes.map((hash) => common.hexToUInt256(hash)),\n        minerTransaction: commonUtils.nullthrows(\n          requestSentContext.transactions[requestSentContext.transactionHashes[0]],\n        ) as MinerTransaction,\n        signature: commonUtils.nullthrows(requestSentContext.signatures[requestSentContext.myIndex]),\n      }),\n    });\n\n    const { secondsPerBlock } = node.blockchain.settings;\n\n    return {\n      context: requestSentContext,\n      // tslint:disable-next-line no-bitwise\n      timerSeconds: secondsPerBlock << (requestSentContext.viewNumber + 1),\n    };\n  }\n\n  if (context instanceof RequestSentContext || context.type === 'backup') {\n    return requestChangeView({\n      context,\n      node,\n      privateKey,\n      consensusContext,\n    });\n  }\n\n  return { context };\n};\n"]}