Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(lib/grandpa): Include equivocatory nodes while creating justification #1911

Merged
merged 46 commits into from
Nov 24, 2021
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
ef6546f
feat: include equivocatory nodes while creating just
EclesioMeloJunior Oct 20, 2021
35fea30
chore: remove unecessary error return
EclesioMeloJunior Oct 20, 2021
1ceb49d
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Oct 22, 2021
64b96fc
chore: undo condition format
EclesioMeloJunior Oct 22, 2021
c2bf981
chore: use equivocatory votes on threshold count
EclesioMeloJunior Oct 26, 2021
5039268
chore: add unsupported subround error
EclesioMeloJunior Oct 26, 2021
0100140
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Oct 26, 2021
e2bafd8
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 1, 2021
b627509
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 2, 2021
d626dae
chore: remove unecessary check
EclesioMeloJunior Nov 2, 2021
b0f3c27
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 3, 2021
7d21850
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 3, 2021
e6299b5
chore: remove eqv votes and use the eqv voters
EclesioMeloJunior Nov 3, 2021
bbe8ba7
Merge branch 'eclesio/include-eqv-votes' of github.com:ChainSafe/goss…
EclesioMeloJunior Nov 3, 2021
0ed4ce5
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 3, 2021
a5f4684
chore: remove unecessary convertion
EclesioMeloJunior Nov 3, 2021
c9c278c
Merge branch 'eclesio/include-eqv-votes' of github.com:ChainSafe/goss…
EclesioMeloJunior Nov 3, 2021
0274ef9
chore: export ErrUnsupportedSubround
EclesioMeloJunior Nov 3, 2021
0b0c524
chore: improve code reading
EclesioMeloJunior Nov 4, 2021
6e5dd0b
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 4, 2021
de8a4ed
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 11, 2021
f58afce
chore: add tests to grandpa.createJustification func
EclesioMeloJunior Nov 15, 2021
56f3d05
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 15, 2021
ad81602
chore: add tests to grandpa message handler files
EclesioMeloJunior Nov 15, 2021
c93a134
chore: fix ci warns
EclesioMeloJunior Nov 16, 2021
7abd69b
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 16, 2021
0301b49
chore: put back a condition to check enough precommits in the justifi…
EclesioMeloJunior Nov 17, 2021
ec61d8e
chore: check if the equivocatory votes are on justification
EclesioMeloJunior Nov 17, 2021
a7d9ecb
chore: fix the fakeAuthorities format
EclesioMeloJunior Nov 17, 2021
f1f4d04
chore: removing parallel and uncoment Err...
EclesioMeloJunior Nov 17, 2021
5730645
chore: adjusting the equivocatory and prevote testing
EclesioMeloJunior Nov 17, 2021
69ee68d
chore: fix lint warns
EclesioMeloJunior Nov 17, 2021
5091650
chore: address nit comments
EclesioMeloJunior Nov 19, 2021
349080b
chore: adjust test asserts
EclesioMeloJunior Nov 19, 2021
32856fa
chore: adjust naming
EclesioMeloJunior Nov 19, 2021
c10dc71
chore: mock time.Unix in tests
EclesioMeloJunior Nov 19, 2021
a0dd708
chore: add test to equivocatory voters on to VerifyBlockJustification
EclesioMeloJunior Nov 19, 2021
37b2ba3
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 22, 2021
add5820
chore: adjust testing
EclesioMeloJunior Nov 22, 2021
e3546d8
chore: check errors
EclesioMeloJunior Nov 23, 2021
9ac7015
chore: changing the style to checking ok variable
EclesioMeloJunior Nov 23, 2021
3a4ee28
Merge branch 'development' into eclesio/include-eqv-votes
EclesioMeloJunior Nov 23, 2021
c675cab
chore: fix conflicts and ignore testing lint warns
EclesioMeloJunior Nov 23, 2021
9490f66
chore: check err
EclesioMeloJunior Nov 23, 2021
f3abef8
chore: fix runtime not found problems
EclesioMeloJunior Nov 24, 2021
1bec0db
chore: removing unused imports
EclesioMeloJunior Nov 24, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 25 additions & 2 deletions lib/grandpa/grandpa.go
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,8 @@ var (
// TODO: make this configurable; currently 1s is same as substrate; total round length is interval * 2 (#1869)
interval = time.Second
logger = log.New("pkg", "grandpa")

errUnsupportedSubround = errors.New("unsupported subround")
qdm12 marked this conversation as resolved.
Show resolved Hide resolved
)

// Service represents the current state of the grandpa protocol
Expand Down Expand Up @@ -697,7 +699,7 @@ func (s *Service) determinePreCommit() (*Vote, error) {
return &pvb, nil
}

