1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | const tslib_1 = require("tslib");
|
4 | const client_common_1 = require("@neo-one/client-common");
|
5 | const node_core_1 = require("@neo-one/node-core");
|
6 | const utils_1 = require("@neo-one/utils");
|
7 | const bignumber_js_1 = tslib_1.__importDefault(require("bignumber.js"));
|
8 | const lodash_1 = tslib_1.__importDefault(require("lodash"));
|
9 | const common_1 = require("./common");
|
10 | const context_1 = require("./context");
|
11 | const 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 | };
|
27 | const 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 | // tslint:disable-next-line no-bitwise
|
46 | timerSeconds: secondsPerBlock << (viewNumber + 1),
|
47 | };
|
48 | };
|
49 | exports.runConsensus = async ({ context, node, options: { privateKey, feeAddress, privateNet }, consensusContext, }) => {
|
50 | if (context.type === 'primary' && !(context instanceof context_1.RequestSentContext)) {
|
51 | let requestSentContext;
|
52 | if (context instanceof context_1.SignatureSentContext) {
|
53 | requestSentContext = context.cloneRequestSent();
|
54 | }
|
55 | else {
|
56 | const nonce = node_core_1.utils.randomUInt64();
|
57 | let mutableTransactions = Object.values(node.memPool);
|
58 | const minerTransaction = await createMinerTransaction({
|
59 | node,
|
60 | feeAddress,
|
61 | transactions: mutableTransactions,
|
62 | nonce,
|
63 | });
|
64 | if (mutableTransactions.length >= node.blockchain.settings.maxTransactionsPerBlock) {
|
65 | const mutableNetworkFees = await Promise.all(mutableTransactions.map(async (transaction) => {
|
66 | const networkFee = await transaction.getNetworkFee(node.blockchain.feeContext);
|
67 | return [transaction, new bignumber_js_1.default(networkFee.toString(10))];
|
68 | }));
|
69 | mutableNetworkFees.sort(([first, a], [second, b]) => b.div(second.size).comparedTo(a.div(first.size)));
|
70 | mutableTransactions = lodash_1.default.take(mutableNetworkFees, node.blockchain.settings.maxTransactionsPerBlock - 1)
|
71 | // tslint:disable-next-line no-unused
|
72 | .map(([transaction, _unused]) => transaction);
|
73 | }
|
74 | mutableTransactions.unshift(minerTransaction);
|
75 | const [previousHeader, validators] = await Promise.all([
|
76 | node.blockchain.header.get({ hashOrIndex: context.previousHash }),
|
77 | node.blockchain.getValidators(mutableTransactions),
|
78 | ]);
|
79 | const newContext = new context_1.RequestSentContext({
|
80 | viewNumber: context.viewNumber,
|
81 | myIndex: context.myIndex,
|
82 | primaryIndex: context.primaryIndex,
|
83 | expectedView: context.expectedView,
|
84 | validators: context.validators,
|
85 | blockReceivedTimeSeconds: context.blockReceivedTimeSeconds,
|
86 | transactions: mutableTransactions.reduce((acc, transaction) => (Object.assign({}, acc, { [transaction.hashHex]: transaction })), {}),
|
87 | signatures: [],
|
88 | header: {
|
89 | type: 'new',
|
90 | previousHash: context.previousHash,
|
91 | transactionHashes: mutableTransactions.map((transaction) => transaction.hashHex),
|
92 | blockIndex: context.blockIndex,
|
93 | nonce,
|
94 | timestamp: Math.max(consensusContext.nowSeconds(), previousHeader.timestamp + 1),
|
95 | nextConsensus: client_common_1.crypto.getConsensusAddress(validators),
|
96 | },
|
97 | });
|
98 | const mutableSignatures = [];
|
99 | mutableSignatures[newContext.myIndex] = client_common_1.crypto.sign({
|
100 | message: newContext.header.message,
|
101 | privateKey,
|
102 | });
|
103 | requestSentContext = newContext.cloneSignatures({ signatures: mutableSignatures });
|
104 | }
|
105 | if (privateNet) {
|
106 | return common_1.checkSignatures({ node, context: requestSentContext });
|
107 | }
|
108 | common_1.signAndRelay({
|
109 | context: requestSentContext,
|
110 | node,
|
111 | privateKey,
|
112 | consensusMessage: new node_core_1.PrepareRequestConsensusMessage({
|
113 | viewNumber: requestSentContext.viewNumber,
|
114 | nonce: requestSentContext.header.consensusData,
|
115 | nextConsensus: requestSentContext.header.nextConsensus,
|
116 | transactionHashes: requestSentContext.transactionHashes.map((hash) => client_common_1.common.hexToUInt256(hash)),
|
117 | minerTransaction: utils_1.utils.nullthrows(requestSentContext.transactions[requestSentContext.transactionHashes[0]]),
|
118 | signature: utils_1.utils.nullthrows(requestSentContext.signatures[requestSentContext.myIndex]),
|
119 | }),
|
120 | });
|
121 | const { secondsPerBlock } = node.blockchain.settings;
|
122 | return {
|
123 | context: requestSentContext,
|
124 | // tslint:disable-next-line no-bitwise
|
125 | timerSeconds: secondsPerBlock << (requestSentContext.viewNumber + 1),
|
126 | };
|
127 | }
|
128 | if (context instanceof context_1.RequestSentContext || context.type === 'backup') {
|
129 | return requestChangeView({
|
130 | context,
|
131 | node,
|
132 | privateKey,
|
133 | consensusContext,
|
134 | });
|
135 | }
|
136 | return { context };
|
137 | };
|
138 |
|
139 | //# 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: ReadonlyArray<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"]}
|