wip
This commit is contained in:
parent
cb708de1b0
commit
eebe305544
|
@ -12,6 +12,7 @@ where
|
||||||
settings: CryptographicProcessorSettings<M::PrivateKey>,
|
settings: CryptographicProcessorSettings<M::PrivateKey>,
|
||||||
membership: Membership<M>,
|
membership: Membership<M>,
|
||||||
rng: R,
|
rng: R,
|
||||||
|
mix_message: M,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
|
@ -28,18 +29,20 @@ where
|
||||||
{
|
{
|
||||||
pub fn new(
|
pub fn new(
|
||||||
settings: CryptographicProcessorSettings<M::PrivateKey>,
|
settings: CryptographicProcessorSettings<M::PrivateKey>,
|
||||||
|
mix_message_settings: M::Settings,
|
||||||
membership: Membership<M>,
|
membership: Membership<M>,
|
||||||
rng: R,
|
rng: R,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
|
let mix_message = M::new(mix_message_settings);
|
||||||
Self {
|
Self {
|
||||||
settings,
|
settings,
|
||||||
membership,
|
membership,
|
||||||
rng,
|
rng,
|
||||||
|
mix_message,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn wrap_message(&mut self, message: &[u8]) -> Result<Vec<u8>, M::Error> {
|
pub fn wrap_message(&mut self, message: &[u8]) -> Result<Vec<u8>, M::Error> {
|
||||||
// TODO: Use the actual Sphinx encoding instead of mock.
|
|
||||||
let public_keys = self
|
let public_keys = self
|
||||||
.membership
|
.membership
|
||||||
.choose_remote_nodes(&mut self.rng, self.settings.num_mix_layers)
|
.choose_remote_nodes(&mut self.rng, self.settings.num_mix_layers)
|
||||||
|
@ -47,10 +50,11 @@ where
|
||||||
.map(|node| node.public_key.clone())
|
.map(|node| node.public_key.clone())
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
M::build_message(message, &public_keys)
|
self.mix_message.build_message(message, &public_keys)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn unwrap_message(&self, message: &[u8]) -> Result<(Vec<u8>, bool), M::Error> {
|
pub fn unwrap_message(&self, message: &[u8]) -> Result<(Vec<u8>, bool), M::Error> {
|
||||||
M::unwrap_message(message, &self.settings.private_key)
|
self.mix_message
|
||||||
|
.unwrap_message(message, &self.settings.private_key)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,12 +59,14 @@ where
|
||||||
pub fn new(
|
pub fn new(
|
||||||
input_stream: S,
|
input_stream: S,
|
||||||
settings: MessageBlendSettings<M>,
|
settings: MessageBlendSettings<M>,
|
||||||
|
mix_message_settings: M::Settings,
|
||||||
membership: Membership<M>,
|
membership: Membership<M>,
|
||||||
scheduler: Scheduler,
|
scheduler: Scheduler,
|
||||||
cryptographic_processor_rng: Rng,
|
cryptographic_processor_rng: Rng,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let cryptographic_processor = CryptographicProcessor::new(
|
let cryptographic_processor = CryptographicProcessor::new(
|
||||||
settings.cryptographic_processor,
|
settings.cryptographic_processor,
|
||||||
|
mix_message_settings,
|
||||||
membership,
|
membership,
|
||||||
cryptographic_processor_rng,
|
cryptographic_processor_rng,
|
||||||
);
|
);
|
||||||
|
@ -132,6 +134,7 @@ where
|
||||||
fn blend(
|
fn blend(
|
||||||
self,
|
self,
|
||||||
message_blend_settings: MessageBlendSettings<M>,
|
message_blend_settings: MessageBlendSettings<M>,
|
||||||
|
mix_message_settings: M::Settings,
|
||||||
membership: Membership<M>,
|
membership: Membership<M>,
|
||||||
scheduler: Scheduler,
|
scheduler: Scheduler,
|
||||||
cryptographic_processor_rng: Rng,
|
cryptographic_processor_rng: Rng,
|
||||||
|
@ -142,6 +145,7 @@ where
|
||||||
MessageBlendStream::new(
|
MessageBlendStream::new(
|
||||||
self,
|
self,
|
||||||
message_blend_settings,
|
message_blend_settings,
|
||||||
|
mix_message_settings,
|
||||||
membership,
|
membership,
|
||||||
scheduler,
|
scheduler,
|
||||||
cryptographic_processor_rng,
|
cryptographic_processor_rng,
|
||||||
|
|
|
@ -1,9 +1,6 @@
|
||||||
use futures::{Stream, StreamExt};
|
use futures::{Stream, StreamExt};
|
||||||
use nomos_mix_message::MixMessage;
|
|
||||||
use rand::{distributions::Uniform, prelude::Distribution, Rng, RngCore};
|
use rand::{distributions::Uniform, prelude::Distribution, Rng, RngCore};
|
||||||
use serde::de::DeserializeOwned;
|
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use std::marker::PhantomData;
|
|
||||||
use std::pin::{pin, Pin};
|
use std::pin::{pin, Pin};
|
||||||
use std::task::{Context, Poll};
|
use std::task::{Context, Poll};
|
||||||
|
|
||||||
|
@ -25,7 +22,7 @@ impl Default for PersistentTransmissionSettings {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Transmit scheduled messages with a persistent rate as a stream.
|
/// Transmit scheduled messages with a persistent rate as a stream.
|
||||||
pub struct PersistentTransmissionStream<S, Rng, M, Scheduler>
|
pub struct PersistentTransmissionStream<S, Rng, Scheduler>
|
||||||
where
|
where
|
||||||
S: Stream,
|
S: Stream,
|
||||||
Rng: RngCore,
|
Rng: RngCore,
|
||||||
|
@ -33,14 +30,13 @@ where
|
||||||
coin: Coin<Rng>,
|
coin: Coin<Rng>,
|
||||||
stream: S,
|
stream: S,
|
||||||
scheduler: Scheduler,
|
scheduler: Scheduler,
|
||||||
_mix_message: PhantomData<M>,
|
drop_message: Vec<u8>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, Rng, M, Scheduler> PersistentTransmissionStream<S, Rng, M, Scheduler>
|
impl<S, Rng, Scheduler> PersistentTransmissionStream<S, Rng, Scheduler>
|
||||||
where
|
where
|
||||||
S: Stream,
|
S: Stream,
|
||||||
Rng: RngCore,
|
Rng: RngCore,
|
||||||
M: MixMessage,
|
|
||||||
Scheduler: Stream<Item = ()>,
|
Scheduler: Stream<Item = ()>,
|
||||||
{
|
{
|
||||||
pub fn new(
|
pub fn new(
|
||||||
|
@ -48,22 +44,22 @@ where
|
||||||
stream: S,
|
stream: S,
|
||||||
scheduler: Scheduler,
|
scheduler: Scheduler,
|
||||||
rng: Rng,
|
rng: Rng,
|
||||||
) -> PersistentTransmissionStream<S, Rng, M, Scheduler> {
|
drop_message: Vec<u8>,
|
||||||
|
) -> PersistentTransmissionStream<S, Rng, Scheduler> {
|
||||||
let coin = Coin::<Rng>::new(rng, settings.drop_message_probability).unwrap();
|
let coin = Coin::<Rng>::new(rng, settings.drop_message_probability).unwrap();
|
||||||
Self {
|
Self {
|
||||||
coin,
|
coin,
|
||||||
stream,
|
stream,
|
||||||
scheduler,
|
scheduler,
|
||||||
_mix_message: Default::default(),
|
drop_message,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, Rng, M, Scheduler> Stream for PersistentTransmissionStream<S, Rng, M, Scheduler>
|
impl<S, Rng, Scheduler> Stream for PersistentTransmissionStream<S, Rng, Scheduler>
|
||||||
where
|
where
|
||||||
S: Stream<Item = Vec<u8>> + Unpin,
|
S: Stream<Item = Vec<u8>> + Unpin,
|
||||||
Rng: RngCore + Unpin,
|
Rng: RngCore + Unpin,
|
||||||
M: MixMessage + Unpin,
|
|
||||||
Scheduler: Stream<Item = ()> + Unpin,
|
Scheduler: Stream<Item = ()> + Unpin,
|
||||||
{
|
{
|
||||||
type Item = Vec<u8>;
|
type Item = Vec<u8>;
|
||||||
|
@ -73,6 +69,7 @@ where
|
||||||
ref mut scheduler,
|
ref mut scheduler,
|
||||||
ref mut stream,
|
ref mut stream,
|
||||||
ref mut coin,
|
ref mut coin,
|
||||||
|
ref drop_message,
|
||||||
..
|
..
|
||||||
} = self.get_mut();
|
} = self.get_mut();
|
||||||
if pin!(scheduler).poll_next_unpin(cx).is_pending() {
|
if pin!(scheduler).poll_next_unpin(cx).is_pending() {
|
||||||
|
@ -81,17 +78,16 @@ 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(M::DROP_MESSAGE.to_vec()))
|
Poll::Ready(Some(drop_message.clone()))
|
||||||
} else {
|
} else {
|
||||||
Poll::Pending
|
Poll::Pending
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait PersistentTransmissionExt<Rng, M, Scheduler>: Stream
|
pub trait PersistentTransmissionExt<Rng, Scheduler>: Stream
|
||||||
where
|
where
|
||||||
Rng: RngCore,
|
Rng: RngCore,
|
||||||
M: MixMessage,
|
|
||||||
Scheduler: Stream<Item = ()>,
|
Scheduler: Stream<Item = ()>,
|
||||||
{
|
{
|
||||||
fn persistent_transmission(
|
fn persistent_transmission(
|
||||||
|
@ -99,20 +95,19 @@ where
|
||||||
settings: PersistentTransmissionSettings,
|
settings: PersistentTransmissionSettings,
|
||||||
rng: Rng,
|
rng: Rng,
|
||||||
scheduler: Scheduler,
|
scheduler: Scheduler,
|
||||||
) -> PersistentTransmissionStream<Self, Rng, M, Scheduler>
|
drop_message: Vec<u8>,
|
||||||
|
) -> PersistentTransmissionStream<Self, Rng, Scheduler>
|
||||||
where
|
where
|
||||||
Self: Sized + Unpin,
|
Self: Sized + Unpin,
|
||||||
{
|
{
|
||||||
PersistentTransmissionStream::new(settings, self, scheduler, rng)
|
PersistentTransmissionStream::new(settings, self, scheduler, rng, drop_message)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S, Rng, M, Scheduler> PersistentTransmissionExt<Rng, M, Scheduler> for S
|
impl<S, Rng, Scheduler> PersistentTransmissionExt<Rng, Scheduler> for S
|
||||||
where
|
where
|
||||||
S: Stream,
|
S: Stream,
|
||||||
Rng: RngCore,
|
Rng: RngCore,
|
||||||
M: MixMessage,
|
|
||||||
M::PublicKey: Clone + Serialize + DeserializeOwned,
|
|
||||||
Scheduler: Stream<Item = ()>,
|
Scheduler: Stream<Item = ()>,
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -150,7 +145,6 @@ enum CoinError {
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use nomos_mix_message::mock::MockMixMessage;
|
|
||||||
use rand::SeedableRng;
|
use rand::SeedableRng;
|
||||||
use rand_chacha::ChaCha8Rng;
|
use rand_chacha::ChaCha8Rng;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
@ -196,15 +190,13 @@ 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: PersistentTransmissionStream<
|
let drop_message = vec![0u8; 10];
|
||||||
_,
|
let mut persistent_transmission_stream: PersistentTransmissionStream<_, _, _> = stream
|
||||||
_,
|
.persistent_transmission(
|
||||||
MockMixMessage,
|
|
||||||
_,
|
|
||||||
> = stream.persistent_transmission(
|
|
||||||
settings,
|
settings,
|
||||||
ChaCha8Rng::from_entropy(),
|
ChaCha8Rng::from_entropy(),
|
||||||
IntervalStream::new(time::interval(expected_emission_interval)).map(|_| ()),
|
IntervalStream::new(time::interval(expected_emission_interval)).map(|_| ()),
|
||||||
|
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();
|
||||||
|
@ -230,14 +222,16 @@ mod tests {
|
||||||
);
|
);
|
||||||
assert_interval!(&mut last_time, lower_bound, upper_bound);
|
assert_interval!(&mut last_time, lower_bound, upper_bound);
|
||||||
|
|
||||||
assert!(MockMixMessage::is_drop_message(
|
assert_eq!(
|
||||||
&persistent_transmission_stream.next().await.unwrap()
|
&persistent_transmission_stream.next().await.unwrap(),
|
||||||
));
|
&drop_message
|
||||||
|
);
|
||||||
assert_interval!(&mut last_time, lower_bound, upper_bound);
|
assert_interval!(&mut last_time, lower_bound, upper_bound);
|
||||||
|
|
||||||
assert!(MockMixMessage::is_drop_message(
|
assert_eq!(
|
||||||
&persistent_transmission_stream.next().await.unwrap()
|
&persistent_transmission_stream.next().await.unwrap(),
|
||||||
));
|
&drop_message
|
||||||
|
);
|
||||||
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
|
||||||
|
|
|
@ -6,6 +6,7 @@ edition = "2021"
|
||||||
[dependencies]
|
[dependencies]
|
||||||
itertools = "0.13"
|
itertools = "0.13"
|
||||||
rand_chacha = "0.3"
|
rand_chacha = "0.3"
|
||||||
|
serde = { version = "1.0.215", features = ["derive"] }
|
||||||
sha2 = "0.10"
|
sha2 = "0.10"
|
||||||
sphinx-packet = "0.2"
|
sphinx-packet = "0.2"
|
||||||
thiserror = "1.0.65"
|
thiserror = "1.0.65"
|
||||||
|
|
|
@ -4,10 +4,13 @@ pub mod sphinx;
|
||||||
pub trait MixMessage {
|
pub trait MixMessage {
|
||||||
type PublicKey;
|
type PublicKey;
|
||||||
type PrivateKey;
|
type PrivateKey;
|
||||||
|
type Settings;
|
||||||
type Error;
|
type Error;
|
||||||
const DROP_MESSAGE: &'static [u8];
|
|
||||||
|
fn new(settings: Self::Settings) -> Self;
|
||||||
|
|
||||||
fn build_message(
|
fn build_message(
|
||||||
|
&self,
|
||||||
payload: &[u8],
|
payload: &[u8],
|
||||||
public_keys: &[Self::PublicKey],
|
public_keys: &[Self::PublicKey],
|
||||||
) -> Result<Vec<u8>, Self::Error>;
|
) -> Result<Vec<u8>, Self::Error>;
|
||||||
|
@ -19,10 +22,14 @@ pub trait MixMessage {
|
||||||
/// If the input message was already fully unwrapped, or if its format is invalid,
|
/// If the input message was already fully unwrapped, or if its format is invalid,
|
||||||
/// this function returns `[Error::InvalidMixMessage]`.
|
/// this function returns `[Error::InvalidMixMessage]`.
|
||||||
fn unwrap_message(
|
fn unwrap_message(
|
||||||
|
&self,
|
||||||
message: &[u8],
|
message: &[u8],
|
||||||
private_key: &Self::PrivateKey,
|
private_key: &Self::PrivateKey,
|
||||||
) -> Result<(Vec<u8>, bool), Self::Error>;
|
) -> Result<(Vec<u8>, bool), Self::Error>;
|
||||||
fn is_drop_message(message: &[u8]) -> bool {
|
|
||||||
message == Self::DROP_MESSAGE
|
fn drop_message(&self) -> &[u8];
|
||||||
|
|
||||||
|
fn is_drop_message(&self, message: &[u8]) -> bool {
|
||||||
|
message == self.drop_message()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,68 +1,99 @@
|
||||||
pub mod error;
|
pub mod error;
|
||||||
|
|
||||||
use error::Error;
|
use error::Error;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::MixMessage;
|
use crate::MixMessage;
|
||||||
// TODO: Remove all the mock below once the actual implementation is integrated to the system.
|
|
||||||
//
|
|
||||||
/// A mock implementation of the Sphinx encoding.
|
/// A mock implementation of the Sphinx encoding.
|
||||||
|
|
||||||
const NODE_ID_SIZE: usize = 32;
|
const NODE_ID_SIZE: usize = 32;
|
||||||
|
|
||||||
// TODO: Move MAX_PAYLOAD_SIZE and MAX_LAYERS to the upper layer (service layer).
|
|
||||||
const MAX_PAYLOAD_SIZE: usize = 2048;
|
|
||||||
const PAYLOAD_PADDING_SEPARATOR: u8 = 0x01;
|
const PAYLOAD_PADDING_SEPARATOR: u8 = 0x01;
|
||||||
const PAYLOAD_PADDING_SEPARATOR_SIZE: usize = 1;
|
const PAYLOAD_PADDING_SEPARATOR_SIZE: usize = std::mem::size_of::<u8>();
|
||||||
const MAX_LAYERS: usize = 5;
|
|
||||||
pub const MESSAGE_SIZE: usize =
|
|
||||||
NODE_ID_SIZE * MAX_LAYERS + MAX_PAYLOAD_SIZE + PAYLOAD_PADDING_SEPARATOR_SIZE;
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct MockMixMessage;
|
pub struct MockMixMessage {
|
||||||
|
settings: MockMixMessageSettings,
|
||||||
|
drop_message: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
|
pub struct MockMixMessageSettings {
|
||||||
|
max_layers: usize,
|
||||||
|
max_payload_size: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MockMixMessageSettings {
|
||||||
|
const fn message_size(&self) -> usize {
|
||||||
|
NODE_ID_SIZE * self.max_layers + self.max_payload_size + PAYLOAD_PADDING_SEPARATOR_SIZE
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MockMixMessage {
|
||||||
|
const fn size(&self) -> usize {
|
||||||
|
self.settings.message_size()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl MixMessage for MockMixMessage {
|
impl MixMessage for MockMixMessage {
|
||||||
type PublicKey = [u8; NODE_ID_SIZE];
|
type PublicKey = [u8; NODE_ID_SIZE];
|
||||||
type PrivateKey = [u8; NODE_ID_SIZE];
|
type PrivateKey = [u8; NODE_ID_SIZE];
|
||||||
|
type Settings = MockMixMessageSettings;
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
const DROP_MESSAGE: &'static [u8] = &[0; MESSAGE_SIZE];
|
|
||||||
|
fn new(settings: Self::Settings) -> Self {
|
||||||
|
let drop_message = vec![0; settings.message_size()];
|
||||||
|
Self {
|
||||||
|
settings,
|
||||||
|
drop_message,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// The length of the encoded message is fixed to [`MESSAGE_SIZE`] bytes.
|
/// The length of the encoded message is fixed to [`MESSAGE_SIZE`] bytes.
|
||||||
/// The [`MAX_LAYERS`] number of [`NodeId`]s are concatenated in front of the payload.
|
/// The [`MAX_LAYERS`] number of [`NodeId`]s are concatenated in front of the payload.
|
||||||
/// The payload is zero-padded to the end.
|
/// The payload is zero-padded to the end.
|
||||||
///
|
|
||||||
fn build_message(
|
fn build_message(
|
||||||
|
&self,
|
||||||
payload: &[u8],
|
payload: &[u8],
|
||||||
public_keys: &[Self::PublicKey],
|
public_keys: &[Self::PublicKey],
|
||||||
) -> Result<Vec<u8>, Self::Error> {
|
) -> Result<Vec<u8>, Self::Error> {
|
||||||
|
let MockMixMessageSettings {
|
||||||
|
max_layers,
|
||||||
|
max_payload_size,
|
||||||
|
} = self.settings;
|
||||||
|
|
||||||
// In this mock, we don't encrypt anything. So, we use public key as just a node ID.
|
// In this mock, we don't encrypt anything. So, we use public key as just a node ID.
|
||||||
let node_ids = public_keys;
|
let node_ids = public_keys;
|
||||||
if node_ids.is_empty() || node_ids.len() > MAX_LAYERS {
|
if node_ids.is_empty() || node_ids.len() > max_layers {
|
||||||
return Err(Error::InvalidNumberOfLayers);
|
return Err(Error::InvalidNumberOfLayers);
|
||||||
}
|
}
|
||||||
if payload.len() > MAX_PAYLOAD_SIZE {
|
if payload.len() > max_payload_size {
|
||||||
return Err(Error::PayloadTooLarge);
|
return Err(Error::PayloadTooLarge);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut message: Vec<u8> = Vec::with_capacity(MESSAGE_SIZE);
|
let mut message: Vec<u8> = Vec::with_capacity(self.size());
|
||||||
|
|
||||||
node_ids.iter().for_each(|node_id| {
|
node_ids.iter().for_each(|node_id| {
|
||||||
message.extend(node_id);
|
message.extend(node_id);
|
||||||
});
|
});
|
||||||
// If there is any remaining layers, fill them with zeros.
|
// If there is any remaining layers, fill them with zeros.
|
||||||
(0..MAX_LAYERS - node_ids.len()).for_each(|_| message.extend(&[0; NODE_ID_SIZE]));
|
(0..max_layers - node_ids.len()).for_each(|_| message.extend(&[0; NODE_ID_SIZE]));
|
||||||
|
|
||||||
// Append payload with padding
|
// Append payload with padding
|
||||||
message.extend(payload);
|
message.extend(payload);
|
||||||
message.push(PAYLOAD_PADDING_SEPARATOR);
|
message.push(PAYLOAD_PADDING_SEPARATOR);
|
||||||
message.extend(std::iter::repeat(0).take(MAX_PAYLOAD_SIZE - payload.len()));
|
message.extend(std::iter::repeat(0).take(max_payload_size - payload.len()));
|
||||||
Ok(message)
|
Ok(message)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn unwrap_message(
|
fn unwrap_message(
|
||||||
|
&self,
|
||||||
message: &[u8],
|
message: &[u8],
|
||||||
private_key: &Self::PrivateKey,
|
private_key: &Self::PrivateKey,
|
||||||
) -> Result<(Vec<u8>, bool), Self::Error> {
|
) -> Result<(Vec<u8>, bool), Self::Error> {
|
||||||
if message.len() != MESSAGE_SIZE {
|
let MockMixMessageSettings { max_layers, .. } = self.settings;
|
||||||
|
|
||||||
|
if message.len() != self.size() {
|
||||||
return Err(Error::InvalidMixMessage);
|
return Err(Error::InvalidMixMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -74,7 +105,7 @@ impl MixMessage for MockMixMessage {
|
||||||
|
|
||||||
// If this is the last layer
|
// If this is the last layer
|
||||||
if message[NODE_ID_SIZE..NODE_ID_SIZE * 2] == [0; NODE_ID_SIZE] {
|
if message[NODE_ID_SIZE..NODE_ID_SIZE * 2] == [0; NODE_ID_SIZE] {
|
||||||
let padded_payload = &message[NODE_ID_SIZE * MAX_LAYERS..];
|
let padded_payload = &message[NODE_ID_SIZE * max_layers..];
|
||||||
// remove the payload padding
|
// remove the payload padding
|
||||||
match padded_payload
|
match padded_payload
|
||||||
.iter()
|
.iter()
|
||||||
|
@ -87,12 +118,16 @@ impl MixMessage for MockMixMessage {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut new_message: Vec<u8> = Vec::with_capacity(MESSAGE_SIZE);
|
let mut new_message: Vec<u8> = Vec::with_capacity(self.size());
|
||||||
new_message.extend(&message[NODE_ID_SIZE..NODE_ID_SIZE * MAX_LAYERS]);
|
new_message.extend(&message[NODE_ID_SIZE..NODE_ID_SIZE * max_layers]);
|
||||||
new_message.extend(&[0; NODE_ID_SIZE]);
|
new_message.extend(&[0; NODE_ID_SIZE]);
|
||||||
new_message.extend(&message[NODE_ID_SIZE * MAX_LAYERS..]); // padded payload
|
new_message.extend(&message[NODE_ID_SIZE * max_layers..]); // padded payload
|
||||||
Ok((new_message, false))
|
Ok((new_message, false))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn drop_message(&self) -> &[u8] {
|
||||||
|
&self.drop_message
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -101,23 +136,28 @@ mod tests {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn message() {
|
fn message() {
|
||||||
|
let mock_message = MockMixMessage::new(MockMixMessageSettings {
|
||||||
|
max_layers: 5,
|
||||||
|
max_payload_size: 100,
|
||||||
|
});
|
||||||
|
|
||||||
let node_ids = (0..3).map(|i| [i; NODE_ID_SIZE]).collect::<Vec<_>>();
|
let node_ids = (0..3).map(|i| [i; NODE_ID_SIZE]).collect::<Vec<_>>();
|
||||||
let payload = [7; 10];
|
let payload = [7; 10];
|
||||||
let message = MockMixMessage::build_message(&payload, &node_ids).unwrap();
|
let message = mock_message.build_message(&payload, &node_ids).unwrap();
|
||||||
assert_eq!(message.len(), MESSAGE_SIZE);
|
assert_eq!(message.len(), mock_message.size());
|
||||||
|
|
||||||
let (message, is_fully_unwrapped) =
|
let (message, is_fully_unwrapped) =
|
||||||
MockMixMessage::unwrap_message(&message, &node_ids[0]).unwrap();
|
mock_message.unwrap_message(&message, &node_ids[0]).unwrap();
|
||||||
assert!(!is_fully_unwrapped);
|
assert!(!is_fully_unwrapped);
|
||||||
assert_eq!(message.len(), MESSAGE_SIZE);
|
assert_eq!(message.len(), mock_message.size());
|
||||||
|
|
||||||
let (message, is_fully_unwrapped) =
|
let (message, is_fully_unwrapped) =
|
||||||
MockMixMessage::unwrap_message(&message, &node_ids[1]).unwrap();
|
mock_message.unwrap_message(&message, &node_ids[1]).unwrap();
|
||||||
assert!(!is_fully_unwrapped);
|
assert!(!is_fully_unwrapped);
|
||||||
assert_eq!(message.len(), MESSAGE_SIZE);
|
assert_eq!(message.len(), mock_message.size());
|
||||||
|
|
||||||
let (unwrapped_payload, is_fully_unwrapped) =
|
let (unwrapped_payload, is_fully_unwrapped) =
|
||||||
MockMixMessage::unwrap_message(&message, &node_ids[2]).unwrap();
|
mock_message.unwrap_message(&message, &node_ids[2]).unwrap();
|
||||||
assert!(is_fully_unwrapped);
|
assert!(is_fully_unwrapped);
|
||||||
assert_eq!(unwrapped_payload, payload);
|
assert_eq!(unwrapped_payload, payload);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
use error::Error;
|
use error::Error;
|
||||||
use packet::{Packet, UnpackedPacket};
|
use packet::{Packet, UnpackedPacket};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::MixMessage;
|
use crate::MixMessage;
|
||||||
|
|
||||||
|
@ -8,21 +9,35 @@ mod layered_cipher;
|
||||||
pub mod packet;
|
pub mod packet;
|
||||||
mod routing;
|
mod routing;
|
||||||
|
|
||||||
pub struct SphinxMessage;
|
pub struct SphinxMessage {
|
||||||
|
settings: SphinxMessageSettings,
|
||||||
|
drop_message: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
|
pub struct SphinxMessageSettings {
|
||||||
|
max_layers: usize,
|
||||||
|
max_payload_size: usize,
|
||||||
|
}
|
||||||
|
|
||||||
const ASYM_KEY_SIZE: usize = 32;
|
const ASYM_KEY_SIZE: usize = 32;
|
||||||
// TODO: Move these constants to the upper layer (service layer).
|
|
||||||
const MAX_PAYLOAD_SIZE: usize = 2048;
|
|
||||||
const MAX_LAYERS: usize = 5;
|
|
||||||
|
|
||||||
impl MixMessage for SphinxMessage {
|
impl MixMessage for SphinxMessage {
|
||||||
type PublicKey = [u8; ASYM_KEY_SIZE];
|
type PublicKey = [u8; ASYM_KEY_SIZE];
|
||||||
type PrivateKey = [u8; ASYM_KEY_SIZE];
|
type PrivateKey = [u8; ASYM_KEY_SIZE];
|
||||||
|
type Settings = SphinxMessageSettings;
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
const DROP_MESSAGE: &'static [u8] = &[0; Packet::size(MAX_LAYERS, MAX_PAYLOAD_SIZE)];
|
fn new(settings: Self::Settings) -> Self {
|
||||||
|
let drop_message = vec![0; Packet::size(settings.max_layers, settings.max_payload_size)];
|
||||||
|
Self {
|
||||||
|
settings,
|
||||||
|
drop_message,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn build_message(
|
fn build_message(
|
||||||
|
&self,
|
||||||
payload: &[u8],
|
payload: &[u8],
|
||||||
public_keys: &[Self::PublicKey],
|
public_keys: &[Self::PublicKey],
|
||||||
) -> Result<Vec<u8>, Self::Error> {
|
) -> Result<Vec<u8>, Self::Error> {
|
||||||
|
@ -31,25 +46,32 @@ impl MixMessage for SphinxMessage {
|
||||||
.iter()
|
.iter()
|
||||||
.map(|k| x25519_dalek::PublicKey::from(*k))
|
.map(|k| x25519_dalek::PublicKey::from(*k))
|
||||||
.collect::<Vec<_>>(),
|
.collect::<Vec<_>>(),
|
||||||
MAX_LAYERS,
|
self.settings.max_layers,
|
||||||
payload,
|
payload,
|
||||||
MAX_PAYLOAD_SIZE,
|
self.settings.max_payload_size,
|
||||||
)?;
|
)?;
|
||||||
Ok(packet.to_bytes())
|
Ok(packet.to_bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn unwrap_message(
|
fn unwrap_message(
|
||||||
|
&self,
|
||||||
message: &[u8],
|
message: &[u8],
|
||||||
private_key: &Self::PrivateKey,
|
private_key: &Self::PrivateKey,
|
||||||
) -> Result<(Vec<u8>, bool), Self::Error> {
|
) -> Result<(Vec<u8>, bool), Self::Error> {
|
||||||
let packet = Packet::from_bytes(message, MAX_LAYERS)?;
|
let packet = Packet::from_bytes(message, self.settings.max_layers)?;
|
||||||
let unpacked_packet =
|
let unpacked_packet = packet.unpack(
|
||||||
packet.unpack(&x25519_dalek::StaticSecret::from(*private_key), MAX_LAYERS)?;
|
&x25519_dalek::StaticSecret::from(*private_key),
|
||||||
|
self.settings.max_layers,
|
||||||
|
)?;
|
||||||
match unpacked_packet {
|
match unpacked_packet {
|
||||||
UnpackedPacket::ToForward(packet) => Ok((packet.to_bytes(), false)),
|
UnpackedPacket::ToForward(packet) => Ok((packet.to_bytes(), false)),
|
||||||
UnpackedPacket::FullyUnpacked(payload) => Ok((payload, true)),
|
UnpackedPacket::FullyUnpacked(payload) => Ok((payload, true)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn drop_message(&self) -> &[u8] {
|
||||||
|
&self.drop_message
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_bytes<'a>(data: &'a [u8], sizes: &[usize]) -> Result<Vec<&'a [u8]>, String> {
|
fn parse_bytes<'a>(data: &'a [u8], sizes: &[usize]) -> Result<Vec<&'a [u8]>, String> {
|
||||||
|
|
Loading…
Reference in New Issue