Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

Beefy on-demand justifications as a custom RequestResponse protocol #12124

Merged
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
46a4231
client/beefy: create communication module and move gossip there
acatangiu Aug 25, 2022
cda48fc
client/beefy: move beefy_protocol_name module to communication
acatangiu Aug 25, 2022
e8157d7
client/beefy: move notification module under communication
acatangiu Aug 25, 2022
f9bf81b
client/beefy: add incoming request_response protocol handler
acatangiu Aug 26, 2022
f68920e
client/beefy: keep track of connected peers and their progress
acatangiu Aug 26, 2022
5fc1bb7
client/beefy: add logic for generating Justif requests
acatangiu Aug 26, 2022
9f05d3a
client/beefy: cancel outdated on-demand justification requests
acatangiu Aug 26, 2022
ac5c3a5
try Andre's suggestion for JustificationEngine
acatangiu Sep 9, 2022
e9ebea7
justif engine add justifs validation
acatangiu Sep 9, 2022
fbd4f57
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 12, 2022
4ab5ab0
client/beefy: impl OnDemandJustificationsEngine async next()
acatangiu Sep 12, 2022
88f8904
move beefy proto name test
acatangiu Sep 13, 2022
b5bf047
client/beefy: initialize OnDemandJustificationsEngine
acatangiu Sep 13, 2022
2e47d2c
client/tests: allow for custom req-resp protocols
acatangiu Sep 13, 2022
fdebcaa
client/beefy: on-demand-justif: implement simple peer selection strategy
acatangiu Sep 14, 2022
9aedbfa
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 15, 2022
2f02fba
client/beefy: fix voter initialization
acatangiu Sep 15, 2022
a9c60df
client/beefy: make sure justif requests are always out for mandatory …
acatangiu Sep 16, 2022
d48e199
client/beefy: add test for on-demand justifications sync
acatangiu Sep 16, 2022
3cb39f7
client/beefy: tweak main loop event processing order
acatangiu Sep 16, 2022
1908941
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 16, 2022
d265845
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 19, 2022
f76dd02
client/beefy: run on-demand-justif-handler under same async task as v…
acatangiu Sep 19, 2022
5496ea8
client/beefy: add test for known-peers
acatangiu Sep 19, 2022
940aaf2
client/beefy: reorg request-response module
acatangiu Sep 19, 2022
dbc2964
client/beefy: add issue references for future work todos
acatangiu Sep 19, 2022
b69471c
client/beefy: consolidate on-demand-justifications engine state machine
acatangiu Sep 20, 2022
b226996
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 21, 2022
68854a8
client/beefy: fix for polkadot companion
acatangiu Sep 21, 2022
683ddc5
client/beefy: implement review suggestions
acatangiu Sep 22, 2022
200cdf0
cargo fmt and clippy
acatangiu Sep 23, 2022
7a8950d
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 26, 2022
447dc2d
fix merge damage
acatangiu Sep 26, 2022
65cc99e
fix rust-doc
acatangiu Sep 26, 2022
9d6cd2f
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Sep 27, 2022
c68cbef
fix merge damage
acatangiu Sep 27, 2022
9a6bccd
fix merge damage
acatangiu Sep 27, 2022
b904741
client/beefy: add test for justif proto name
acatangiu Oct 3, 2022
fd50ee7
Merge branch 'master' of github.com:paritytech/substrate into beefy-c…
acatangiu Oct 3, 2022
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
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions client/beefy/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ sc-consensus = { version = "0.10.0-dev", path = "../consensus/common" }
sc-finality-grandpa = { version = "0.10.0-dev", path = "../../client/finality-grandpa" }
sc-keystore = { version = "4.0.0-dev", path = "../keystore" }
sc-network = { version = "0.10.0-dev", path = "../network" }
sc-network-common = { version = "0.10.0-dev", path = "../network/common" }
sc-network-gossip = { version = "0.10.0-dev", path = "../network-gossip" }
sc-utils = { version = "4.0.0-dev", path = "../utils" }
sp-api = { version = "4.0.0-dev", path = "../../primitives/api" }
Expand Down
6 changes: 1 addition & 5 deletions client/beefy/rpc/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ use jsonrpsee::{
};
use log::warn;

