mirror of
https://github.com/serai-dex/serai.git
synced 2024-12-26 13:39:37 +00:00
307 lines
8.4 KiB
Rust
307 lines
8.4 KiB
Rust
use curve25519_dalek::edwards::EdwardsPoint;
|
|
|
|
use crate::{
|
|
hash,
|
|
serialize::*,
|
|
bulletproofs::Bulletproofs, clsag::Clsag,
|
|
};
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub enum Input {
|
|
Gen(u64),
|
|
|
|
ToKey {
|
|
amount: u64,
|
|
key_offsets: Vec<u64>,
|
|
key_image: EdwardsPoint
|
|
}
|
|
}
|
|
|
|
impl Input {
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
match self {
|
|
Input::Gen(height) => {
|
|
w.write_all(&[255])?;
|
|
write_varint(height, w)
|
|
},
|
|
|
|
Input::ToKey { amount, key_offsets, key_image } => {
|
|
w.write_all(&[2])?;
|
|
write_varint(amount, w)?;
|
|
write_vec(write_varint, key_offsets, w)?;
|
|
write_point(key_image, w)
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(r: &mut R) -> std::io::Result<Input> {
|
|
let mut variant = [0];
|
|
r.read_exact(&mut variant)?;
|
|
Ok(
|
|
match variant[0] {
|
|
255 => Input::Gen(read_varint(r)?),
|
|
2 => Input::ToKey {
|
|
amount: read_varint(r)?,
|
|
key_offsets: read_vec(read_varint, r)?,
|
|
key_image: read_point(r)?
|
|
},
|
|
_ => Err(std::io::Error::new(std::io::ErrorKind::Other, "Tried to deserialize unknown/unused input type"))?
|
|
}
|
|
)
|
|
}
|
|
}
|
|
|
|
// Doesn't bother moving to an enum for the unused Script classes
|
|
#[derive(Clone, Debug)]
|
|
pub struct Output {
|
|
pub amount: u64,
|
|
pub key: EdwardsPoint,
|
|
pub tag: Option<u8>
|
|
}
|
|
|
|
impl Output {
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
write_varint(&self.amount, w)?;
|
|
w.write_all(&[2 + (if self.tag.is_some() { 1 } else { 0 })])?;
|
|
write_point(&self.key, w)?;
|
|
if let Some(tag) = self.tag {
|
|
w.write_all(&[tag])?;
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(r: &mut R) -> std::io::Result<Output> {
|
|
let amount = read_varint(r)?;
|
|
let mut tag = [0];
|
|
r.read_exact(&mut tag)?;
|
|
if (tag[0] != 2) && (tag[0] != 3) {
|
|
Err(std::io::Error::new(std::io::ErrorKind::Other, "Tried to deserialize unknown/unused output type"))?;
|
|
}
|
|
|
|
Ok(
|
|
Output {
|
|
amount,
|
|
key: read_point(r)?,
|
|
tag: if tag[0] == 3 { r.read_exact(&mut tag)?; Some(tag[0]) } else { None }
|
|
}
|
|
)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct TransactionPrefix {
|
|
pub version: u64,
|
|
pub unlock_time: u64,
|
|
pub inputs: Vec<Input>,
|
|
pub outputs: Vec<Output>,
|
|
pub extra: Vec<u8>
|
|
}
|
|
|
|
impl TransactionPrefix {
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
write_varint(&self.version, w)?;
|
|
write_varint(&self.unlock_time, w)?;
|
|
write_vec(Input::serialize, &self.inputs, w)?;
|
|
write_vec(Output::serialize, &self.outputs, w)?;
|
|
write_varint(&self.extra.len().try_into().unwrap(), w)?;
|
|
w.write_all(&self.extra)
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(r: &mut R) -> std::io::Result<TransactionPrefix> {
|
|
let mut prefix = TransactionPrefix {
|
|
version: read_varint(r)?,
|
|
unlock_time: read_varint(r)?,
|
|
inputs: read_vec(Input::deserialize, r)?,
|
|
outputs: read_vec(Output::deserialize, r)?,
|
|
extra: vec![]
|
|
};
|
|
|
|
let len = read_varint(r)?;
|
|
prefix.extra.resize(len.try_into().unwrap(), 0);
|
|
r.read_exact(&mut prefix.extra)?;
|
|
|
|
Ok(prefix)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct RctBase {
|
|
pub fee: u64,
|
|
pub ecdh_info: Vec<[u8; 8]>,
|
|
pub commitments: Vec<EdwardsPoint>
|
|
}
|
|
|
|
impl RctBase {
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W, rct_type: u8) -> std::io::Result<()> {
|
|
w.write_all(&[rct_type])?;
|
|
write_varint(&self.fee, w)?;
|
|
for ecdh in &self.ecdh_info {
|
|
w.write_all(ecdh)?;
|
|
}
|
|
write_raw_vec(write_point, &self.commitments, w)
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(outputs: usize, r: &mut R) -> std::io::Result<(RctBase, u8)> {
|
|
let mut rct_type = [0];
|
|
r.read_exact(&mut rct_type)?;
|
|
Ok((
|
|
RctBase {
|
|
fee: read_varint(r)?,
|
|
ecdh_info: (0 .. outputs).map(
|
|
|_| { let mut ecdh = [0; 8]; r.read_exact(&mut ecdh).map(|_| ecdh) }
|
|
).collect::<Result<_, _>>()?,
|
|
commitments: read_raw_vec(read_point, outputs, r)?
|
|
},
|
|
rct_type[0]
|
|
))
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub enum RctPrunable {
|
|
Null,
|
|
Clsag {
|
|
bulletproofs: Vec<Bulletproofs>,
|
|
clsags: Vec<Clsag>,
|
|
pseudo_outs: Vec<EdwardsPoint>
|
|
}
|
|
}
|
|
|
|
impl RctPrunable {
|
|
pub fn rct_type(&self) -> u8 {
|
|
match self {
|
|
RctPrunable::Null => 0,
|
|
RctPrunable::Clsag { .. } => 5
|
|
}
|
|
}
|
|
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
match self {
|
|
RctPrunable::Null => Ok(()),
|
|
RctPrunable::Clsag { bulletproofs, clsags, pseudo_outs } => {
|
|
write_vec(Bulletproofs::serialize, &bulletproofs, w)?;
|
|
write_raw_vec(Clsag::serialize, &clsags, w)?;
|
|
write_raw_vec(write_point, &pseudo_outs, w)
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(
|
|
rct_type: u8,
|
|
decoys: &[usize],
|
|
outputs: usize,
|
|
r: &mut R
|
|
) -> std::io::Result<RctPrunable> {
|
|
Ok(
|
|
match rct_type {
|
|
0 => RctPrunable::Null,
|
|
5 => RctPrunable::Clsag {
|
|
// TODO: Can the amount of outputs be calculated from the BPs for any validly formed TX?
|
|
bulletproofs: read_vec(Bulletproofs::deserialize, r)?,
|
|
clsags: (0 .. decoys.len()).map(|o| Clsag::deserialize(decoys[o], r)).collect::<Result<_, _>>()?,
|
|
pseudo_outs: read_raw_vec(read_point, outputs, r)?
|
|
},
|
|
_ => Err(std::io::Error::new(std::io::ErrorKind::Other, "Tried to deserialize unknown RCT type"))?
|
|
}
|
|
)
|
|
}
|
|
|
|
pub fn signature_serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
match self {
|
|
RctPrunable::Null => panic!("Serializing RctPrunable::Null for a signature"),
|
|
RctPrunable::Clsag { bulletproofs, .. } => bulletproofs.iter().map(|bp| bp.signature_serialize(w)).collect(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct RctSignatures {
|
|
pub base: RctBase,
|
|
pub prunable: RctPrunable
|
|
}
|
|
|
|
impl RctSignatures {
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
self.base.serialize(w, self.prunable.rct_type())?;
|
|
self.prunable.serialize(w)
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(decoys: Vec<usize>, outputs: usize, r: &mut R) -> std::io::Result<RctSignatures> {
|
|
let base = RctBase::deserialize(outputs, r)?;
|
|
Ok(RctSignatures { base: base.0, prunable: RctPrunable::deserialize(base.1, &decoys, outputs, r)? })
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct Transaction {
|
|
pub prefix: TransactionPrefix,
|
|
pub rct_signatures: RctSignatures
|
|
}
|
|
|
|
impl Transaction {
|
|
pub fn serialize<W: std::io::Write>(&self, w: &mut W) -> std::io::Result<()> {
|
|
self.prefix.serialize(w)?;
|
|
self.rct_signatures.serialize(w)
|
|
}
|
|
|
|
pub fn deserialize<R: std::io::Read>(r: &mut R) -> std::io::Result<Transaction> {
|
|
let prefix = TransactionPrefix::deserialize(r)?;
|
|
Ok(
|
|
Transaction {
|
|
rct_signatures: RctSignatures::deserialize(
|
|
prefix.inputs.iter().map(|input| match input {
|
|
Input::Gen(_) => 0,
|
|
Input::ToKey { key_offsets, .. } => key_offsets.len()
|
|
}).collect(),
|
|
prefix.outputs.len(),
|
|
r
|
|
)?,
|
|
prefix
|
|
}
|
|
)
|
|
}
|
|
|
|
pub fn hash(&self) -> [u8; 32] {
|
|
let mut serialized = Vec::with_capacity(2048);
|
|
if self.prefix.version == 1 {
|
|
self.serialize(&mut serialized).unwrap();
|
|
hash(&serialized)
|
|
} else {
|
|
let mut sig_hash = Vec::with_capacity(96);
|
|
|
|
self.prefix.serialize(&mut serialized).unwrap();
|
|
sig_hash.extend(hash(&serialized));
|
|
serialized.clear();
|
|
|
|
self.rct_signatures.base.serialize(
|
|
&mut serialized,
|
|
self.rct_signatures.prunable.rct_type()
|
|
).unwrap();
|
|
sig_hash.extend(hash(&serialized));
|
|
serialized.clear();
|
|
|
|
self.rct_signatures.prunable.serialize(&mut serialized).unwrap();
|
|
sig_hash.extend(hash(&serialized));
|
|
|
|
hash(&sig_hash)
|
|
}
|
|
}
|
|
|
|
pub fn signature_hash(&self) -> [u8; 32] {
|
|
let mut serialized = Vec::with_capacity(2048);
|
|
let mut sig_hash = Vec::with_capacity(96);
|
|
|
|
self.prefix.serialize(&mut serialized).unwrap();
|
|
sig_hash.extend(hash(&serialized));
|
|
serialized.clear();
|
|
|
|
self.rct_signatures.base.serialize(&mut serialized, self.rct_signatures.prunable.rct_type()).unwrap();
|
|
sig_hash.extend(hash(&serialized));
|
|
serialized.clear();
|
|
|
|
self.rct_signatures.prunable.signature_serialize(&mut serialized).unwrap();
|
|
sig_hash.extend(&hash(&serialized));
|
|
|
|
hash(&sig_hash)
|
|
}
|
|
}
|