Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
P
polkadot-sdk
Manage
Activity
Members
Labels
Plan
Issues
0
Issue boards
Milestones
Iterations
Wiki
Requirements
Code
Merge requests
0
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Locked files
Build
Pipelines
Jobs
Pipeline schedules
Test cases
Artifacts
Deploy
Releases
Package Registry
Model registry
Operate
Environments
Terraform modules
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
Code review analytics
Issue analytics
Insights
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
parity
Mirrored projects
polkadot-sdk
Commits
1ae121ff
Commit
1ae121ff
authored
3 years ago
by
asynchronous rob
Committed by
GitHub
3 years ago
Browse files
Options
Downloads
Patches
Plain Diff
Ignore redundant dispute messages (#4854)
* ignore duplicate dispute votes * fmt and TODOs * tests * fmt
parent
bb69c82c
No related merge requests found
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
polkadot/node/core/dispute-coordinator/src/real/initialized.rs
+36
-11
36 additions, 11 deletions
...dot/node/core/dispute-coordinator/src/real/initialized.rs
polkadot/node/core/dispute-coordinator/src/real/tests.rs
+109
-0
109 additions, 0 deletions
polkadot/node/core/dispute-coordinator/src/real/tests.rs
with
145 additions
and
11 deletions
polkadot/node/core/dispute-coordinator/src/real/initialized.rs
+
36
−
11
View file @
1ae121ff
...
...
@@ -779,14 +779,21 @@ impl Initialized {
// There is one exception: A sufficiently sophisticated attacker could prevent
// us from seeing the backing votes by witholding arbitrary blocks, and hence we do
// not have a `CandidateReceipt` available.
let
mut
votes
=
match
overlay_db
let
(
mut
votes
,
mut
votes_changed
)
=
match
overlay_db
.load_candidate_votes
(
session
,
&
candidate_hash
)
?
.map
(
CandidateVotes
::
from
)
{
Some
(
votes
)
=>
votes
,
Some
(
votes
)
=>
(
votes
,
false
),
None
=>
if
let
MaybeCandidateReceipt
::
Provides
(
candidate_receipt
)
=
candidate_receipt
{
CandidateVotes
{
candidate_receipt
,
valid
:
Vec
::
new
(),
invalid
:
Vec
::
new
()
}
(
CandidateVotes
{
candidate_receipt
,
valid
:
Vec
::
new
(),
invalid
:
Vec
::
new
(),
},
true
,
)
}
else
{
tracing
::
warn!
(
target
:
LOG_TARGET
,
...
...
@@ -841,22 +848,34 @@ impl Initialized {
match
statement
.statement
()
.clone
()
{
DisputeStatement
::
Valid
(
valid_kind
)
=>
{
self
.metrics
.on_valid_vote
();
insert_into_statement_vec
(
let
fresh
=
insert_into_statement_vec
(
&
mut
votes
.valid
,
valid_kind
,
*
val_index
,
statement
.validator_signature
()
.clone
(),
);
if
!
fresh
{
continue
}
votes_changed
=
true
;
self
.metrics
.on_valid_vote
();
},
DisputeStatement
::
Invalid
(
invalid_kind
)
=>
{
self
.metrics
.on_invalid_vote
();
insert_into_statement_vec
(
let
fresh
=
insert_into_statement_vec
(
&
mut
votes
.invalid
,
invalid_kind
,
*
val_index
,
statement
.validator_signature
()
.clone
(),
);
if
!
fresh
{
continue
}
votes_changed
=
true
;
self
.metrics
.on_invalid_vote
();
},
}
}
...
...
@@ -871,7 +890,7 @@ impl Initialized {
// Potential spam:
if
!
is_confirmed
{
let
mut
free_spam_slots
=
false
;
let
mut
free_spam_slots
=
statements
.is_empty
()
;
for
(
statement
,
index
)
in
statements
.iter
()
{
free_spam_slots
|
=
statement
.statement
()
.is_backing
()
||
self
.spam_slots
.add_unconfirmed
(
session
,
candidate_hash
,
*
index
);
...
...
@@ -988,7 +1007,10 @@ impl Initialized {
overlay_db
.write_recent_disputes
(
recent_disputes
);
}
overlay_db
.write_candidate_votes
(
session
,
candidate_hash
,
votes
.into
());
// Only write when votes have changed.
if
votes_changed
{
overlay_db
.write_candidate_votes
(
session
,
candidate_hash
,
votes
.into
());
}
Ok
(
ImportStatementsResult
::
ValidImport
)
}
...
...
@@ -1145,18 +1167,21 @@ impl MuxedMessage {
}
}
// Returns 'true' if no other vote by that validator was already
// present and 'false' otherwise. Same semantics as `HashSet`.
fn
insert_into_statement_vec
<
T
>
(
vec
:
&
mut
Vec
<
(
T
,
ValidatorIndex
,
ValidatorSignature
)
>
,
tag
:
T
,
val_index
:
ValidatorIndex
,
val_signature
:
ValidatorSignature
,
)
{
)
->
bool
{
let
pos
=
match
vec
.binary_search_by_key
(
&
val_index
,
|
x
|
x
.1
)
{
Ok
(
_
)
=>
return
,
// no duplicates needed.
Ok
(
_
)
=>
return
false
,
// no duplicates needed.
Err
(
p
)
=>
p
,
};
vec
.insert
(
pos
,
(
tag
,
val_index
,
val_signature
));
true
}
#[derive(Debug,
Clone)]
...
...
This diff is collapsed.
Click to expand it.
polkadot/node/core/dispute-coordinator/src/real/tests.rs
+
109
−
0
View file @
1ae121ff
...
...
@@ -1548,3 +1548,112 @@ fn negative_issue_local_statement_only_triggers_import() {
})
});
}
#[test]
fn
empty_import_still_writes_candidate_receipt
()
{
test_harness
(|
mut
test_state
,
mut
virtual_overseer
|
{
Box
::
pin
(
async
move
{
let
session
=
1
;
test_state
.handle_resume_sync
(
&
mut
virtual_overseer
,
session
)
.await
;
let
candidate_receipt
=
make_valid_candidate_receipt
();
let
candidate_hash
=
candidate_receipt
.hash
();
test_state
.activate_leaf_at_session
(
&
mut
virtual_overseer
,
session
,
1
)
.await
;
let
(
tx
,
rx
)
=
oneshot
::
channel
();
virtual_overseer
.send
(
FromOverseer
::
Communication
{
msg
:
DisputeCoordinatorMessage
::
ImportStatements
{
candidate_hash
,
candidate_receipt
:
candidate_receipt
.clone
(),
session
,
statements
:
Vec
::
new
(),
pending_confirmation
:
tx
,
},
})
.await
;
rx
.await
.unwrap
();
let
backend
=
DbBackend
::
new
(
test_state
.db
.clone
(),
test_state
.config
.column_config
());
let
votes
=
backend
.load_candidate_votes
(
session
,
&
candidate_hash
)
.unwrap
()
.unwrap
();
assert_eq!
(
votes
.invalid
.len
(),
0
);
assert_eq!
(
votes
.valid
.len
(),
0
);
assert_eq!
(
votes
.candidate_receipt
,
candidate_receipt
);
virtual_overseer
.send
(
FromOverseer
::
Signal
(
OverseerSignal
::
Conclude
))
.await
;
assert!
(
virtual_overseer
.try_recv
()
.await
.is_none
());
test_state
})
});
}
#[test]
fn
redundant_votes_ignored
()
{
test_harness
(|
mut
test_state
,
mut
virtual_overseer
|
{
Box
::
pin
(
async
move
{
let
session
=
1
;
test_state
.handle_resume_sync
(
&
mut
virtual_overseer
,
session
)
.await
;
let
candidate_receipt
=
make_valid_candidate_receipt
();
let
candidate_hash
=
candidate_receipt
.hash
();
test_state
.activate_leaf_at_session
(
&
mut
virtual_overseer
,
session
,
1
)
.await
;
let
valid_vote
=
test_state
.issue_statement_with_index
(
1
,
candidate_hash
,
session
,
true
)
.await
;
let
valid_vote_2
=
test_state
.issue_statement_with_index
(
1
,
candidate_hash
,
session
,
true
)
.await
;
assert!
(
valid_vote
.validator_signature
()
!=
valid_vote_2
.validator_signature
());
let
(
tx
,
rx
)
=
oneshot
::
channel
();
virtual_overseer
.send
(
FromOverseer
::
Communication
{
msg
:
DisputeCoordinatorMessage
::
ImportStatements
{
candidate_hash
,
candidate_receipt
:
candidate_receipt
.clone
(),
session
,
statements
:
vec!
[(
valid_vote
.clone
(),
ValidatorIndex
(
1
))],
pending_confirmation
:
tx
,
},
})
.await
;
rx
.await
.unwrap
();
let
(
tx
,
rx
)
=
oneshot
::
channel
();
virtual_overseer
.send
(
FromOverseer
::
Communication
{
msg
:
DisputeCoordinatorMessage
::
ImportStatements
{
candidate_hash
,
candidate_receipt
:
candidate_receipt
.clone
(),
session
,
statements
:
vec!
[(
valid_vote_2
,
ValidatorIndex
(
1
))],
pending_confirmation
:
tx
,
},
})
.await
;
rx
.await
.unwrap
();
let
backend
=
DbBackend
::
new
(
test_state
.db
.clone
(),
test_state
.config
.column_config
());
let
votes
=
backend
.load_candidate_votes
(
session
,
&
candidate_hash
)
.unwrap
()
.unwrap
();
assert_eq!
(
votes
.invalid
.len
(),
0
);
assert_eq!
(
votes
.valid
.len
(),
1
);
assert_eq!
(
&
votes
.valid
[
0
]
.2
,
valid_vote
.validator_signature
());
virtual_overseer
.send
(
FromOverseer
::
Signal
(
OverseerSignal
::
Conclude
))
.await
;
assert!
(
virtual_overseer
.try_recv
()
.await
.is_none
());
test_state
})
});
}
This diff is collapsed.
Click to expand it.
Preview
0%
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment