mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-07 10:12:19 +00:00
060a311a14
* comments on core package declarations * fixed sentence * cap * no extra line * fixed tests * gazelle
123 lines
4.7 KiB
Go
123 lines
4.7 KiB
Go
// Package attestations tracks the life-cycle of the latest attestations
|
|
// from each validator. It also contains libraries to create attestation
|
|
// message, verify attestation correctness and slashing conditions.
|
|
package attestations
|
|
|
|
import (
|
|
"encoding/binary"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
)
|
|
|
|
// CreateAttestationMsg hashes parentHashes + shardID + slotNumber +
|
|
// shardBlockHash + justifiedSlot into a message to use for verifying
|
|
// with aggregated public key and signature.
|
|
func CreateAttestationMsg(
|
|
blockHash []byte,
|
|
slot uint64,
|
|
shardID uint64,
|
|
justifiedSlot uint64,
|
|
forkVersion uint64,
|
|
) [32]byte {
|
|
msg := make([]byte, binary.MaxVarintLen64)
|
|
binary.BigEndian.PutUint64(msg, forkVersion)
|
|
binary.PutUvarint(msg, slot%params.BeaconConfig().EpochLength)
|
|
binary.PutUvarint(msg, shardID)
|
|
msg = append(msg, blockHash...)
|
|
binary.PutUvarint(msg, justifiedSlot)
|
|
return hashutil.Hash(msg)
|
|
}
|
|
|
|
// Key generates the blake2b hash of the following attestation fields:
|
|
// slotNumber + shardID + blockHash + obliqueParentHash
|
|
// This is used for attestation table look up in localDB.
|
|
func Key(att *pb.AttestationData) [32]byte {
|
|
key := make([]byte, binary.MaxVarintLen64)
|
|
binary.PutUvarint(key, att.Slot)
|
|
binary.PutUvarint(key, att.Shard)
|
|
key = append(key, att.ShardBlockRootHash32...)
|
|
return hashutil.Hash(key)
|
|
}
|
|
|
|
// VerifyProposerAttestation verifies the proposer's attestation of the block.
|
|
// Proposers broadcast the attestation along with the block to its peers.
|
|
func VerifyProposerAttestation(att *pb.AttestationData, pubKey [32]byte, proposerShardID uint64) error {
|
|
// Verify the attestation attached with block response.
|
|
// Get proposer index and shardID.
|
|
attestationMsg := CreateAttestationMsg(
|
|
att.ShardBlockRootHash32,
|
|
att.Slot,
|
|
proposerShardID,
|
|
att.JustifiedSlot,
|
|
params.BeaconConfig().GenesisForkVersion,
|
|
)
|
|
_ = attestationMsg
|
|
_ = pubKey
|
|
// TODO(#258): use attestationMsg to verify against signature
|
|
// and public key. Return error if incorrect.
|
|
return nil
|
|
}
|
|
|
|
// ContainsValidator checks if the validator is included in the attestation.
|
|
// TODO(#569): Modify method to accept a single index rather than a bitfield.
|
|
func ContainsValidator(attesterBitfield []byte, bitfield []byte) bool {
|
|
for i := 0; i < len(bitfield); i++ {
|
|
if bitfield[i]&attesterBitfield[i] != 0 {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// IsDoubleVote checks if both of the attestations have been used to vote for the same slot.
|
|
// Spec:
|
|
// def is_double_vote(attestation_data_1: AttestationData,
|
|
// attestation_data_2: AttestationData) -> bool
|
|
// """
|
|
// Assumes ``attestation_data_1`` is distinct from ``attestation_data_2``.
|
|
// Returns True if the provided ``AttestationData`` are slashable
|
|
// due to a 'double vote'.
|
|
// """
|
|
// target_epoch_1 = attestation_data_1.slot // EPOCH_LENGTH
|
|
// target_epoch_2 = attestation_data_2.slot // EPOCH_LENGTH
|
|
// return target_epoch_1 == target_epoch_2
|
|
func IsDoubleVote(attestation1 *pb.AttestationData, attestation2 *pb.AttestationData) bool {
|
|
epochLength := params.BeaconConfig().EpochLength
|
|
return attestation1.Slot/epochLength == attestation2.Slot/epochLength
|
|
}
|
|
|
|
// IsSurroundVote checks if the data provided by the attestations fulfill the conditions for
|
|
// a surround vote.
|
|
// Spec:
|
|
// def is_surround_vote(attestation_data_1: AttestationData,
|
|
// attestation_data_2: AttestationData) -> bool:
|
|
// """
|
|
// Assumes ``attestation_data_1`` is distinct from ``attestation_data_2``.
|
|
// Returns True if the provided ``AttestationData`` are slashable
|
|
// due to a 'surround vote'.
|
|
// Note: parameter order matters as this function only checks
|
|
// that ``attestation_data_1`` surrounds ``attestation_data_2``.
|
|
// """
|
|
// source_epoch_1 = attestation_data_1.justified_slot // EPOCH_LENGTH
|
|
// source_epoch_2 = attestation_data_2.justified_slot // EPOCH_LENGTH
|
|
// target_epoch_1 = attestation_data_1.slot // EPOCH_LENGTH
|
|
// target_epoch_2 = attestation_data_2.slot // EPOCH_LENGTH
|
|
// return (
|
|
// (source_epoch_1 < source_epoch_2) and
|
|
// (source_epoch_2 + 1 == target_epoch_2) and
|
|
// (target_epoch_2 < target_epoch_1)
|
|
// )
|
|
func IsSurroundVote(attestation1 *pb.AttestationData, attestation2 *pb.AttestationData) bool {
|
|
epochLength := params.BeaconConfig().EpochLength
|
|
sourceEpoch1 := attestation1.JustifiedSlot / epochLength
|
|
sourceEpoch2 := attestation2.JustifiedSlot / epochLength
|
|
targetEpoch1 := attestation1.Slot / epochLength
|
|
targetEpoch2 := attestation2.Slot / epochLength
|
|
|
|
return sourceEpoch1 < sourceEpoch2 &&
|
|
sourceEpoch2+1 == targetEpoch2 &&
|
|
targetEpoch2 < targetEpoch1
|
|
}
|