1 | import { common, crypto, InvalidFormatError, IOHelper, } from '@neo-one/client-common-esnext-esm';
|
2 | import { BlockBase } from './BlockBase';
|
3 | import { MerkleTree } from './crypto';
|
4 | import { VerifyError } from './errors';
|
5 | import { Header } from './Header';
|
6 | import { ScriptContainerType } from './ScriptContainer';
|
7 | import { deserializeTransactionWireBase, TransactionType, } from './transaction';
|
8 | import { BinaryReader, utils } from './utils';
|
9 | export class Block extends BlockBase {
|
10 | constructor({ version, previousHash, timestamp, index, consensusData, nextConsensus, script, hash, transactions, merkleRoot = MerkleTree.computeRoot(transactions.map((transaction) => transaction.hash)), }) {
|
11 | super({
|
12 | version,
|
13 | previousHash,
|
14 | merkleRoot,
|
15 | timestamp,
|
16 | index,
|
17 | consensusData,
|
18 | nextConsensus,
|
19 | script,
|
20 | hash,
|
21 | });
|
22 | this.sizeExclusive = utils.lazy(() => IOHelper.sizeOfArray(this.transactions, (transaction) => transaction.size));
|
23 | this.headerInternal = utils.lazy(() => new Header({
|
24 | version: this.version,
|
25 | previousHash: this.previousHash,
|
26 | merkleRoot: this.merkleRoot,
|
27 | timestamp: this.timestamp,
|
28 | index: this.index,
|
29 | consensusData: this.consensusData,
|
30 | nextConsensus: this.nextConsensus,
|
31 | script: this.script,
|
32 | }));
|
33 | this.transactions = transactions;
|
34 | }
|
35 | static async calculateNetworkFee(context, transactions) {
|
36 | const fees = await Promise.all(transactions.map(async (transaction) => transaction.getNetworkFee(context)));
|
37 | return fees.reduce((acc, fee) => acc.add(fee), utils.ZERO);
|
38 | }
|
39 | static deserializeWireBase(options) {
|
40 | const { reader } = options;
|
41 | const blockBase = super.deserializeBlockBaseWireBase(options);
|
42 | const transactions = reader.readArray(() => deserializeTransactionWireBase(options), 0x10000);
|
43 | if (transactions.length === 0) {
|
44 | throw new InvalidFormatError('Expected at least one transcaction in the block');
|
45 | }
|
46 | const merkleRoot = MerkleTree.computeRoot(transactions.map((transaction) => transaction.hash));
|
47 | if (!common.uInt256Equal(merkleRoot, blockBase.merkleRoot)) {
|
48 | throw new InvalidFormatError('Invalid merkle root');
|
49 | }
|
50 | return new this({
|
51 | version: blockBase.version,
|
52 | previousHash: blockBase.previousHash,
|
53 | merkleRoot: blockBase.merkleRoot,
|
54 | timestamp: blockBase.timestamp,
|
55 | index: blockBase.index,
|
56 | consensusData: blockBase.consensusData,
|
57 | nextConsensus: blockBase.nextConsensus,
|
58 | script: blockBase.script,
|
59 | transactions,
|
60 | });
|
61 | }
|
62 | static deserializeWire(options) {
|
63 | return this.deserializeWireBase({
|
64 | context: options.context,
|
65 | reader: new BinaryReader(options.buffer),
|
66 | });
|
67 | }
|
68 | get header() {
|
69 | return this.headerInternal();
|
70 | }
|
71 | clone({ transactions, script, }) {
|
72 | return new Block({
|
73 | version: this.version,
|
74 | previousHash: this.previousHash,
|
75 | merkleRoot: this.merkleRoot,
|
76 | timestamp: this.timestamp,
|
77 | index: this.index,
|
78 | consensusData: this.consensusData,
|
79 | nextConsensus: this.nextConsensus,
|
80 | transactions,
|
81 | script,
|
82 | });
|
83 | }
|
84 | async getNetworkFee(context) {
|
85 | return Block.calculateNetworkFee(context, this.transactions);
|
86 | }
|
87 | getSystemFee(context) {
|
88 | return this.transactions.reduce((acc, transaction) => acc.add(transaction.getSystemFee(context)), utils.ZERO);
|
89 | }
|
90 | async verify(options) {
|
91 | const { completely = false } = options;
|
92 | if (this.transactions.length === 0 ||
|
93 | this.transactions[0].type !== TransactionType.Miner ||
|
94 | this.transactions.slice(1).some((transaction) => transaction.type === TransactionType.Miner)) {
|
95 | throw new VerifyError('Invalid miner transaction in block.');
|
96 | }
|
97 | await Promise.all([this.verifyBase(options), completely ? this.verifyComplete(options) : Promise.resolve()]);
|
98 | }
|
99 | serializeWireBase(writer) {
|
100 | super.serializeWireBase(writer);
|
101 | writer.writeArray(this.transactions, (transaction) => transaction.serializeWireBase(writer));
|
102 | }
|
103 | async serializeJSON(context) {
|
104 | const blockBaseJSON = super.serializeBlockBaseJSON(context);
|
105 | return {
|
106 | version: blockBaseJSON.version,
|
107 | hash: blockBaseJSON.hash,
|
108 | previousblockhash: blockBaseJSON.previousblockhash,
|
109 | merkleroot: blockBaseJSON.merkleroot,
|
110 | time: blockBaseJSON.time,
|
111 | index: blockBaseJSON.index,
|
112 | nonce: blockBaseJSON.nonce,
|
113 | nextconsensus: blockBaseJSON.nextconsensus,
|
114 | script: blockBaseJSON.script,
|
115 | tx: await Promise.all(this.transactions.map((transaction) => transaction.serializeJSON(context))),
|
116 | size: blockBaseJSON.size,
|
117 | confirmations: blockBaseJSON.confirmations,
|
118 | };
|
119 | }
|
120 | async verifyBase({ genesisBlock, tryGetBlock, tryGetHeader, verifyScript, }) {
|
121 | if (common.uInt256Equal(this.hash, genesisBlock.hash)) {
|
122 | return;
|
123 | }
|
124 | const existingBlock = await tryGetBlock({ hashOrIndex: this.hash });
|
125 | if (existingBlock !== undefined) {
|
126 | return;
|
127 | }
|
128 | const previousHeader = await tryGetHeader({
|
129 | hashOrIndex: this.previousHash,
|
130 | });
|
131 | if (previousHeader === undefined) {
|
132 | throw new VerifyError('Previous header does not exist.');
|
133 | }
|
134 | if (previousHeader.index + 1 !== this.index) {
|
135 | throw new VerifyError('Previous index + 1 does not match index.');
|
136 | }
|
137 | if (previousHeader.timestamp >= this.timestamp) {
|
138 | throw new VerifyError('Previous timestamp is greater than block.');
|
139 | }
|
140 | const { failureMessage } = await verifyScript({
|
141 | scriptContainer: { type: ScriptContainerType.Block, value: this },
|
142 | hash: previousHeader.nextConsensus,
|
143 | witness: this.script,
|
144 | });
|
145 | if (failureMessage !== undefined) {
|
146 | throw new VerifyError(failureMessage);
|
147 | }
|
148 | }
|
149 | async verifyComplete(options) {
|
150 | await Promise.all([
|
151 | this.verifyConsensus(options),
|
152 | this.verifyTransactions(options),
|
153 | this.verifyNetworkFee(options),
|
154 | ]);
|
155 | }
|
156 | async verifyConsensus({ getValidators }) {
|
157 | const validators = await getValidators(this.transactions);
|
158 | if (!common.uInt160Equal(this.nextConsensus, crypto.getConsensusAddress(validators))) {
|
159 | throw new VerifyError('Invalid next consensus address');
|
160 | }
|
161 | }
|
162 | async verifyTransactions(options) {
|
163 | const results = await Promise.all(this.transactions.map(async (transaction) => transaction.verify({
|
164 | isSpent: options.isSpent,
|
165 | getAsset: options.getAsset,
|
166 | getOutput: options.getOutput,
|
167 | tryGetAccount: options.tryGetAccount,
|
168 | calculateClaimAmount: options.calculateClaimAmount,
|
169 | standbyValidators: options.standbyValidators,
|
170 | getAllValidators: options.getAllValidators,
|
171 | verifyScript: options.verifyScript,
|
172 | currentHeight: options.currentHeight,
|
173 | governingToken: options.governingToken,
|
174 | utilityToken: options.utilityToken,
|
175 | fees: options.fees,
|
176 | registerValidatorFee: options.registerValidatorFee,
|
177 | })));
|
178 | const failureResults = results.filter((verifyResults) => verifyResults.some(({ failureMessage }) => failureMessage !== undefined));
|
179 | if (failureResults.length > 0) {
|
180 | const failureResult = failureResults[0].find(({ failureMessage }) => failureMessage !== undefined);
|
181 | if (failureResult !== undefined && failureResult.failureMessage !== undefined) {
|
182 | throw new VerifyError(failureResult.failureMessage);
|
183 | }
|
184 | }
|
185 | }
|
186 | async verifyNetworkFee(options) {
|
187 | const networkFee = await this.getNetworkFee({
|
188 | getOutput: options.getOutput,
|
189 | governingToken: options.governingToken,
|
190 | utilityToken: options.utilityToken,
|
191 | fees: options.fees,
|
192 | registerValidatorFee: options.registerValidatorFee,
|
193 | });
|
194 | const minerTransaction = this.transactions.find((transaction) => transaction.type === TransactionType.Miner);
|
195 | if (minerTransaction === undefined) {
|
196 | throw new VerifyError('Missing miner transaction');
|
197 | }
|
198 | const minerTransactionNetworkFee = minerTransaction.outputs.reduce((acc, output) => acc.add(output.value), utils.ZERO);
|
199 | if (!networkFee.eq(minerTransactionNetworkFee)) {
|
200 | throw new VerifyError('Miner output does not equal network fee.');
|
201 | }
|
202 | }
|
203 | }
|
204 |
|
205 | //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["Block.ts"],"names":[],"mappings":"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","file":"neo-one-node-core/src/Block.js","sourcesContent":["import {\n  BinaryWriter,\n  BlockJSON,\n  common,\n  crypto,\n  ECPoint,\n  InvalidFormatError,\n  IOHelper,\n  TransactionJSON,\n  UInt160,\n  UInt256,\n} from '@neo-one/client-common-esnext-esm';\nimport BN from 'bn.js';\nimport { Account, AccountKey } from './Account';\nimport { Asset, AssetKey } from './Asset';\nimport { BlockBase } from './BlockBase';\nimport { MerkleTree } from './crypto';\nimport { VerifyError } from './errors';\nimport { Header, HeaderKey } from './Header';\nimport { ScriptContainerType } from './ScriptContainer';\nimport {\n  DeserializeWireBaseOptions,\n  DeserializeWireOptions,\n  SerializableJSON,\n  SerializableWire,\n  SerializeJSONContext,\n} from './Serializable';\nimport {\n  deserializeTransactionWireBase,\n  FeeContext,\n  Input,\n  Output,\n  OutputKey,\n  RegisterTransaction,\n  Transaction,\n  TransactionType,\n} from './transaction';\nimport { BinaryReader, utils } from './utils';\nimport { Validator } from './Validator';\nimport { VerifyScript } from './vm';\nimport { Witness } from './Witness';\n\nexport interface BlockAdd {\n  readonly version?: number;\n  readonly previousHash: UInt256;\n  readonly merkleRoot?: UInt256;\n  readonly timestamp: number;\n  readonly index: number;\n  readonly consensusData: BN;\n  readonly nextConsensus: UInt160;\n  readonly script?: Witness;\n  readonly hash?: UInt256;\n  readonly transactions: ReadonlyArray<Transaction>;\n}\n\nexport interface BlockKey {\n  readonly hashOrIndex: UInt256 | number;\n}\n\nexport interface BlockVerifyOptions {\n  readonly genesisBlock: Block;\n  readonly tryGetBlock: (block: BlockKey) => Promise<Block | undefined>;\n  readonly tryGetHeader: (header: HeaderKey) => Promise<Header | undefined>;\n  readonly isSpent: (key: OutputKey) => Promise<boolean>;\n  readonly getAsset: (key: AssetKey) => Promise<Asset>;\n  readonly getOutput: (key: OutputKey) => Promise<Output>;\n  readonly tryGetAccount: (key: AccountKey) => Promise<Account | undefined>;\n  readonly getValidators: (transactions: ReadonlyArray<Transaction>) => Promise<ReadonlyArray<ECPoint>>;\n  readonly standbyValidators: ReadonlyArray<ECPoint>;\n  readonly getAllValidators: () => Promise<ReadonlyArray<Validator>>;\n  readonly calculateClaimAmount: (inputs: ReadonlyArray<Input>) => Promise<BN>;\n  readonly verifyScript: VerifyScript;\n  readonly currentHeight: number;\n  readonly governingToken: RegisterTransaction;\n  readonly utilityToken: RegisterTransaction;\n  readonly fees: { [K in TransactionType]?: BN };\n  readonly registerValidatorFee: BN;\n  readonly completely?: boolean;\n}\n\nexport class Block extends BlockBase implements SerializableWire<Block>, SerializableJSON<BlockJSON> {\n  public static async calculateNetworkFee(context: FeeContext, transactions: ReadonlyArray<Transaction>): Promise<BN> {\n    const fees = await Promise.all(transactions.map(async (transaction) => transaction.getNetworkFee(context)));\n\n    return fees.reduce((acc, fee) => acc.add(fee), utils.ZERO);\n  }\n\n  public static deserializeWireBase(options: DeserializeWireBaseOptions): Block {\n    const { reader } = options;\n    const blockBase = super.deserializeBlockBaseWireBase(options);\n    const transactions = reader.readArray(() => deserializeTransactionWireBase(options), 0x10000);\n\n    if (transactions.length === 0) {\n      throw new InvalidFormatError('Expected at least one transcaction in the block');\n    }\n\n    const merkleRoot = MerkleTree.computeRoot(transactions.map((transaction) => transaction.hash));\n\n    if (!common.uInt256Equal(merkleRoot, blockBase.merkleRoot)) {\n      throw new InvalidFormatError('Invalid merkle root');\n    }\n\n    return new this({\n      version: blockBase.version,\n      previousHash: blockBase.previousHash,\n      merkleRoot: blockBase.merkleRoot,\n      timestamp: blockBase.timestamp,\n      index: blockBase.index,\n      consensusData: blockBase.consensusData,\n      nextConsensus: blockBase.nextConsensus,\n      script: blockBase.script,\n      transactions,\n    });\n  }\n\n  public static deserializeWire(options: DeserializeWireOptions): Block {\n    return this.deserializeWireBase({\n      context: options.context,\n      reader: new BinaryReader(options.buffer),\n    });\n  }\n\n  public readonly transactions: ReadonlyArray<Transaction>;\n  protected readonly sizeExclusive = utils.lazy(() =>\n    IOHelper.sizeOfArray(this.transactions, (transaction) => transaction.size),\n  );\n  private readonly headerInternal = utils.lazy(\n    () =>\n      new Header({\n        version: this.version,\n        previousHash: this.previousHash,\n        merkleRoot: this.merkleRoot,\n        timestamp: this.timestamp,\n        index: this.index,\n        consensusData: this.consensusData,\n        nextConsensus: this.nextConsensus,\n        script: this.script,\n      }),\n  );\n\n  public constructor({\n    version,\n    previousHash,\n    timestamp,\n    index,\n    consensusData,\n    nextConsensus,\n    script,\n    hash,\n    transactions,\n    merkleRoot = MerkleTree.computeRoot(transactions.map((transaction) => transaction.hash)),\n  }: BlockAdd) {\n    super({\n      version,\n      previousHash,\n      merkleRoot,\n      timestamp,\n      index,\n      consensusData,\n      nextConsensus,\n      script,\n      hash,\n    });\n\n    this.transactions = transactions;\n  }\n\n  public get header(): Header {\n    return this.headerInternal();\n  }\n\n  public clone({\n    transactions,\n    script,\n  }: {\n    readonly transactions: ReadonlyArray<Transaction>;\n    readonly script: Witness;\n  }): Block {\n    return new Block({\n      version: this.version,\n      previousHash: this.previousHash,\n      merkleRoot: this.merkleRoot,\n      timestamp: this.timestamp,\n      index: this.index,\n      consensusData: this.consensusData,\n      nextConsensus: this.nextConsensus,\n      transactions,\n      script,\n    });\n  }\n\n  public async getNetworkFee(context: FeeContext): Promise<BN> {\n    return Block.calculateNetworkFee(context, this.transactions);\n  }\n\n  public getSystemFee(context: FeeContext): BN {\n    return this.transactions.reduce((acc, transaction) => acc.add(transaction.getSystemFee(context)), utils.ZERO);\n  }\n\n  public async verify(options: BlockVerifyOptions): Promise<void> {\n    const { completely = false } = options;\n\n    if (\n      this.transactions.length === 0 ||\n      this.transactions[0].type !== TransactionType.Miner ||\n      this.transactions.slice(1).some((transaction) => transaction.type === TransactionType.Miner)\n    ) {\n      throw new VerifyError('Invalid miner transaction in block.');\n    }\n\n    await Promise.all([this.verifyBase(options), completely ? this.verifyComplete(options) : Promise.resolve()]);\n  }\n\n  public serializeWireBase(writer: BinaryWriter): void {\n    super.serializeWireBase(writer);\n    writer.writeArray(this.transactions, (transaction) => transaction.serializeWireBase(writer));\n  }\n\n  public async serializeJSON(context: SerializeJSONContext): Promise<BlockJSON> {\n    const blockBaseJSON = super.serializeBlockBaseJSON(context);\n\n    return {\n      version: blockBaseJSON.version,\n      hash: blockBaseJSON.hash,\n      previousblockhash: blockBaseJSON.previousblockhash,\n      merkleroot: blockBaseJSON.merkleroot,\n      time: blockBaseJSON.time,\n      index: blockBaseJSON.index,\n      nonce: blockBaseJSON.nonce,\n      nextconsensus: blockBaseJSON.nextconsensus,\n      script: blockBaseJSON.script,\n      tx: await Promise.all(\n        this.transactions.map(\n          (transaction) => transaction.serializeJSON(context) as TransactionJSON | PromiseLike<TransactionJSON>,\n        ),\n      ),\n      size: blockBaseJSON.size,\n      confirmations: blockBaseJSON.confirmations,\n    };\n  }\n\n  private async verifyBase({\n    genesisBlock,\n    tryGetBlock,\n    tryGetHeader,\n    verifyScript,\n  }: BlockVerifyOptions): Promise<void> {\n    if (common.uInt256Equal(this.hash, genesisBlock.hash)) {\n      return;\n    }\n\n    const existingBlock = await tryGetBlock({ hashOrIndex: this.hash });\n    if (existingBlock !== undefined) {\n      return;\n    }\n\n    const previousHeader = await tryGetHeader({\n      hashOrIndex: this.previousHash,\n    });\n\n    if (previousHeader === undefined) {\n      throw new VerifyError('Previous header does not exist.');\n    }\n\n    if (previousHeader.index + 1 !== this.index) {\n      throw new VerifyError('Previous index + 1 does not match index.');\n    }\n\n    if (previousHeader.timestamp >= this.timestamp) {\n      throw new VerifyError('Previous timestamp is greater than block.');\n    }\n\n    const { failureMessage } = await verifyScript({\n      scriptContainer: { type: ScriptContainerType.Block, value: this },\n      hash: previousHeader.nextConsensus,\n      witness: this.script,\n    });\n\n    if (failureMessage !== undefined) {\n      throw new VerifyError(failureMessage);\n    }\n  }\n\n  private async verifyComplete(options: BlockVerifyOptions): Promise<void> {\n    await Promise.all([\n      this.verifyConsensus(options),\n      this.verifyTransactions(options),\n      this.verifyNetworkFee(options),\n    ]);\n  }\n\n  private async verifyConsensus({ getValidators }: BlockVerifyOptions): Promise<void> {\n    const validators = await getValidators(this.transactions);\n    if (!common.uInt160Equal(this.nextConsensus, crypto.getConsensusAddress(validators))) {\n      throw new VerifyError('Invalid next consensus address');\n    }\n  }\n\n  private async verifyTransactions(options: BlockVerifyOptions): Promise<void> {\n    const results = await Promise.all(\n      this.transactions.map(async (transaction) =>\n        transaction.verify({\n          isSpent: options.isSpent,\n          getAsset: options.getAsset,\n          getOutput: options.getOutput,\n          tryGetAccount: options.tryGetAccount,\n          calculateClaimAmount: options.calculateClaimAmount,\n          standbyValidators: options.standbyValidators,\n          getAllValidators: options.getAllValidators,\n          verifyScript: options.verifyScript,\n          currentHeight: options.currentHeight,\n          governingToken: options.governingToken,\n          utilityToken: options.utilityToken,\n          fees: options.fees,\n          registerValidatorFee: options.registerValidatorFee,\n        }),\n      ),\n    );\n    const failureResults = results.filter((verifyResults) =>\n      verifyResults.some(({ failureMessage }) => failureMessage !== undefined),\n    );\n    if (failureResults.length > 0) {\n      const failureResult = failureResults[0].find(({ failureMessage }) => failureMessage !== undefined);\n      if (failureResult !== undefined && failureResult.failureMessage !== undefined) {\n        throw new VerifyError(failureResult.failureMessage);\n      }\n    }\n  }\n\n  private async verifyNetworkFee(options: BlockVerifyOptions): Promise<void> {\n    const networkFee = await this.getNetworkFee({\n      getOutput: options.getOutput,\n      governingToken: options.governingToken,\n      utilityToken: options.utilityToken,\n      fees: options.fees,\n      registerValidatorFee: options.registerValidatorFee,\n    });\n\n    const minerTransaction = this.transactions.find((transaction) => transaction.type === TransactionType.Miner);\n\n    if (minerTransaction === undefined) {\n      throw new VerifyError('Missing miner transaction');\n    }\n\n    const minerTransactionNetworkFee = minerTransaction.outputs.reduce(\n      (acc, output) => acc.add(output.value),\n      utils.ZERO,\n    );\n\n    if (!networkFee.eq(minerTransactionNetworkFee)) {\n      throw new VerifyError('Miner output does not equal network fee.');\n    }\n  }\n}\n"]}
|