use std::io; use transcript::{Transcript, RecommendedTranscript}; use group::GroupEncoding; use frost::curve::Ciphersuite; use crate::coins::{Output, Coin}; #[derive(Clone, PartialEq, Eq, Debug)] pub struct Payment { pub address: C::Address, pub data: Option>, pub amount: u64, } impl Payment { pub fn transcript(&self, transcript: &mut T) { transcript.domain_separate(b"payment"); transcript.append_message(b"address", self.address.to_string().as_bytes()); if let Some(data) = self.data.as_ref() { transcript.append_message(b"data", data); } transcript.append_message(b"amount", self.amount.to_le_bytes()); } pub fn write(&self, writer: &mut W) -> io::Result<()> { let address: Vec = self .address .clone() .try_into() .map_err(|_| io::Error::new(io::ErrorKind::Other, "address couldn't be serialized"))?; writer.write_all(&u32::try_from(address.len()).unwrap().to_le_bytes())?; writer.write_all(&address)?; writer.write_all(&[u8::from(self.data.is_some())])?; if let Some(data) = &self.data { writer.write_all(&u32::try_from(data.len()).unwrap().to_le_bytes())?; writer.write_all(data)?; } writer.write_all(&self.amount.to_le_bytes()) } pub fn read(reader: &mut R) -> io::Result { let mut buf = [0; 4]; reader.read_exact(&mut buf)?; let mut address = vec![0; usize::try_from(u32::from_le_bytes(buf)).unwrap()]; reader.read_exact(&mut address)?; let address = C::Address::try_from(address) .map_err(|_| io::Error::new(io::ErrorKind::Other, "invalid address"))?; let mut buf = [0; 1]; reader.read_exact(&mut buf)?; let data = if buf[0] == 1 { let mut buf = [0; 4]; reader.read_exact(&mut buf)?; let mut data = vec![0; usize::try_from(u32::from_le_bytes(buf)).unwrap()]; reader.read_exact(&mut data)?; Some(data) } else { None }; let mut buf = [0; 8]; reader.read_exact(&mut buf)?; let amount = u64::from_le_bytes(buf); Ok(Payment { address, data, amount }) } } #[derive(Clone, PartialEq, Eq, Debug)] pub struct Plan { pub key: ::G, pub inputs: Vec, pub payments: Vec>, pub change: Option<::G>, } impl Plan { pub fn transcript(&self) -> RecommendedTranscript { let mut transcript = RecommendedTranscript::new(b"Serai Processor Plan ID"); transcript.domain_separate(b"meta"); transcript.append_message(b"network", C::ID); transcript.append_message(b"key", self.key.to_bytes()); transcript.domain_separate(b"inputs"); for input in &self.inputs { transcript.append_message(b"input", input.id()); } transcript.domain_separate(b"payments"); for payment in &self.payments { payment.transcript(&mut transcript); } if let Some(change) = self.change { transcript.append_message(b"change", change.to_bytes()); } transcript } pub fn id(&self) -> [u8; 32] { let challenge = self.transcript().challenge(b"id"); let mut res = [0; 32]; res.copy_from_slice(&challenge[.. 32]); res } pub fn write(&self, writer: &mut W) -> io::Result<()> { writer.write_all(self.key.to_bytes().as_ref())?; writer.write_all(&u32::try_from(self.inputs.len()).unwrap().to_le_bytes())?; for input in &self.inputs { input.write(writer)?; } writer.write_all(&u32::try_from(self.payments.len()).unwrap().to_le_bytes())?; for payment in &self.payments { payment.write(writer)?; } writer.write_all(&[u8::from(self.change.is_some())])?; if let Some(change) = &self.change { writer.write_all(change.to_bytes().as_ref())?; } Ok(()) } pub fn read(reader: &mut R) -> io::Result { let key = C::Curve::read_G(reader)?; let mut inputs = vec![]; let mut buf = [0; 4]; reader.read_exact(&mut buf)?; for _ in 0 .. u32::from_le_bytes(buf) { inputs.push(C::Output::read(reader)?); } let mut payments = vec![]; reader.read_exact(&mut buf)?; for _ in 0 .. u32::from_le_bytes(buf) { payments.push(Payment::::read(reader)?); } let mut buf = [0; 1]; reader.read_exact(&mut buf)?; let change = if buf[0] == 1 { Some(C::Curve::read_G(reader)?) } else { None }; Ok(Plan { key, inputs, payments, change }) } }