From 1b5573b494d3dd1124237e01d7ed868bd6c6e31b Mon Sep 17 00:00:00 2001 From: th4s Date: Wed, 17 Jan 2024 14:48:26 +0100 Subject: [PATCH] Add RandomOT traits and implement them for KOS15 (#91) * Add RandomOT traits and implement them for KOS15 * Fix trait comments * Add part of feedback * Add test for KOS15 RandomOT * Improve comments 1/2 Adhere to new wording for OT Co-authored-by: dan * Improve comments 2/2 Adhere to new wording for OT. Co-authored-by: dan --------- Co-authored-by: dan --- ot/mpz-ot-core/src/kos/receiver.rs | 5 ++ ot/mpz-ot-core/src/kos/sender.rs | 5 ++ ot/mpz-ot/src/kos/mod.rs | 42 ++++++++++++++++- ot/mpz-ot/src/kos/receiver.rs | 76 ++++++++++++++++++++++++++++-- ot/mpz-ot/src/kos/sender.rs | 75 +++++++++++++++++++++++++---- ot/mpz-ot/src/lib.rs | 49 +++++++++++++++++++ 6 files changed, 240 insertions(+), 12 deletions(-) diff --git a/ot/mpz-ot-core/src/kos/receiver.rs b/ot/mpz-ot-core/src/kos/receiver.rs index 491b3911..b7b8fb59 100644 --- a/ot/mpz-ot-core/src/kos/receiver.rs +++ b/ot/mpz-ot-core/src/kos/receiver.rs @@ -581,6 +581,11 @@ impl ReceiverKeys { }) .collect()) } + + /// Returns the choices and the keys + pub fn take_choices_and_keys(self) -> (Vec, Vec) { + (self.choices, self.keys) + } } struct PayloadRecordNoDelta { diff --git a/ot/mpz-ot-core/src/kos/sender.rs b/ot/mpz-ot-core/src/kos/sender.rs index ea93d90c..8e692727 100644 --- a/ot/mpz-ot-core/src/kos/sender.rs +++ b/ot/mpz-ot-core/src/kos/sender.rs @@ -448,6 +448,11 @@ impl SenderKeys { }, }) } + + /// Returns the keys + pub fn take_keys(self) -> Vec<[Block; 2]> { + self.keys + } } /// The sender's state. diff --git a/ot/mpz-ot/src/kos/mod.rs b/ot/mpz-ot/src/kos/mod.rs index c9ffb8b1..e6c92b23 100644 --- a/ot/mpz-ot/src/kos/mod.rs +++ b/ot/mpz-ot/src/kos/mod.rs @@ -53,7 +53,7 @@ mod tests { use crate::{ mock::{mock_ot_pair, MockOTReceiver, MockOTSender}, - OTReceiver, OTSender, OTSetup, VerifiableOTReceiver, + OTReceiver, OTSender, OTSetup, RandomOTReceiver, RandomOTSender, VerifiableOTReceiver, }; #[fixture] @@ -146,6 +146,46 @@ mod tests { assert_eq!(received, expected); } + #[tokio::test] + async fn test_kos_random() { + let (sender_channel, receiver_channel) = MemoryDuplex::new(); + + let (mut sender_sink, mut sender_stream) = sender_channel.split(); + let (mut receiver_sink, mut receiver_stream) = receiver_channel.split(); + + let (mut sender, mut receiver) = setup( + SenderConfig::default(), + ReceiverConfig::default(), + &mut sender_sink, + &mut sender_stream, + &mut receiver_sink, + &mut receiver_stream, + 10, + ) + .await; + + let (sender_res, receiver_res) = tokio::join!( + RandomOTSender::send_random(&mut sender, &mut sender_sink, &mut sender_stream, 10), + RandomOTReceiver::receive_random( + &mut receiver, + &mut receiver_sink, + &mut receiver_stream, + 10 + ) + ); + + let sender_output: Vec<[Block; 2]> = sender_res.unwrap(); + let (choices, receiver_output): (Vec, Vec) = receiver_res.unwrap(); + + let expected = sender_output + .into_iter() + .zip(choices) + .map(|(output, choice)| output[choice as usize]) + .collect::>(); + + assert_eq!(receiver_output, expected); + } + #[rstest] #[tokio::test] async fn test_kos_bytes(data: Vec<[Block; 2]>, choices: Vec) { diff --git a/ot/mpz-ot/src/kos/receiver.rs b/ot/mpz-ot/src/kos/receiver.rs index ed3661e0..1562e76a 100644 --- a/ot/mpz-ot/src/kos/receiver.rs +++ b/ot/mpz-ot/src/kos/receiver.rs @@ -1,22 +1,25 @@ use async_trait::async_trait; use futures::SinkExt; use itybity::{FromBitIterator, IntoBitIterator}; -use mpz_core::{cointoss, Block, ProtocolMessage}; +use mpz_core::{cointoss, prg::Prg, Block, ProtocolMessage}; use mpz_ot_core::kos::{ msgs::Message, receiver_state as state, Receiver as ReceiverCore, ReceiverConfig, CSP, SSP, }; use enum_try_as_inner::EnumTryAsInner; use rand::{thread_rng, Rng}; +use rand_core::{RngCore, SeedableRng}; use utils_aio::{ non_blocking_backend::{Backend, NonBlockingBackend}, sink::IoSink, stream::{ExpectStreamExt, IoStream}, }; -use crate::{OTError, OTReceiver, OTSender, OTSetup, VerifiableOTReceiver, VerifiableOTSender}; - use super::{into_base_sink, into_base_stream, ReceiverError, ReceiverVerifyError}; +use crate::{ + OTError, OTReceiver, OTSender, OTSetup, RandomOTReceiver, VerifiableOTReceiver, + VerifiableOTSender, +}; #[derive(Debug, EnumTryAsInner)] #[derive_err(Debug)] @@ -303,6 +306,34 @@ where } } +#[async_trait] +impl RandomOTReceiver for Receiver +where + BaseOT: ProtocolMessage + Send, +{ + async fn receive_random< + Si: IoSink> + Send + Unpin, + St: IoStream> + Send + Unpin, + >( + &mut self, + _sink: &mut Si, + _stream: &mut St, + count: usize, + ) -> Result<(Vec, Vec), OTError> { + let receiver = self + .state + .try_as_extension_mut() + .map_err(ReceiverError::from)?; + + let (choices, random_outputs) = receiver + .keys(count) + .map_err(ReceiverError::from)? + .take_choices_and_keys(); + + Ok((choices, random_outputs)) + } +} + #[async_trait] impl OTReceiver for Receiver where @@ -350,6 +381,45 @@ where } } +#[async_trait] +impl RandomOTReceiver for Receiver +where + BaseOT: ProtocolMessage + Send, +{ + async fn receive_random< + Si: IoSink> + Send + Unpin, + St: IoStream> + Send + Unpin, + >( + &mut self, + _sink: &mut Si, + _stream: &mut St, + count: usize, + ) -> Result<(Vec, Vec<[u8; N]>), OTError> { + let receiver = self + .state + .try_as_extension_mut() + .map_err(ReceiverError::from)?; + + let (choices, random_outputs) = receiver + .keys(count) + .map_err(ReceiverError::from)? + .take_choices_and_keys(); + + Ok(( + choices, + random_outputs + .into_iter() + .map(|block| { + let mut prg = Prg::from_seed(block); + let mut out = [0_u8; N]; + prg.fill_bytes(&mut out); + out + }) + .collect(), + )) + } +} + #[async_trait] impl VerifiableOTReceiver for Receiver where diff --git a/ot/mpz-ot/src/kos/sender.rs b/ot/mpz-ot/src/kos/sender.rs index a1a21411..407ddadd 100644 --- a/ot/mpz-ot/src/kos/sender.rs +++ b/ot/mpz-ot/src/kos/sender.rs @@ -1,25 +1,24 @@ -use crate::{ - kos::SenderError, CommittedOTReceiver, CommittedOTSender, OTError, OTReceiver, OTSender, - OTSetup, -}; - use async_trait::async_trait; +use enum_try_as_inner::EnumTryAsInner; use futures_util::SinkExt; use itybity::IntoBits; -use mpz_core::{cointoss, Block, ProtocolMessage}; +use mpz_core::{cointoss, prg::Prg, Block, ProtocolMessage}; use mpz_ot_core::kos::{ msgs::Message, sender_state as state, Sender as SenderCore, SenderConfig, CSP, SSP, }; use rand::{thread_rng, Rng}; +use rand_core::{RngCore, SeedableRng}; use utils_aio::{ non_blocking_backend::{Backend, NonBlockingBackend}, sink::IoSink, stream::{ExpectStreamExt, IoStream}, }; -use enum_try_as_inner::EnumTryAsInner; - use super::{into_base_sink, into_base_stream}; +use crate::{ + kos::SenderError, CommittedOTReceiver, CommittedOTSender, OTError, OTReceiver, OTSender, + OTSetup, RandomOTSender, +}; #[derive(Debug, EnumTryAsInner)] #[derive_err(Debug)] @@ -336,6 +335,30 @@ where } } +#[async_trait] +impl RandomOTSender<[Block; 2]> for Sender +where + BaseOT: ProtocolMessage + Send, +{ + async fn send_random< + Si: IoSink> + Send + Unpin, + St: IoStream> + Send + Unpin, + >( + &mut self, + _sink: &mut Si, + _stream: &mut St, + count: usize, + ) -> Result, OTError> { + let sender = self + .state + .try_as_extension_mut() + .map_err(SenderError::from)?; + + let random_outputs = sender.keys(count).map_err(SenderError::from)?; + Ok(random_outputs.take_keys()) + } +} + #[async_trait] impl OTSender<[[u8; N]; 2]> for Sender where @@ -375,6 +398,42 @@ where } } +#[async_trait] +impl RandomOTSender<[[u8; N]; 2]> for Sender +where + BaseOT: ProtocolMessage + Send, +{ + async fn send_random< + Si: IoSink> + Send + Unpin, + St: IoStream> + Send + Unpin, + >( + &mut self, + _sink: &mut Si, + _stream: &mut St, + count: usize, + ) -> Result, OTError> { + let sender = self + .state + .try_as_extension_mut() + .map_err(SenderError::from)?; + + let random_outputs = sender.keys(count).map_err(SenderError::from)?; + + let prng = |block| { + let mut prg = Prg::from_seed(block); + let mut out = [0_u8; N]; + prg.fill_bytes(&mut out); + out + }; + + Ok(random_outputs + .take_keys() + .into_iter() + .map(|[a, b]| [prng(a), prng(b)]) + .collect()) + } +} + #[async_trait] impl CommittedOTSender<[Block; 2]> for Sender where diff --git a/ot/mpz-ot/src/lib.rs b/ot/mpz-ot/src/lib.rs index 9da8cd7e..8ff88faf 100644 --- a/ot/mpz-ot/src/lib.rs +++ b/ot/mpz-ot/src/lib.rs @@ -68,6 +68,30 @@ where ) -> Result<(), OTError>; } +/// A random OT sender. +#[async_trait] +pub trait RandomOTSender: ProtocolMessage +where + T: Send + Sync, +{ + /// Outputs pairs of random messages. + /// + /// # Arguments + /// + /// * `sink` - The IO sink to the receiver. + /// * `stream` - The IO stream from the receiver. + /// * `count` - The number of pairs of random messages to output. + async fn send_random< + Si: IoSink + Send + Unpin, + St: IoStream + Send + Unpin, + >( + &mut self, + sink: &mut Si, + stream: &mut St, + count: usize, + ) -> Result, OTError>; +} + /// An oblivious transfer receiver. #[async_trait] pub trait OTReceiver: ProtocolMessage @@ -90,6 +114,31 @@ where ) -> Result, OTError>; } +/// A random OT receiver. +#[async_trait] +pub trait RandomOTReceiver: ProtocolMessage +where + T: Send + Sync, + U: Send + Sync, +{ + /// Outputs the choice bits and the corresponding messages. + /// + /// # Arguments + /// + /// * `sink` - The IO sink to the sender. + /// * `stream` - The IO stream from the sender. + /// * `count` - The number of random messages to receive. + async fn receive_random< + Si: IoSink + Send + Unpin, + St: IoStream + Send + Unpin, + >( + &mut self, + sink: &mut Si, + stream: &mut St, + count: usize, + ) -> Result<(Vec, Vec), OTError>; +} + /// An oblivious transfer sender that is committed to its messages and can reveal them /// to the receiver to verify them. #[async_trait]