// Copyright 2019 The dexon-consensus Authors
// This file is part of the dexon-consensus library.
//
// The dexon-consensus library is free software: you can redistribute it
// and/or modify it under the terms of the GNU Lesser General Public License as
// published by the Free Software Foundation, either version 3 of the License,
// or (at your option) any later version.
//
// The dexon-consensus library is distributed in the hope that it will be
// useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser
// General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the dexon-consensus library. If not, see
// <http://www.gnu.org/licenses/>.
package utils
import (
"errors"
"github.com/dexon-foundation/dexon-consensus/core/types"
typesDKG "github.com/dexon-foundation/dexon-consensus/core/types/dkg"
)
var (
// ErrInvalidDKGMasterPublicKey means the DKG MasterPublicKey is invalid.
ErrInvalidDKGMasterPublicKey = errors.New("invalid DKG master public key")
// ErrPayloadNotEmpty means the payload of block is not empty.
ErrPayloadNotEmpty = errors.New("payload not empty")
)
// NeedPenaltyDKGPrivateShare checks if the proposer of dkg private share
// should be penalized.
func NeedPenaltyDKGPrivateShare(
complaint *typesDKG.Complaint, mpk *typesDKG.MasterPublicKey) (bool, error) {
if complaint.IsNack() {
return false, nil
}
if mpk.ProposerID != complaint.PrivateShare.ProposerID {
return false, nil
}
ok, err := VerifyDKGMasterPublicKeySignature(mpk)
if err != nil {
return false, err
}
if !ok {
return false, ErrInvalidDKGMasterPublicKey
}
ok, err = VerifyDKGComplaintSignature(complaint)
if err != nil {
return false, err
}
if !ok {
return false, nil
}
ok, err = mpk.PublicKeyShares.VerifyPrvShare(
typesDKG.NewID(complaint.PrivateShare.ReceiverID),
&complaint.PrivateShare.PrivateShare)
if err != nil {
return false, err
}
return !ok, nil
}
// NeedPenaltyForkVote checks if two votes are fork vote.
func NeedPenaltyForkVote(vote1, vote2 *types.Vote) (bool, error) {
if vote1.ProposerID != vote2.ProposerID ||
vote1.Type != vote2.Type ||
vote1.Period != vote2.Period ||
vote1.Position != vote2.Position ||
vote1.BlockHash == vote2.BlockHash {
return false, nil
}
ok, err := VerifyVoteSignature(vote1)
if err != nil {
return false, err
}
if !ok {
return false, nil
}
ok, err = VerifyVoteSignature(vote2)
if err != nil {
return false, err
}
if !ok {
return false, nil
}
return true, nil
}
// NeedPenaltyForkBlock checks if two blocks are fork block.
func NeedPenaltyForkBlock(block1, block2 *types.Block) (bool, error) {
if block1.ProposerID != block2.ProposerID ||
block1.Position != block2.Position ||
block1.Hash == block2.Hash {
return false, nil
}
if len(block1.Payload) != 0 || len(block2.Payload) != 0 {
return false, ErrPayloadNotEmpty
}
verifyBlock := func(block *types.Block) (bool, error) {
err := VerifyBlockSignatureWithoutPayload(block)
switch err {
case nil:
return true, nil
case ErrIncorrectSignature:
return false, nil
case ErrIncorrectHash:
return false, nil
default:
return false, err
}
}
ok, err := verifyBlock(block1)
if err != nil {
return false, err
}
if !ok {
return false, nil
}
ok, err = verifyBlock(block2)
if err != nil {
return false, err
}
if !ok {
return false, nil
}
return true, nil
}