UNPKG

22.1 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const client_common_1 = require("@neo-one/client-common");
4const node_core_1 = require("@neo-one/node-core");
5const utils_1 = require("@neo-one/utils");
6const common_1 = require("./common");
7const context_1 = require("./context");
8const handleChangeView = ({ context: contextIn, node, payload, consensusContext, message, }) => {
9 let context = contextIn;
10 const viewNumber = message.newViewNumber;
11 if (viewNumber > context.expectedView[payload.validatorIndex]) {
12 const mutableExpectedView = [...context.expectedView];
13 mutableExpectedView[payload.validatorIndex] = viewNumber;
14 context = context.cloneExpectedView({ expectedView: mutableExpectedView });
15 if (common_1.checkExpectedView({ context, viewNumber })) {
16 return common_1.initializeConsensus({
17 node,
18 context,
19 viewNumber,
20 consensusContext,
21 });
22 }
23 }
24 return { context };
25};
26const TEN_MINUTES_IN_SECONDS = 10 * 60;
27const handlePrepareRequest = async ({ context: contextIn, node, privateKey, payload, consensusContext, message, }) => {
28 let context = contextIn;
29 if (context.type !== 'backup' ||
30 context instanceof context_1.RequestReceivedContext ||
31 payload.validatorIndex !== context.primaryIndex ||
32 payload.timestamp > consensusContext.nowSeconds() + TEN_MINUTES_IN_SECONDS) {
33 return { context };
34 }
35 const header = await node.blockchain.header.get({
36 hashOrIndex: context.previousHash,
37 });
38 if (payload.timestamp <= header.timestamp) {
39 return { context };
40 }
41 const mutableSignatures = [];
42 mutableSignatures[payload.validatorIndex] = message.signature;
43 const newContext = new context_1.RequestReceivedContext({
44 viewNumber: context.viewNumber,
45 myIndex: context.myIndex,
46 primaryIndex: context.primaryIndex,
47 expectedView: context.expectedView,
48 validators: context.validators,
49 blockReceivedTimeSeconds: context.blockReceivedTimeSeconds,
50 transactions: {},
51 signatures: mutableSignatures,
52 header: {
53 type: 'new',
54 previousHash: context.previousHash,
55 transactionHashes: message.transactionHashes.map((hash) => client_common_1.common.uInt256ToHex(hash)),
56 blockIndex: context.blockIndex,
57 nonce: message.nonce,
58 timestamp: payload.timestamp,
59 nextConsensus: message.nextConsensus,
60 },
61 });
62 const verified = client_common_1.crypto.verify({
63 message: newContext.header.message,
64 signature: message.signature,
65 publicKey: context.validators[payload.validatorIndex],
66 });
67 if (!verified) {
68 return { context };
69 }
70 let nextContext = newContext;
71 for (const hash of newContext.transactionHashes.slice(1)) {
72 const transaction = node.memPool[hash];
73 if (transaction !== undefined) {
74 const res = await common_1.addTransaction({
75 context: nextContext,
76 node,
77 privateKey,
78 transaction,
79 verify: false,
80 consensusContext,
81 });
82 if (!(res.context instanceof context_1.RequestReceivedContext)) {
83 return res;
84 }
85 nextContext = res.context;
86 }
87 }
88 const result = await common_1.addTransaction({
89 context: nextContext,
90 node,
91 privateKey,
92 transaction: message.minerTransaction,
93 verify: true,
94 consensusContext,
95 });
96 if (!(result.context instanceof context_1.RequestReceivedContext)) {
97 return result;
98 }
99 context = result.context;
100 node.syncMemPool();
101 return { context };
102};
103const handlePrepareResponse = async ({ context, node, payload, message, }) => {
104 if (context instanceof context_1.HeaderContext &&
105 context.signatures[payload.validatorIndex] === undefined &&
106 client_common_1.crypto.verify({
107 message: context.header.message,
108 signature: message.signature,
109 publicKey: context.validators[payload.validatorIndex],
110 })) {
111 const mutableSignatures = [...context.signatures];
112 mutableSignatures[payload.validatorIndex] = message.signature;
113 const newContext = context.cloneSignatures({ signatures: mutableSignatures });
114 return common_1.checkSignatures({ context: newContext, node });
115 }
116 return { context };
117};
118exports.handleConsensusPayload = async ({ context, node, privateKey, payload, consensusContext, }) => {
119 const { consensusMessage } = payload;
120 if (payload.validatorIndex === context.myIndex ||
121 payload.version !== context.version ||
122 !client_common_1.common.uInt256Equal(payload.previousHash, context.previousHash) ||
123 payload.blockIndex !== context.blockIndex ||
124 payload.validatorIndex >= context.validators.length ||
125 (consensusMessage.type !== node_core_1.ConsensusMessageType.ChangeView && consensusMessage.viewNumber !== context.viewNumber)) {
126 return { context };
127 }
128 switch (consensusMessage.type) {
129 case node_core_1.ConsensusMessageType.ChangeView:
130 return handleChangeView({
131 context,
132 node,
133 payload,
134 consensusContext,
135 message: consensusMessage,
136 });
137 case node_core_1.ConsensusMessageType.PrepareRequest:
138 return handlePrepareRequest({
139 context,
140 node,
141 privateKey,
142 payload,
143 consensusContext,
144 message: consensusMessage,
145 });
146 case node_core_1.ConsensusMessageType.PrepareResponse:
147 return handlePrepareResponse({
148 context,
149 node,
150 payload,
151 message: consensusMessage,
152 });
153 default:
154 utils_1.utils.assertNever(consensusMessage);
155 throw new Error('For TS');
156 }
157};
158
159//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["handleConsensusPayload.ts"],"names":[],"mappings":"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","file":"neo-one-node-consensus/src/handleConsensusPayload.js","sourcesContent":["import { common, crypto, PrivateKey } from '@neo-one/client-common';\nimport {\n  ChangeViewConsensusMessage,\n  ConsensusMessageType,\n  ConsensusPayload,\n  Node,\n  PrepareRequestConsensusMessage,\n  PrepareResponseConsensusMessage,\n  Transaction,\n} from '@neo-one/node-core';\nimport { utils as commonUtils } from '@neo-one/utils';\nimport { addTransaction, checkExpectedView, checkSignatures, initializeConsensus } from './common';\nimport { ConsensusContext } from './ConsensusContext';\nimport { Context, HeaderContext, RequestReceivedContext } from './context';\nimport { Result } from './types';\n\nconst handleChangeView = ({\n  context: contextIn,\n  node,\n  payload,\n  consensusContext,\n  message,\n}: {\n  readonly context: Context;\n  readonly node: Node;\n  readonly payload: ConsensusPayload;\n  readonly consensusContext: ConsensusContext;\n  readonly message: ChangeViewConsensusMessage;\n}): Result<Context> => {\n  let context = contextIn;\n  const viewNumber = message.newViewNumber;\n  if (viewNumber > context.expectedView[payload.validatorIndex]) {\n    const mutableExpectedView = [...context.expectedView];\n    mutableExpectedView[payload.validatorIndex] = viewNumber;\n    context = context.cloneExpectedView({ expectedView: mutableExpectedView });\n    if (checkExpectedView({ context, viewNumber })) {\n      return initializeConsensus({\n        node,\n        context,\n        viewNumber,\n        consensusContext,\n      });\n    }\n  }\n\n  return { context };\n};\n\nconst TEN_MINUTES_IN_SECONDS = 10 * 60;\n\nconst handlePrepareRequest = async ({\n  context: contextIn,\n  node,\n  privateKey,\n  payload,\n  consensusContext,\n  message,\n}: {\n  readonly context: Context;\n  readonly node: Node;\n  readonly privateKey: PrivateKey;\n  readonly payload: ConsensusPayload;\n  readonly consensusContext: ConsensusContext;\n  readonly message: PrepareRequestConsensusMessage;\n}): Promise<Result<Context>> => {\n  let context = contextIn;\n  if (\n    context.type !== 'backup' ||\n    context instanceof RequestReceivedContext ||\n    payload.validatorIndex !== context.primaryIndex ||\n    payload.timestamp > consensusContext.nowSeconds() + TEN_MINUTES_IN_SECONDS\n  ) {\n    return { context };\n  }\n  const header = await node.blockchain.header.get({\n    hashOrIndex: context.previousHash,\n  });\n\n  if (payload.timestamp <= header.timestamp) {\n    return { context };\n  }\n\n  const mutableSignatures = [];\n  mutableSignatures[payload.validatorIndex] = message.signature;\n  const newContext = new RequestReceivedContext({\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: {},\n    signatures: mutableSignatures,\n    header: {\n      type: 'new',\n      previousHash: context.previousHash,\n      transactionHashes: message.transactionHashes.map((hash) => common.uInt256ToHex(hash)),\n\n      blockIndex: context.blockIndex,\n      nonce: message.nonce,\n      timestamp: payload.timestamp,\n      nextConsensus: message.nextConsensus,\n    },\n  });\n\n  const verified = crypto.verify({\n    message: newContext.header.message,\n    signature: message.signature,\n    publicKey: context.validators[payload.validatorIndex],\n  });\n\n  if (!verified) {\n    return { context };\n  }\n\n  let nextContext = newContext;\n  // tslint:disable-next-line no-loop-statement\n  for (const hash of newContext.transactionHashes.slice(1)) {\n    const transaction = node.memPool[hash] as Transaction | undefined;\n    if (transaction !== undefined) {\n      const res = await addTransaction({\n        context: nextContext,\n        node,\n        privateKey,\n        transaction,\n        verify: false,\n        consensusContext,\n      });\n\n      if (!(res.context instanceof RequestReceivedContext)) {\n        return res;\n      }\n      // eslint-disable-next-line\n      nextContext = res.context;\n    }\n  }\n\n  const result = await addTransaction({\n    context: nextContext,\n    node,\n    privateKey,\n    transaction: message.minerTransaction,\n    verify: true,\n    consensusContext,\n  });\n\n  if (!(result.context instanceof RequestReceivedContext)) {\n    return result;\n  }\n  // eslint-disable-next-line\n  context = result.context;\n  node.syncMemPool();\n\n  return { context };\n};\n\nconst handlePrepareResponse = async ({\n  context,\n  node,\n  payload,\n  message,\n}: {\n  readonly context: Context;\n  readonly node: Node;\n  readonly payload: ConsensusPayload;\n  readonly message: PrepareResponseConsensusMessage;\n}): Promise<Result<Context>> => {\n  if (\n    context instanceof HeaderContext &&\n    context.signatures[payload.validatorIndex] === undefined &&\n    crypto.verify({\n      message: context.header.message,\n      signature: message.signature,\n      publicKey: context.validators[payload.validatorIndex],\n    })\n  ) {\n    const mutableSignatures = [...context.signatures];\n    mutableSignatures[payload.validatorIndex] = message.signature;\n    const newContext = context.cloneSignatures({ signatures: mutableSignatures });\n\n    return checkSignatures({ context: newContext, node });\n  }\n\n  return { context };\n};\n\nexport const handleConsensusPayload = async ({\n  context,\n  node,\n  privateKey,\n  payload,\n  consensusContext,\n}: {\n  readonly context: Context;\n  readonly node: Node;\n  readonly privateKey: PrivateKey;\n  readonly payload: ConsensusPayload;\n  readonly consensusContext: ConsensusContext;\n}): Promise<Result<Context>> => {\n  const { consensusMessage } = payload;\n  if (\n    payload.validatorIndex === context.myIndex ||\n    payload.version !== context.version ||\n    !common.uInt256Equal(payload.previousHash, context.previousHash) ||\n    payload.blockIndex !== context.blockIndex ||\n    payload.validatorIndex >= context.validators.length ||\n    (consensusMessage.type !== ConsensusMessageType.ChangeView && consensusMessage.viewNumber !== context.viewNumber)\n  ) {\n    return { context };\n  }\n\n  switch (consensusMessage.type) {\n    case ConsensusMessageType.ChangeView:\n      return handleChangeView({\n        context,\n        node,\n        payload,\n        consensusContext,\n        message: consensusMessage,\n      });\n\n    case ConsensusMessageType.PrepareRequest:\n      return handlePrepareRequest({\n        context,\n        node,\n        privateKey,\n        payload,\n        consensusContext,\n        message: consensusMessage,\n      });\n\n    case ConsensusMessageType.PrepareResponse:\n      return handlePrepareResponse({\n        context,\n        node,\n        payload,\n        message: consensusMessage,\n      });\n\n    default:\n      commonUtils.assertNever(consensusMessage);\n      throw new Error('For TS');\n  }\n};\n"]}