UNPKG

35.4 kBJavaScriptView Raw
1import { common, crypto, InvalidFormatError, IOHelper, } from '@neo-one/client-common-esnext-esm';
2import { BlockBase } from './BlockBase';
3import { MerkleTree } from './crypto';
4import { VerifyError } from './errors';
5import { Header } from './Header';
6import { ScriptContainerType } from './ScriptContainer';
7import { deserializeTransactionWireBase, TransactionType, } from './transaction';
8import { BinaryReader, utils } from './utils';
9export 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"]}