Mix: Integrate packet encoding

This commit is contained in:
Youngjoon Lee 2024-11-02 17:36:29 +09:00
parent 5c8b84d3e1
commit 906fe6e2e5
No known key found for this signature in database
GPG Key ID: 167546E2D1712F8C
27 changed files with 387 additions and 203 deletions

View File

@ -83,9 +83,6 @@ pub struct MixArgs {
#[clap(long = "mix-node-key", env = "MIX_NODE_KEY")] #[clap(long = "mix-node-key", env = "MIX_NODE_KEY")]
mix_node_key: Option<String>, mix_node_key: Option<String>,
#[clap(long = "mix-membership", env = "MIX_MEMBERSHIP", num_args = 1.., value_delimiter = ',')]
pub mix_membership: Option<Vec<Multiaddr>>,
#[clap(long = "mix-peering-degree", env = "MIX_PEERING_DEGREE")] #[clap(long = "mix-peering-degree", env = "MIX_PEERING_DEGREE")]
mix_peering_degree: Option<usize>, mix_peering_degree: Option<usize>,
@ -256,7 +253,6 @@ pub fn update_mix(
let MixArgs { let MixArgs {
mix_addr, mix_addr,
mix_node_key, mix_node_key,
mix_membership,
mix_peering_degree, mix_peering_degree,
mix_num_mix_layers, mix_num_mix_layers,
} = mix_args; } = mix_args;
@ -270,10 +266,6 @@ pub fn update_mix(
mix.backend.node_key = SecretKey::try_from_bytes(key_bytes.as_mut_slice())?; mix.backend.node_key = SecretKey::try_from_bytes(key_bytes.as_mut_slice())?;
} }
if let Some(membership) = mix_membership {
mix.backend.membership = membership;
}
if let Some(peering_degree) = mix_peering_degree { if let Some(peering_degree) = mix_peering_degree {
mix.backend.peering_degree = peering_degree; mix.backend.peering_degree = peering_degree;
} }

View File

@ -5,6 +5,7 @@ edition = "2021"
[dependencies] [dependencies]
cached = "0.53" cached = "0.53"
thiserror = "1"
tokio = { version = "1", features = ["time", "sync", "macros"] } tokio = { version = "1", features = ["time", "sync", "macros"] }
tracing = "0.1" tracing = "0.1"
rand = "0.8" rand = "0.8"
@ -12,7 +13,7 @@ serde = { version = "1.0", features = ["derive"] }
nomos-mix-message = { path = "../message" } nomos-mix-message = { path = "../message" }
futures = "0.3" futures = "0.3"
rand_chacha = "0.3" rand_chacha = "0.3"
multiaddr = "0.18.2"
[dev-dependencies] [dev-dependencies]
tokio = { version = "1", features = ["rt-multi-thread"] } tokio = { version = "1", features = ["rt-multi-thread"] }

View File

@ -0,0 +1,7 @@
#[derive(thiserror::Error, Debug)]
pub enum Error {
#[error("Not sufficient nodes")]
NotSufficientNodes,
#[error("Mix message error: {0}")]
MixMessageError(#[from] nomos_mix_message::Error),
}

View File

@ -1,2 +1,4 @@
pub mod error;
pub mod membership;
pub mod message_blend; pub mod message_blend;
pub mod persistent_transmission; pub mod persistent_transmission;

View File

@ -0,0 +1,24 @@
use multiaddr::Multiaddr;
use rand::{seq::SliceRandom, Rng};
use serde::{Deserialize, Serialize};
#[derive(Clone, Debug)]
pub struct Membership {
remote_nodes: Vec<Node>,
}
#[derive(Clone, Debug, Serialize, Deserialize)]
pub struct Node {
pub address: Multiaddr,
pub public_key: [u8; 32],
}
impl Membership {
pub fn new(remote_nodes: Vec<Node>) -> Self {
Self { remote_nodes }
}
pub fn choose_nodes<R: Rng>(&self, rng: &mut R, amount: usize) -> Vec<&Node> {
self.remote_nodes.choose_multiple(rng, amount).collect()
}
}

View File

@ -1,32 +1,60 @@
use nomos_mix_message::{new_message, unwrap_message}; use nomos_mix_message::MessageBuilder;
use rand::SeedableRng;
use rand_chacha::ChaCha12Rng;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::{error::Error, membership::Membership};
/// [`CryptographicProcessor`] is responsible for wrapping and unwrapping messages /// [`CryptographicProcessor`] is responsible for wrapping and unwrapping messages
/// for the message indistinguishability. /// for the message indistinguishability.
pub(crate) struct CryptographicProcessor { pub(crate) struct CryptographicProcessor {
settings: CryptographicProcessorSettings, settings: CryptographicProcessorSettings,
message_builder: MessageBuilder,
membership: Membership,
rng: ChaCha12Rng,
} }
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
pub struct CryptographicProcessorSettings { pub struct CryptographicProcessorSettings {
pub num_mix_layers: usize, pub num_mix_layers: usize,
pub max_num_mix_layers: usize,
pub max_payload_size: usize,
pub private_key: [u8; 32],
} }
impl CryptographicProcessor { impl CryptographicProcessor {
pub(crate) fn new(settings: CryptographicProcessorSettings) -> Self { pub(crate) fn new(
Self { settings } settings: CryptographicProcessorSettings,
message_builder: MessageBuilder,
membership: Membership,
) -> Self {
Self {
settings,
message_builder,
membership,
rng: ChaCha12Rng::from_entropy(),
}
} }
pub(crate) fn wrap_message(&self, message: &[u8]) -> Result<Vec<u8>, nomos_mix_message::Error> { pub(crate) fn wrap_message(&mut self, message: &[u8]) -> Result<Vec<u8>, Error> {
// TODO: Use the actual Sphinx encoding instead of mock. let public_keys = self.choose_public_keys(self.settings.num_mix_layers);
// TODO: Select `num_mix_layers` random nodes from the membership. if public_keys.len() < self.settings.num_mix_layers {
new_message(message, self.settings.num_mix_layers.try_into().unwrap()) return Err(Error::NotSufficientNodes);
}
Ok(self.message_builder.new_message(public_keys, message)?)
} }
pub(crate) fn unwrap_message( pub(crate) fn unwrap_message(&self, message: &[u8]) -> Result<(Vec<u8>, bool), Error> {
&self, Ok(self
message: &[u8], .message_builder
) -> Result<(Vec<u8>, bool), nomos_mix_message::Error> { .unpack_message(message, self.settings.private_key)?)
unwrap_message(message) }
fn choose_public_keys(&mut self, amount: usize) -> Vec<[u8; 32]> {
self.membership
.choose_nodes(&mut self.rng, amount)
.iter()
.map(|node| node.public_key)
.collect()
} }
} }

View File

@ -3,12 +3,17 @@ mod temporal;
pub use crypto::CryptographicProcessorSettings; pub use crypto::CryptographicProcessorSettings;
use futures::StreamExt; use futures::StreamExt;
use nomos_mix_message::MessageBuilder;
pub use temporal::TemporalProcessorSettings; pub use temporal::TemporalProcessorSettings;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use tokio::sync::mpsc; use tokio::sync::mpsc;
use crate::message_blend::{crypto::CryptographicProcessor, temporal::TemporalProcessor}; use crate::{
error::Error,
membership::Membership,
message_blend::{crypto::CryptographicProcessor, temporal::TemporalProcessor},
};
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
pub struct MessageBlendSettings { pub struct MessageBlendSettings {
@ -38,6 +43,8 @@ pub struct MessageBlend {
impl MessageBlend { impl MessageBlend {
pub fn new( pub fn new(
settings: MessageBlendSettings, settings: MessageBlendSettings,
message_builder: MessageBuilder,
membership: Membership,
new_message_receiver: mpsc::UnboundedReceiver<Vec<u8>>, new_message_receiver: mpsc::UnboundedReceiver<Vec<u8>>,
inbound_message_receiver: mpsc::UnboundedReceiver<Vec<u8>>, inbound_message_receiver: mpsc::UnboundedReceiver<Vec<u8>>,
outbound_message_sender: mpsc::UnboundedSender<Vec<u8>>, outbound_message_sender: mpsc::UnboundedSender<Vec<u8>>,
@ -48,7 +55,11 @@ impl MessageBlend {
inbound_message_receiver, inbound_message_receiver,
outbound_message_sender, outbound_message_sender,
fully_unwrapped_message_sender, fully_unwrapped_message_sender,
cryptographic_processor: CryptographicProcessor::new(settings.cryptographic_processor), cryptographic_processor: CryptographicProcessor::new(
settings.cryptographic_processor,
message_builder,
membership,
),
temporal_processor: TemporalProcessor::<_>::new(settings.temporal_processor), temporal_processor: TemporalProcessor::<_>::new(settings.temporal_processor),
} }
} }
@ -93,7 +104,7 @@ impl MessageBlend {
fully_unwrapped, fully_unwrapped,
}); });
} }
Err(nomos_mix_message::Error::MsgUnwrapNotAllowed) => { Err(Error::MixMessageError(nomos_mix_message::Error::MsgUnwrapNotAllowed)) => {
tracing::debug!("Message cannot be unwrapped by this node"); tracing::debug!("Message cannot be unwrapped by this node");
} }
Err(e) => { Err(e) => {

View File

@ -1,5 +1,4 @@
use futures::Stream; use futures::Stream;
use nomos_mix_message::DROP_MESSAGE;
use rand::{distributions::Uniform, prelude::Distribution, Rng, SeedableRng}; use rand::{distributions::Uniform, prelude::Distribution, Rng, SeedableRng};
use rand_chacha::ChaCha12Rng; use rand_chacha::ChaCha12Rng;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
@ -35,6 +34,7 @@ where
{ {
interval: Interval, interval: Interval,
coin: Coin<ChaCha12Rng>, coin: Coin<ChaCha12Rng>,
drop_message: Vec<u8>,
stream: S, stream: S,
} }
@ -44,6 +44,7 @@ where
{ {
pub fn new( pub fn new(
settings: PersistentTransmissionSettings, settings: PersistentTransmissionSettings,
drop_message: Vec<u8>,
stream: S, stream: S,
) -> PersistentTransmissionStream<S> { ) -> PersistentTransmissionStream<S> {
let interval = time::interval(Duration::from_secs_f64( let interval = time::interval(Duration::from_secs_f64(
@ -57,6 +58,7 @@ where
Self { Self {
interval, interval,
coin, coin,
drop_message,
stream, stream,
} }
} }
@ -73,6 +75,7 @@ where
ref mut interval, ref mut interval,
ref mut stream, ref mut stream,
ref mut coin, ref mut coin,
ref drop_message,
.. ..
} = self.get_mut(); } = self.get_mut();
if pin!(interval).poll_tick(cx).is_pending() { if pin!(interval).poll_tick(cx).is_pending() {
@ -81,7 +84,7 @@ where
if let Poll::Ready(Some(item)) = pin!(stream).poll_next(cx) { if let Poll::Ready(Some(item)) = pin!(stream).poll_next(cx) {
Poll::Ready(Some(item)) Poll::Ready(Some(item))
} else if coin.flip() { } else if coin.flip() {
Poll::Ready(Some(DROP_MESSAGE.to_vec())) Poll::Ready(Some(drop_message.clone()))
} else { } else {
Poll::Pending Poll::Pending
} }
@ -92,11 +95,12 @@ pub trait PersistentTransmissionExt: Stream {
fn persistent_transmission( fn persistent_transmission(
self, self,
settings: PersistentTransmissionSettings, settings: PersistentTransmissionSettings,
drop_message: Vec<u8>,
) -> PersistentTransmissionStream<Self> ) -> PersistentTransmissionStream<Self>
where where
Self: Sized + Unpin, Self: Sized + Unpin,
{ {
PersistentTransmissionStream::new(settings, self) PersistentTransmissionStream::new(settings, drop_message, self)
} }
} }
@ -111,6 +115,7 @@ impl<S> PersistentTransmissionExt for S where S: Stream {}
/// * `emission_sender` - The channel to emit messages /// * `emission_sender` - The channel to emit messages
pub async fn persistent_transmission( pub async fn persistent_transmission(
settings: PersistentTransmissionSettings, settings: PersistentTransmissionSettings,
drop_message: Vec<u8>,
schedule_receiver: mpsc::UnboundedReceiver<Vec<u8>>, schedule_receiver: mpsc::UnboundedReceiver<Vec<u8>>,
emission_sender: mpsc::UnboundedSender<Vec<u8>>, emission_sender: mpsc::UnboundedSender<Vec<u8>>,
) { ) {
@ -138,7 +143,7 @@ pub async fn persistent_transmission(
Err(TryRecvError::Empty) => { Err(TryRecvError::Empty) => {
// If the coin is head, emit the drop message. // If the coin is head, emit the drop message.
if coin.flip() { if coin.flip() {
if let Err(e) = emission_sender.send(DROP_MESSAGE.to_vec()) { if let Err(e) = emission_sender.send(drop_message.clone()) {
tracing::error!( tracing::error!(
"Failed to send drop message to the transmission channel: {e:?}" "Failed to send drop message to the transmission channel: {e:?}"
); );
@ -228,8 +233,10 @@ mod tests {
let upper_bound = expected_emission_interval + torelance; let upper_bound = expected_emission_interval + torelance;
// Start the persistent transmission and schedule messages // Start the persistent transmission and schedule messages
let drop_message = vec![0];
tokio::spawn(persistent_transmission( tokio::spawn(persistent_transmission(
settings, settings,
drop_message.clone(),
schedule_receiver, schedule_receiver,
emission_sender, emission_sender,
)); ));
@ -248,16 +255,10 @@ mod tests {
assert_eq!(emission_receiver.recv().await.unwrap(), vec![3]); assert_eq!(emission_receiver.recv().await.unwrap(), vec![3]);
assert_interval!(&mut last_time, lower_bound, upper_bound); assert_interval!(&mut last_time, lower_bound, upper_bound);
assert_eq!( assert_eq!(emission_receiver.recv().await.unwrap(), drop_message);
emission_receiver.recv().await.unwrap(),
DROP_MESSAGE.to_vec()
);
assert_interval!(&mut last_time, lower_bound, upper_bound); assert_interval!(&mut last_time, lower_bound, upper_bound);
assert_eq!( assert_eq!(emission_receiver.recv().await.unwrap(), drop_message);
emission_receiver.recv().await.unwrap(),
DROP_MESSAGE.to_vec()
);
assert_interval!(&mut last_time, lower_bound, upper_bound); assert_interval!(&mut last_time, lower_bound, upper_bound);
// Schedule a new message and check if it is emitted at the next interval // Schedule a new message and check if it is emitted at the next interval
@ -282,7 +283,9 @@ mod tests {
let lower_bound = expected_emission_interval - torelance; let lower_bound = expected_emission_interval - torelance;
let upper_bound = expected_emission_interval + torelance; let upper_bound = expected_emission_interval + torelance;
// prepare stream // prepare stream
let mut persistent_transmission_stream = stream.persistent_transmission(settings); let drop_message = vec![0];
let mut persistent_transmission_stream =
stream.persistent_transmission(settings, drop_message.clone());
// Messages must be scheduled in non-blocking manner. // Messages must be scheduled in non-blocking manner.
schedule_sender.send(vec![1]).unwrap(); schedule_sender.send(vec![1]).unwrap();
schedule_sender.send(vec![2]).unwrap(); schedule_sender.send(vec![2]).unwrap();
@ -309,13 +312,13 @@ mod tests {
assert_eq!( assert_eq!(
persistent_transmission_stream.next().await.unwrap(), persistent_transmission_stream.next().await.unwrap(),
DROP_MESSAGE.to_vec() drop_message
); );
assert_interval!(&mut last_time, lower_bound, upper_bound); assert_interval!(&mut last_time, lower_bound, upper_bound);
assert_eq!( assert_eq!(
persistent_transmission_stream.next().await.unwrap(), persistent_transmission_stream.next().await.unwrap(),
DROP_MESSAGE.to_vec() drop_message
); );
assert_interval!(&mut last_time, lower_bound, upper_bound); assert_interval!(&mut last_time, lower_bound, upper_bound);

View File

@ -5,66 +5,106 @@ mod routing;
pub use error::Error; pub use error::Error;
use sha2::{Digest, Sha256}; use packet::{Packet, UnpackedPacket};
pub const MSG_SIZE: usize = 2048; pub struct MessageBuilder {
pub const DROP_MESSAGE: [u8; MSG_SIZE] = [0; MSG_SIZE]; max_layers: usize,
max_payload_size: usize,
drop_message: Vec<u8>,
}
// TODO: Remove all the mock below once the actual implementation is integrated to the system. #[repr(u8)]
// pub enum MessageFlag {
/// A mock implementation of the Sphinx encoding. Drop = 0x00,
/// Data = 0x01,
/// The length of the encoded message is fixed to [`MSG_SIZE`] bytes. }
/// The first byte of the encoded message is the number of remaining layers to be unwrapped.
/// The remaining bytes are the payload that is zero-padded to the end. impl MessageBuilder {
pub fn new_message(payload: &[u8], num_layers: u8) -> Result<Vec<u8>, Error> { pub fn new(max_layers: usize, max_payload_size: usize) -> Result<Self, Error> {
if payload.len() > MSG_SIZE - 1 { Ok(Self {
return Err(Error::PayloadTooLarge); max_layers,
max_payload_size,
drop_message: Self::new_drop_message(max_layers, max_payload_size)?,
})
} }
let mut message: Vec<u8> = Vec::with_capacity(MSG_SIZE); pub fn new_message(
message.push(num_layers); &self,
message.extend(payload); recipient_pubkeys: Vec<[u8; 32]>,
message.extend(std::iter::repeat(0).take(MSG_SIZE - message.len())); payload: &[u8],
Ok(message) ) -> Result<Vec<u8>, Error> {
} let recipient_pubkeys = recipient_pubkeys
.into_iter()
.map(x25519_dalek::PublicKey::from)
.collect::<Vec<_>>();
let packet = Packet::build(
&recipient_pubkeys,
self.max_layers,
payload,
self.max_payload_size,
)?;
Ok(Self::concat_flag(MessageFlag::Data, &packet.to_bytes()))
}
/// SHA-256 hash of the message fn new_drop_message(max_layers: usize, max_payload_size: usize) -> Result<Vec<u8>, Error> {
pub fn message_id(message: &[u8]) -> Vec<u8> { let dummy_packet = Packet::build(
let mut hasher = Sha256::new(); &[x25519_dalek::PublicKey::from(
hasher.update(message); &x25519_dalek::EphemeralSecret::random(),
hasher.finalize().to_vec() )],
} max_layers,
&[0u8; 1],
max_payload_size,
)?;
Ok(Self::concat_flag(
MessageFlag::Drop,
&dummy_packet.to_bytes(),
))
}
/// Unwrap the message one layer. pub fn drop_message(&self) -> Vec<u8> {
/// self.drop_message.clone()
/// This function returns the unwrapped message and a boolean indicating whether the message was fully unwrapped. }
/// (False if the message still has layers to be unwrapped, true otherwise)
/// pub fn is_drop_message(message: &[u8]) -> bool {
/// If the input message was already fully unwrapped, or if ititss format is invalid, Self::check_flag(MessageFlag::Drop, message).is_ok()
/// this function returns `[Error::InvalidMixMessage]`. }
pub fn unwrap_message(message: &[u8]) -> Result<(Vec<u8>, bool), Error> {
if message.is_empty() { pub fn unpack_message(
&self,
message: &[u8],
private_key: [u8; 32],
) -> Result<(Vec<u8>, bool), Error> {
let message = Self::check_flag(MessageFlag::Data, message)?;
let packet = Packet::from_bytes(message, self.max_layers)?;
let private_key = x25519_dalek::StaticSecret::from(private_key);
Ok(match packet.unpack(&private_key, self.max_layers)? {
UnpackedPacket::ToForward(m) => {
(Self::concat_flag(MessageFlag::Data, &m.to_bytes()), false)
}
UnpackedPacket::FullyUnpacked(m) => (m, true),
})
}
pub fn message_size(&self) -> usize {
self.drop_message.len()
}
fn concat_flag(flag: MessageFlag, bytes: &[u8]) -> Vec<u8> {
concat_bytes(&[&[flag as u8], bytes])
}
fn check_flag(flag: MessageFlag, message: &[u8]) -> Result<&[u8], Error> {
if message.first() != Some(&(flag as u8)) {
return Err(Error::InvalidMixMessage); return Err(Error::InvalidMixMessage);
} }
if message.len() == 1 {
match message[0] { Ok(&[])
0 => Err(Error::InvalidMixMessage), } else {
1 => Ok((message[1..].to_vec(), true)), Ok(&message[1..])
n => {
let mut unwrapped: Vec<u8> = Vec::with_capacity(message.len());
unwrapped.push(n - 1);
unwrapped.extend(&message[1..]);
Ok((unwrapped, false))
} }
} }
} }
/// Check if the message is a drop message.
pub fn is_drop_message(message: &[u8]) -> bool {
message == DROP_MESSAGE
}
pub(crate) fn concat_bytes(bytes_list: &[&[u8]]) -> Vec<u8> { pub(crate) fn concat_bytes(bytes_list: &[&[u8]]) -> Vec<u8> {
let mut buf = Vec::with_capacity(bytes_list.iter().map(|bytes| bytes.len()).sum()); let mut buf = Vec::with_capacity(bytes_list.iter().map(|bytes| bytes.len()).sum());
bytes_list bytes_list
@ -87,3 +127,20 @@ pub(crate) fn parse_bytes<'a>(data: &'a [u8], sizes: &[usize]) -> Result<Vec<&'a
}) })
.collect() .collect()
} }
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn message_builder() {
let builder = MessageBuilder::new(5, 2048).unwrap();
let drop_message = builder.drop_message();
assert_eq!(drop_message.len(), builder.message_size());
let data_message = builder.new_message(vec![[1u8; 32]], &[10u8; 100]).unwrap();
assert!(data_message != drop_message);
assert_eq!(data_message.len(), builder.message_size());
}
}

View File

@ -9,6 +9,7 @@ futures = "0.3.30"
futures-timer = "3.0.3" futures-timer = "3.0.3"
libp2p = "0.53" libp2p = "0.53"
tracing = "0.1" tracing = "0.1"
sha2 = "0.10"
nomos-mix = { path = "../core" } nomos-mix = { path = "../core" }
nomos-mix-message = { path = "../message" } nomos-mix-message = { path = "../message" }

View File

@ -12,7 +12,8 @@ use libp2p::{
}, },
Multiaddr, PeerId, Multiaddr, PeerId,
}; };
use nomos_mix_message::{is_drop_message, message_id}; use nomos_mix_message::MessageBuilder;
use sha2::{Digest, Sha256};
use crate::{ use crate::{
error::Error, error::Error,
@ -61,12 +62,12 @@ impl Behaviour {
/// Publish a message (data or drop) to all connected peers /// Publish a message (data or drop) to all connected peers
pub fn publish(&mut self, message: Vec<u8>) -> Result<(), Error> { pub fn publish(&mut self, message: Vec<u8>) -> Result<(), Error> {
if is_drop_message(&message) { if MessageBuilder::is_drop_message(&message) {
// Bypass deduplication for the drop message // Bypass deduplication for the drop message
return self.forward_message(message, None); return self.forward_message(message, None);
} }
let msg_id = message_id(&message); let msg_id = Self::message_id(&message);
// If the message was already seen, don't forward it again // If the message was already seen, don't forward it again
if self.duplicate_cache.cache_get(&msg_id).is_some() { if self.duplicate_cache.cache_get(&msg_id).is_some() {
return Ok(()); return Ok(());
@ -110,6 +111,12 @@ impl Behaviour {
Ok(()) Ok(())
} }
fn message_id(message: &[u8]) -> Vec<u8> {
let mut hasher = Sha256::new();
hasher.update(message);
hasher.finalize().to_vec()
}
fn add_negotiated_peer(&mut self, peer_id: PeerId, connection_id: ConnectionId) -> bool { fn add_negotiated_peer(&mut self, peer_id: PeerId, connection_id: ConnectionId) -> bool {
tracing::debug!( tracing::debug!(
"Adding to connected_peers: peer_id:{:?}, connection_id:{:?}", "Adding to connected_peers: peer_id:{:?}, connection_id:{:?}",
@ -191,14 +198,14 @@ impl NetworkBehaviour for Behaviour {
// A message was forwarded from the peer. // A message was forwarded from the peer.
ToBehaviour::Message(message) => { ToBehaviour::Message(message) => {
// Ignore drop message // Ignore drop message
if is_drop_message(&message) { if MessageBuilder::is_drop_message(&message) {
return; return;
} }
// Add the message to the cache. If it was already seen, ignore it. // Add the message to the cache. If it was already seen, ignore it.
if self if self
.duplicate_cache .duplicate_cache
.cache_set(message_id(&message), ()) .cache_set(Self::message_id(&message), ())
.is_some() .is_some()
{ {
return; return;

View File

@ -13,7 +13,6 @@ use libp2p::{
}, },
Stream, StreamProtocol, Stream, StreamProtocol,
}; };
use nomos_mix_message::MSG_SIZE;
use crate::behaviour::Config; use crate::behaviour::Config;
@ -248,15 +247,21 @@ impl ConnectionHandler for MixConnectionHandler {
/// Write a message to the stream /// Write a message to the stream
async fn send_msg(mut stream: Stream, msg: Vec<u8>) -> io::Result<Stream> { async fn send_msg(mut stream: Stream, msg: Vec<u8>) -> io::Result<Stream> {
let msg_len = (msg.len() as u64).to_be_bytes();
stream.write_all(&msg_len).await?;
stream.write_all(&msg).await?; stream.write_all(&msg).await?;
stream.flush().await?; stream.flush().await?;
Ok(stream) Ok(stream)
} }
/// Read a fixed-length message from the stream /// Read a message from the stream
// TODO: Consider handling variable-length messages
async fn recv_msg(mut stream: Stream) -> io::Result<(Stream, Vec<u8>)> { async fn recv_msg(mut stream: Stream) -> io::Result<(Stream, Vec<u8>)> {
let mut buf = vec![0; MSG_SIZE]; let msg_len = {
let mut buf = [0u8; 8];
stream.read_exact(&mut buf).await?;
u64::from_be_bytes(buf) as usize
};
let mut buf = vec![0u8; msg_len];
stream.read_exact(&mut buf).await?; stream.read_exact(&mut buf).await?;
Ok((stream, buf)) Ok((stream, buf))
} }

View File

@ -14,7 +14,7 @@ mod test {
swarm::{dummy, NetworkBehaviour, SwarmEvent}, swarm::{dummy, NetworkBehaviour, SwarmEvent},
Multiaddr, PeerId, Swarm, SwarmBuilder, Multiaddr, PeerId, Swarm, SwarmBuilder,
}; };
use nomos_mix_message::MSG_SIZE; use nomos_mix_message::MessageFlag;
use tokio::select; use tokio::select;
use crate::{behaviour::Config, error::Error, Behaviour, Event}; use crate::{behaviour::Config, error::Error, Behaviour, Event};
@ -43,7 +43,7 @@ mod test {
// Swamr2 publishes a message. // Swamr2 publishes a message.
let task = async { let task = async {
let msg = vec![1; MSG_SIZE]; let msg = vec![MessageFlag::Data as u8; 1024];
let mut msg_published = false; let mut msg_published = false;
let mut publish_try_interval = tokio::time::interval(Duration::from_secs(1)); let mut publish_try_interval = tokio::time::interval(Duration::from_secs(1));
loop { loop {
@ -98,7 +98,7 @@ mod test {
// Expect all publish attempts to fail with [`Error::NoPeers`] // Expect all publish attempts to fail with [`Error::NoPeers`]
// because swarm2 doesn't have any peers that support the mix protocol. // because swarm2 doesn't have any peers that support the mix protocol.
let msg = vec![1; MSG_SIZE]; let msg = vec![MessageFlag::Data as u8; 1024];
let mut publish_try_interval = tokio::time::interval(Duration::from_secs(1)); let mut publish_try_interval = tokio::time::interval(Duration::from_secs(1));
let mut publish_try_count = 0; let mut publish_try_count = 0;
loop { loop {

View File

@ -41,6 +41,7 @@ time = "0.3"
[dev-dependencies] [dev-dependencies]
blake2 = { version = "0.10" } blake2 = { version = "0.10" }
x25519-dalek = { version = "2", features = ["getrandom", "static_secrets"] }
[features] [features]
default = ["libp2p"] default = ["libp2p"]

View File

@ -1,6 +1,7 @@
use cryptarchia_consensus::LeaderConfig; use cryptarchia_consensus::LeaderConfig;
// std // std
use nomos_da_network_service::backends::libp2p::common::DaNetworkBackendSettings; use nomos_da_network_service::backends::libp2p::common::DaNetworkBackendSettings;
use nomos_mix::membership::Node;
use nomos_mix::message_blend::{ use nomos_mix::message_blend::{
CryptographicProcessorSettings, MessageBlendSettings, TemporalProcessorSettings, CryptographicProcessorSettings, MessageBlendSettings, TemporalProcessorSettings,
}; };
@ -193,7 +194,9 @@ pub fn new_node(
genesis_state: &LedgerState, genesis_state: &LedgerState,
time_config: &TimeConfig, time_config: &TimeConfig,
swarm_config: &SwarmConfig, swarm_config: &SwarmConfig,
mix_config: &Libp2pMixBackendSettings, mix_backend_settings: &Libp2pMixBackendSettings,
mix_private_key: &x25519_dalek::StaticSecret,
mix_membership: Vec<Node>,
db_path: PathBuf, db_path: PathBuf,
blobs_dir: &PathBuf, blobs_dir: &PathBuf,
initial_peers: Vec<Multiaddr>, initial_peers: Vec<Multiaddr>,
@ -210,14 +213,20 @@ pub fn new_node(
}, },
}, },
mix: MixConfig { mix: MixConfig {
backend: mix_config.clone(), backend: mix_backend_settings.clone(),
persistent_transmission: Default::default(), persistent_transmission: Default::default(),
message_blend: MessageBlendSettings { message_blend: MessageBlendSettings {
cryptographic_processor: CryptographicProcessorSettings { num_mix_layers: 1 }, cryptographic_processor: CryptographicProcessorSettings {
num_mix_layers: 1,
max_num_mix_layers: 1,
max_payload_size: 2048,
private_key: mix_private_key.to_bytes(),
},
temporal_processor: TemporalProcessorSettings { temporal_processor: TemporalProcessorSettings {
max_delay_seconds: 2, max_delay_seconds: 2,
}, },
}, },
membership: mix_membership,
}, },
da_network: DaNetworkConfig { da_network: DaNetworkConfig {
backend: DaNetworkBackendSettings { backend: DaNetworkBackendSettings {
@ -308,35 +317,35 @@ pub fn new_node(
.unwrap() .unwrap()
} }
pub fn new_mix_configs(listening_addresses: Vec<Multiaddr>) -> Vec<Libp2pMixBackendSettings> { pub fn new_mix_configs(
let mut configs = listening_addresses listening_addresses: Vec<Multiaddr>,
) -> (
Vec<(Libp2pMixBackendSettings, x25519_dalek::StaticSecret)>,
Vec<Node>,
) {
let node_settings = listening_addresses
.iter() .iter()
.map(|listening_address| Libp2pMixBackendSettings { .map(|listening_address| {
(
Libp2pMixBackendSettings {
listening_address: listening_address.clone(), listening_address: listening_address.clone(),
node_key: ed25519::SecretKey::generate(), node_key: ed25519::SecretKey::generate(),
membership: Vec::new(),
peering_degree: 1, peering_degree: 1,
},
x25519_dalek::StaticSecret::random(),
)
}) })
.collect::<Vec<_>>(); .collect::<Vec<_>>();
let membership = configs let membership = node_settings
.iter() .iter()
.map(|c| { .map(|(backend_setting, private_key)| Node {
let peer_id = PeerId::from_public_key( address: backend_setting.listening_address.clone(),
&ed25519::Keypair::from(c.node_key.clone()).public().into(), public_key: x25519_dalek::PublicKey::from(private_key).to_bytes(),
);
c.listening_address
.clone()
.with_p2p(peer_id)
.unwrap_or_else(|orig_addr| orig_addr)
}) })
.collect::<Vec<_>>(); .collect::<Vec<_>>();
configs (node_settings, membership)
.iter_mut()
.for_each(|c| c.membership = membership.clone());
configs
} }
// Client node is only created for asyncroniously interact with nodes in the test. // Client node is only created for asyncroniously interact with nodes in the test.

View File

@ -91,7 +91,7 @@ fn test_indexer() {
port: 7772, port: 7772,
..Default::default() ..Default::default()
}; };
let mix_configs = new_mix_configs(vec![ let (mix_configs, mix_membership) = new_mix_configs(vec![
Multiaddr::from_str("/ip4/127.0.0.1/udp/7781/quic-v1").unwrap(), Multiaddr::from_str("/ip4/127.0.0.1/udp/7781/quic-v1").unwrap(),
Multiaddr::from_str("/ip4/127.0.0.1/udp/7782/quic-v1").unwrap(), Multiaddr::from_str("/ip4/127.0.0.1/udp/7782/quic-v1").unwrap(),
]); ]);
@ -122,7 +122,9 @@ fn test_indexer() {
&genesis_state, &genesis_state,
&time_config, &time_config,
&swarm_config1, &swarm_config1,
&mix_configs[0], &mix_configs[0].0,
&mix_configs[0].1,
mix_membership.clone(),
NamedTempFile::new().unwrap().path().to_path_buf(), NamedTempFile::new().unwrap().path().to_path_buf(),
&blobs_dir, &blobs_dir,
vec![node_address(&swarm_config2)], vec![node_address(&swarm_config2)],
@ -150,7 +152,9 @@ fn test_indexer() {
&genesis_state, &genesis_state,
&time_config, &time_config,
&swarm_config2, &swarm_config2,
&mix_configs[1], &mix_configs[1].0,
&mix_configs[1].1,
mix_membership.clone(),
NamedTempFile::new().unwrap().path().to_path_buf(), NamedTempFile::new().unwrap().path().to_path_buf(),
&blobs_dir, &blobs_dir,
vec![node_address(&swarm_config1)], vec![node_address(&swarm_config1)],

View File

@ -70,7 +70,7 @@ fn test_verifier() {
port: 7774, port: 7774,
..Default::default() ..Default::default()
}; };
let mix_configs = new_mix_configs(vec![ let (mix_configs, mix_membership) = new_mix_configs(vec![
Multiaddr::from_str("/ip4/127.0.0.1/udp/7783/quic-v1").unwrap(), Multiaddr::from_str("/ip4/127.0.0.1/udp/7783/quic-v1").unwrap(),
Multiaddr::from_str("/ip4/127.0.0.1/udp/7784/quic-v1").unwrap(), Multiaddr::from_str("/ip4/127.0.0.1/udp/7784/quic-v1").unwrap(),
]); ]);
@ -103,7 +103,9 @@ fn test_verifier() {
&genesis_state, &genesis_state,
&time_config, &time_config,
&swarm_config1, &swarm_config1,
&mix_configs[0], &mix_configs[0].0,
&mix_configs[0].1,
mix_membership.clone(),
NamedTempFile::new().unwrap().path().to_path_buf(), NamedTempFile::new().unwrap().path().to_path_buf(),
&blobs_dir, &blobs_dir,
vec![node_address(&swarm_config2)], vec![node_address(&swarm_config2)],
@ -131,7 +133,9 @@ fn test_verifier() {
&genesis_state, &genesis_state,
&time_config, &time_config,
&swarm_config2, &swarm_config2,
&mix_configs[1], &mix_configs[1].0,
&mix_configs[1].1,
mix_membership.clone(),
NamedTempFile::new().unwrap().path().to_path_buf(), NamedTempFile::new().unwrap().path().to_path_buf(),
&blobs_dir, &blobs_dir,
vec![node_address(&swarm_config1)], vec![node_address(&swarm_config1)],

View File

@ -19,6 +19,7 @@ serde = { version = "1.0", features = ["derive"] }
tokio = { version = "1", features = ["macros", "sync"] } tokio = { version = "1", features = ["macros", "sync"] }
tokio-stream = "0.1" tokio-stream = "0.1"
tracing = "0.1" tracing = "0.1"
x25519-dalek = { version = "2", features = ["static_secrets"] }
[features] [features]
default = [] default = []

View File

@ -6,11 +6,11 @@ use libp2p::{
core::transport::ListenerId, core::transport::ListenerId,
identity::{ed25519, Keypair}, identity::{ed25519, Keypair},
swarm::SwarmEvent, swarm::SwarmEvent,
Multiaddr, PeerId, Swarm, SwarmBuilder, TransportError, Multiaddr, Swarm, SwarmBuilder, TransportError,
}; };
use nomos_libp2p::{secret_key_serde, DialError, DialOpts, Protocol}; use nomos_libp2p::{secret_key_serde, DialError, DialOpts};
use nomos_mix::membership::Membership;
use overwatch_rs::overwatch::handle::OverwatchHandle; use overwatch_rs::overwatch::handle::OverwatchHandle;
use rand::seq::IteratorRandom;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use tokio::{ use tokio::{
sync::{broadcast, mpsc}, sync::{broadcast, mpsc},
@ -34,7 +34,6 @@ pub struct Libp2pMixBackendSettings {
// A key for deriving PeerId and establishing secure connections (TLS 1.3 by QUIC) // A key for deriving PeerId and establishing secure connections (TLS 1.3 by QUIC)
#[serde(with = "secret_key_serde", default = "ed25519::SecretKey::generate")] #[serde(with = "secret_key_serde", default = "ed25519::SecretKey::generate")]
pub node_key: ed25519::SecretKey, pub node_key: ed25519::SecretKey,
pub membership: Vec<Multiaddr>,
pub peering_degree: usize, pub peering_degree: usize,
} }
@ -44,12 +43,15 @@ const CHANNEL_SIZE: usize = 64;
impl MixBackend for Libp2pMixBackend { impl MixBackend for Libp2pMixBackend {
type Settings = Libp2pMixBackendSettings; type Settings = Libp2pMixBackendSettings;
fn new(config: Self::Settings, overwatch_handle: OverwatchHandle) -> Self { fn new(
config: Self::Settings,
overwatch_handle: OverwatchHandle,
membership: Membership,
) -> Self {
let (swarm_message_sender, swarm_message_receiver) = mpsc::channel(CHANNEL_SIZE); let (swarm_message_sender, swarm_message_receiver) = mpsc::channel(CHANNEL_SIZE);
let (incoming_message_sender, _) = broadcast::channel(CHANNEL_SIZE); let (incoming_message_sender, _) = broadcast::channel(CHANNEL_SIZE);
let keypair = Keypair::from(ed25519::Keypair::from(config.node_key.clone())); let keypair = Keypair::from(ed25519::Keypair::from(config.node_key.clone()));
let local_peer_id = keypair.public().to_peer_id();
let mut swarm = MixSwarm::new( let mut swarm = MixSwarm::new(
keypair, keypair,
swarm_message_receiver, swarm_message_receiver,
@ -63,20 +65,12 @@ impl MixBackend for Libp2pMixBackend {
}); });
// Randomly select peering_degree number of peers, and dial to them // Randomly select peering_degree number of peers, and dial to them
// TODO: Consider moving the peer seelction to the nomos_mix_network::Behaviour membership
config .choose_nodes(&mut rand::thread_rng(), config.peering_degree)
.membership
.iter() .iter()
.filter(|addr| match extract_peer_id(addr) { .for_each(|node| {
Some(peer_id) => peer_id != local_peer_id, if let Err(e) = swarm.dial(node.address.clone()) {
None => false, tracing::error!("failed to dial to {:?}: {:?}", node.address, e);
})
.choose_multiple(&mut rand::thread_rng(), config.peering_degree)
.iter()
.cloned()
.for_each(|addr| {
if let Err(e) = swarm.dial(addr.clone()) {
tracing::error!("failed to dial to {:?}: {:?}", addr, e);
} }
}); });
@ -193,13 +187,3 @@ impl MixSwarm {
} }
} }
} }
fn extract_peer_id(multiaddr: &Multiaddr) -> Option<PeerId> {
multiaddr.iter().find_map(|protocol| {
if let Protocol::P2p(peer_id) = protocol {
Some(peer_id)
} else {
None
}
})
}

View File

@ -4,6 +4,7 @@ pub mod libp2p;
use std::{fmt::Debug, pin::Pin}; use std::{fmt::Debug, pin::Pin};
use futures::Stream; use futures::Stream;
use nomos_mix::membership::Membership;
use overwatch_rs::overwatch::handle::OverwatchHandle; use overwatch_rs::overwatch::handle::OverwatchHandle;
/// A trait for mix backends that send messages to the mix network. /// A trait for mix backends that send messages to the mix network.
@ -11,7 +12,11 @@ use overwatch_rs::overwatch::handle::OverwatchHandle;
pub trait MixBackend { pub trait MixBackend {
type Settings: Clone + Debug + Send + Sync + 'static; type Settings: Clone + Debug + Send + Sync + 'static;
fn new(config: Self::Settings, overwatch_handle: OverwatchHandle) -> Self; fn new(
config: Self::Settings,
overwatch_handle: OverwatchHandle,
membership: Membership,
) -> Self;
/// Publish a message to the mix network. /// Publish a message to the mix network.
async fn publish(&self, msg: Vec<u8>); async fn publish(&self, msg: Vec<u8>);
/// Listen to messages received from the mix network. /// Listen to messages received from the mix network.

View File

@ -9,9 +9,11 @@ use futures::StreamExt;
use network::NetworkAdapter; use network::NetworkAdapter;
use nomos_core::wire; use nomos_core::wire;
use nomos_mix::{ use nomos_mix::{
message_blend::{MessageBlend, MessageBlendSettings}, membership::{Membership, Node},
message_blend::{CryptographicProcessorSettings, MessageBlend, MessageBlendSettings},
persistent_transmission::{persistent_transmission, PersistentTransmissionSettings}, persistent_transmission::{persistent_transmission, PersistentTransmissionSettings},
}; };
use nomos_mix_message::MessageBuilder;
use nomos_network::NetworkService; use nomos_network::NetworkService;
use overwatch_rs::services::{ use overwatch_rs::services::{
handle::ServiceStateHandle, handle::ServiceStateHandle,
@ -38,6 +40,7 @@ where
backend: Backend, backend: Backend,
service_state: ServiceStateHandle<Self>, service_state: ServiceStateHandle<Self>,
network_relay: Relay<NetworkService<Network::Backend>>, network_relay: Relay<NetworkService<Network::Backend>>,
membership: Membership,
} }
impl<Backend, Network> ServiceData for MixService<Backend, Network> impl<Backend, Network> ServiceData for MixService<Backend, Network>
@ -64,14 +67,18 @@ where
Clone + Debug + Serialize + DeserializeOwned + Send + Sync + 'static, Clone + Debug + Serialize + DeserializeOwned + Send + Sync + 'static,
{ {
fn init(service_state: ServiceStateHandle<Self>) -> Result<Self, overwatch_rs::DynError> { fn init(service_state: ServiceStateHandle<Self>) -> Result<Self, overwatch_rs::DynError> {
let mix_config = service_state.settings_reader.get_updated_settings();
let membership = mix_config.membership();
let network_relay = service_state.overwatch_handle.relay(); let network_relay = service_state.overwatch_handle.relay();
Ok(Self { Ok(Self {
backend: <Backend as MixBackend>::new( backend: <Backend as MixBackend>::new(
service_state.settings_reader.get_updated_settings().backend, service_state.settings_reader.get_updated_settings().backend,
service_state.overwatch_handle.clone(), service_state.overwatch_handle.clone(),
membership.clone(),
), ),
service_state, service_state,
network_relay, network_relay,
membership,
}) })
} }
@ -80,19 +87,30 @@ where
mut service_state, mut service_state,
mut backend, mut backend,
network_relay, network_relay,
membership,
} = self; } = self;
let mix_config = service_state.settings_reader.get_updated_settings(); let mix_config = service_state.settings_reader.get_updated_settings();
let network_relay = network_relay.connect().await?; let network_relay = network_relay.connect().await?;
let network_adapter = Network::new(network_relay); let network_adapter = Network::new(network_relay);
// Create message builder used in all tiers.
let CryptographicProcessorSettings {
max_num_mix_layers,
max_payload_size,
..
} = mix_config.message_blend.cryptographic_processor;
let message_builder = MessageBuilder::new(max_num_mix_layers, max_payload_size).unwrap();
// Spawn Persistent Transmission // Spawn Persistent Transmission
let (transmission_schedule_sender, transmission_schedule_receiver) = let (transmission_schedule_sender, transmission_schedule_receiver) =
mpsc::unbounded_channel(); mpsc::unbounded_channel();
let (emission_sender, mut emission_receiver) = mpsc::unbounded_channel(); let (emission_sender, mut emission_receiver) = mpsc::unbounded_channel();
let drop_message = message_builder.drop_message();
tokio::spawn(async move { tokio::spawn(async move {
persistent_transmission( persistent_transmission(
mix_config.persistent_transmission, mix_config.persistent_transmission,
drop_message,
transmission_schedule_receiver, transmission_schedule_receiver,
emission_sender, emission_sender,
) )
@ -107,6 +125,8 @@ where
tokio::spawn(async move { tokio::spawn(async move {
MessageBlend::new( MessageBlend::new(
mix_config.message_blend, mix_config.message_blend,
message_builder,
membership.clone(),
new_message_receiver, new_message_receiver,
processor_inbound_receiver, processor_inbound_receiver,
// Connect the outputs of Message Blend to Persistent Transmission // Connect the outputs of Message Blend to Persistent Transmission
@ -200,6 +220,24 @@ pub struct MixConfig<BackendSettings> {
pub backend: BackendSettings, pub backend: BackendSettings,
pub persistent_transmission: PersistentTransmissionSettings, pub persistent_transmission: PersistentTransmissionSettings,
pub message_blend: MessageBlendSettings, pub message_blend: MessageBlendSettings,
pub membership: Vec<Node>,
}
impl<BackendSettings> MixConfig<BackendSettings> {
// TODO: This step can be redesigned once we can load membership info from the chain state.
fn membership(&self) -> Membership {
let local_public_key = x25519_dalek::PublicKey::from(&x25519_dalek::StaticSecret::from(
self.message_blend.cryptographic_processor.private_key,
))
.to_bytes();
let remote_nodes = self
.membership
.iter()
.filter(|node| node.public_key != local_public_key)
.cloned()
.collect();
Membership::new(remote_nodes)
}
} }
/// A message that is handled by [`MixService`]. /// A message that is handled by [`MixService`].

View File

@ -9,6 +9,7 @@ clap = { version = "4", features = ["derive"] }
nomos-executor = { path = "../../nodes/nomos-executor" } nomos-executor = { path = "../../nodes/nomos-executor" }
nomos-libp2p = { path = "../../nomos-libp2p" } nomos-libp2p = { path = "../../nomos-libp2p" }
nomos-node = { path = "../../nodes/nomos-node" } nomos-node = { path = "../../nodes/nomos-node" }
nomos-mix = { path = "../../nomos-mix/core" }
nomos-tracing = { path = "../../nomos-tracing" } nomos-tracing = { path = "../../nomos-tracing" }
nomos-tracing-service = { path = "../../nomos-services/tracing" } nomos-tracing-service = { path = "../../nomos-services/tracing" }
rand = "0.8" rand = "0.8"

View File

@ -1,7 +1,8 @@
// std // std
use std::{collections::HashMap, net::Ipv4Addr, str::FromStr}; use std::{collections::HashMap, net::Ipv4Addr, str::FromStr};
// crates // crates
use nomos_libp2p::{Multiaddr, PeerId, Protocol}; use nomos_libp2p::{Multiaddr, PeerId};
use nomos_mix::membership::Node;
use nomos_tracing::{logging::loki::LokiConfig, tracing::otlp::OtlpTracingConfig}; use nomos_tracing::{logging::loki::LokiConfig, tracing::otlp::OtlpTracingConfig};
use nomos_tracing_service::{FilterLayer, LoggerLayer, TracingSettings}; use nomos_tracing_service::{FilterLayer, LoggerLayer, TracingSettings};
use rand::{thread_rng, Rng}; use rand::{thread_rng, Rng};
@ -91,7 +92,7 @@ pub fn create_node_configs(
let host_network_init_peers = update_network_init_peers(hosts.clone()); let host_network_init_peers = update_network_init_peers(hosts.clone());
let host_da_peer_addresses = update_da_peer_addresses(hosts.clone(), peer_addresses); let host_da_peer_addresses = update_da_peer_addresses(hosts.clone(), peer_addresses);
let host_mix_membership = let host_mix_membership =
update_mix_membership(hosts.clone(), mix_configs[0].backend.membership.clone()); update_mix_membership(hosts.clone(), mix_configs[0].membership.clone());
let new_peer_addresses: HashMap<PeerId, Multiaddr> = host_da_peer_addresses let new_peer_addresses: HashMap<PeerId, Multiaddr> = host_da_peer_addresses
.clone() .clone()
@ -122,7 +123,7 @@ pub fn create_node_configs(
let mut mix_config = mix_configs[i].to_owned(); let mut mix_config = mix_configs[i].to_owned();
mix_config.backend.listening_address = mix_config.backend.listening_address =
Multiaddr::from_str(&format!("/ip4/0.0.0.0/udp/{}/quic-v1", host.mix_port)).unwrap(); Multiaddr::from_str(&format!("/ip4/0.0.0.0/udp/{}/quic-v1", host.mix_port)).unwrap();
mix_config.backend.membership = host_mix_membership.clone(); mix_config.membership = host_mix_membership.clone();
// Tracing config. // Tracing config.
let tracing_config = let tracing_config =
@ -170,32 +171,19 @@ fn update_da_peer_addresses(
.collect() .collect()
} }
fn update_mix_membership(hosts: Vec<Host>, membership: Vec<Multiaddr>) -> Vec<Multiaddr> { fn update_mix_membership(hosts: Vec<Host>, membership: Vec<Node>) -> Vec<Node> {
membership membership
.into_iter() .into_iter()
.zip(hosts) .zip(hosts)
.map(|(addr, host)| { .map(|(mut node, host)| {
Multiaddr::from_str(&format!( node.address =
"/ip4/{}/udp/{}/quic-v1/p2p/{}", Multiaddr::from_str(&format!("/ip4/{}/udp/{}/quic-v1", host.ip, host.mix_port))
host.ip, .unwrap();
host.mix_port, node
extract_peer_id(&addr).unwrap(),
))
.unwrap()
}) })
.collect() .collect()
} }
fn extract_peer_id(multiaddr: &Multiaddr) -> Option<PeerId> {
multiaddr.iter().find_map(|protocol| {
if let Protocol::P2p(peer_id) = protocol {
Some(peer_id)
} else {
None
}
})
}
fn tracing_config_for_grafana(params: TracingParams, identifier: String) -> GeneralTracingConfig { fn tracing_config_for_grafana(params: TracingParams, identifier: String) -> GeneralTracingConfig {
GeneralTracingConfig { GeneralTracingConfig {
tracing_settings: TracingSettings { tracing_settings: TracingSettings {

View File

@ -50,6 +50,7 @@ criterion = { version = "0.5", features = ["async_tokio"] }
nomos-cli = { path = "../nomos-cli" } nomos-cli = { path = "../nomos-cli" }
time = "0.3" time = "0.3"
tracing = "0.1" tracing = "0.1"
x25519-dalek = { version = "2.0.1", features = ["getrandom", "static_secrets"] }
[[test]] [[test]]
name = "test_cryptarchia_happy_path" name = "test_cryptarchia_happy_path"

View File

@ -159,11 +159,17 @@ pub fn create_executor_config(config: GeneralConfig) -> Config {
backend: config.mix_config.backend, backend: config.mix_config.backend,
persistent_transmission: Default::default(), persistent_transmission: Default::default(),
message_blend: MessageBlendSettings { message_blend: MessageBlendSettings {
cryptographic_processor: CryptographicProcessorSettings { num_mix_layers: 1 }, cryptographic_processor: CryptographicProcessorSettings {
num_mix_layers: 1,
max_num_mix_layers: 1,
max_payload_size: 2048,
private_key: config.mix_config.private_key.to_bytes(),
},
temporal_processor: TemporalProcessorSettings { temporal_processor: TemporalProcessorSettings {
max_delay_seconds: 2, max_delay_seconds: 2,
}, },
}, },
membership: config.mix_config.membership,
}, },
cryptarchia: CryptarchiaSettings { cryptarchia: CryptarchiaSettings {
leader_config: config.consensus_config.leader_config, leader_config: config.consensus_config.leader_config,

View File

@ -244,11 +244,17 @@ pub fn create_validator_config(config: GeneralConfig) -> Config {
backend: config.mix_config.backend, backend: config.mix_config.backend,
persistent_transmission: Default::default(), persistent_transmission: Default::default(),
message_blend: MessageBlendSettings { message_blend: MessageBlendSettings {
cryptographic_processor: CryptographicProcessorSettings { num_mix_layers: 1 }, cryptographic_processor: CryptographicProcessorSettings {
num_mix_layers: 1,
max_num_mix_layers: 1,
max_payload_size: 2048,
private_key: config.mix_config.private_key.to_bytes(),
},
temporal_processor: TemporalProcessorSettings { temporal_processor: TemporalProcessorSettings {
max_delay_seconds: 2, max_delay_seconds: 2,
}, },
}, },
membership: config.mix_config.membership,
}, },
cryptarchia: CryptarchiaSettings { cryptarchia: CryptarchiaSettings {
leader_config: config.consensus_config.leader_config, leader_config: config.consensus_config.leader_config,

View File

@ -1,17 +1,20 @@
use std::str::FromStr; use std::str::FromStr;
use nomos_libp2p::{ed25519, Multiaddr}; use nomos_libp2p::{ed25519, Multiaddr};
use nomos_mix::membership::Node;
use nomos_mix_service::backends::libp2p::Libp2pMixBackendSettings; use nomos_mix_service::backends::libp2p::Libp2pMixBackendSettings;
use crate::{get_available_port, secret_key_to_peer_id}; use crate::get_available_port;
#[derive(Clone)] #[derive(Clone)]
pub struct GeneralMixConfig { pub struct GeneralMixConfig {
pub backend: Libp2pMixBackendSettings, pub backend: Libp2pMixBackendSettings,
pub private_key: x25519_dalek::StaticSecret,
pub membership: Vec<Node>,
} }
pub fn create_mix_configs(ids: &[[u8; 32]]) -> Vec<GeneralMixConfig> { pub fn create_mix_configs(ids: &[[u8; 32]]) -> Vec<GeneralMixConfig> {
let mut configs: Vec<GeneralMixConfig> = ids let mut configs = ids
.iter() .iter()
.map(|id| { .map(|id| {
let mut node_key_bytes = *id; let mut node_key_bytes = *id;
@ -26,33 +29,28 @@ pub fn create_mix_configs(ids: &[[u8; 32]]) -> Vec<GeneralMixConfig> {
)) ))
.unwrap(), .unwrap(),
node_key, node_key,
membership: Vec::new(),
peering_degree: 1, peering_degree: 1,
}, },
private_key: x25519_dalek::StaticSecret::random(),
membership: Vec::new(),
} }
}) })
.collect(); .collect::<Vec<_>>();
let membership = mix_membership(&configs); let membership = mix_membership(&configs);
configs
configs.iter_mut().for_each(|config| { .iter_mut()
config.backend.membership = membership.clone(); .for_each(|config| config.membership = membership.clone());
});
configs configs
} }
fn mix_membership(configs: &[GeneralMixConfig]) -> Vec<Multiaddr> { fn mix_membership(configs: &[GeneralMixConfig]) -> Vec<Node> {
configs configs
.iter() .iter()
.map(|config| { .map(|config| Node {
let peer_id = secret_key_to_peer_id(config.backend.node_key.clone()); address: config.backend.listening_address.clone(),
config public_key: x25519_dalek::PublicKey::from(&config.private_key).to_bytes(),
.backend
.listening_address
.clone()
.with_p2p(peer_id)
.unwrap_or_else(|orig_addr| orig_addr)
}) })
.collect() .collect()
} }