package validator import ( "context" "fmt" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/v4/beacon-chain/builder" consensus_types "github.com/prysmaticlabs/prysm/v4/consensus-types" consensusblocks "github.com/prysmaticlabs/prysm/v4/consensus-types/blocks" "github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces" "github.com/prysmaticlabs/prysm/v4/encoding/bytesutil" enginev1 "github.com/prysmaticlabs/prysm/v4/proto/engine/v1" ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v4/runtime/version" "github.com/sirupsen/logrus" "google.golang.org/protobuf/proto" ) type unblinder struct { b interfaces.SignedBeaconBlock blobs []*ethpb.SignedBlindedBlobSidecar builder builder.BlockBuilder } func newUnblinder(b interfaces.SignedBeaconBlock, blobs []*ethpb.SignedBlindedBlobSidecar, builder builder.BlockBuilder) (*unblinder, error) { if err := consensusblocks.BeaconBlockIsNil(b); err != nil { return nil, err } if builder == nil { return nil, errors.New("nil builder provided") } return &unblinder{ b: b, blobs: blobs, builder: builder, }, nil } func (u *unblinder) unblindBuilderBlock(ctx context.Context) (interfaces.SignedBeaconBlock, []*ethpb.SignedBlobSidecar, error) { if !u.b.IsBlinded() || u.b.Version() < version.Bellatrix { return u.b, nil, nil } if u.b.IsBlinded() && !u.builder.Configured() { return nil, nil, errors.New("builder not configured") } psb, err := u.blindedProtoBlock() if err != nil { return nil, nil, errors.Wrap(err, "could not get blinded proto block") } sb, err := consensusblocks.NewSignedBeaconBlock(psb) if err != nil { return nil, nil, errors.Wrap(err, "could not create signed block") } if err = copyBlockData(u.b, sb); err != nil { return nil, nil, errors.Wrap(err, "could not copy block data") } h, err := u.b.Block().Body().Execution() if err != nil { return nil, nil, errors.Wrap(err, "could not get execution") } if err = sb.SetExecution(h); err != nil { return nil, nil, errors.Wrap(err, "could not set execution") } payload, blobsBundle, err := u.builder.SubmitBlindedBlock(ctx, sb, u.blobs) if err != nil { return nil, nil, errors.Wrap(err, "could not submit blinded block") } headerRoot, err := h.HashTreeRoot() if err != nil { return nil, nil, errors.Wrap(err, "could not get header root") } payloadRoot, err := payload.HashTreeRoot() if err != nil { return nil, nil, errors.Wrap(err, "could not get payload root") } if headerRoot != payloadRoot { return nil, nil, fmt.Errorf("header and payload root do not match, consider disconnect from relay to avoid further issues, "+ "%#x != %#x", headerRoot, payloadRoot) } bb, err := u.protoBlock() if err != nil { return nil, nil, errors.Wrap(err, "could not get proto block") } wb, err := consensusblocks.NewSignedBeaconBlock(bb) if err != nil { return nil, nil, errors.Wrap(err, "could not create signed block") } if err = copyBlockData(sb, wb); err != nil { return nil, nil, errors.Wrap(err, "could not copy block data") } if err = wb.SetExecution(payload); err != nil { return nil, nil, errors.Wrap(err, "could not set execution") } txs, err := payload.Transactions() if err != nil { return nil, nil, errors.Wrap(err, "could not get transactions from payload") } if wb.Version() >= version.Bellatrix && blobsBundle != nil { log.WithField("blobCount", len(blobsBundle.Blobs)) } log.WithFields(logrus.Fields{ "blockHash": fmt.Sprintf("%#x", h.BlockHash()), "feeRecipient": fmt.Sprintf("%#x", h.FeeRecipient()), "gasUsed": h.GasUsed(), "slot": u.b.Block().Slot(), "txs": len(txs), }).Info("Retrieved full payload from builder") bundle, err := unblindBlobsSidecars(u.blobs, blobsBundle) if err != nil { return nil, nil, errors.Wrap(err, "could not unblind blobs sidecars") } return wb, bundle, nil } func unblindBlobsSidecars(blindSidecars []*ethpb.SignedBlindedBlobSidecar, bundle *enginev1.BlobsBundle) ([]*ethpb.SignedBlobSidecar, error) { if bundle == nil { return nil, nil } sidecars := make([]*ethpb.SignedBlobSidecar, len(blindSidecars)) for i, b := range blindSidecars { sidecars[i] = ðpb.SignedBlobSidecar{ Message: ðpb.BlobSidecar{ BlockRoot: bytesutil.SafeCopyBytes(b.Message.BlockRoot), Index: b.Message.Index, Slot: b.Message.Slot, BlockParentRoot: bytesutil.SafeCopyBytes(b.Message.BlockParentRoot), ProposerIndex: b.Message.ProposerIndex, Blob: bytesutil.SafeCopyBytes(bundle.Blobs[i]), KzgCommitment: bytesutil.SafeCopyBytes(b.Message.KzgCommitment), KzgProof: bytesutil.SafeCopyBytes(b.Message.KzgProof), }, Signature: bytesutil.SafeCopyBytes(b.Signature), } } return sidecars, nil } func copyBlockData(src interfaces.SignedBeaconBlock, dst interfaces.SignedBeaconBlock) error { agg, err := src.Block().Body().SyncAggregate() if err != nil { return errors.Wrap(err, "could not get sync aggregate") } parentRoot := src.Block().ParentRoot() stateRoot := src.Block().StateRoot() randaoReveal := src.Block().Body().RandaoReveal() graffiti := src.Block().Body().Graffiti() sig := src.Signature() blsToExecChanges, err := src.Block().Body().BLSToExecutionChanges() if err != nil && !errors.Is(err, consensus_types.ErrUnsupportedField) { return errors.Wrap(err, "could not get bls to execution changes") } kzgCommitments, err := src.Block().Body().BlobKzgCommitments() if err != nil && !errors.Is(err, consensus_types.ErrUnsupportedField) { return errors.Wrap(err, "could not get blob kzg commitments") } dst.SetSlot(src.Block().Slot()) dst.SetProposerIndex(src.Block().ProposerIndex()) dst.SetParentRoot(parentRoot[:]) dst.SetStateRoot(stateRoot[:]) dst.SetRandaoReveal(randaoReveal[:]) dst.SetEth1Data(src.Block().Body().Eth1Data()) dst.SetGraffiti(graffiti[:]) dst.SetProposerSlashings(src.Block().Body().ProposerSlashings()) dst.SetAttesterSlashings(src.Block().Body().AttesterSlashings()) dst.SetAttestations(src.Block().Body().Attestations()) dst.SetDeposits(src.Block().Body().Deposits()) dst.SetVoluntaryExits(src.Block().Body().VoluntaryExits()) if err = dst.SetSyncAggregate(agg); err != nil { return errors.Wrap(err, "could not set sync aggregate") } dst.SetSignature(sig[:]) if err = dst.SetBLSToExecutionChanges(blsToExecChanges); err != nil && !errors.Is(err, consensus_types.ErrUnsupportedField) { return errors.Wrap(err, "could not set bls to execution changes") } if err = dst.SetBlobKzgCommitments(kzgCommitments); err != nil && !errors.Is(err, consensus_types.ErrUnsupportedField) { return errors.Wrap(err, "could not set bls to execution changes") } return nil } func (u *unblinder) blindedProtoBlock() (proto.Message, error) { switch u.b.Version() { case version.Bellatrix: return ðpb.SignedBlindedBeaconBlockBellatrix{ Block: ðpb.BlindedBeaconBlockBellatrix{ Body: ðpb.BlindedBeaconBlockBodyBellatrix{}, }, }, nil case version.Capella: return ðpb.SignedBlindedBeaconBlockCapella{ Block: ðpb.BlindedBeaconBlockCapella{ Body: ðpb.BlindedBeaconBlockBodyCapella{}, }, }, nil case version.Deneb: return ðpb.SignedBlindedBeaconBlockDeneb{ Message: ðpb.BlindedBeaconBlockDeneb{ Body: ðpb.BlindedBeaconBlockBodyDeneb{}, }, }, nil default: return nil, fmt.Errorf("invalid version %s", version.String(u.b.Version())) } } func (u *unblinder) protoBlock() (proto.Message, error) { switch u.b.Version() { case version.Bellatrix: return ðpb.SignedBeaconBlockBellatrix{ Block: ðpb.BeaconBlockBellatrix{ Body: ðpb.BeaconBlockBodyBellatrix{}, }, }, nil case version.Capella: return ðpb.SignedBeaconBlockCapella{ Block: ðpb.BeaconBlockCapella{ Body: ðpb.BeaconBlockBodyCapella{}, }, }, nil case version.Deneb: return ðpb.SignedBeaconBlockDeneb{ Block: ðpb.BeaconBlockDeneb{ Body: ðpb.BeaconBlockBodyDeneb{}, }, }, nil default: return nil, fmt.Errorf("invalid version %s", version.String(u.b.Version())) } }