aboutsummaryrefslogtreecommitdiffstats
path: root/core/utils/penalty-helper.go
blob: 2b2456c62af1b9f618caea0a70035bb7f1c00f85 (plain) (blame)
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
}