mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-14 23:05:09 +00:00
154 lines
4.4 KiB
Rust
154 lines
4.4 KiB
Rust
|
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<C: Coin> {
|
||
|
pub address: C::Address,
|
||
|
pub data: Option<Vec<u8>>,
|
||
|
pub amount: u64,
|
||
|
}
|
||
|
|
||
|
impl<C: Coin> Payment<C> {
|
||
|
pub fn transcript<T: 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<W: io::Write>(&self, writer: &mut W) -> io::Result<()> {
|
||
|
let address: Vec<u8> = 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<R: io::Read>(reader: &mut R) -> io::Result<Self> {
|
||
|
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<C: Coin> {
|
||
|
pub key: <C::Curve as Ciphersuite>::G,
|
||
|
pub inputs: Vec<C::Output>,
|
||
|
pub payments: Vec<Payment<C>>,
|
||
|
pub change: Option<<C::Curve as Ciphersuite>::G>,
|
||
|
}
|
||
|
|
||
|
impl<C: Coin> Plan<C> {
|
||
|
pub fn transcript(&self) -> RecommendedTranscript {
|
||
|
let mut transcript = RecommendedTranscript::new(b"Serai Processor Plan ID");
|
||
|
transcript.domain_separate(b"meta");
|
||
|
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<W: io::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<R: io::Read>(reader: &mut R) -> io::Result<Self> {
|
||
|
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::<C>::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 })
|
||
|
}
|
||
|
}
|