package client import ( "context" "fmt" fssz "github.com/ferranbt/fastssz" emptypb "github.com/golang/protobuf/ptypes/empty" types "github.com/prysmaticlabs/eth2-types" "github.com/prysmaticlabs/prysm/beacon-chain/core/altair" "github.com/prysmaticlabs/prysm/beacon-chain/core/signing" "github.com/prysmaticlabs/prysm/config/params" "github.com/prysmaticlabs/prysm/crypto/bls" "github.com/prysmaticlabs/prysm/encoding/bytesutil" "github.com/prysmaticlabs/prysm/monitoring/tracing" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" validatorpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1/validator-client" "github.com/prysmaticlabs/prysm/time/slots" "github.com/sirupsen/logrus" "go.opencensus.io/trace" ) // SubmitSyncCommitteeMessage submits the sync committee message to the beacon chain. func (v *validator) SubmitSyncCommitteeMessage(ctx context.Context, slot types.Slot, pubKey [48]byte) { ctx, span := trace.StartSpan(ctx, "validator.SubmitSyncCommitteeMessage") defer span.End() span.AddAttributes(trace.StringAttribute("validator", fmt.Sprintf("%#x", pubKey))) v.waitOneThirdOrValidBlock(ctx, slot) res, err := v.validatorClient.GetSyncMessageBlockRoot(ctx, &emptypb.Empty{}) if err != nil { log.WithError(err).Error("Could not request sync message block root to sign") tracing.AnnotateError(span, err) return } duty, err := v.duty(pubKey) if err != nil { log.WithError(err).Error("Could not fetch validator assignment") return } d, err := v.domainData(ctx, slots.ToEpoch(slot), params.BeaconConfig().DomainSyncCommittee[:]) if err != nil { log.WithError(err).Error("Could not get sync committee domain data") return } sszRoot := types.SSZBytes(res.Root) r, err := signing.ComputeSigningRoot(&sszRoot, d.SignatureDomain) if err != nil { log.WithError(err).Error("Could not get sync committee message signing root") return } sig, err := v.keyManager.Sign(ctx, &validatorpb.SignRequest{ PublicKey: pubKey[:], SigningRoot: r[:], SignatureDomain: d.SignatureDomain, }) if err != nil { log.WithError(err).Error("Could not sign sync committee message") return } msg := ðpb.SyncCommitteeMessage{ Slot: slot, BlockRoot: res.Root, ValidatorIndex: duty.ValidatorIndex, Signature: sig.Marshal(), } if _, err := v.validatorClient.SubmitSyncMessage(ctx, msg); err != nil { log.WithError(err).Error("Could not submit sync committee message") return } log.WithFields(logrus.Fields{ "slot": msg.Slot, "blockRoot": fmt.Sprintf("%#x", bytesutil.Trunc(msg.BlockRoot)), "validatorIndex": msg.ValidatorIndex, }).Info("Submitted new sync message") } // SubmitSignedContributionAndProof submits the signed sync committee contribution and proof to the beacon chain. func (v *validator) SubmitSignedContributionAndProof(ctx context.Context, slot types.Slot, pubKey [48]byte) { ctx, span := trace.StartSpan(ctx, "validator.SubmitSignedContributionAndProof") defer span.End() span.AddAttributes(trace.StringAttribute("validator", fmt.Sprintf("%#x", pubKey))) duty, err := v.duty(pubKey) if err != nil { log.Errorf("Could not fetch validator assignment: %v", err) return } indexRes, err := v.validatorClient.GetSyncSubcommitteeIndex(ctx, ðpb.SyncSubcommitteeIndexRequest{ PublicKey: pubKey[:], Slot: slot, }) if err != nil { log.Errorf("Could not get sync subcommittee index: %v", err) return } if len(indexRes.Indices) == 0 { log.Debug("Empty subcommittee index list, do nothing") return } selectionProofs, err := v.selectionProofs(ctx, slot, pubKey, indexRes) if err != nil { log.Errorf("Could not get selection proofs: %v", err) return } v.waitToSlotTwoThirds(ctx, slot) for i, comIdx := range indexRes.Indices { isAggregator, err := altair.IsSyncCommitteeAggregator(selectionProofs[i]) if err != nil { log.Errorf("Could check in aggregator: %v", err) return } if !isAggregator { continue } subCommitteeSize := params.BeaconConfig().SyncCommitteeSize / params.BeaconConfig().SyncCommitteeSubnetCount subnet := uint64(comIdx) / subCommitteeSize contribution, err := v.validatorClient.GetSyncCommitteeContribution(ctx, ðpb.SyncCommitteeContributionRequest{ Slot: slot, PublicKey: pubKey[:], SubnetId: subnet, }) if err != nil { log.Errorf("Could not get sync committee contribution: %v", err) return } if contribution.AggregationBits.Count() == 0 { log.WithFields(logrus.Fields{ "slot": slot, "pubkey": pubKey, "subnet": subnet, }).Warn("Sync contribution for validator has no bits set.") continue } contributionAndProof := ðpb.ContributionAndProof{ AggregatorIndex: duty.ValidatorIndex, Contribution: contribution, SelectionProof: selectionProofs[i], } sig, err := v.signContributionAndProof(ctx, pubKey, contributionAndProof) if err != nil { log.Errorf("Could not sign contribution and proof: %v", err) return } if _, err := v.validatorClient.SubmitSignedContributionAndProof(ctx, ðpb.SignedContributionAndProof{ Message: contributionAndProof, Signature: sig, }); err != nil { log.Errorf("Could not submit signed contribution and proof: %v", err) return } log.WithFields(logrus.Fields{ "slot": contributionAndProof.Contribution.Slot, "blockRoot": fmt.Sprintf("%#x", bytesutil.Trunc(contributionAndProof.Contribution.BlockRoot)), "subcommitteeIndex": contributionAndProof.Contribution.SubcommitteeIndex, "aggregatorIndex": contributionAndProof.AggregatorIndex, "bitsCount": contributionAndProof.Contribution.AggregationBits.Count(), }).Info("Submitted new sync contribution and proof") } } // Signs and returns selection proofs per validator for slot and pub key. func (v *validator) selectionProofs(ctx context.Context, slot types.Slot, pubKey [48]byte, indexRes *ethpb.SyncSubcommitteeIndexResponse) ([][]byte, error) { selectionProofs := make([][]byte, len(indexRes.Indices)) cfg := params.BeaconConfig() size := cfg.SyncCommitteeSize subCount := cfg.SyncCommitteeSubnetCount for i, index := range indexRes.Indices { subSize := size / subCount subnet := uint64(index) / subSize selectionProof, err := v.signSyncSelectionData(ctx, pubKey, subnet, slot) if err != nil { return nil, err } selectionProofs[i] = selectionProof } return selectionProofs, nil } // Signs input slot with domain sync committee selection proof. This is used to create the signature for sync committee selection. func (v *validator) signSyncSelectionData(ctx context.Context, pubKey [48]byte, index uint64, slot types.Slot) (signature []byte, err error) { domain, err := v.domainData(ctx, slots.ToEpoch(slot), params.BeaconConfig().DomainSyncCommitteeSelectionProof[:]) if err != nil { return nil, err } data := ðpb.SyncAggregatorSelectionData{ Slot: slot, SubcommitteeIndex: index, } sig, err := v.computeAndSign(ctx, data, pubKey, domain.SignatureDomain) if err != nil { return nil, err } return sig.Marshal(), nil } // This returns the signature of validator signing over sync committee contribution and proof object. func (v *validator) signContributionAndProof(ctx context.Context, pubKey [48]byte, c *ethpb.ContributionAndProof) ([]byte, error) { d, err := v.domainData(ctx, slots.ToEpoch(c.Contribution.Slot), params.BeaconConfig().DomainContributionAndProof[:]) if err != nil { return nil, err } sig, err := v.computeAndSign(ctx, c, pubKey, d.SignatureDomain) if err != nil { return nil, err } return sig.Marshal(), nil } // This computes the signing root of hash tree root capable object `obj` and signs it using public key `pubKey` along with the signature domain `sigDomain`. func (v *validator) computeAndSign(ctx context.Context, obj fssz.HashRoot, pubKey [48]byte, sigDomain []byte) (bls.Signature, error) { root, err := signing.ComputeSigningRoot(obj, sigDomain) if err != nil { return nil, err } return v.keyManager.Sign(ctx, &validatorpb.SignRequest{ PublicKey: pubKey[:], SigningRoot: root[:], SignatureDomain: sigDomain, }) }