use beefy_gadget::notification::{BeefyBestBlockStream, BeefyVersionedFinalityProofStream};
use beefy_gadget::communication::notification::{BeefyBestBlockStream, BeefyVersionedFinalityProofStream};

mod notification;

Expand Down Expand Up @@ -164,10 +164,6 @@ where
mod tests {
use super::*;

use beefy_gadget::{
justification::BeefyVersionedFinalityProof,
notification::{BeefyBestBlockStream, BeefyVersionedFinalityProofSender},
};
use beefy_primitives::{known_payload_ids, Payload, SignedCommitment};
use codec::{Decode, Encode};
use jsonrpsee::{types::EmptyParams, RpcModule};
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.

use std::{collections::BTreeMap, time::Duration};
use std::{collections::BTreeMap, sync::Arc, time::Duration};

use sc_network::PeerId;
use sc_network_gossip::{MessageIntent, ValidationResult, Validator, ValidatorContext};
Expand All @@ -28,13 +28,12 @@ use log::{debug, trace};
use parking_lot::{Mutex, RwLock};
use wasm_timer::Instant;

use crate::{communication::peers::KnownPeers, keystore::BeefyKeystore};
use beefy_primitives::{
crypto::{Public, Signature},
VoteMessage,
};

use crate::keystore::BeefyKeystore;

// Timeout for rebroadcasting messages.
const REBROADCAST_AFTER: Duration = Duration::from_secs(60 * 5);

Expand Down Expand Up @@ -103,17 +102,19 @@ where
topic: B::Hash,
known_votes: RwLock<KnownVotes<B>>,
next_rebroadcast: Mutex<Instant>,
known_peers: Arc<Mutex<KnownPeers<B>>>,
}

