mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-08 20:09:54 +00:00
Use multiple LibP2P topics
We still need a peering protocol... Hopefully, we can read peers off of the Substrate node's DHT.
This commit is contained in:
parent
25066437da
commit
be48dcc4a4
2 changed files with 100 additions and 20 deletions
|
@ -58,6 +58,16 @@ pub enum P2pMessageKind {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl P2pMessageKind {
|
impl P2pMessageKind {
|
||||||
|
fn genesis(&self) -> Option<[u8; 32]> {
|
||||||
|
match self {
|
||||||
|
P2pMessageKind::KeepAlive => None,
|
||||||
|
P2pMessageKind::Tributary(genesis) => Some(*genesis),
|
||||||
|
P2pMessageKind::Heartbeat(genesis) => Some(*genesis),
|
||||||
|
P2pMessageKind::Block(genesis) => Some(*genesis),
|
||||||
|
P2pMessageKind::CosignedBlock => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn serialize(&self) -> Vec<u8> {
|
fn serialize(&self) -> Vec<u8> {
|
||||||
match self {
|
match self {
|
||||||
P2pMessageKind::KeepAlive => vec![0],
|
P2pMessageKind::KeepAlive => vec![0],
|
||||||
|
@ -119,14 +129,17 @@ pub struct Message<P: P2p> {
|
||||||
pub trait P2p: Send + Sync + Clone + fmt::Debug + TributaryP2p {
|
pub trait P2p: Send + Sync + Clone + fmt::Debug + TributaryP2p {
|
||||||
type Id: Send + Sync + Clone + Copy + fmt::Debug;
|
type Id: Send + Sync + Clone + Copy + fmt::Debug;
|
||||||
|
|
||||||
async fn send_raw(&self, to: Self::Id, msg: Vec<u8>);
|
async fn subscribe(&self, genesis: [u8; 32]);
|
||||||
async fn broadcast_raw(&self, msg: Vec<u8>);
|
async fn unsubscribe(&self, genesis: [u8; 32]);
|
||||||
|
|
||||||
|
async fn send_raw(&self, to: Self::Id, genesis: Option<[u8; 32]>, msg: Vec<u8>);
|
||||||
|
async fn broadcast_raw(&self, genesis: Option<[u8; 32]>, msg: Vec<u8>);
|
||||||
async fn receive_raw(&self) -> (Self::Id, Vec<u8>);
|
async fn receive_raw(&self) -> (Self::Id, Vec<u8>);
|
||||||
|
|
||||||
async fn send(&self, to: Self::Id, kind: P2pMessageKind, msg: Vec<u8>) {
|
async fn send(&self, to: Self::Id, kind: P2pMessageKind, msg: Vec<u8>) {
|
||||||
let mut actual_msg = kind.serialize();
|
let mut actual_msg = kind.serialize();
|
||||||
actual_msg.extend(msg);
|
actual_msg.extend(msg);
|
||||||
self.send_raw(to, actual_msg).await;
|
self.send_raw(to, kind.genesis(), actual_msg).await;
|
||||||
}
|
}
|
||||||
async fn broadcast(&self, kind: P2pMessageKind, msg: Vec<u8>) {
|
async fn broadcast(&self, kind: P2pMessageKind, msg: Vec<u8>) {
|
||||||
let mut actual_msg = kind.serialize();
|
let mut actual_msg = kind.serialize();
|
||||||
|
@ -143,7 +156,7 @@ pub trait P2p: Send + Sync + Clone + fmt::Debug + TributaryP2p {
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
*/
|
*/
|
||||||
self.broadcast_raw(actual_msg).await;
|
self.broadcast_raw(kind.genesis(), actual_msg).await;
|
||||||
}
|
}
|
||||||
async fn receive(&self) -> Message<Self> {
|
async fn receive(&self) -> Message<Self> {
|
||||||
let (sender, kind, msg) = loop {
|
let (sender, kind, msg) = loop {
|
||||||
|
@ -185,10 +198,11 @@ struct Behavior {
|
||||||
|
|
||||||
#[allow(clippy::type_complexity)]
|
#[allow(clippy::type_complexity)]
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct LibP2p(
|
pub struct LibP2p {
|
||||||
Arc<Mutex<mpsc::UnboundedSender<Vec<u8>>>>,
|
subscribe: Arc<Mutex<mpsc::UnboundedSender<(bool, [u8; 32])>>>,
|
||||||
Arc<Mutex<mpsc::UnboundedReceiver<(PeerId, Vec<u8>)>>>,
|
broadcast: Arc<Mutex<mpsc::UnboundedSender<(Option<[u8; 32]>, Vec<u8>)>>>,
|
||||||
);
|
receive: Arc<Mutex<mpsc::UnboundedReceiver<(PeerId, Vec<u8>)>>>,
|
||||||
|
}
|
||||||
impl fmt::Debug for LibP2p {
|
impl fmt::Debug for LibP2p {
|
||||||
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
fmt.debug_struct("LibP2p").finish_non_exhaustive()
|
fmt.debug_struct("LibP2p").finish_non_exhaustive()
|
||||||
|
@ -240,8 +254,7 @@ impl LibP2p {
|
||||||
)
|
)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
// Uses a single topic to prevent being a BTC validator only connected to ETH validators,
|
// Subscribe to the base topic
|
||||||
// unable to communicate with other BTC validators
|
|
||||||
let topic = IdentTopic::new(LIBP2P_TOPIC);
|
let topic = IdentTopic::new(LIBP2P_TOPIC);
|
||||||
gossipsub.subscribe(&topic).unwrap();
|
gossipsub.subscribe(&topic).unwrap();
|
||||||
|
|
||||||
|
@ -274,6 +287,11 @@ impl LibP2p {
|
||||||
|
|
||||||
let (broadcast_send, mut broadcast_recv) = mpsc::unbounded_channel();
|
let (broadcast_send, mut broadcast_recv) = mpsc::unbounded_channel();
|
||||||
let (receive_send, receive_recv) = mpsc::unbounded_channel();
|
let (receive_send, receive_recv) = mpsc::unbounded_channel();
|
||||||
|
let (subscribe_send, mut subscribe_recv) = mpsc::unbounded_channel();
|
||||||
|
|
||||||
|
fn topic_for_genesis(genesis: [u8; 32]) -> IdentTopic {
|
||||||
|
IdentTopic::new(format!("{LIBP2P_TOPIC}-{}", hex::encode(genesis)))
|
||||||
|
}
|
||||||
|
|
||||||
tokio::spawn({
|
tokio::spawn({
|
||||||
let mut time_of_last_p2p_message = Instant::now();
|
let mut time_of_last_p2p_message = Instant::now();
|
||||||
|
@ -282,12 +300,19 @@ impl LibP2p {
|
||||||
async fn broadcast_raw(
|
async fn broadcast_raw(
|
||||||
p2p: &mut Swarm<Behavior>,
|
p2p: &mut Swarm<Behavior>,
|
||||||
time_of_last_p2p_message: &mut Instant,
|
time_of_last_p2p_message: &mut Instant,
|
||||||
|
genesis: Option<[u8; 32]>,
|
||||||
msg: Vec<u8>,
|
msg: Vec<u8>,
|
||||||
) {
|
) {
|
||||||
// Update the time of last message
|
// Update the time of last message
|
||||||
*time_of_last_p2p_message = Instant::now();
|
*time_of_last_p2p_message = Instant::now();
|
||||||
|
|
||||||
match p2p.behaviour_mut().gossipsub.publish(IdentTopic::new(LIBP2P_TOPIC), msg.clone()) {
|
let topic = if let Some(genesis) = genesis {
|
||||||
|
topic_for_genesis(genesis)
|
||||||
|
} else {
|
||||||
|
IdentTopic::new(LIBP2P_TOPIC)
|
||||||
|
};
|
||||||
|
|
||||||
|
match p2p.behaviour_mut().gossipsub.publish(topic, msg.clone()) {
|
||||||
Err(PublishError::SigningError(e)) => panic!("signing error when broadcasting: {e}"),
|
Err(PublishError::SigningError(e)) => panic!("signing error when broadcasting: {e}"),
|
||||||
Err(PublishError::InsufficientPeers) => {
|
Err(PublishError::InsufficientPeers) => {
|
||||||
log::warn!("failed to send p2p message due to insufficient peers")
|
log::warn!("failed to send p2p message due to insufficient peers")
|
||||||
|
@ -307,12 +332,32 @@ impl LibP2p {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
biased;
|
biased;
|
||||||
|
|
||||||
|
// Subscribe to any new topics
|
||||||
|
topic = subscribe_recv.recv() => {
|
||||||
|
let (subscribe, topic) = topic.expect("subscribe_recv closed. are we shutting down?");
|
||||||
|
if subscribe {
|
||||||
|
swarm
|
||||||
|
.behaviour_mut()
|
||||||
|
.gossipsub
|
||||||
|
.subscribe(&topic_for_genesis(topic))
|
||||||
|
.unwrap();
|
||||||
|
} else {
|
||||||
|
swarm
|
||||||
|
.behaviour_mut()
|
||||||
|
.gossipsub
|
||||||
|
.unsubscribe(&topic_for_genesis(topic))
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Handle any queued outbound messages
|
// Handle any queued outbound messages
|
||||||
msg = broadcast_recv.recv() => {
|
msg = broadcast_recv.recv() => {
|
||||||
|
let (genesis, msg) = msg.expect("broadcast_recv closed. are we shutting down?");
|
||||||
broadcast_raw(
|
broadcast_raw(
|
||||||
&mut swarm,
|
&mut swarm,
|
||||||
&mut time_of_last_p2p_message,
|
&mut time_of_last_p2p_message,
|
||||||
msg.expect("broadcast_recv closed. are we shutting down?")
|
genesis,
|
||||||
|
msg,
|
||||||
).await;
|
).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -363,6 +408,7 @@ impl LibP2p {
|
||||||
broadcast_raw(
|
broadcast_raw(
|
||||||
&mut swarm,
|
&mut swarm,
|
||||||
&mut time_of_last_p2p_message,
|
&mut time_of_last_p2p_message,
|
||||||
|
None,
|
||||||
P2pMessageKind::KeepAlive.serialize()
|
P2pMessageKind::KeepAlive.serialize()
|
||||||
).await;
|
).await;
|
||||||
}
|
}
|
||||||
|
@ -371,7 +417,11 @@ impl LibP2p {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
LibP2p(Arc::new(Mutex::new(broadcast_send)), Arc::new(Mutex::new(receive_recv)))
|
LibP2p {
|
||||||
|
subscribe: Arc::new(Mutex::new(subscribe_send)),
|
||||||
|
broadcast: Arc::new(Mutex::new(broadcast_send)),
|
||||||
|
receive: Arc::new(Mutex::new(receive_recv)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -379,18 +429,41 @@ impl LibP2p {
|
||||||
impl P2p for LibP2p {
|
impl P2p for LibP2p {
|
||||||
type Id = PeerId;
|
type Id = PeerId;
|
||||||
|
|
||||||
async fn send_raw(&self, _: Self::Id, msg: Vec<u8>) {
|
async fn subscribe(&self, genesis: [u8; 32]) {
|
||||||
self.broadcast_raw(msg).await;
|
self
|
||||||
|
.subscribe
|
||||||
|
.lock()
|
||||||
|
.await
|
||||||
|
.send((true, genesis))
|
||||||
|
.expect("subscribe_send closed. are we shutting down?");
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn broadcast_raw(&self, msg: Vec<u8>) {
|
async fn unsubscribe(&self, genesis: [u8; 32]) {
|
||||||
self.0.lock().await.send(msg).expect("broadcast_send closed. are we shutting down?");
|
self
|
||||||
|
.subscribe
|
||||||
|
.lock()
|
||||||
|
.await
|
||||||
|
.send((false, genesis))
|
||||||
|
.expect("subscribe_send closed. are we shutting down?");
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn send_raw(&self, _: Self::Id, genesis: Option<[u8; 32]>, msg: Vec<u8>) {
|
||||||
|
self.broadcast_raw(genesis, msg).await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn broadcast_raw(&self, genesis: Option<[u8; 32]>, msg: Vec<u8>) {
|
||||||
|
self
|
||||||
|
.broadcast
|
||||||
|
.lock()
|
||||||
|
.await
|
||||||
|
.send((genesis, msg))
|
||||||
|
.expect("broadcast_send closed. are we shutting down?");
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: We only have a single handle call this. Differentiate Send/Recv to remove this constant
|
// TODO: We only have a single handle call this. Differentiate Send/Recv to remove this constant
|
||||||
// lock acquisition?
|
// lock acquisition?
|
||||||
async fn receive_raw(&self) -> (Self::Id, Vec<u8>) {
|
async fn receive_raw(&self) -> (Self::Id, Vec<u8>) {
|
||||||
self.1.lock().await.recv().await.expect("receive_recv closed. are we shutting down?")
|
self.receive.lock().await.recv().await.expect("receive_recv closed. are we shutting down?")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -473,6 +546,9 @@ pub async fn handle_p2p_task<D: Db, P: P2p>(
|
||||||
let (send, mut recv) = mpsc::unbounded_channel();
|
let (send, mut recv) = mpsc::unbounded_channel();
|
||||||
channels.write().await.insert(genesis, send);
|
channels.write().await.insert(genesis, send);
|
||||||
|
|
||||||
|
// Subscribe to the topic for this tributary
|
||||||
|
p2p.subscribe(genesis).await;
|
||||||
|
|
||||||
// Per-Tributary P2P message handler
|
// Per-Tributary P2P message handler
|
||||||
tokio::spawn({
|
tokio::spawn({
|
||||||
let p2p = p2p.clone();
|
let p2p = p2p.clone();
|
||||||
|
@ -595,6 +671,7 @@ pub async fn handle_p2p_task<D: Db, P: P2p>(
|
||||||
TributaryEvent::TributaryRetired(set) => {
|
TributaryEvent::TributaryRetired(set) => {
|
||||||
if let Some(genesis) = set_to_genesis.remove(&set) {
|
if let Some(genesis) = set_to_genesis.remove(&set) {
|
||||||
channels.write().await.remove(&genesis);
|
channels.write().await.remove(&genesis);
|
||||||
|
p2p.unsubscribe(genesis).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,11 +62,14 @@ impl LocalP2p {
|
||||||
impl P2p for LocalP2p {
|
impl P2p for LocalP2p {
|
||||||
type Id = usize;
|
type Id = usize;
|
||||||
|
|
||||||
async fn send_raw(&self, to: Self::Id, msg: Vec<u8>) {
|
async fn subscribe(&self, _genesis: [u8; 32]) {}
|
||||||
|
async fn unsubscribe(&self, _genesis: [u8; 32]) {}
|
||||||
|
|
||||||
|
async fn send_raw(&self, to: Self::Id, _genesis: Option<[u8; 32]>, msg: Vec<u8>) {
|
||||||
self.1.write().await.1[to].push_back((self.0, msg));
|
self.1.write().await.1[to].push_back((self.0, msg));
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn broadcast_raw(&self, msg: Vec<u8>) {
|
async fn broadcast_raw(&self, _genesis: Option<[u8; 32]>, msg: Vec<u8>) {
|
||||||
// Content-based deduplication
|
// Content-based deduplication
|
||||||
let mut lock = self.1.write().await;
|
let mut lock = self.1.write().await;
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue