2023-04-08 01:01:10 +00:00
|
|
|
package forkchoice
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-05-14 22:12:24 +00:00
|
|
|
|
2023-09-29 21:42:07 +00:00
|
|
|
"github.com/Giulio2002/bls"
|
2023-05-23 18:58:34 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes/solid"
|
2023-09-29 21:42:07 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/fork"
|
2023-05-13 21:44:07 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state"
|
2023-09-29 21:42:07 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/pool"
|
2023-04-08 01:01:10 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
|
|
|
)
|
|
|
|
|
2023-09-29 21:42:07 +00:00
|
|
|
func (f *ForkChoiceStore) OnAttesterSlashing(attesterSlashing *cltypes.AttesterSlashing, test bool) error {
|
|
|
|
if f.operationsPool.AttesterSlashingsPool.Has(pool.ComputeKeyForAttesterSlashing(attesterSlashing)) {
|
|
|
|
return nil
|
|
|
|
}
|
2023-04-08 01:01:10 +00:00
|
|
|
f.mu.Lock()
|
2023-09-29 21:42:07 +00:00
|
|
|
// Check if this attestation is even slashable.
|
2023-04-08 01:01:10 +00:00
|
|
|
attestation1 := attesterSlashing.Attestation_1
|
|
|
|
attestation2 := attesterSlashing.Attestation_2
|
|
|
|
if !cltypes.IsSlashableAttestationData(attestation1.Data, attestation2.Data) {
|
2023-09-29 21:42:07 +00:00
|
|
|
f.mu.Unlock()
|
2023-04-08 01:01:10 +00:00
|
|
|
return fmt.Errorf("attestation data is not slashable")
|
|
|
|
}
|
|
|
|
// Retrieve justified state
|
2023-10-21 21:10:58 +00:00
|
|
|
s, err := f.forkGraph.GetState(f.justifiedCheckpoint.BlockRoot(), false)
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
2023-09-29 21:42:07 +00:00
|
|
|
f.mu.Unlock()
|
2023-04-08 01:01:10 +00:00
|
|
|
return err
|
|
|
|
}
|
2023-05-04 13:18:42 +00:00
|
|
|
if s == nil {
|
2023-09-29 21:42:07 +00:00
|
|
|
f.mu.Unlock()
|
2023-04-08 01:01:10 +00:00
|
|
|
return fmt.Errorf("justified checkpoint state not accessible")
|
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
attestation1PublicKeys, err := getIndexedAttestationPublicKeys(s, attestation1)
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
2023-09-29 21:42:07 +00:00
|
|
|
f.mu.Unlock()
|
|
|
|
return err
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
attestation2PublicKeys, err := getIndexedAttestationPublicKeys(s, attestation2)
|
|
|
|
if err != nil {
|
|
|
|
f.mu.Unlock()
|
|
|
|
return err
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
domain1, err := s.GetDomain(s.BeaconConfig().DomainBeaconAttester, attestation1.Data.Target().Epoch())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to get the domain: %v", err)
|
|
|
|
}
|
|
|
|
domain2, err := s.GetDomain(s.BeaconConfig().DomainBeaconAttester, attestation2.Data.Target().Epoch())
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
2023-09-29 21:42:07 +00:00
|
|
|
return fmt.Errorf("unable to get the domain: %v", err)
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
f.mu.Unlock()
|
|
|
|
|
|
|
|
if !test {
|
|
|
|
// Verify validity of slashings (1)
|
|
|
|
signingRoot, err := fork.ComputeSigningRoot(attestation1.Data, domain1)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to get signing root: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
valid, err := bls.VerifyAggregate(attestation1.Signature[:], signingRoot[:], attestation1PublicKeys)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error while validating signature: %v", err)
|
|
|
|
}
|
|
|
|
if !valid {
|
|
|
|
return fmt.Errorf("invalid aggregate signature")
|
|
|
|
}
|
|
|
|
// Verify validity of slashings (2)
|
|
|
|
signingRoot, err = fork.ComputeSigningRoot(attestation2.Data, domain2)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to get signing root: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
valid, err = bls.VerifyAggregate(attestation2.Signature[:], signingRoot[:], attestation2PublicKeys)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error while validating signature: %v", err)
|
|
|
|
}
|
|
|
|
if !valid {
|
|
|
|
return fmt.Errorf("invalid aggregate signature")
|
|
|
|
}
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
var anySlashed bool
|
2023-05-23 18:58:34 +00:00
|
|
|
for _, index := range solid.IntersectionOfSortedSets(attestation1.AttestingIndices, attestation2.AttestingIndices) {
|
2024-01-06 20:49:23 +00:00
|
|
|
f.setUnequivocating(index)
|
2023-09-29 21:42:07 +00:00
|
|
|
if !anySlashed {
|
|
|
|
v, err := s.ValidatorForValidatorIndex(int(index))
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to retrieve state: %v", err)
|
|
|
|
}
|
|
|
|
if v.IsSlashable(state.Epoch(s)) {
|
|
|
|
anySlashed = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if anySlashed {
|
|
|
|
f.operationsPool.AttesterSlashingsPool.Insert(pool.ComputeKeyForAttesterSlashing(attesterSlashing), attesterSlashing)
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
|
|
|
|
func getIndexedAttestationPublicKeys(b *state.CachingBeaconState, att *cltypes.IndexedAttestation) ([][]byte, error) {
|
|
|
|
inds := att.AttestingIndices
|
|
|
|
if inds.Length() == 0 || !solid.IsUint64SortedSet(inds) {
|
|
|
|
return nil, fmt.Errorf("isValidIndexedAttestation: attesting indices are not sorted or are null")
|
|
|
|
}
|
|
|
|
pks := make([][]byte, 0, inds.Length())
|
|
|
|
if err := solid.RangeErr[uint64](inds, func(_ int, v uint64, _ int) error {
|
|
|
|
val, err := b.ValidatorForValidatorIndex(int(v))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
pk := val.PublicKey()
|
|
|
|
pks = append(pks, pk[:])
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return pks, nil
|
|
|
|
}
|