support electra block proposals for internal BN validators (#6187)

This commit is contained in:
tersec 2024-04-09 10:04:33 +00:00 committed by GitHub
parent b8eb51852c
commit 6ce5d5814c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
10 changed files with 116 additions and 42 deletions

View File

@ -105,7 +105,6 @@ type
processingDelay*: Opt[Duration] processingDelay*: Opt[Duration]
lastValidAttestedBlock*: Opt[BlockSlot] lastValidAttestedBlock*: Opt[BlockSlot]
# TODO stew/sequtils2
template findIt*(s: openArray, predicate: untyped): int = template findIt*(s: openArray, predicate: untyped): int =
var res = -1 var res = -1
for i, it {.inject.} in s: for i, it {.inject.} in s:

View File

@ -168,7 +168,6 @@ func get_attesting_indices*(shufflingRef: ShufflingRef,
committee_index: CommitteeIndex, committee_index: CommitteeIndex,
bits: CommitteeValidatorsBits): bits: CommitteeValidatorsBits):
seq[ValidatorIndex] = seq[ValidatorIndex] =
# TODO sequtils2 mapIt
for idx in get_attesting_indices(shufflingRef, slot, committee_index, bits): for idx in get_attesting_indices(shufflingRef, slot, committee_index, bits):
result.add(idx) result.add(idx)

View File