// isFinalisable returns true is the round is finalisable, false otherwise.
// isFinalisable returns true if the round is finalisable, false otherwise.
func (s *Service) isFinalisable(round uint64) (bool, error) {
var pvb Vote
var err error
Expand Down Expand Up @@ -815,16 +817,20 @@ func (s *Service) createJustification(bfc common.Hash, stage Subround) ([]Signed
spc *sync.Map
err error
just []SignedVote
eqv map[ed25519.PublicKeyBytes][]*SignedVote
)

switch stage {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should probably throw an error if you pass an unsupported Subround.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added

case prevote:
spc = s.prevotes
eqv = s.pvEquivocations
case precommit:
spc = s.precommits
eqv = s.pcEquivocations
default:
return nil, fmt.Errorf("%w: %v", errUnsupportedSubround, stage)
}

// TODO: use equivacatory votes to create justification as well (#1667)
spc.Range(func(_, value interface{}) bool {
pc := value.(*SignedVote)
var isDescendant bool
Expand All @@ -846,6 +852,23 @@ func (s *Service) createJustification(bfc common.Hash, stage Subround) ([]Signed
return nil, err
}

for _, votes := range eqv {
for _, vote := range votes {
var signedVote SignedVote = *vote
qdm12 marked this conversation as resolved.
Show resolved Hide resolved
isDescendant, err := s.blockState.IsDescendantOf(bfc, signedVote.Vote.Hash)

if err != nil {
return nil, err
}

if !isDescendant {
continue
}

just = append(just, signedVote)
}
noot marked this conversation as resolved.
Show resolved Hide resolved
}

return just, nil
}

Expand Down
33 changes: 28 additions & 5 deletions lib/grandpa/message_handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -273,6 +273,11 @@ func (h *MessageHandler) verifyCommitMessageJustification(fm *CommitMessage) err
if isDescendant {
count++
}

eqPreCommitVotes, ok := h.grandpa.pcEquivocations[just.AuthorityID]
if ok {
count += len(eqPreCommitVotes)
}
noot marked this conversation as resolved.
Show resolved Hide resolved
}

// confirm total # signatures >= grandpa threshold
Expand All @@ -297,6 +302,11 @@ func (h *MessageHandler) verifyPreVoteJustification(msg *CatchUpResponse) (commo
}

votes[just.Vote.Hash]++

eqPreVotes, ok := h.grandpa.pvEquivocations[just.AuthorityID]
if ok {
votes[just.Vote.Hash] += uint64(len(eqPreVotes))
}
noot marked this conversation as resolved.
Show resolved Hide resolved
}

var prevote common.Hash
Expand Down Expand Up @@ -326,6 +336,11 @@ func (h *MessageHandler) verifyPreCommitJustification(msg *CatchUpResponse) erro
if just.Vote.Hash == msg.Hash && just.Vote.Number == msg.Number {
count++
}

eqPreCommitVotes, ok := h.grandpa.pcEquivocations[just.AuthorityID]
if ok {
count += len(eqPreCommitVotes)
}
noot marked this conversation as resolved.
Show resolved Hide resolved
}

if uint64(count) < h.grandpa.state.threshold() {
Expand Down Expand Up @@ -363,6 +378,7 @@ func (h *MessageHandler) verifyJustification(just *SignedVote, round, setID uint

// verify authority in justification set
authFound := false

for _, auth := range h.grandpa.authorities() {
justKey, err := just.AuthorityID.Encode()
if err != nil {
Expand Down Expand Up @@ -414,9 +430,7 @@ func (s *Service) VerifyBlockJustification(hash common.Hash, justification []byt
"sig count", len(fj.Commit.Precommits),
)

if len(fj.Commit.Precommits) < (2 * len(auths) / 3) {
return ErrMinVotesNotMet
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you can leave this check in, as it will quickly check if there aren't enough precommits in the justification to ever be valid

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done!

}
var equivocatoryVotersCount int

for _, just := range fj.Commit.Precommits {
// check if vote was for descendant of committed block
Expand All @@ -429,13 +443,18 @@ func (s *Service) VerifyBlockJustification(hash common.Hash, justification []byt
return ErrPrecommitBlockMismatch
}

// check for equivocatory votes
eqPreCommitVotes, ok := s.pcEquivocations[just.AuthorityID]
if ok {
equivocatoryVotersCount += len(eqPreCommitVotes)
}
noot marked this conversation as resolved.
Show resolved Hide resolved

pk, err := ed25519.NewPublicKey(just.AuthorityID[:])
if err != nil {
return err
}

ok := isInAuthSet(pk, auths)
if !ok {
if !isInAuthSet(pk, auths) {
return ErrAuthorityNotInSet
}

Expand All @@ -460,6 +479,10 @@ func (s *Service) VerifyBlockJustification(hash common.Hash, justification []byt
}
}

if len(fj.Commit.Precommits)+equivocatoryVotersCount <= (2 * len(auths) / 3) {
qdm12 marked this conversation as resolved.
Show resolved Hide resolved
return ErrMinVotesNotMet
}

err = s.blockState.SetFinalisedHash(hash, fj.Round, setID)
if err != nil {
return err
Expand Down
8 changes: 4 additions & 4 deletions lib/grandpa/vote_message.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,14 +38,14 @@ func (s *Service) receiveMessages(ctx context.Context) {
for {
select {
case msg, ok := <-s.in:
if msg == nil || msg.msg == nil {
continue
}

if !ok {
return
}

if msg == nil || msg.msg == nil {
kishansagathiya marked this conversation as resolved.
Show resolved Hide resolved
continue
}

logger.Trace("received vote message", "msg", msg.msg, "from", msg.from)
vm := msg.msg

Expand Down