mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-08 20:09:54 +00:00
Start handling P2P messages
This defines the tart of a very complex series of locks I'm really unhappy with. At the same time, there's not immediately a better solution. This also should work without issue.
This commit is contained in:
parent
f2d9d70068
commit
ad5522d854
12 changed files with 199 additions and 95 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -1309,6 +1309,7 @@ dependencies = [
|
||||||
"blake2",
|
"blake2",
|
||||||
"ciphersuite",
|
"ciphersuite",
|
||||||
"flexible-transcript",
|
"flexible-transcript",
|
||||||
|
"futures",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
"modular-frost",
|
"modular-frost",
|
||||||
|
|
|
@ -42,4 +42,5 @@ log = "0.4"
|
||||||
tokio = { version = "1", features = ["full"] }
|
tokio = { version = "1", features = ["full"] }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
|
futures = "0.3"
|
||||||
tributary = { package = "tributary-chain", path = "./tributary", features = ["tests"] }
|
tributary = { package = "tributary-chain", path = "./tributary", features = ["tests"] }
|
||||||
|
|
|
@ -52,16 +52,19 @@ async fn run<D: Db, Pro: Processor, P: P2p>(
|
||||||
serai: Serai,
|
serai: Serai,
|
||||||
) {
|
) {
|
||||||
let add_new_tributary = |db, spec: TributarySpec| async {
|
let add_new_tributary = |db, spec: TributarySpec| async {
|
||||||
|
// Save it to the database
|
||||||
MainDb(db).add_active_tributary(&spec);
|
MainDb(db).add_active_tributary(&spec);
|
||||||
|
// Add it to the queue
|
||||||
|
// If we reboot before this is read from the queue, the fact it was saved to the database
|
||||||
|
// means it'll be handled on reboot
|
||||||
NEW_TRIBUTARIES.write().await.push_back(spec);
|
NEW_TRIBUTARIES.write().await.push_back(spec);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Handle new Substrate blocks
|
||||||
{
|
{
|
||||||
let mut substrate_db = substrate::SubstrateDb::new(raw_db.clone());
|
let mut substrate_db = substrate::SubstrateDb::new(raw_db.clone());
|
||||||
let mut last_substrate_block = substrate_db.last_block();
|
let mut last_substrate_block = substrate_db.last_block();
|
||||||
|
|
||||||
let p2p = p2p.clone();
|
|
||||||
|
|
||||||
let key = key.clone();
|
let key = key.clone();
|
||||||
let mut processor = processor.clone();
|
let mut processor = processor.clone();
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
|
@ -70,7 +73,6 @@ async fn run<D: Db, Pro: Processor, P: P2p>(
|
||||||
&mut substrate_db,
|
&mut substrate_db,
|
||||||
&key,
|
&key,
|
||||||
add_new_tributary,
|
add_new_tributary,
|
||||||
&p2p,
|
|
||||||
&mut processor,
|
&mut processor,
|
||||||
&serai,
|
&serai,
|
||||||
&mut last_substrate_block,
|
&mut last_substrate_block,
|
||||||
|
@ -87,15 +89,14 @@ async fn run<D: Db, Pro: Processor, P: P2p>(
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handle the Tributaries
|
||||||
{
|
{
|
||||||
struct ActiveTributary<D: Db, P: P2p> {
|
struct ActiveTributary<D: Db, P: P2p> {
|
||||||
spec: TributarySpec,
|
spec: TributarySpec,
|
||||||
tributary: Tributary<D, Transaction, P>,
|
tributary: Tributary<D, Transaction, P>,
|
||||||
}
|
}
|
||||||
|
let tributaries = Arc::new(RwLock::new(HashMap::<[u8; 32], ActiveTributary<D, P>>::new()));
|
||||||
|
|
||||||
let mut tributaries = HashMap::<[u8; 32], ActiveTributary<D, P>>::new();
|
|
||||||
|
|
||||||
// TODO: Use a db on a distinct volume
|
|
||||||
async fn add_tributary<D: Db, P: P2p>(
|
async fn add_tributary<D: Db, P: P2p>(
|
||||||
db: D,
|
db: D,
|
||||||
key: Zeroizing<<Ristretto as Ciphersuite>::F>,
|
key: Zeroizing<<Ristretto as Ciphersuite>::F>,
|
||||||
|
@ -104,6 +105,7 @@ async fn run<D: Db, Pro: Processor, P: P2p>(
|
||||||
spec: TributarySpec,
|
spec: TributarySpec,
|
||||||
) {
|
) {
|
||||||
let tributary = Tributary::<_, Transaction, _>::new(
|
let tributary = Tributary::<_, Transaction, _>::new(
|
||||||
|
// TODO: Use a db on a distinct volume
|
||||||
db,
|
db,
|
||||||
spec.genesis(),
|
spec.genesis(),
|
||||||
spec.start_time(),
|
spec.start_time(),
|
||||||
|
@ -117,40 +119,85 @@ async fn run<D: Db, Pro: Processor, P: P2p>(
|
||||||
tributaries.insert(tributary.genesis(), ActiveTributary { spec, tributary });
|
tributaries.insert(tributary.genesis(), ActiveTributary { spec, tributary });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Reload active tributaries from the database
|
||||||
// TODO: Can MainDb take a borrow?
|
// TODO: Can MainDb take a borrow?
|
||||||
for spec in MainDb(raw_db.clone()).active_tributaries().1 {
|
for spec in MainDb(raw_db.clone()).active_tributaries().1 {
|
||||||
add_tributary(raw_db.clone(), key.clone(), p2p.clone(), &mut tributaries, spec).await;
|
add_tributary(
|
||||||
|
raw_db.clone(),
|
||||||
|
key.clone(),
|
||||||
|
p2p.clone(),
|
||||||
|
&mut *tributaries.write().await,
|
||||||
|
spec,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handle new Tributary blocks
|
||||||
let mut tributary_db = tributary::TributaryDb::new(raw_db.clone());
|
let mut tributary_db = tributary::TributaryDb::new(raw_db.clone());
|
||||||
tokio::spawn(async move {
|
{
|
||||||
loop {
|
let tributaries = tributaries.clone();
|
||||||
// The following handle_new_blocks function may take an arbitrary amount of time
|
let p2p = p2p.clone();
|
||||||
// If registering a new tributary waited for a lock on the tributaries table, the substrate
|
tokio::spawn(async move {
|
||||||
// scanner may wait on a lock for an arbitrary amount of time
|
loop {
|
||||||
// By instead using the distinct NEW_TRIBUTARIES, there should be minimal
|
// The following handle_new_blocks function may take an arbitrary amount of time
|
||||||
// competition/blocking
|
// If registering a new tributary waited for a lock on the tributaries table, the
|
||||||
{
|
// substrate scanner may wait on a lock for an arbitrary amount of time
|
||||||
let mut new_tributaries = NEW_TRIBUTARIES.write().await;
|
// By instead using the distinct NEW_TRIBUTARIES, there should be minimal
|
||||||
while let Some(spec) = new_tributaries.pop_front() {
|
// competition/blocking
|
||||||
add_tributary(raw_db.clone(), key.clone(), p2p.clone(), &mut tributaries, spec).await;
|
{
|
||||||
|
let mut new_tributaries = NEW_TRIBUTARIES.write().await;
|
||||||
|
while let Some(spec) = new_tributaries.pop_front() {
|
||||||
|
add_tributary(
|
||||||
|
raw_db.clone(),
|
||||||
|
key.clone(),
|
||||||
|
p2p.clone(),
|
||||||
|
// This is a short-lived write acquisition, which is why it should be fine
|
||||||
|
&mut *tributaries.write().await,
|
||||||
|
spec,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unknown-length read acquisition. This would risk screwing over the P2P process EXCEPT
|
||||||
|
// they both use read locks. Accordingly, they can co-exist
|
||||||
|
for ActiveTributary { spec, tributary } in tributaries.read().await.values() {
|
||||||
|
tributary::scanner::handle_new_blocks::<_, _, P>(
|
||||||
|
&mut tributary_db,
|
||||||
|
&key,
|
||||||
|
&mut processor,
|
||||||
|
spec,
|
||||||
|
tributary,
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
|
sleep(Duration::from_secs(3)).await;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle P2P messages
|
||||||
|
{
|
||||||
|
tokio::spawn(async move {
|
||||||
|
loop {
|
||||||
|
let msg = p2p.receive().await;
|
||||||
|
match msg.kind {
|
||||||
|
P2pMessageKind::Tributary(genesis) => {
|
||||||
|
let tributaries_read = tributaries.read().await;
|
||||||
|
let Some(tributary) = tributaries_read.get(&genesis) else {
|
||||||
|
log::debug!("received p2p message for unknown network");
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
if tributary.tributary.handle_message(&msg.msg).await {
|
||||||
|
P2p::broadcast(&p2p, msg.kind, msg.msg).await;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
for ActiveTributary { spec, tributary } in tributaries.values() {
|
}
|
||||||
tributary::scanner::handle_new_blocks::<_, _, P>(
|
|
||||||
&mut tributary_db,
|
|
||||||
&key,
|
|
||||||
&mut processor,
|
|
||||||
spec,
|
|
||||||
tributary,
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
}
|
|
||||||
|
|
||||||
sleep(Duration::from_secs(3)).await;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
use core::fmt::Debug;
|
use core::fmt::Debug;
|
||||||
use std::{
|
use std::{
|
||||||
sync::{Arc, RwLock},
|
sync::{Arc, RwLock},
|
||||||
|
io::Read,
|
||||||
collections::VecDeque,
|
collections::VecDeque,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -10,33 +11,81 @@ pub use tributary::P2p as TributaryP2p;
|
||||||
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)]
|
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug)]
|
||||||
pub enum P2pMessageKind {
|
pub enum P2pMessageKind {
|
||||||
Tributary,
|
Tributary([u8; 32]),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl P2pMessageKind {
|
impl P2pMessageKind {
|
||||||
fn to_byte(self) -> u8 {
|
fn serialize(&self) -> Vec<u8> {
|
||||||
match self {
|
match self {
|
||||||
P2pMessageKind::Tributary => 0,
|
P2pMessageKind::Tributary(genesis) => {
|
||||||
|
let mut res = vec![0];
|
||||||
|
res.extend(genesis);
|
||||||
|
res
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn from_byte(byte: u8) -> Option<P2pMessageKind> {
|
fn read<R: Read>(reader: &mut R) -> Option<P2pMessageKind> {
|
||||||
match byte {
|
let mut kind = [0; 1];
|
||||||
0 => Some(P2pMessageKind::Tributary),
|
reader.read_exact(&mut kind).ok()?;
|
||||||
|
match kind[0] {
|
||||||
|
0 => Some({
|
||||||
|
let mut genesis = [0; 32];
|
||||||
|
reader.read_exact(&mut genesis).ok()?;
|
||||||
|
P2pMessageKind::Tributary(genesis)
|
||||||
|
}),
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO
|
#[derive(Clone, Debug)]
|
||||||
#[async_trait]
|
pub struct Message<P: P2p> {
|
||||||
pub trait P2p: Send + Sync + Clone + Debug + TributaryP2p {
|
pub sender: P::Id,
|
||||||
async fn broadcast(&self, kind: P2pMessageKind, msg: Vec<u8>);
|
pub kind: P2pMessageKind,
|
||||||
async fn receive(&self) -> Option<(P2pMessageKind, Vec<u8>)>;
|
pub msg: Vec<u8>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
pub trait P2p: Send + Sync + Clone + Debug + TributaryP2p {
|
||||||
|
type Id: Send + Sync + Clone + Debug;
|
||||||
|
|
||||||
|
async fn send_raw(&self, to: Self::Id, msg: Vec<u8>);
|
||||||
|
async fn broadcast_raw(&self, msg: Vec<u8>);
|
||||||
|
async fn receive_raw(&self) -> (Self::Id, Vec<u8>);
|
||||||
|
|
||||||
|
async fn send(&self, to: Self::Id, kind: P2pMessageKind, msg: Vec<u8>) {
|
||||||
|
let mut actual_msg = kind.serialize();
|
||||||
|
actual_msg.extend(msg);
|
||||||
|
self.send_raw(to, actual_msg).await;
|
||||||
|
}
|
||||||
|
async fn broadcast(&self, kind: P2pMessageKind, msg: Vec<u8>) {
|
||||||
|
let mut actual_msg = kind.serialize();
|
||||||
|
actual_msg.extend(msg);
|
||||||
|
self.broadcast_raw(actual_msg).await;
|
||||||
|
}
|
||||||
|
async fn receive(&self) -> Message<Self> {
|
||||||
|
let (sender, kind, msg) = loop {
|
||||||
|
let (sender, msg) = self.receive_raw().await;
|
||||||
|
if msg.is_empty() {
|
||||||
|
log::error!("empty p2p message from {sender:?}");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut msg_ref = msg.as_ref();
|
||||||
|
let Some(kind) = P2pMessageKind::read::<&[u8]>(&mut msg_ref) else {
|
||||||
|
log::error!("invalid p2p message kind from {sender:?}");
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
break (sender, kind, msg_ref.to_vec());
|
||||||
|
};
|
||||||
|
Message { sender, kind, msg }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::type_complexity)]
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct LocalP2p(usize, Arc<RwLock<Vec<VecDeque<Vec<u8>>>>>);
|
pub struct LocalP2p(usize, Arc<RwLock<Vec<VecDeque<(usize, Vec<u8>)>>>>);
|
||||||
|
|
||||||
impl LocalP2p {
|
impl LocalP2p {
|
||||||
pub fn new(validators: usize) -> Vec<LocalP2p> {
|
pub fn new(validators: usize) -> Vec<LocalP2p> {
|
||||||
|
@ -51,29 +100,35 @@ impl LocalP2p {
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl P2p for LocalP2p {
|
impl P2p for LocalP2p {
|
||||||
async fn broadcast(&self, kind: P2pMessageKind, mut msg: Vec<u8>) {
|
type Id = usize;
|
||||||
msg.insert(0, kind.to_byte());
|
|
||||||
|
async fn send_raw(&self, to: Self::Id, msg: Vec<u8>) {
|
||||||
|
self.1.write().unwrap()[to].push_back((self.0, msg));
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn broadcast_raw(&self, msg: Vec<u8>) {
|
||||||
for (i, msg_queue) in self.1.write().unwrap().iter_mut().enumerate() {
|
for (i, msg_queue) in self.1.write().unwrap().iter_mut().enumerate() {
|
||||||
if i == self.0 {
|
if i == self.0 {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
msg_queue.push_back(msg.clone());
|
msg_queue.push_back((self.0, msg.clone()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive(&self) -> Option<(P2pMessageKind, Vec<u8>)> {
|
async fn receive_raw(&self) -> (Self::Id, Vec<u8>) {
|
||||||
let mut msg = self.1.write().unwrap()[self.0].pop_front()?;
|
// This is a cursed way to implement an async read from a Vec
|
||||||
if msg.is_empty() {
|
loop {
|
||||||
log::error!("empty p2p message");
|
if let Some(res) = self.1.write().unwrap()[self.0].pop_front() {
|
||||||
return None;
|
return res;
|
||||||
|
}
|
||||||
|
tokio::time::sleep(std::time::Duration::from_millis(100)).await;
|
||||||
}
|
}
|
||||||
Some((P2pMessageKind::from_byte(msg.remove(0))?, msg))
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl TributaryP2p for LocalP2p {
|
impl TributaryP2p for LocalP2p {
|
||||||
async fn broadcast(&self, msg: Vec<u8>) {
|
async fn broadcast(&self, genesis: [u8; 32], msg: Vec<u8>) {
|
||||||
<Self as P2p>::broadcast(self, P2pMessageKind::Tributary, msg).await
|
<Self as P2p>::broadcast(self, P2pMessageKind::Tributary(genesis), msg).await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@ use serai_db::DbTxn;
|
||||||
|
|
||||||
use processor_messages::{SubstrateContext, key_gen::KeyGenId, CoordinatorMessage};
|
use processor_messages::{SubstrateContext, key_gen::KeyGenId, CoordinatorMessage};
|
||||||
|
|
||||||
use crate::{Db, P2p, processor::Processor, tributary::TributarySpec};
|
use crate::{Db, processor::Processor, tributary::TributarySpec};
|
||||||
|
|
||||||
mod db;
|
mod db;
|
||||||
pub use db::*;
|
pub use db::*;
|
||||||
|
@ -63,7 +63,6 @@ async fn handle_new_set<
|
||||||
// We already have a unique event ID based on block, event index (where event index is
|
// We already have a unique event ID based on block, event index (where event index is
|
||||||
// the one generated in this handle_block function)
|
// the one generated in this handle_block function)
|
||||||
// We could use that on this end and the processor end?
|
// We could use that on this end and the processor end?
|
||||||
// TODO: Should this be handled in the Tributary code?
|
|
||||||
processor
|
processor
|
||||||
.send(CoordinatorMessage::KeyGen(
|
.send(CoordinatorMessage::KeyGen(
|
||||||
processor_messages::key_gen::CoordinatorMessage::GenerateKey {
|
processor_messages::key_gen::CoordinatorMessage::GenerateKey {
|
||||||
|
@ -212,12 +211,10 @@ async fn handle_block<
|
||||||
Fut: Future<Output = ()>,
|
Fut: Future<Output = ()>,
|
||||||
ANT: Clone + Fn(D, TributarySpec) -> Fut,
|
ANT: Clone + Fn(D, TributarySpec) -> Fut,
|
||||||
Pro: Processor,
|
Pro: Processor,
|
||||||
P: P2p,
|
|
||||||
>(
|
>(
|
||||||
db: &mut SubstrateDb<D>,
|
db: &mut SubstrateDb<D>,
|
||||||
key: &Zeroizing<<Ristretto as Ciphersuite>::F>,
|
key: &Zeroizing<<Ristretto as Ciphersuite>::F>,
|
||||||
add_new_tributary: ANT,
|
add_new_tributary: ANT,
|
||||||
p2p: &P,
|
|
||||||
processor: &mut Pro,
|
processor: &mut Pro,
|
||||||
serai: &Serai,
|
serai: &Serai,
|
||||||
block: Block,
|
block: Block,
|
||||||
|
@ -235,7 +232,6 @@ async fn handle_block<
|
||||||
// stable)
|
// stable)
|
||||||
if !SubstrateDb::<D>::handled_event(&db.0, hash, event_id) {
|
if !SubstrateDb::<D>::handled_event(&db.0, hash, event_id) {
|
||||||
if let ValidatorSetsEvent::NewSet { set } = new_set {
|
if let ValidatorSetsEvent::NewSet { set } = new_set {
|
||||||
// TODO2: Use a DB on a dedicated volume
|
|
||||||
handle_new_set(&db.0, key, add_new_tributary.clone(), processor, serai, &block, set)
|
handle_new_set(&db.0, key, add_new_tributary.clone(), processor, serai, &block, set)
|
||||||
.await?;
|
.await?;
|
||||||
} else {
|
} else {
|
||||||
|
@ -283,12 +279,10 @@ pub async fn handle_new_blocks<
|
||||||
Fut: Future<Output = ()>,
|
Fut: Future<Output = ()>,
|
||||||
ANT: Clone + Fn(D, TributarySpec) -> Fut,
|
ANT: Clone + Fn(D, TributarySpec) -> Fut,
|
||||||
Pro: Processor,
|
Pro: Processor,
|
||||||
P: P2p,
|
|
||||||
>(
|
>(
|
||||||
db: &mut SubstrateDb<D>,
|
db: &mut SubstrateDb<D>,
|
||||||
key: &Zeroizing<<Ristretto as Ciphersuite>::F>,
|
key: &Zeroizing<<Ristretto as Ciphersuite>::F>,
|
||||||
add_new_tributary: ANT,
|
add_new_tributary: ANT,
|
||||||
p2p: &P,
|
|
||||||
processor: &mut Pro,
|
processor: &mut Pro,
|
||||||
serai: &Serai,
|
serai: &Serai,
|
||||||
last_block: &mut u64,
|
last_block: &mut u64,
|
||||||
|
@ -306,7 +300,6 @@ pub async fn handle_new_blocks<
|
||||||
db,
|
db,
|
||||||
key,
|
key,
|
||||||
add_new_tributary.clone(),
|
add_new_tributary.clone(),
|
||||||
p2p,
|
|
||||||
processor,
|
processor,
|
||||||
serai,
|
serai,
|
||||||
if b == latest_number {
|
if b == latest_number {
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
use std::time::{Duration, SystemTime};
|
use std::time::{Duration, SystemTime};
|
||||||
|
|
||||||
use zeroize::Zeroizing;
|
use zeroize::Zeroizing;
|
||||||
|
|
||||||
use rand_core::{RngCore, CryptoRng, OsRng};
|
use rand_core::{RngCore, CryptoRng, OsRng};
|
||||||
|
use futures::{task::Poll, poll};
|
||||||
|
|
||||||
use ciphersuite::{
|
use ciphersuite::{
|
||||||
group::{ff::Field, GroupEncoding},
|
group::{ff::Field, GroupEncoding},
|
||||||
|
@ -95,11 +95,12 @@ pub async fn run_tributaries(
|
||||||
) {
|
) {
|
||||||
loop {
|
loop {
|
||||||
for (p2p, tributary) in tributaries.iter_mut() {
|
for (p2p, tributary) in tributaries.iter_mut() {
|
||||||
while let Some(msg) = p2p.receive().await {
|
while let Poll::Ready(msg) = poll!(p2p.receive()) {
|
||||||
match msg.0 {
|
match msg.kind {
|
||||||
P2pMessageKind::Tributary => {
|
P2pMessageKind::Tributary(genesis) => {
|
||||||
if tributary.handle_message(&msg.1).await {
|
assert_eq!(genesis, tributary.genesis());
|
||||||
p2p.broadcast(msg.0, msg.1).await;
|
if tributary.handle_message(&msg.msg).await {
|
||||||
|
p2p.broadcast(msg.kind, msg.msg).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -163,10 +164,11 @@ async fn tributary_test() {
|
||||||
let timeout = SystemTime::now() + Duration::from_secs(65);
|
let timeout = SystemTime::now() + Duration::from_secs(65);
|
||||||
while (blocks < 10) && (SystemTime::now().duration_since(timeout).is_err()) {
|
while (blocks < 10) && (SystemTime::now().duration_since(timeout).is_err()) {
|
||||||
for (p2p, tributary) in tributaries.iter_mut() {
|
for (p2p, tributary) in tributaries.iter_mut() {
|
||||||
while let Some(msg) = p2p.receive().await {
|
while let Poll::Ready(msg) = poll!(p2p.receive()) {
|
||||||
match msg.0 {
|
match msg.kind {
|
||||||
P2pMessageKind::Tributary => {
|
P2pMessageKind::Tributary(genesis) => {
|
||||||
tributary.handle_message(&msg.1).await;
|
assert_eq!(genesis, tributary.genesis());
|
||||||
|
tributary.handle_message(&msg.msg).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -187,10 +189,11 @@ async fn tributary_test() {
|
||||||
|
|
||||||
// Handle all existing messages
|
// Handle all existing messages
|
||||||
for (p2p, tributary) in tributaries.iter_mut() {
|
for (p2p, tributary) in tributaries.iter_mut() {
|
||||||
while let Some(msg) = p2p.receive().await {
|
while let Poll::Ready(msg) = poll!(p2p.receive()) {
|
||||||
match msg.0 {
|
match msg.kind {
|
||||||
P2pMessageKind::Tributary => {
|
P2pMessageKind::Tributary(genesis) => {
|
||||||
tributary.handle_message(&msg.1).await;
|
assert_eq!(genesis, tributary.genesis());
|
||||||
|
tributary.handle_message(&msg.msg).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@ async fn dkg_test() {
|
||||||
let keys = new_keys(&mut OsRng);
|
let keys = new_keys(&mut OsRng);
|
||||||
let spec = new_spec(&mut OsRng, &keys);
|
let spec = new_spec(&mut OsRng, &keys);
|
||||||
|
|
||||||
let mut tributaries = new_tributaries(&keys, &spec).await;
|
let tributaries = new_tributaries(&keys, &spec).await;
|
||||||
|
|
||||||
// Run the tributaries in the background
|
// Run the tributaries in the background
|
||||||
tokio::spawn(run_tributaries(tributaries.clone()));
|
tokio::spawn(run_tributaries(tributaries.clone()));
|
||||||
|
|
|
@ -19,7 +19,7 @@ async fn tx_test() {
|
||||||
let keys = new_keys(&mut OsRng);
|
let keys = new_keys(&mut OsRng);
|
||||||
let spec = new_spec(&mut OsRng, &keys);
|
let spec = new_spec(&mut OsRng, &keys);
|
||||||
|
|
||||||
let mut tributaries = new_tributaries(&keys, &spec).await;
|
let tributaries = new_tributaries(&keys, &spec).await;
|
||||||
|
|
||||||
// Run the tributaries in the background
|
// Run the tributaries in the background
|
||||||
tokio::spawn(run_tributaries(tributaries.clone()));
|
tokio::spawn(run_tributaries(tributaries.clone()));
|
||||||
|
|
|
@ -84,10 +84,6 @@ impl<D: Db, T: Transaction> Blockchain<D, T> {
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn genesis(&self) -> [u8; 32] {
|
|
||||||
self.genesis
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn tip(&self) -> [u8; 32] {
|
pub(crate) fn tip(&self) -> [u8; 32] {
|
||||||
self.tip
|
self.tip
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,8 @@ use ::tendermint::{
|
||||||
|
|
||||||
use serai_db::Db;
|
use serai_db::Db;
|
||||||
|
|
||||||
|
use tokio::sync::RwLock as AsyncRwLock;
|
||||||
|
|
||||||
mod merkle;
|
mod merkle;
|
||||||
pub(crate) use merkle::*;
|
pub(crate) use merkle::*;
|
||||||
|
|
||||||
|
@ -72,22 +74,23 @@ pub trait ReadWrite: Sized {
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
pub trait P2p: 'static + Send + Sync + Clone + Debug {
|
pub trait P2p: 'static + Send + Sync + Clone + Debug {
|
||||||
async fn broadcast(&self, msg: Vec<u8>);
|
async fn broadcast(&self, genesis: [u8; 32], msg: Vec<u8>);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl<P: P2p> P2p for Arc<P> {
|
impl<P: P2p> P2p for Arc<P> {
|
||||||
async fn broadcast(&self, msg: Vec<u8>) {
|
async fn broadcast(&self, genesis: [u8; 32], msg: Vec<u8>) {
|
||||||
(*self).broadcast(msg).await
|
(*self).broadcast(genesis, msg).await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct Tributary<D: Db, T: Transaction, P: P2p> {
|
pub struct Tributary<D: Db, T: Transaction, P: P2p> {
|
||||||
|
genesis: [u8; 32],
|
||||||
network: TendermintNetwork<D, T, P>,
|
network: TendermintNetwork<D, T, P>,
|
||||||
|
|
||||||
synced_block: SyncedBlockSender<TendermintNetwork<D, T, P>>,
|
synced_block: SyncedBlockSender<TendermintNetwork<D, T, P>>,
|
||||||
messages: MessageSender<TendermintNetwork<D, T, P>>,
|
messages: Arc<AsyncRwLock<MessageSender<TendermintNetwork<D, T, P>>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
||||||
|
@ -121,7 +124,7 @@ impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
||||||
TendermintMachine::new(network.clone(), block_number, start_time, proposal).await;
|
TendermintMachine::new(network.clone(), block_number, start_time, proposal).await;
|
||||||
tokio::task::spawn(machine.run());
|
tokio::task::spawn(machine.run());
|
||||||
|
|
||||||
Some(Self { network, synced_block, messages })
|
Some(Self { genesis, network, synced_block, messages: Arc::new(AsyncRwLock::new(messages)) })
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn block_time() -> u32 {
|
pub fn block_time() -> u32 {
|
||||||
|
@ -129,7 +132,7 @@ impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn genesis(&self) -> [u8; 32] {
|
pub fn genesis(&self) -> [u8; 32] {
|
||||||
self.network.blockchain.read().unwrap().genesis()
|
self.genesis
|
||||||
}
|
}
|
||||||
pub fn block_number(&self) -> u32 {
|
pub fn block_number(&self) -> u32 {
|
||||||
self.network.blockchain.read().unwrap().block_number()
|
self.network.blockchain.read().unwrap().block_number()
|
||||||
|
@ -153,12 +156,14 @@ impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns if the transaction was valid.
|
// Returns if the transaction was valid.
|
||||||
pub async fn add_transaction(&mut self, tx: T) -> bool {
|
// Safe to be &self since the only meaningful usage of self is self.network.blockchain which
|
||||||
|
// successfully acquires its own write lock.
|
||||||
|
pub async fn add_transaction(&self, tx: T) -> bool {
|
||||||
let mut to_broadcast = vec![TRANSACTION_MESSAGE];
|
let mut to_broadcast = vec![TRANSACTION_MESSAGE];
|
||||||
tx.write(&mut to_broadcast).unwrap();
|
tx.write(&mut to_broadcast).unwrap();
|
||||||
let res = self.network.blockchain.write().unwrap().add_transaction(true, tx);
|
let res = self.network.blockchain.write().unwrap().add_transaction(true, tx);
|
||||||
if res {
|
if res {
|
||||||
self.network.p2p.broadcast(to_broadcast).await;
|
self.network.p2p.broadcast(self.genesis, to_broadcast).await;
|
||||||
}
|
}
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
@ -189,7 +194,9 @@ impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return true if the message should be rebroadcasted.
|
// Return true if the message should be rebroadcasted.
|
||||||
pub async fn handle_message(&mut self, msg: &[u8]) -> bool {
|
// Safe to be &self since the only usage of self is on self.network.blockchain and self.messages,
|
||||||
|
// both which successfully acquire their own write locks and don't rely on each other
|
||||||
|
pub async fn handle_message(&self, msg: &[u8]) -> bool {
|
||||||
match msg.first() {
|
match msg.first() {
|
||||||
Some(&TRANSACTION_MESSAGE) => {
|
Some(&TRANSACTION_MESSAGE) => {
|
||||||
let Ok(tx) = T::read::<&[u8]>(&mut &msg[1 ..]) else {
|
let Ok(tx) = T::read::<&[u8]>(&mut &msg[1 ..]) else {
|
||||||
|
@ -212,7 +219,7 @@ impl<D: Db, T: Transaction, P: P2p> Tributary<D, T, P> {
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
self.messages.send(msg).await.unwrap();
|
self.messages.write().await.send(msg).await.unwrap();
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -256,7 +256,7 @@ impl<D: Db, T: Transaction, P: P2p> Network for TendermintNetwork<D, T, P> {
|
||||||
async fn broadcast(&mut self, msg: SignedMessageFor<Self>) {
|
async fn broadcast(&mut self, msg: SignedMessageFor<Self>) {
|
||||||
let mut to_broadcast = vec![TENDERMINT_MESSAGE];
|
let mut to_broadcast = vec![TENDERMINT_MESSAGE];
|
||||||
to_broadcast.extend(msg.encode());
|
to_broadcast.extend(msg.encode());
|
||||||
self.p2p.broadcast(to_broadcast).await
|
self.p2p.broadcast(self.genesis, to_broadcast).await
|
||||||
}
|
}
|
||||||
async fn slash(&mut self, validator: Self::ValidatorId) {
|
async fn slash(&mut self, validator: Self::ValidatorId) {
|
||||||
// TODO: Handle this slash
|
// TODO: Handle this slash
|
||||||
|
|
|
@ -263,6 +263,7 @@ pub trait Network: Send + Sync {
|
||||||
/// Trigger a slash for the validator in question who was definitively malicious.
|
/// Trigger a slash for the validator in question who was definitively malicious.
|
||||||
///
|
///
|
||||||
/// The exact process of triggering a slash is undefined and left to the network as a whole.
|
/// The exact process of triggering a slash is undefined and left to the network as a whole.
|
||||||
|
// TODO: We need to provide some evidence for this.
|
||||||
async fn slash(&mut self, validator: Self::ValidatorId);
|
async fn slash(&mut self, validator: Self::ValidatorId);
|
||||||
|
|
||||||
/// Validate a block.
|
/// Validate a block.
|
||||||
|
|
Loading…
Reference in a new issue