2019-02-05 13:47:25 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
// Validator client proposer functions.
|
|
|
|
import (
|
|
|
|
"context"
|
2019-02-20 18:58:34 +00:00
|
|
|
"encoding/binary"
|
2019-02-05 13:47:25 +00:00
|
|
|
"fmt"
|
|
|
|
|
2019-04-05 14:58:33 +00:00
|
|
|
"github.com/gogo/protobuf/proto"
|
2019-02-13 23:49:06 +00:00
|
|
|
ptypes "github.com/gogo/protobuf/types"
|
2019-02-05 13:47:25 +00:00
|
|
|
pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-02-25 02:09:45 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
2019-03-12 23:39:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/forkutil"
|
2019-02-28 03:55:47 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2019-02-23 06:06:20 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2019-03-18 15:45:28 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2019-02-28 03:55:47 +00:00
|
|
|
"go.opencensus.io/trace"
|
2019-02-05 13:47:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// ProposeBlock A new beacon block for a given slot. This method collects the
|
|
|
|
// previous beacon block, any pending deposits, and ETH1 data from the beacon
|
|
|
|
// chain node to construct the new block. The new block is then processed with
|
|
|
|
// the state root computation, and finally signed by the validator before being
|
|
|
|
// sent back to the beacon node for broadcasting.
|
2019-04-18 17:23:38 +00:00
|
|
|
func (v *validator) ProposeBlock(ctx context.Context, slot uint64, idx string) {
|
2019-03-14 14:13:57 +00:00
|
|
|
if slot == params.BeaconConfig().GenesisSlot {
|
|
|
|
log.Info("Assigned to genesis slot, skipping proposal")
|
|
|
|
return
|
|
|
|
}
|
2019-02-28 03:55:47 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "validator.ProposeBlock")
|
|
|
|
defer span.End()
|
2019-04-18 17:23:38 +00:00
|
|
|
span.AddAttributes(trace.StringAttribute("validator", fmt.Sprintf("%#x", v.keys[idx].PublicKey.Marshal())))
|
|
|
|
truncatedPk := idx
|
|
|
|
if len(idx) > 12 {
|
|
|
|
truncatedPk = idx[:12]
|
|
|
|
}
|
|
|
|
log.WithFields(logrus.Fields{"validator": truncatedPk}).Info("Performing a beacon block proposal...")
|
2019-02-05 13:47:25 +00:00
|
|
|
// 1. Fetch data from Beacon Chain node.
|
|
|
|
// Get current head beacon block.
|
|
|
|
headBlock, err := v.beaconClient.CanonicalHead(ctx, &ptypes.Empty{})
|
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).Error("Failed to fetch CanonicalHead")
|
2019-02-05 13:47:25 +00:00
|
|
|
return
|
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
parentTreeRoot, err := hashutil.HashBeaconBlock(headBlock)
|
2019-02-05 13:47:25 +00:00
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).Error("Failed to hash parent block")
|
2019-02-05 13:47:25 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-03-18 15:45:28 +00:00
|
|
|
// Get validator ETH1 deposits which have not been included in the beacon chain.
|
2019-02-05 13:47:25 +00:00
|
|
|
pDepResp, err := v.beaconClient.PendingDeposits(ctx, &ptypes.Empty{})
|
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).Error("Failed to get pendings deposits")
|
2019-02-05 13:47:25 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get ETH1 data.
|
|
|
|
eth1DataResp, err := v.beaconClient.Eth1Data(ctx, &ptypes.Empty{})
|
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).Error("Failed to get ETH1 data")
|
2019-02-05 13:47:25 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-02-20 18:58:34 +00:00
|
|
|
// Retrieve the current fork data from the beacon node.
|
|
|
|
fork, err := v.beaconClient.ForkData(ctx, &ptypes.Empty{})
|
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).Error("Failed to get fork data from beacon node's state")
|
2019-02-20 18:58:34 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
// Then, we generate a RandaoReveal by signing the block's slot information using
|
|
|
|
// the validator's private key.
|
|
|
|
// epoch_signature = bls_sign(
|
|
|
|
// privkey=validator.privkey,
|
|
|
|
// message_hash=int_to_bytes32(slot_to_epoch(block.slot)),
|
|
|
|
// domain=get_domain(
|
|
|
|
// fork=fork, # `fork` is the fork object at the slot `block.slot`
|
|
|
|
// epoch=slot_to_epoch(block.slot),
|
|
|
|
// domain_type=DOMAIN_RANDAO,
|
|
|
|
// )
|
|
|
|
// )
|
|
|
|
epoch := slot / params.BeaconConfig().SlotsPerEpoch
|
|
|
|
buf := make([]byte, 32)
|
|
|
|
binary.LittleEndian.PutUint64(buf, epoch)
|
2019-03-12 23:39:13 +00:00
|
|
|
domain := forkutil.DomainVersion(fork, epoch, params.BeaconConfig().DomainRandao)
|
2019-04-18 17:23:38 +00:00
|
|
|
epochSignature := v.keys[idx].SecretKey.Sign(buf, domain)
|
2019-02-20 18:58:34 +00:00
|
|
|
|
2019-02-16 00:36:40 +00:00
|
|
|
// Fetch pending attestations seen by the beacon node.
|
2019-02-25 02:09:45 +00:00
|
|
|
attResp, err := v.proposerClient.PendingAttestations(ctx, &pb.PendingAttestationsRequest{
|
|
|
|
FilterReadyForInclusion: true,
|
2019-03-09 00:29:47 +00:00
|
|
|
ProposalBlockSlot: slot,
|
2019-02-25 02:09:45 +00:00
|
|
|
})
|
2019-02-16 00:36:40 +00:00
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).Error("Failed to fetch pending attestations from the beacon node")
|
2019-02-16 00:36:40 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-02-05 13:47:25 +00:00
|
|
|
// 2. Construct block.
|
|
|
|
block := &pbp2p.BeaconBlock{
|
2019-02-19 20:24:00 +00:00
|
|
|
Slot: slot,
|
2019-02-26 03:42:31 +00:00
|
|
|
ParentRootHash32: parentTreeRoot[:],
|
2019-02-20 18:58:34 +00:00
|
|
|
RandaoReveal: epochSignature.Marshal(),
|
2019-02-19 20:24:00 +00:00
|
|
|
Eth1Data: eth1DataResp.Eth1Data,
|
2019-02-05 13:47:25 +00:00
|
|
|
Body: &pbp2p.BeaconBlockBody{
|
2019-02-16 00:36:40 +00:00
|
|
|
Attestations: attResp.PendingAttestations,
|
2019-02-05 13:47:25 +00:00
|
|
|
ProposerSlashings: nil, // TODO(1438): Add after operations pool
|
|
|
|
AttesterSlashings: nil, // TODO(1438): Add after operations pool
|
|
|
|
Deposits: pDepResp.PendingDeposits,
|
2019-02-18 16:52:16 +00:00
|
|
|
VoluntaryExits: nil, // TODO(1323): Add validator exits
|
2019-02-05 13:47:25 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// 3. Compute state root transition from parent block to the new block.
|
2019-04-05 14:58:33 +00:00
|
|
|
resp, err := v.proposerClient.ComputeStateRoot(ctx, block)
|
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"block": proto.MarshalTextString(block),
|
|
|
|
"validator": truncatedPk,
|
|
|
|
}).WithError(err).Error("Not proposing! Unable to compute state root")
|
2019-04-05 14:58:33 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
block.StateRootHash32 = resp.GetStateRoot()
|
2019-02-05 13:47:25 +00:00
|
|
|
|
|
|
|
// 4. Sign the complete block.
|
|
|
|
// TODO(1366): BLS sign block
|
|
|
|
block.Signature = nil
|
|
|
|
|
|
|
|
// 5. Broadcast to the network via beacon chain node.
|
|
|
|
blkResp, err := v.proposerClient.ProposeBlock(ctx, block)
|
|
|
|
if err != nil {
|
2019-04-18 17:23:38 +00:00
|
|
|
log.WithError(err).WithFields(logrus.Fields{
|
|
|
|
"validator": truncatedPk,
|
|
|
|
}).Error("Failed to propose block")
|
2019-02-05 13:47:25 +00:00
|
|
|
return
|
|
|
|
}
|
2019-04-05 01:07:15 +00:00
|
|
|
span.AddAttributes(
|
|
|
|
trace.StringAttribute("blockRoot", fmt.Sprintf("%#x", blkResp.BlockRootHash32)),
|
|
|
|
trace.Int64Attribute("numDeposits", int64(len(block.Body.Deposits))),
|
|
|
|
trace.Int64Attribute("numAttestations", int64(len(block.Body.Attestations))),
|
|
|
|
)
|
2019-03-18 15:45:28 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
2019-04-19 15:41:22 +00:00
|
|
|
"slot": block.Slot - params.BeaconConfig().GenesisSlot,
|
2019-04-18 17:23:38 +00:00
|
|
|
"blockRoot": fmt.Sprintf("%#x", blkResp.BlockRootHash32),
|
2019-03-18 15:45:28 +00:00
|
|
|
"numAttestations": len(block.Body.Attestations),
|
|
|
|
"numDeposits": len(block.Body.Deposits),
|
2019-04-19 15:41:22 +00:00
|
|
|
"validator": truncatedPk,
|
2019-04-18 17:23:38 +00:00
|
|
|
}).Info("Proposed new beacon block")
|
2019-02-05 13:47:25 +00:00
|
|
|
}
|