impl<B> GossipValidator<B>
where
B: Block,
{
pub fn new() -> GossipValidator<B> {
pub fn new(known_peers: Arc<Mutex<KnownPeers<B>>>) -> GossipValidator<B> {
GossipValidator {
topic: topic::<B>(),
known_votes: RwLock::new(KnownVotes::new()),
next_rebroadcast: Mutex::new(Instant::now() + REBROADCAST_AFTER),
known_peers,
}
}

Expand Down Expand Up @@ -165,6 +166,7 @@ where

if BeefyKeystore::verify(&msg.id, &msg.signature, &msg.commitment.encode()) {
self.known_votes.write().add_known(&round, msg_hash);
self.known_peers.lock().note_vote_for(*sender, round);
return ValidationResult::ProcessAndKeep(self.topic)
} else {
// TODO: report peer
Expand Down Expand Up @@ -271,7 +273,7 @@ mod tests {

#[test]
fn note_and_drop_round_works() {
let gv = GossipValidator::<Block>::new();
let gv = GossipValidator::<Block>::new(Arc::new(Mutex::new(KnownPeers::new())));

gv.note_round(1u64);

Expand All @@ -298,7 +300,7 @@ mod tests {

#[test]
fn note_same_round_twice() {
let gv = GossipValidator::<Block>::new();
let gv = GossipValidator::<Block>::new(Arc::new(Mutex::new(KnownPeers::new())));

gv.note_round(3u64);
gv.note_round(7u64);
Expand Down Expand Up @@ -355,7 +357,7 @@ mod tests {

#[test]
fn should_avoid_verifying_signatures_twice() {
let gv = GossipValidator::<Block>::new();
let gv = GossipValidator::<Block>::new(Arc::new(Mutex::new(KnownPeers::new())));
let sender = sc_network::PeerId::random();
let mut context = TestContext;

Expand Down Expand Up @@ -391,7 +393,7 @@ mod tests {

#[test]
fn messages_allowed_and_expired() {
let gv = GossipValidator::<Block>::new();
let gv = GossipValidator::<Block>::new(Arc::new(Mutex::new(KnownPeers::new())));
let sender = sc_network::PeerId::random();
let topic = Default::default();
let intent = MessageIntent::Broadcast;
Expand Down Expand Up @@ -434,7 +436,7 @@ mod tests {

#[test]
fn messages_rebroadcast() {
let gv = GossipValidator::<Block>::new();
let gv = GossipValidator::<Block>::new(Arc::new(Mutex::new(KnownPeers::new())));
let sender = sc_network::PeerId::random();
let topic = Default::default();

Expand Down
106 changes: 106 additions & 0 deletions client/beefy/src/communication/mod.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,106 @@
// This file is part of Substrate.

// Copyright (C) 2022 Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0

// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// This program 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 General Public License for more details.

// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.

//! Communication streams for the BEEFY networking protocols.

pub mod notification;
pub mod request_response;

pub(crate) mod gossip;
pub(crate) mod peers;

pub(crate) mod beefy_protocol_name {
use array_bytes::bytes2hex;
use sc_network::ProtocolName;

/// BEEFY votes gossip protocol name suffix.
const GOSSIP_NAME: &str = "/beefy/1";
/// BEEFY justifications protocol name suffix.
const JUSTIFICATIONS_NAME: &str = "/beefy/justifications/1";

/// Old names for the gossip protocol, used for backward compatibility.
pub(super) const LEGACY_NAMES: [&str; 1] = ["/paritytech/beefy/1"];
acatangiu marked this conversation as resolved.
Show resolved Hide resolved

/// Name of the votes gossip protocol used by BEEFY.
///
/// Must be registered towards the networking in order for BEEFY voter to properly function.
pub fn gossip_protocol_name<Hash: AsRef<[u8]>>(
genesis_hash: Hash,
fork_id: Option<&str>,
) -> ProtocolName {
let genesis_hash = genesis_hash.as_ref();
if let Some(fork_id) = fork_id {
format!("/{}/{}{}", bytes2hex("", genesis_hash), fork_id, GOSSIP_NAME).into()
} else {
format!("/{}{}", bytes2hex("", genesis_hash), GOSSIP_NAME).into()
}
}

/// Name of the BEEFY justifications request-response protocol.
pub fn justifications_protocol_name<Hash: AsRef<[u8]>>(
genesis_hash: Hash,
fork_id: Option<&str>,
) -> ProtocolName {
let genesis_hash = genesis_hash.as_ref();
if let Some(fork_id) = fork_id {
format!("/{}/{}{}", bytes2hex("", genesis_hash), fork_id, JUSTIFICATIONS_NAME).into()
} else {
format!("/{}{}", bytes2hex("", genesis_hash), JUSTIFICATIONS_NAME).into()
}
}
}

/// Returns the configuration value to put in
/// [`sc_network::config::NetworkConfiguration::extra_sets`].
/// For standard protocol name see [`beefy_protocol_name::gossip_protocol_name`].
pub fn beefy_peers_set_config(
gossip_protocol_name: sc_network::ProtocolName,
) -> sc_network::config::NonDefaultSetConfig {
let mut cfg = sc_network::config::NonDefaultSetConfig::new(gossip_protocol_name, 1024 * 1024);

cfg.allow_non_reserved(25, 25);
cfg.add_fallback_names(beefy_protocol_name::LEGACY_NAMES.iter().map(|&n| n.into()).collect());
cfg
}

#[cfg(test)]
mod tests {
use super::*;

use sp_core::H256;

#[test]
fn beefy_gossip_protocol_name() {
use beefy_protocol_name::gossip_protocol_name;
// Create protocol name using random genesis hash.
let genesis_hash = H256::random();
let expected = format!("/{}/beefy/1", array_bytes::bytes2hex("", genesis_hash.as_ref()));
let proto_name = gossip_protocol_name(&genesis_hash, None);
assert_eq!(proto_name.to_string(), expected);

// Create protocol name using hardcoded genesis hash. Verify exact representation.
let genesis_hash = [
50, 4, 60, 123, 58, 106, 216, 246, 194, 188, 139, 193, 33, 212, 202, 171, 9, 55, 123,
94, 8, 43, 12, 251, 187, 57, 173, 19, 188, 74, 205, 147,
];
let expected =
"/32043c7b3a6ad8f6c2bc8bc121d4caab09377b5e082b0cfbbb39ad13bc4acd93/beefy/1".to_string();
let proto_name = gossip_protocol_name(&genesis_hash, None);
acatangiu marked this conversation as resolved.
Show resolved Hide resolved
assert_eq!(proto_name.to_string(), expected);
}
}
131 changes: 131 additions & 0 deletions client/beefy/src/communication/peers.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
// This file is part of Substrate.

// Copyright (C) 2022 Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0

// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// This program 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 General Public License for more details.

// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.

//! Logic for keeping track of BEEFY peers.

// TODO (issue #12296): replace this naive peer tracking with generic one that infers data
// from multiple network protocols.

use sc_network::PeerId;
use sp_runtime::traits::{Block, NumberFor, Zero};
use std::collections::{HashMap, VecDeque};

struct PeerData<B: Block> {
last_voted_on: NumberFor<B>,
}

impl<B: Block> Default for PeerData<B> {
fn default() -> Self {
PeerData { last_voted_on: Zero::zero() }
}
}

/// Keep a simple map of connected peers
/// and the most recent voting round they participated in.
pub struct KnownPeers<B: Block> {
live: HashMap<PeerId, PeerData<B>>,
}

impl<B: Block> KnownPeers<B> {
pub fn new() -> Self {
Self { live: HashMap::new() }
}

/// Add new connected `peer`.
pub fn add_new(&mut self, peer: PeerId) {
self.live.entry(peer).or_default();
}

/// Note vote round number for `peer`.
pub fn note_vote_for(&mut self, peer: PeerId, round: NumberFor<B>) {
let data = self.live.entry(peer).or_default();
data.last_voted_on = round.max(data.last_voted_on);
}

/// Remove connected `peer`.
pub fn remove(&mut self, peer: &PeerId) {
self.live.remove(peer);
}

/// Return _filtered and cloned_ list of peers that have voted on `block` or higher.
pub fn at_least_at_block(&self, block: NumberFor<B>) -> VecDeque<PeerId> {
self.live
.iter()
.filter_map(|(k, v)| if v.last_voted_on >= block { Some(k) } else { None })
acatangiu marked this conversation as resolved.
Show resolved Hide resolved
.cloned()
.collect()
}

/// Answer whether `peer` is part of `KnownPeers` set.
pub fn contains(&self, peer: &PeerId) -> bool {
self.live.contains_key(peer)
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn should_track_known_peers_progress() {
let (alice, bob, charlie) = (PeerId::random(), PeerId::random(), PeerId::random());
let mut peers = KnownPeers::<sc_network_test::Block>::new();
assert!(peers.live.is_empty());

// Alice and Bob new connected peers.
peers.add_new(alice);
peers.add_new(bob);
// 'Tracked' Bob seen voting for 5.
peers.note_vote_for(bob, 5);
// Previously unseen Charlie now seen voting for 10.
peers.note_vote_for(charlie, 10);

assert_eq!(peers.live.len(), 3);
assert!(peers.contains(&alice));
assert!(peers.contains(&bob));
assert!(peers.contains(&charlie));

// Get peers at block >= 5
let at_5 = peers.at_least_at_block(5);
// Should be Bob and Charlie
assert_eq!(at_5.len(), 2);
assert!(at_5.contains(&bob));
assert!(at_5.contains(&charlie));

// 'Tracked' Alice seen voting for 10.
peers.note_vote_for(alice, 10);

// Get peers at block >= 9
let at_9 = peers.at_least_at_block(9);
// Should be Charlie and Alice
assert_eq!(at_9.len(), 2);
assert!(at_9.contains(&charlie));
assert!(at_9.contains(&alice));

// Remove Alice
peers.remove(&alice);
assert_eq!(peers.live.len(), 2);
assert!(!peers.contains(&alice));

// Get peers at block >= 9
let at_9 = peers.at_least_at_block(9);
// Now should be just Charlie
assert_eq!(at_9.len(), 1);
assert!(at_9.contains(&charlie));
}
}
Loading