@ -88,6 +88,7 @@ RestJson.useDefaultSerializationFor(
GetForkScheduleResponse, GetForkScheduleResponse,
GetGenesisResponse, GetGenesisResponse,
GetHeaderResponseDeneb, GetHeaderResponseDeneb,
GetHeaderResponseElectra,
GetKeystoresResponse, GetKeystoresResponse,
GetNextWithdrawalsResponse, GetNextWithdrawalsResponse,
GetPoolAttesterSlashingsResponse, GetPoolAttesterSlashingsResponse,
@ -253,8 +254,10 @@ RestJson.useDefaultSerializationFor(
electra.SignedBeaconBlock, electra.SignedBeaconBlock,
electra_mev.BlindedBeaconBlock, electra_mev.BlindedBeaconBlock,
electra_mev.BlindedBeaconBlockBody, electra_mev.BlindedBeaconBlockBody,
electra_mev.BuilderBid,
electra_mev.ExecutionPayloadAndBlobsBundle, electra_mev.ExecutionPayloadAndBlobsBundle,
electra_mev.SignedBlindedBeaconBlock, electra_mev.SignedBlindedBeaconBlock,
electra_mev.SignedBuilderBid,
phase0.BeaconBlock, phase0.BeaconBlock,
phase0.BeaconBlockBody, phase0.BeaconBlockBody,
phase0.BeaconState, phase0.BeaconState,

View File

@ -548,6 +548,7 @@ type
GetForkScheduleResponse* = DataEnclosedObject[seq[Fork]] GetForkScheduleResponse* = DataEnclosedObject[seq[Fork]]
GetGenesisResponse* = DataEnclosedObject[RestGenesis] GetGenesisResponse* = DataEnclosedObject[RestGenesis]
GetHeaderResponseDeneb* = DataVersionEnclosedObject[deneb_mev.SignedBuilderBid] GetHeaderResponseDeneb* = DataVersionEnclosedObject[deneb_mev.SignedBuilderBid]
GetHeaderResponseElectra* = DataVersionEnclosedObject[electra_mev.SignedBuilderBid]
GetNetworkIdentityResponse* = DataEnclosedObject[RestNetworkIdentity] GetNetworkIdentityResponse* = DataEnclosedObject[RestNetworkIdentity]
GetPeerCountResponse* = DataMetaEnclosedObject[RestPeerCount] GetPeerCountResponse* = DataMetaEnclosedObject[RestPeerCount]
GetPeerResponse* = DataMetaEnclosedObject[RestNodePeer] GetPeerResponse* = DataMetaEnclosedObject[RestNodePeer]

View File

@ -13,6 +13,14 @@ import
export chronos, client, rest_types, eth2_rest_serialization export chronos, client, rest_types, eth2_rest_serialization
proc getHeaderElectra*(slot: Slot,
parent_hash: Eth2Digest,
pubkey: ValidatorPubKey
): RestPlainResponse {.
rest, endpoint: "/eth/v1/builder/header/{slot}/{parent_hash}/{pubkey}",
meth: MethodGet, connection: {Dedicated, Close}.}
## https://github.com/ethereum/builder-specs/blob/v0.4.0/apis/builder/header.yaml
proc submitBlindedBlock*(body: electra_mev.SignedBlindedBeaconBlock proc submitBlindedBlock*(body: electra_mev.SignedBlindedBeaconBlock
): RestPlainResponse {. ): RestPlainResponse {.
rest, endpoint: "/eth/v1/builder/blinded_blocks", rest, endpoint: "/eth/v1/builder/blinded_blocks",

View File

@ -371,7 +371,8 @@ proc verify_contribution_and_proof_signature*(
# https://github.com/ethereum/builder-specs/blob/v0.4.0/specs/bellatrix/builder.md#signing # https://github.com/ethereum/builder-specs/blob/v0.4.0/specs/bellatrix/builder.md#signing
func compute_builder_signing_root( func compute_builder_signing_root(
fork: Fork, fork: Fork,
msg: deneb_mev.BuilderBid | ValidatorRegistrationV1): Eth2Digest = msg: deneb_mev.BuilderBid | electra_mev.BuilderBid |
ValidatorRegistrationV1): Eth2Digest =
# Uses genesis fork version regardless # Uses genesis fork version regardless
doAssert fork.current_version == fork.previous_version doAssert fork.current_version == fork.previous_version
@ -386,7 +387,7 @@ proc get_builder_signature*(
blsSign(privkey, signing_root.data) blsSign(privkey, signing_root.data)
proc verify_builder_signature*( proc verify_builder_signature*(
fork: Fork, msg: deneb_mev.BuilderBid, fork: Fork, msg: deneb_mev.BuilderBid | electra_mev.BuilderBid,
pubkey: ValidatorPubKey | CookedPubKey, signature: SomeSig): bool = pubkey: ValidatorPubKey | CookedPubKey, signature: SomeSig): bool =
let signing_root = compute_builder_signing_root(fork, msg) let signing_root = compute_builder_signing_root(fork, msg)
blsVerify(pubkey, signing_root.data, signature) blsVerify(pubkey, signing_root.data, signature)

View File

@ -573,7 +573,6 @@ proc process_sync_aggregate*(
# Apply participant and proposer rewards # Apply participant and proposer rewards
let indices = get_sync_committee_cache(state, cache).current_sync_committee let indices = get_sync_committee_cache(state, cache).current_sync_committee
# TODO could use a sequtils2 zipIt
for i in 0 ..< min( for i in 0 ..< min(
state.current_sync_committee.pubkeys.len, state.current_sync_committee.pubkeys.len,
sync_aggregate.sync_committee_bits.len): sync_aggregate.sync_committee_bits.len):

View File

@ -565,7 +565,8 @@ proc makeBeaconBlockForHeadAndSlot*(
kzg_commitments = Opt.none(KzgCommitments)) kzg_commitments = Opt.none(KzgCommitments))
proc getBlindedExecutionPayload[ proc getBlindedExecutionPayload[
EPH: deneb_mev.BlindedExecutionPayloadAndBlobsBundle]( EPH: deneb_mev.BlindedExecutionPayloadAndBlobsBundle |
electra_mev.BlindedExecutionPayloadAndBlobsBundle](
node: BeaconNode, payloadBuilderClient: RestClientRef, slot: Slot, node: BeaconNode, payloadBuilderClient: RestClientRef, slot: Slot,
executionBlockHash: Eth2Digest, pubkey: ValidatorPubKey): executionBlockHash: Eth2Digest, pubkey: ValidatorPubKey):
Future[BlindedBlockResult[EPH]] {.async: (raises: [CancelledError, RestError]).} = Future[BlindedBlockResult[EPH]] {.async: (raises: [CancelledError, RestError]).} =
@ -587,6 +588,22 @@ proc getBlindedExecutionPayload[
"Unable to decode Deneb blinded header: " & $res.error & "Unable to decode Deneb blinded header: " & $res.error &
" with HTTP status " & $response.status & ", Content-Type " & " with HTTP status " & $response.status & ", Content-Type " &
$response.contentType & " and content " & $response.data) $response.contentType & " and content " & $response.data)
elif EPH is electra_mev.BlindedExecutionPayloadAndBlobsBundle:
let
response = awaitWithTimeout(
payloadBuilderClient.getHeaderElectra(
slot, executionBlockHash, pubkey),
BUILDER_PROPOSAL_DELAY_TOLERANCE):
return err "Timeout obtaining Electra blinded header from builder"
res = decodeBytes(
GetHeaderResponseElectra, response.data, response.contentType)
blindedHeader = res.valueOr:
return err(
"Unable to decode Electra blinded header: " & $res.error &
" with HTTP status " & $response.status & ", Content-Type " &
$response.contentType & " and content " & $response.data)
else: else:
static: doAssert false static: doAssert false
@ -599,15 +616,12 @@ proc getBlindedExecutionPayload[
blindedHeader.data.message.pubkey, blindedHeader.data.signature): blindedHeader.data.message.pubkey, blindedHeader.data.signature):
return err "getBlindedExecutionPayload: signature verification failed" return err "getBlindedExecutionPayload: signature verification failed"
when EPH is deneb_mev.BlindedExecutionPayloadAndBlobsBundle: template builderBid: untyped = blindedHeader.data.message
template builderBid: untyped = blindedHeader.data.message return ok(BuilderBid[EPH](
return ok(BuilderBid[EPH]( blindedBlckPart: EPH(
blindedBlckPart: EPH( execution_payload_header: builderBid.header,
execution_payload_header: builderBid.header, blob_kzg_commitments: builderBid.blob_kzg_commitments),
blob_kzg_commitments: builderBid.blob_kzg_commitments), executionPayloadValue: builderBid.value))
executionPayloadValue: builderBid.value))
else:
static: doAssert false
from ./message_router_mev import from ./message_router_mev import
copyFields, getFieldNames, unblindAndRouteBlockMEV copyFields, getFieldNames, unblindAndRouteBlockMEV
@ -634,6 +648,28 @@ proc constructSignableBlindedBlock[T: deneb_mev.SignedBlindedBeaconBlock](
blindedBlock blindedBlock
proc constructSignableBlindedBlock[T: electra_mev.SignedBlindedBeaconBlock](
blck: electra.BeaconBlock,
blindedBundle: electra_mev.BlindedExecutionPayloadAndBlobsBundle): T =
# Leaves signature field default, to be filled in by caller
const
blckFields = getFieldNames(typeof(blck))
blckBodyFields = getFieldNames(typeof(blck.body))
var blindedBlock: T
# https://github.com/ethereum/builder-specs/blob/v0.4.0/specs/bellatrix/validator.md#block-proposal
copyFields(blindedBlock.message, blck, blckFields)
copyFields(blindedBlock.message.body, blck.body, blckBodyFields)
assign(
blindedBlock.message.body.execution_payload_header,
blindedBundle.execution_payload_header)
assign(
blindedBlock.message.body.blob_kzg_commitments,
blindedBundle.blob_kzg_commitments)
blindedBlock
func constructPlainBlindedBlock[T: deneb_mev.BlindedBeaconBlock]( func constructPlainBlindedBlock[T: deneb_mev.BlindedBeaconBlock](
blck: ForkyBeaconBlock, blck: ForkyBeaconBlock,
blindedBundle: deneb_mev.BlindedExecutionPayloadAndBlobsBundle): T = blindedBundle: deneb_mev.BlindedExecutionPayloadAndBlobsBundle): T =
@ -658,7 +694,9 @@ func constructPlainBlindedBlock[T: deneb_mev.BlindedBeaconBlock](
blindedBlock blindedBlock
proc blindedBlockCheckSlashingAndSign[T: deneb_mev.SignedBlindedBeaconBlock]( proc blindedBlockCheckSlashingAndSign[
T: deneb_mev.SignedBlindedBeaconBlock |
electra_mev.SignedBlindedBeaconBlock](
node: BeaconNode, slot: Slot, validator: AttachedValidator, node: BeaconNode, slot: Slot, validator: AttachedValidator,
validator_index: ValidatorIndex, nonsignedBlindedBlock: T): validator_index: ValidatorIndex, nonsignedBlindedBlock: T):
Future[Result[T, string]] {.async: (raises: [CancelledError]).} = Future[Result[T, string]] {.async: (raises: [CancelledError]).} =
@ -690,17 +728,21 @@ proc blindedBlockCheckSlashingAndSign[T: deneb_mev.SignedBlindedBeaconBlock](
return ok blindedBlock return ok blindedBlock
proc getUnsignedBlindedBeaconBlock[T: deneb_mev.SignedBlindedBeaconBlock]( proc getUnsignedBlindedBeaconBlock[
T: deneb_mev.SignedBlindedBeaconBlock |
electra_mev.SignedBlindedBeaconBlock](
node: BeaconNode, slot: Slot, node: BeaconNode, slot: Slot,
validator_index: ValidatorIndex, forkedBlock: ForkedBeaconBlock, validator_index: ValidatorIndex, forkedBlock: ForkedBeaconBlock,
executionPayloadHeader: capella.ExecutionPayloadHeader | executionPayloadHeader: deneb_mev.BlindedExecutionPayloadAndBlobsBundle |
deneb_mev.BlindedExecutionPayloadAndBlobsBundle): electra_mev.BlindedExecutionPayloadAndBlobsBundle):
Result[T, string] = Result[T, string] =
withBlck(forkedBlock): withBlck(forkedBlock):
when consensusFork >= ConsensusFork.Deneb: when consensusFork >= ConsensusFork.Deneb:
when not ( when not (
(T is deneb_mev.SignedBlindedBeaconBlock and (T is deneb_mev.SignedBlindedBeaconBlock and
consensusFork == ConsensusFork.Deneb)): consensusFork == ConsensusFork.Deneb) or
(T is electra_mev.SignedBlindedBeaconBlock and
consensusFork == ConsensusFork.Electra)):
return err("getUnsignedBlindedBeaconBlock: mismatched block/payload types") return err("getUnsignedBlindedBeaconBlock: mismatched block/payload types")
else: else:
return ok constructSignableBlindedBlock[T]( return ok constructSignableBlindedBlock[T](
@ -709,8 +751,8 @@ proc getUnsignedBlindedBeaconBlock[T: deneb_mev.SignedBlindedBeaconBlock](
return err("getUnsignedBlindedBeaconBlock: attempt to construct pre-Deneb blinded block") return err("getUnsignedBlindedBeaconBlock: attempt to construct pre-Deneb blinded block")
proc getBlindedBlockParts[ proc getBlindedBlockParts[
EPH: capella.ExecutionPayloadHeader | EPH: deneb_mev.BlindedExecutionPayloadAndBlobsBundle |
deneb_mev.BlindedExecutionPayloadAndBlobsBundle]( electra_mev.BlindedExecutionPayloadAndBlobsBundle](
node: BeaconNode, payloadBuilderClient: RestClientRef, head: BlockRef, node: BeaconNode, payloadBuilderClient: RestClientRef, head: BlockRef,
pubkey: ValidatorPubKey, slot: Slot, randao: ValidatorSig, pubkey: ValidatorPubKey, slot: Slot, randao: ValidatorSig,
validator_index: ValidatorIndex, graffiti: GraffitiBytes): validator_index: ValidatorIndex, graffiti: GraffitiBytes):
@ -755,18 +797,7 @@ proc getBlindedBlockParts[
# #
# This doesn't have withdrawals, which each node has regardless of engine or # This doesn't have withdrawals, which each node has regardless of engine or
# builder API. makeBeaconBlockForHeadAndSlot fills it in later. # builder API. makeBeaconBlockForHeadAndSlot fills it in later.
when EPH is capella.ExecutionPayloadHeader: when EPH is deneb_mev.BlindedExecutionPayloadAndBlobsBundle:
type PayloadType = capella.ExecutionPayloadForSigning
template actualEPH: untyped = executionPayloadHeader.get.blindedBlckPart
let withdrawals_root =
Opt.some executionPayloadHeader.get.blindedBlckPart.withdrawals_root
const kzg_commitments = Opt.none KzgCommitments
var shimExecutionPayload: PayloadType
copyFields(
shimExecutionPayload.executionPayload,
executionPayloadHeader.get.blindedBlckPart, getFieldNames(EPH))
elif EPH is deneb_mev.BlindedExecutionPayloadAndBlobsBundle:
type PayloadType = deneb.ExecutionPayloadForSigning type PayloadType = deneb.ExecutionPayloadForSigning
template actualEPH: untyped = template actualEPH: untyped =
executionPayloadHeader.get.blindedBlckPart.execution_payload_header executionPayloadHeader.get.blindedBlckPart.execution_payload_header
@ -780,6 +811,20 @@ proc getBlindedBlockParts[
deneb_mev.BlindedExecutionPayloadAndBlobsBundle.execution_payload_header deneb_mev.BlindedExecutionPayloadAndBlobsBundle.execution_payload_header
copyFields( copyFields(
shimExecutionPayload.executionPayload, actualEPH, getFieldNames(DenebEPH)) shimExecutionPayload.executionPayload, actualEPH, getFieldNames(DenebEPH))
elif EPH is electra_mev.BlindedExecutionPayloadAndBlobsBundle:
type PayloadType = electra.ExecutionPayloadForSigning
template actualEPH: untyped =
executionPayloadHeader.get.blindedBlckPart.execution_payload_header
let
withdrawals_root = Opt.some actualEPH.withdrawals_root
kzg_commitments = Opt.some(
executionPayloadHeader.get.blindedBlckPart.blob_kzg_commitments)
var shimExecutionPayload: PayloadType
type ElectraEPH =
electra_mev.BlindedExecutionPayloadAndBlobsBundle.execution_payload_header
copyFields(
shimExecutionPayload.executionPayload, actualEPH, getFieldNames(ElectraEPH))
else: else:
static: doAssert false static: doAssert false
@ -803,7 +848,9 @@ proc getBlindedBlockParts[
forkedBlck.consensusBlockValue, forkedBlck.consensusBlockValue,
forkedBlck.blck)) forkedBlck.blck))
proc getBuilderBid[SBBB: deneb_mev.SignedBlindedBeaconBlock]( proc getBuilderBid[
SBBB: deneb_mev.SignedBlindedBeaconBlock |
electra_mev.SignedBlindedBeaconBlock](
node: BeaconNode, payloadBuilderClient: RestClientRef, head: BlockRef, node: BeaconNode, payloadBuilderClient: RestClientRef, head: BlockRef,
validator_pubkey: ValidatorPubKey, slot: Slot, randao: ValidatorSig, validator_pubkey: ValidatorPubKey, slot: Slot, randao: ValidatorSig,
graffitiBytes: GraffitiBytes, validator_index: ValidatorIndex): graffitiBytes: GraffitiBytes, validator_index: ValidatorIndex):
@ -812,6 +859,8 @@ proc getBuilderBid[SBBB: deneb_mev.SignedBlindedBeaconBlock](
## Used by the BN's own validators, but not the REST server ## Used by the BN's own validators, but not the REST server
when SBBB is deneb_mev.SignedBlindedBeaconBlock: when SBBB is deneb_mev.SignedBlindedBeaconBlock:
type EPH = deneb_mev.BlindedExecutionPayloadAndBlobsBundle type EPH = deneb_mev.BlindedExecutionPayloadAndBlobsBundle
elif SBBB is electra_mev.SignedBlindedBeaconBlock:
type EPH = electra_mev.BlindedExecutionPayloadAndBlobsBundle
else: else:
static: doAssert false static: doAssert false
@ -842,7 +891,9 @@ proc getBuilderBid[SBBB: deneb_mev.SignedBlindedBeaconBlock](
proc proposeBlockMEV( proc proposeBlockMEV(
node: BeaconNode, payloadBuilderClient: RestClientRef, node: BeaconNode, payloadBuilderClient: RestClientRef,
blindedBlock: deneb_mev.SignedBlindedBeaconBlock): blindedBlock:
deneb_mev.SignedBlindedBeaconBlock |
electra_mev.SignedBlindedBeaconBlock):
Future[Result[BlockRef, string]] {.async: (raises: [CancelledError]).} = Future[Result[BlockRef, string]] {.async: (raises: [CancelledError]).} =
let unblindedBlockRef = await node.unblindAndRouteBlockMEV( let unblindedBlockRef = await node.unblindAndRouteBlockMEV(
payloadBuilderClient, blindedBlock) payloadBuilderClient, blindedBlock)
@ -1198,10 +1249,7 @@ proc proposeBlock(node: BeaconNode,
genesis_validators_root, node.config.localBlockValueBoost) genesis_validators_root, node.config.localBlockValueBoost)
return withConsensusFork(node.dag.cfg.consensusForkAtEpoch(slot.epoch)): return withConsensusFork(node.dag.cfg.consensusForkAtEpoch(slot.epoch)):
when consensusFork >= ConsensusFork.Electra: when consensusFork >= ConsensusFork.Deneb:
debugRaiseAssert "can't propose electra block"
return default(BlockRef)
elif consensusFork >= ConsensusFork.Deneb:
proposeBlockContinuation( proposeBlockContinuation(
consensusFork.SignedBlindedBeaconBlock, consensusFork.SignedBlindedBeaconBlock,
consensusFork.ExecutionPayloadForSigning) consensusFork.ExecutionPayloadForSigning)

View File

@ -33,11 +33,13 @@ macro copyFields*(
dst: untyped, src: untyped, fieldNames: static[seq[string]]): untyped = dst: untyped, src: untyped, fieldNames: static[seq[string]]): untyped =
result = newStmtList() result = newStmtList()
for name in fieldNames: for name in fieldNames:
debugRaiseAssert "deposit_receipts_root and exits_root are not currently filled in anywhere properly, so blinded electra proposals will fail"
if name notin [ if name notin [
# These fields are the ones which vary between the blinded and # These fields are the ones which vary between the blinded and
# unblinded objects, and can't simply be copied. # unblinded objects, and can't simply be copied.
"transactions_root", "execution_payload", "transactions_root", "execution_payload",
"execution_payload_header", "body", "withdrawals_root"]: "execution_payload_header", "body", "withdrawals_root",
"deposit_receipts_root", "exits_root"]:
# TODO use stew/assign2 # TODO use stew/assign2
result.add newAssignment( result.add newAssignment(
newDotExpr(dst, ident(name)), newDotExpr(src, ident(name))) newDotExpr(dst, ident(name)), newDotExpr(src, ident(name)))

View File

@ -502,7 +502,8 @@ proc getBlockSignature*(v: AttachedValidator, fork: Fork,
block_root: Eth2Digest, block_root: Eth2Digest,
blck: ForkedBeaconBlock | ForkedBlindedBeaconBlock | blck: ForkedBeaconBlock | ForkedBlindedBeaconBlock |
ForkedMaybeBlindedBeaconBlock | ForkedMaybeBlindedBeaconBlock |
deneb_mev.BlindedBeaconBlock deneb_mev.BlindedBeaconBlock |
electra_mev.BlindedBeaconBlock
): Future[SignatureResult] ): Future[SignatureResult]
{.async: (raises: [CancelledError]).} = {.async: (raises: [CancelledError]).} =
type SomeBlockBody = type SomeBlockBody =
@ -578,6 +579,19 @@ proc getBlockSignature*(v: AttachedValidator, fork: Fork,
Web3SignerForkedBeaconBlock(kind: ConsensusFork.Deneb, Web3SignerForkedBeaconBlock(kind: ConsensusFork.Deneb,
data: blck.toBeaconBlockHeader), data: blck.toBeaconBlockHeader),
proofs) proofs)
elif blck is electra_mev.BlindedBeaconBlock:
case v.data.remoteType
of RemoteSignerType.Web3Signer:
Web3SignerRequest.init(fork, genesis_validators_root,
Web3SignerForkedBeaconBlock(kind: ConsensusFork.Electra,
data: blck.toBeaconBlockHeader))
of RemoteSignerType.VerifyingWeb3Signer:
let proofs = blockPropertiesProofs(
blck.body, electraIndex)
Web3SignerRequest.init(fork, genesis_validators_root,
Web3SignerForkedBeaconBlock(kind: ConsensusFork.Electra,
data: blck.toBeaconBlockHeader),
proofs)
elif blck is ForkedMaybeBlindedBeaconBlock: elif blck is ForkedMaybeBlindedBeaconBlock:
withForkyMaybeBlindedBlck(blck): withForkyMaybeBlindedBlck(blck):
# TODO why isn't this a case statement # TODO why isn't this a case statement