332 lines
12 KiB
Nim
332 lines
12 KiB
Nim
# Nimbus
|
|
# Copyright (c) 2024 Status Research & Development GmbH
|
|
# Licensed under either of
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
|
|
# at your option.
|
|
# This file may not be copied, modified, or distributed except according to
|
|
# those terms.
|
|
|
|
import
|
|
std/sequtils,
|
|
chronicles,
|
|
../nimbus/constants,
|
|
../nimbus/core/chain,
|
|
./config,
|
|
../nimbus/utils/era_helpers,
|
|
kzg4844/kzg,
|
|
web3,
|
|
web3/[engine_api, primitives, conversions],
|
|
beacon_chain/spec/digest,
|
|
beacon_chain/el/el_conf,
|
|
beacon_chain/el/el_manager,
|
|
beacon_chain/el/engine_api_conversions,
|
|
beacon_chain/spec/[forks, state_transition_block],
|
|
beacon_chain/spec/eth2_apis/[rest_types, rest_beacon_calls],
|
|
beacon_chain/networking/network_metadata,
|
|
eth/async_utils
|
|
|
|
var running* {.volatile.} = true
|
|
|
|
# Load the EL block, from CL ( either head or CL root )
|
|
template getCLBlockFromBeaconChain(
|
|
client: RestClientRef, blockIdent: BlockIdent, clConfig: RuntimeConfig
|
|
): (ForkedSignedBeaconBlock, bool) =
|
|
let clBlock =
|
|
try:
|
|
awaitWithTimeout(client.getBlockV2(blockIdent, clConfig), 30.seconds):
|
|
error "Failed to get CL head"
|
|
quit(QuitFailure)
|
|
except CatchableError as exc:
|
|
error "Error getting CL head", error = exc.msg
|
|
quit(QuitFailure)
|
|
|
|
# Constructing the EL block from the CL block
|
|
var blck: ForkedSignedBeaconBlock
|
|
if clBlock.isSome():
|
|
let blck = clBlock.get()[]
|
|
|
|
(blck, true)
|
|
else:
|
|
(blck, false)
|
|
|
|
# Load the EL block, from CL ( either head or CL root )
|
|
# Also returns the availability of the block as a boolean
|
|
template getELBlockFromBeaconChain(
|
|
client: RestClientRef, blockIdent: BlockIdent, clConfig: RuntimeConfig
|
|
): (EthBlock, bool) =
|
|
let (clBlock, isAvailable) = getCLBlockFromBeaconChain(client, blockIdent, clConfig)
|
|
|
|
# Constructing the EL block from the CL block
|
|
var eth1block: EthBlock
|
|
if isAvailable:
|
|
withBlck(clBlock.asTrusted()):
|
|
eth1Block = getEthBlock(forkyBlck.message).valueOr:
|
|
error "Failed to get EL block from CL head"
|
|
quit(QuitFailure)
|
|
|
|
(eth1Block, true)
|
|
else:
|
|
(eth1Block, false)
|
|
|
|
# Load the network configuration based on the network id
|
|
template loadNetworkConfig(conf: NRpcConf): (RuntimeConfig, uint64, uint64) =
|
|
case conf.networkId
|
|
of MainNet:
|
|
(getMetadataForNetwork("mainnet").cfg, 15537393'u64, 4700013'u64)
|
|
of SepoliaNet:
|
|
(getMetadataForNetwork("sepolia").cfg, 1450408'u64, 115193'u64)
|
|
of HoleskyNet:
|
|
(getMetadataForNetwork("holesky").cfg, 0'u64, 0'u64)
|
|
else:
|
|
error "Unsupported network", network = conf.networkId
|
|
quit(QuitFailure)
|
|
|
|
# Slot Finding Mechanism
|
|
# First it sets the initial lower bound to `firstSlotAfterMerge` + number of blocks after Era1
|
|
# Then it iterates over the slots to find the current slot number, along with reducing the
|
|
# search space by calculating the difference between the `blockNumber` and the `block_number` from the executionPayload
|
|
# of the slot, then adding the difference to the importedSlot. This pushes the lower bound more,
|
|
# making the search way smaller
|
|
template findSlot(
|
|
client: RestClientRef,
|
|
currentBlockNumber: uint64,
|
|
lastEra1Block: uint64,
|
|
firstSlotAfterMerge: uint64,
|
|
): uint64 =
|
|
var importedSlot = (currentBlockNumber - lastEra1Block) + firstSlotAfterMerge
|
|
notice "Finding slot number corresponding to block", importedSlot = importedSlot
|
|
|
|
var clNum = 0'u64
|
|
while running and clNum < currentBlockNumber:
|
|
let (blk, stat) =
|
|
client.getELBlockFromBeaconChain(BlockIdent.init(Slot(importedSlot)), clConfig)
|
|
if not stat:
|
|
importedSlot += 1
|
|
continue
|
|
|
|
clNum = blk.header.number
|
|
# decreasing the lower bound with each iteration
|
|
importedSlot += currentBlockNumber - clNum
|
|
|
|
notice "Found the slot to start with", slot = importedSlot
|
|
importedSlot
|
|
|
|
# The main procedure to sync the EL with the help of CL
|
|
# Takes blocks from the CL and sends them to the EL via the engineAPI
|
|
proc syncToEngineApi(conf: NRpcConf) {.async.} =
|
|
let
|
|
# Load the network configuration, jwt secret and engine api url
|
|
(clConfig, lastEra1Block, firstSlotAfterMerge) = loadNetworkConfig(conf)
|
|
jwtSecret =
|
|
if conf.jwtSecret.isSome():
|
|
loadJwtSecret(Opt.some(conf.jwtSecret.get()))
|
|
else:
|
|
Opt.none(seq[byte])
|
|
engineUrl = EngineApiUrl.init(conf.elEngineApi, jwtSecret)
|
|
|
|
# Create the client for the engine api
|
|
# And exchange the capabilities for a test communication
|
|
web3 = await engineUrl.newWeb3()
|
|
rpcClient = web3.provider
|
|
|
|
try:
|
|
let data = await rpcClient.exchangeCapabilities(
|
|
@[
|
|
"engine_forkchoiceUpdatedV1", "engine_getPayloadBodiesByHash",
|
|
"engine_getPayloadBodiesByRangeV1", "engine_getPayloadV1", "engine_newPayloadV1",
|
|
]
|
|
)
|
|
notice "Communication with the EL Success", data = data
|
|
except CatchableError as exc:
|
|
error "Error connecting to the EL Engine API", error = exc.msg
|
|
quit(QuitFailure)
|
|
|
|
# Get the latest block number from the EL rest api
|
|
template elBlockNumber(): uint64 =
|
|
try:
|
|
uint64(await rpcClient.eth_blockNumber())
|
|
except CatchableError as exc:
|
|
error "Error getting block number", error = exc.msg
|
|
0'u64
|
|
|
|
# Load the EL state detials and create the beaconAPI client
|
|
var
|
|
currentBlockNumber = elBlockNumber() + 1
|
|
curBlck: ForkedSignedBeaconBlock
|
|
client = RestClientRef.new(conf.beaconApi).valueOr:
|
|
error "Cannot connect to Beacon Api", url = conf.beaconApi
|
|
quit(QuitFailure)
|
|
|
|
notice "Current block number", number = currentBlockNumber
|
|
|
|
# Check for pre-merge situation
|
|
if currentBlockNumber <= lastEra1Block:
|
|
notice "Pre-merge, nrpc syncer works post-merge",
|
|
blocknumber = currentBlockNumber, lastPoWBlock = lastEra1Block
|
|
quit(QuitSuccess)
|
|
|
|
# Load the latest state from the CL
|
|
var
|
|
(finalizedBlck, _) = client.getELBlockFromBeaconChain(
|
|
BlockIdent.init(BlockIdentType.Finalized), clConfig
|
|
)
|
|
(headBlck, _) =
|
|
client.getELBlockFromBeaconChain(BlockIdent.init(BlockIdentType.Head), clConfig)
|
|
|
|
# Check if the EL is already in sync or ahead of the CL
|
|
if headBlck.header.number <= currentBlockNumber:
|
|
notice "CL head is behind of EL head, or in sync", head = headBlck.header.number
|
|
quit(QuitSuccess)
|
|
|
|
var
|
|
importedSlot =
|
|
findSlot(client, currentBlockNumber, lastEra1Block, firstSlotAfterMerge)
|
|
finalizedHash = Eth2Digest.fromHex("0x00")
|
|
headHash: Eth2Digest
|
|
|
|
template sendFCU(clblk: ForkedSignedBeaconBlock) =
|
|
withBlck(clblk):
|
|
let
|
|
state = ForkchoiceStateV1(
|
|
headBlockHash: headHash.asBlockHash,
|
|
safeBlockHash: finalizedHash.asBlockHash,
|
|
finalizedBlockHash: finalizedHash.asBlockHash,
|
|
)
|
|
payloadAttributes =
|
|
when consensusFork <= ConsensusFork.Bellatrix:
|
|
Opt.none(PayloadAttributesV1)
|
|
elif consensusFork == ConsensusFork.Capella:
|
|
Opt.none(PayloadAttributesV2)
|
|
elif consensusFork == ConsensusFork.Deneb or
|
|
consensusFork == ConsensusFork.Electra or
|
|
consensusFork == ConsensusFork.Fulu:
|
|
Opt.none(PayloadAttributesV3)
|
|
else:
|
|
static: doAssert(false, "Unsupported consensus fork")
|
|
Opt.none(PayloadAttributesV3)
|
|
|
|
# Make the forkchoiceUpdated call based, after loading attributes based on the consensus fork
|
|
let fcuResponse = await rpcClient.forkchoiceUpdated(state, payloadAttributes)
|
|
debug "forkchoiceUpdated", state = state, response = fcuResponse
|
|
info "forkchoiceUpdated Request sent", response = fcuResponse.payloadStatus.status
|
|
|
|
while running and currentBlockNumber < headBlck.header.number:
|
|
var isAvailable = false
|
|
(curBlck, isAvailable) =
|
|
client.getCLBlockFromBeaconChain(BlockIdent.init(Slot(importedSlot)), clConfig)
|
|
|
|
if not isAvailable:
|
|
importedSlot += 1
|
|
continue
|
|
|
|
importedSlot += 1
|
|
withBlck(curBlck):
|
|
# Don't include blocks before bellatrix, as it doesn't have payload
|
|
when consensusFork >= ConsensusFork.Bellatrix:
|
|
# Load the execution payload for all blocks after the bellatrix upgrade
|
|
let payload = forkyBlck.message.body.asEngineExecutionPayload()
|
|
var payloadResponse: engine_api.PayloadStatusV1
|
|
|
|
# Make the newPayload call based on the consensus fork
|
|
# Before Deneb calls are made without versioned hashes
|
|
# Thus calls will be same for Bellatrix and Capella forks
|
|
# And for Deneb, we will pass the versioned hashes
|
|
when consensusFork <= ConsensusFork.Capella:
|
|
payloadResponse = await rpcClient.newPayload(payload)
|
|
debug "Payload status", response = payloadResponse, payload = payload
|
|
elif consensusFork == ConsensusFork.Deneb:
|
|
# Calculate the versioned hashes from the kzg commitments
|
|
let versioned_hashes = mapIt(
|
|
forkyBlck.message.body.blob_kzg_commitments,
|
|
engine_api.VersionedHash(kzg_commitment_to_versioned_hash(it)),
|
|
)
|
|
payloadResponse = await rpcClient.newPayload(
|
|
payload, versioned_hashes, forkyBlck.message.parent_root.to(Hash32)
|
|
)
|
|
debug "Payload status",
|
|
response = payloadResponse,
|
|
payload = payload,
|
|
versionedHashes = versioned_hashes
|
|
elif consensusFork == ConsensusFork.Electra or
|
|
consensusFork == ConsensusFork.Fulu:
|
|
# Calculate the versioned hashes from the kzg commitments
|
|
let versioned_hashes = mapIt(
|
|
forkyBlck.message.body.blob_kzg_commitments,
|
|
engine_api.VersionedHash(kzg_commitment_to_versioned_hash(it)),
|
|
)
|
|
# Execution Requests for Electra
|
|
let execution_requests = [
|
|
SSZ.encode(forkyBlck.message.body.execution_requests.deposits),
|
|
SSZ.encode(forkyBlck.message.body.execution_requests.withdrawals),
|
|
SSZ.encode(forkyBlck.message.body.execution_requests.consolidations),
|
|
]
|
|
# TODO: Update to `newPayload()` once nim-web3 is updated
|
|
payloadResponse = await rpcClient.engine_newPayloadV4(
|
|
payload,
|
|
versioned_hashes,
|
|
forkyBlck.message.parent_root.to(Hash32),
|
|
execution_requests,
|
|
)
|
|
debug "Payload status",
|
|
response = payloadResponse,
|
|
payload = payload,
|
|
versionedHashes = versioned_hashes,
|
|
executionRequests = execution_requests
|
|
else:
|
|
static: doAssert(false, "Unsupported consensus fork")
|
|
|
|
info "newPayload Request sent",
|
|
blockNumber = int(payload.blockNumber), response = payloadResponse.status
|
|
|
|
# Load the head hash from the execution payload, for forkchoice
|
|
headHash = forkyBlck.message.body.execution_payload.block_hash
|
|
|
|
# Update the finalized hash
|
|
# This is updated after the fcu call is made
|
|
# So that head - head mod 32 is maintained
|
|
# i.e finalized have to be mod slots per epoch == 0
|
|
let blknum = forkyBlck.message.body.execution_payload.block_number
|
|
if blknum < finalizedBlck.header.number and blknum mod 32 == 0:
|
|
finalizedHash = headHash
|
|
# Make the forkchoicestate based on the the last
|
|
# `new_payload` call and the state received from the EL JSON-RPC API
|
|
# And generate the PayloadAttributes based on the consensus fork
|
|
sendFCU(curBlck)
|
|
elif blknum >= finalizedBlck.header.number:
|
|
# If the real finalized block is crossed, then upate the finalized hash to the real one
|
|
(finalizedBlck, _) = client.getELBlockFromBeaconChain(
|
|
BlockIdent.init(BlockIdentType.Finalized), clConfig
|
|
)
|
|
finalizedHash = finalizedBlck.header.blockHash.asEth2Digest
|
|
|
|
# Update the current block number from EL rest api
|
|
# Shows that the fcu call has succeeded
|
|
currentBlockNumber = elBlockNumber()
|
|
(headBlck, _) =
|
|
client.getELBlockFromBeaconChain(BlockIdent.init(BlockIdentType.Head), clConfig)
|
|
|
|
# fcU call for the last remaining payloads
|
|
sendFCU(curBlck)
|
|
|
|
when isMainModule:
|
|
## Ctrl+C handling
|
|
proc controlCHandler() {.noconv.} =
|
|
when defined(windows):
|
|
# workaround for https://github.com/nim-lang/Nim/issues/4057
|
|
setupForeignThreadGc()
|
|
running = false
|
|
|
|
setControlCHook(controlCHandler)
|
|
|
|
## Show logs on stdout until we get the user's logging choice
|
|
discard defaultChroniclesStream.output.open(stdout)
|
|
|
|
## Processing command line arguments
|
|
let conf = makeConfig()
|
|
setLogLevel(conf.logLevel)
|
|
|
|
case conf.cmd
|
|
of NRpcCmd.`sync`:
|
|
waitFor syncToEngineApi(conf)
|