You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
191 lines
6.9 KiB
191 lines
6.9 KiB
package node
|
|
|
|
import (
|
|
"encoding/binary"
|
|
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
"github.com/harmony-one/bls/ffi/go/bls"
|
|
"github.com/harmony-one/harmony/consensus/quorum"
|
|
"github.com/harmony-one/harmony/core/types"
|
|
bls_cosi "github.com/harmony-one/harmony/crypto/bls"
|
|
"github.com/harmony-one/harmony/internal/ctxerror"
|
|
"github.com/harmony-one/harmony/internal/utils"
|
|
"github.com/harmony-one/harmony/shard"
|
|
)
|
|
|
|
const (
|
|
maxPendingCrossLinkSize = 1000
|
|
crossLinkBatchSize = 10
|
|
)
|
|
|
|
// VerifyBlockCrossLinks verifies the cross links of the block
|
|
func (node *Node) VerifyBlockCrossLinks(block *types.Block) error {
|
|
if len(block.Header().CrossLinks()) == 0 {
|
|
utils.Logger().Debug().Msgf("[CrossLinkVerification] Zero CrossLinks in the header")
|
|
return nil
|
|
}
|
|
|
|
crossLinks := &types.CrossLinks{}
|
|
err := rlp.DecodeBytes(block.Header().CrossLinks(), crossLinks)
|
|
if err != nil {
|
|
return ctxerror.New("[CrossLinkVerification] failed to decode cross links",
|
|
"blockHash", block.Hash(),
|
|
"crossLinks", len(block.Header().CrossLinks()),
|
|
).WithCause(err)
|
|
}
|
|
|
|
if !crossLinks.IsSorted() {
|
|
return ctxerror.New("[CrossLinkVerification] cross links are not sorted",
|
|
"blockHash", block.Hash(),
|
|
"crossLinks", len(block.Header().CrossLinks()),
|
|
)
|
|
}
|
|
|
|
for _, crossLink := range *crossLinks {
|
|
cl, err := node.Blockchain().ReadCrossLink(crossLink.ShardID(), crossLink.BlockNum())
|
|
if err == nil && cl != nil {
|
|
// Add slash for exist same blocknum but different crosslink
|
|
return ctxerror.New("crosslink already exist!")
|
|
}
|
|
if err = node.VerifyCrossLink(crossLink); err != nil {
|
|
return ctxerror.New("cannot VerifyBlockCrossLinks",
|
|
"blockHash", block.Hash(),
|
|
"blockNum", block.Number(),
|
|
"crossLinkShard", crossLink.ShardID(),
|
|
"crossLinkBlock", crossLink.BlockNum(),
|
|
"numTx", len(block.Transactions()),
|
|
).WithCause(err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ProcessCrossLinkMessage verify and process Node/CrossLink message into crosslink when it's valid
|
|
func (node *Node) ProcessCrossLinkMessage(msgPayload []byte) {
|
|
if node.NodeConfig.ShardID == 0 {
|
|
pendingCLs, err := node.Blockchain().ReadPendingCrossLinks()
|
|
if err == nil && len(pendingCLs) >= maxPendingCrossLinkSize {
|
|
utils.Logger().Debug().
|
|
Msgf("[ProcessingCrossLink] Pending Crosslink reach maximum size: %d", len(pendingCLs))
|
|
return
|
|
}
|
|
|
|
var crosslinks []types.CrossLink
|
|
err = rlp.DecodeBytes(msgPayload, &crosslinks)
|
|
if err != nil {
|
|
utils.Logger().Error().
|
|
Err(err).
|
|
Msg("[ProcessingCrossLink] Crosslink Message Broadcast Unable to Decode")
|
|
return
|
|
}
|
|
|
|
candidates := []types.CrossLink{}
|
|
utils.Logger().Debug().
|
|
Msgf("[ProcessingCrossLink] Received crosslinks: %d", len(crosslinks))
|
|
|
|
for i, cl := range crosslinks {
|
|
if i > crossLinkBatchSize {
|
|
break
|
|
}
|
|
exist, err := node.Blockchain().ReadCrossLink(cl.ShardID(), cl.Number().Uint64())
|
|
if err == nil && exist != nil {
|
|
// TODO: leader add double sign checking
|
|
utils.Logger().Err(err).
|
|
Msgf("[ProcessingCrossLink] Cross Link already exists, pass. Beacon Epoch: %d, Block num: %d, Epoch: %d, shardID %d", node.Blockchain().CurrentHeader().Epoch(), cl.Number(), cl.Epoch(), cl.ShardID())
|
|
continue
|
|
}
|
|
|
|
if err = node.VerifyCrossLink(cl); err != nil {
|
|
utils.Logger().Err(err).
|
|
Msgf("[ProcessingCrossLink] Failed to verify new cross link for blockNum %d epochNum %d shard %d skipped: %v", cl.BlockNum(), cl.Epoch().Uint64(), cl.ShardID(), cl)
|
|
continue
|
|
}
|
|
|
|
candidates = append(candidates, cl)
|
|
utils.Logger().Debug().
|
|
Msgf("[ProcessingCrossLink] Committing for shardID %d, blockNum %d", cl.ShardID(), cl.Number().Uint64())
|
|
}
|
|
node.pendingCLMutex.Lock()
|
|
Len, _ := node.Blockchain().AddPendingCrossLinks(candidates)
|
|
node.pendingCLMutex.Unlock()
|
|
utils.Logger().Debug().
|
|
Msgf("[ProcessingCrossLink] Add pending crosslinks, total pending: %d", Len)
|
|
}
|
|
}
|
|
|
|
// VerifyCrossLink verifies the header is valid
|
|
func (node *Node) VerifyCrossLink(cl types.CrossLink) error {
|
|
if node.Blockchain().ShardID() != shard.BeaconChainShardID {
|
|
return ctxerror.New("[VerifyCrossLink] Shard chains should not verify cross links")
|
|
}
|
|
|
|
if cl.BlockNum() <= 1 {
|
|
return ctxerror.New("[VerifyCrossLink] CrossLink BlockNumber should greater than 1")
|
|
}
|
|
|
|
if !node.Blockchain().Config().IsCrossLink(cl.Epoch()) {
|
|
return ctxerror.New("[VerifyCrossLink] CrossLink Epoch should >= cross link starting epoch", "crossLinkEpoch", cl.Epoch(), "cross_link_starting_eoch", node.Blockchain().Config().CrossLinkEpoch)
|
|
}
|
|
|
|
// Verify signature of the new cross link header
|
|
// TODO: check whether to recalculate shard state
|
|
shardState, err := node.Blockchain().ReadShardState(cl.Epoch())
|
|
committee := shardState.FindCommitteeByID(cl.ShardID())
|
|
|
|
if err != nil || committee == nil {
|
|
return ctxerror.New("[VerifyCrossLink] Failed to read shard state for cross link", "beaconEpoch", node.Blockchain().CurrentHeader().Epoch(), "epoch", cl.Epoch(), "shardID", cl.ShardID(), "blockNum", cl.BlockNum()).WithCause(err)
|
|
}
|
|
var committerKeys []*bls.PublicKey
|
|
|
|
parseKeysSuccess := true
|
|
for _, member := range committee.Slots {
|
|
committerKey := new(bls.PublicKey)
|
|
err = member.BlsPublicKey.ToLibBLSPublicKey(committerKey)
|
|
if err != nil {
|
|
parseKeysSuccess = false
|
|
break
|
|
}
|
|
committerKeys = append(committerKeys, committerKey)
|
|
}
|
|
if !parseKeysSuccess {
|
|
return ctxerror.New("[VerifyCrossLink] cannot convert BLS public key", "shardID", cl.ShardID(), "blockNum", cl.BlockNum()).WithCause(err)
|
|
}
|
|
|
|
mask, err := bls_cosi.NewMask(committerKeys, nil)
|
|
if err != nil {
|
|
return ctxerror.New("[VerifyCrossLink] cannot create group sig mask", "shardID", cl.ShardID(), "blockNum", cl.BlockNum()).WithCause(err)
|
|
}
|
|
if err := mask.SetMask(cl.Bitmap()); err != nil {
|
|
return ctxerror.New("[VerifyCrossLink] cannot set group sig mask bits", "shardID", cl.ShardID(), "blockNum", cl.BlockNum()).WithCause(err)
|
|
}
|
|
|
|
decider := quorum.NewDecider(quorum.SuperMajorityStake)
|
|
decider.SetShardIDProvider(func() (uint32, error) {
|
|
return cl.ShardID(), nil
|
|
})
|
|
decider.SetMyPublicKeyProvider(func() (*bls.PublicKey, error) {
|
|
return nil, nil
|
|
})
|
|
if _, err := decider.SetVoters(committee.Slots, false); err != nil {
|
|
return ctxerror.New("[VerifyCrossLink] Cannot SetVoters for committee", "shardID", cl.ShardID())
|
|
}
|
|
if !decider.IsQuorumAchievedByMask(mask, false) {
|
|
return ctxerror.New("[VerifyCrossLink] Not enough voting power for crosslink", "shardID", cl.ShardID())
|
|
}
|
|
|
|
aggSig := bls.Sign{}
|
|
sig := cl.Signature()
|
|
err = aggSig.Deserialize(sig[:])
|
|
if err != nil {
|
|
return ctxerror.New("[VerifyCrossLink] unable to deserialize multi-signature from payload").WithCause(err)
|
|
}
|
|
|
|
hash := cl.Hash()
|
|
blockNumBytes := make([]byte, 8)
|
|
binary.LittleEndian.PutUint64(blockNumBytes, cl.BlockNum())
|
|
commitPayload := append(blockNumBytes, hash[:]...)
|
|
if !aggSig.VerifyHash(mask.AggregatePublic, commitPayload) {
|
|
return ctxerror.New("[VerifyCrossLink] Failed to verify the signature for cross link", "shardID", cl.ShardID(), "blockNum", cl.BlockNum())
|
|
}
|
|
return nil
|
|
}
|
|
|