1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
// 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")
)
// 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
}
verifyBlock := func(block *types.Block) (bool, error) {
err := VerifyBlockSignature(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
}
|