BlockHeader.java
/*
* Copyright Hyperledger Besu Contributors.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
* an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*
* SPDX-License-Identifier: Apache-2.0
*/
package org.hyperledger.besu.ethereum.core;
import org.hyperledger.besu.datatypes.Address;
import org.hyperledger.besu.datatypes.BlobGas;
import org.hyperledger.besu.datatypes.Hash;
import org.hyperledger.besu.datatypes.Wei;
import org.hyperledger.besu.ethereum.rlp.RLPInput;
import org.hyperledger.besu.ethereum.rlp.RLPOutput;
import org.hyperledger.besu.evm.log.LogsBloomFilter;
import java.util.Objects;
import java.util.function.Supplier;
import com.google.common.base.Suppliers;
import org.apache.tuweni.bytes.Bytes;
import org.apache.tuweni.bytes.Bytes32;
/** A mined Ethereum block header. */
public class BlockHeader extends SealableBlockHeader
implements org.hyperledger.besu.plugin.data.BlockHeader {
public static final int MAX_EXTRA_DATA_BYTES = 32;
public static final long GENESIS_BLOCK_NUMBER = 0L;
private final long nonce;
private final Supplier<Hash> hash;
private final Supplier<ParsedExtraData> parsedExtraData;
public BlockHeader(
final Hash parentHash,
final Hash ommersHash,
final Address coinbase,
final Hash stateRoot,
final Hash transactionsRoot,
final Hash receiptsRoot,
final LogsBloomFilter logsBloom,
final Difficulty difficulty,
final long number,
final long gasLimit,
final long gasUsed,
final long timestamp,
final Bytes extraData,
final Wei baseFee,
final Bytes32 mixHashOrPrevRandao,
final long nonce,
final Hash withdrawalsRoot,
final Long blobGasUsed,
final BlobGas excessBlobGas,
final Bytes32 parentBeaconBlockRoot,
final Hash depositsRoot,
final Hash exitsRoot,
final BlockHeaderFunctions blockHeaderFunctions) {
super(
parentHash,
ommersHash,
coinbase,
stateRoot,
transactionsRoot,
receiptsRoot,
logsBloom,
difficulty,
number,
gasLimit,
gasUsed,
timestamp,
extraData,
baseFee,
mixHashOrPrevRandao,
withdrawalsRoot,
blobGasUsed,
excessBlobGas,
parentBeaconBlockRoot,
depositsRoot,
exitsRoot);
this.nonce = nonce;
this.hash = Suppliers.memoize(() -> blockHeaderFunctions.hash(this));
this.parsedExtraData = Suppliers.memoize(() -> blockHeaderFunctions.parseExtraData(this));
}
/**
* Returns the block mixed hash.
*
* @return the block mixed hash
*/
@Override
public Hash getMixHash() {
return Hash.wrap(mixHashOrPrevRandao);
}
/**
* Returns the block nonce.
*
* @return the block nonce
*/
@Override
public long getNonce() {
return nonce;
}
/**
* Returns the block extra data field, as parsed by the {@link BlockHeaderFunctions}.
*
* @return the block extra data field
*/
public ParsedExtraData getParsedExtraData() {
return parsedExtraData.get();
}
/**
* Returns the block header hash.
*
* @return the block header hash
*/
public Hash getHash() {
return hash.get();
}
@Override
public Hash getBlockHash() {
return hash.get();
}
/**
* Write an RLP representation.
*
* @param out The RLP output to write to
*/
public void writeTo(final RLPOutput out) {
out.startList();
out.writeBytes(parentHash);
out.writeBytes(ommersHash);
out.writeBytes(coinbase);
out.writeBytes(stateRoot);
out.writeBytes(transactionsRoot);
out.writeBytes(receiptsRoot);
out.writeBytes(logsBloom);
out.writeUInt256Scalar(difficulty);
out.writeLongScalar(number);
out.writeLongScalar(gasLimit);
out.writeLongScalar(gasUsed);
out.writeLongScalar(timestamp);
out.writeBytes(extraData);
out.writeBytes(mixHashOrPrevRandao);
out.writeLong(nonce);
if (baseFee != null) {
out.writeUInt256Scalar(baseFee);
}
if (withdrawalsRoot != null) {
out.writeBytes(withdrawalsRoot);
}
if (excessBlobGas != null && blobGasUsed != null) {
out.writeLongScalar(blobGasUsed);
out.writeUInt64Scalar(excessBlobGas);
}
if (parentBeaconBlockRoot != null) {
out.writeBytes(parentBeaconBlockRoot);
}
if (depositsRoot != null) {
out.writeBytes(depositsRoot);
}
if (exitsRoot != null) {
out.writeBytes(exitsRoot);
}
out.endList();
}
public static BlockHeader readFrom(
final RLPInput input, final BlockHeaderFunctions blockHeaderFunctions) {
input.enterList();
final Hash parentHash = Hash.wrap(input.readBytes32());
final Hash ommersHash = Hash.wrap(input.readBytes32());
final Address coinbase = Address.readFrom(input);
final Hash stateRoot = Hash.wrap(input.readBytes32());
final Hash transactionsRoot = Hash.wrap(input.readBytes32());
final Hash receiptsRoot = Hash.wrap(input.readBytes32());
final LogsBloomFilter logsBloom = LogsBloomFilter.readFrom(input);
final Difficulty difficulty = Difficulty.of(input.readUInt256Scalar());
final long number = input.readLongScalar();
final long gasLimit = input.readLongScalar();
final long gasUsed = input.readLongScalar();
final long timestamp = input.readLongScalar();
final Bytes extraData = input.readBytes();
final Bytes32 mixHashOrPrevRandao = input.readBytes32();
final long nonce = input.readLong();
final Wei baseFee = !input.isEndOfCurrentList() ? Wei.of(input.readUInt256Scalar()) : null;
final Hash withdrawalHashRoot =
!(input.isEndOfCurrentList() || input.isZeroLengthString())
? Hash.wrap(input.readBytes32())
: null;
final Long blobGasUsed = !input.isEndOfCurrentList() ? input.readLongScalar() : null;
final BlobGas excessBlobGas =
!input.isEndOfCurrentList() ? BlobGas.of(input.readUInt64Scalar()) : null;
final Bytes32 parentBeaconBlockRoot = !input.isEndOfCurrentList() ? input.readBytes32() : null;
final Hash depositHashRoot =
!input.isEndOfCurrentList() ? Hash.wrap(input.readBytes32()) : null;
final Hash exitsHashRoot = !input.isEndOfCurrentList() ? Hash.wrap(input.readBytes32()) : null;
input.leaveList();
return new BlockHeader(
parentHash,
ommersHash,
coinbase,
stateRoot,
transactionsRoot,
receiptsRoot,
logsBloom,
difficulty,
number,
gasLimit,
gasUsed,
timestamp,
extraData,
baseFee,
mixHashOrPrevRandao,
nonce,
withdrawalHashRoot,
blobGasUsed,
excessBlobGas,
parentBeaconBlockRoot,
depositHashRoot,
exitsHashRoot,
blockHeaderFunctions);
}
@Override
public boolean equals(final Object obj) {
if (obj == this) {
return true;
}
if (!(obj instanceof BlockHeader other)) {
return false;
}
return getHash().equals(other.getHash());
}
@Override
public int hashCode() {
return Objects.hash(getHash());
}
@Override
public String toString() {
final StringBuilder sb = new StringBuilder();
sb.append("BlockHeader{");
sb.append("number=").append(number).append(", ");
sb.append("hash=").append(getHash()).append(", ");
sb.append("parentHash=").append(parentHash).append(", ");
sb.append("ommersHash=").append(ommersHash).append(", ");
sb.append("coinbase=").append(coinbase).append(", ");
sb.append("stateRoot=").append(stateRoot).append(", ");
sb.append("transactionsRoot=").append(transactionsRoot).append(", ");
sb.append("receiptsRoot=").append(receiptsRoot).append(", ");
sb.append("logsBloom=").append(logsBloom).append(", ");
sb.append("difficulty=").append(difficulty).append(", ");
sb.append("gasLimit=").append(gasLimit).append(", ");
sb.append("gasUsed=").append(gasUsed).append(", ");
sb.append("timestamp=").append(timestamp).append(", ");
sb.append("extraData=").append(extraData).append(", ");
sb.append("baseFee=").append(baseFee).append(", ");
sb.append("mixHashOrPrevRandao=").append(mixHashOrPrevRandao).append(", ");
sb.append("nonce=").append(nonce).append(", ");
if (withdrawalsRoot != null) {
sb.append("withdrawalsRoot=").append(withdrawalsRoot).append(", ");
}
if (blobGasUsed != null && excessBlobGas != null) {
sb.append("blobGasUsed=").append(blobGasUsed).append(", ");
sb.append("excessBlobGas=").append(excessBlobGas).append(", ");
}
if (parentBeaconBlockRoot != null) {
sb.append("parentBeaconBlockRoot=").append(parentBeaconBlockRoot).append(", ");
}
if (depositsRoot != null) {
sb.append("depositsRoot=").append(depositsRoot);
}
if (exitsRoot != null) {
sb.append("exitsRoot=").append(exitsRoot);
}
return sb.append("}").toString();
}
public static org.hyperledger.besu.ethereum.core.BlockHeader convertPluginBlockHeader(
final org.hyperledger.besu.plugin.data.BlockHeader pluginBlockHeader,
final BlockHeaderFunctions blockHeaderFunctions) {
return new org.hyperledger.besu.ethereum.core.BlockHeader(
Hash.fromHexString(pluginBlockHeader.getParentHash().toHexString()),
Hash.fromHexString(pluginBlockHeader.getOmmersHash().toHexString()),
Address.fromHexString(pluginBlockHeader.getCoinbase().toHexString()),
Hash.fromHexString(pluginBlockHeader.getStateRoot().toHexString()),
Hash.fromHexString(pluginBlockHeader.getTransactionsRoot().toHexString()),
Hash.fromHexString(pluginBlockHeader.getReceiptsRoot().toHexString()),
LogsBloomFilter.fromHexString(pluginBlockHeader.getLogsBloom().toHexString()),
Difficulty.fromHexString(pluginBlockHeader.getDifficulty().toHexString()),
pluginBlockHeader.getNumber(),
pluginBlockHeader.getGasLimit(),
pluginBlockHeader.getGasUsed(),
pluginBlockHeader.getTimestamp(),
pluginBlockHeader.getExtraData(),
pluginBlockHeader.getBaseFee().map(Wei::fromQuantity).orElse(null),
pluginBlockHeader.getPrevRandao().orElse(null),
pluginBlockHeader.getNonce(),
pluginBlockHeader
.getWithdrawalsRoot()
.map(h -> Hash.fromHexString(h.toHexString()))
.orElse(null),
pluginBlockHeader.getBlobGasUsed().map(Long::longValue).orElse(null),
pluginBlockHeader.getExcessBlobGas().map(BlobGas::fromQuantity).orElse(null),
pluginBlockHeader.getParentBeaconBlockRoot().orElse(null),
pluginBlockHeader
.getDepositsRoot()
.map(h -> Hash.fromHexString(h.toHexString()))
.orElse(null),
pluginBlockHeader.getExitsRoot().map(h -> Hash.fromHexString(h.toHexString())).orElse(null),
blockHeaderFunctions);
}
@Override
public String toLogString() {
return getNumber() + " (" + getHash() + ")";
}
}