#![cfg_attr(docsrs, feature(doc_auto_cfg))] #![doc = include_str!("../README.md")] #![deny(missing_docs)] #![cfg_attr(not(feature = "std"), no_std)] use std_shims::vec::Vec; use zeroize::{Zeroize, Zeroizing}; use curve25519_dalek::{Scalar, EdwardsPoint}; use monero_serai::{ io::write_varint, primitives::{Commitment, keccak256, keccak256_to_scalar}, ringct::EncryptedAmount, transaction::Input, }; pub use monero_serai::*; pub use monero_rpc as rpc; pub use monero_address as address; mod view_pair; pub use view_pair::{ViewPair, GuaranteedViewPair}; /// Structures and functionality for working with transactions' extra fields. pub mod extra; pub(crate) use extra::{PaymentId, Extra}; pub(crate) mod output; pub use output::WalletOutput; mod scan; pub use scan::{Scanner, GuaranteedScanner}; mod decoys; pub use decoys::OutputWithDecoys; /// Structs and functionality for sending transactions. pub mod send; #[cfg(test)] mod tests; #[derive(Clone, PartialEq, Eq, Zeroize)] struct SharedKeyDerivations { // Hs("view_tag" || 8Ra || o) view_tag: u8, // Hs(uniqueness || 8Ra || o) where uniqueness may be empty shared_key: Scalar, } impl SharedKeyDerivations { // https://gist.github.com/kayabaNerve/8066c13f1fe1573286ba7a2fd79f6100 fn uniqueness(inputs: &[Input]) -> [u8; 32] { let mut u = b"uniqueness".to_vec(); for input in inputs { match input { // If Gen, this should be the only input, making this loop somewhat pointless // This works and even if there were somehow multiple inputs, it'd be a false negative Input::Gen(height) => { write_varint(height, &mut u).unwrap(); } Input::ToKey { key_image, .. } => u.extend(key_image.compress().to_bytes()), } } keccak256(u) } #[allow(clippy::needless_pass_by_value)] fn output_derivations( uniqueness: Option<[u8; 32]>, ecdh: Zeroizing, o: usize, ) -> Zeroizing { // 8Ra let mut output_derivation = Zeroizing::new( Zeroizing::new(Zeroizing::new(ecdh.mul_by_cofactor()).compress().to_bytes()).to_vec(), ); // || o { let output_derivation: &mut Vec = output_derivation.as_mut(); write_varint(&o, output_derivation).unwrap(); } let view_tag = keccak256([b"view_tag".as_ref(), &output_derivation].concat())[0]; // uniqueness || let output_derivation = if let Some(uniqueness) = uniqueness { Zeroizing::new([uniqueness.as_ref(), &output_derivation].concat()) } else { output_derivation }; Zeroizing::new(SharedKeyDerivations { view_tag, shared_key: keccak256_to_scalar(&output_derivation), }) } // H(8Ra || 0x8d) #[allow(clippy::needless_pass_by_value)] fn payment_id_xor(ecdh: Zeroizing) -> [u8; 8] { // 8Ra let output_derivation = Zeroizing::new( Zeroizing::new(Zeroizing::new(ecdh.mul_by_cofactor()).compress().to_bytes()).to_vec(), ); let mut payment_id_xor = [0; 8]; payment_id_xor .copy_from_slice(&keccak256([output_derivation.as_ref(), [0x8d].as_ref()].concat())[.. 8]); payment_id_xor } fn commitment_mask(&self) -> Scalar { let mut mask = b"commitment_mask".to_vec(); mask.extend(self.shared_key.as_bytes()); let res = keccak256_to_scalar(&mask); mask.zeroize(); res } fn compact_amount_encryption(&self, amount: u64) -> [u8; 8] { let mut amount_mask = Zeroizing::new(b"amount".to_vec()); amount_mask.extend(self.shared_key.to_bytes()); let mut amount_mask = keccak256(&amount_mask); let mut amount_mask_8 = [0; 8]; amount_mask_8.copy_from_slice(&amount_mask[.. 8]); amount_mask.zeroize(); (amount ^ u64::from_le_bytes(amount_mask_8)).to_le_bytes() } fn decrypt(&self, enc_amount: &EncryptedAmount) -> Commitment { match enc_amount { // TODO: Add a test vector for this EncryptedAmount::Original { mask, amount } => { let mask_shared_sec = keccak256(self.shared_key.as_bytes()); let mask = Scalar::from_bytes_mod_order(*mask) - Scalar::from_bytes_mod_order(mask_shared_sec); let amount_shared_sec = keccak256(mask_shared_sec); let amount_scalar = Scalar::from_bytes_mod_order(*amount) - Scalar::from_bytes_mod_order(amount_shared_sec); // d2b from rctTypes.cpp let amount = u64::from_le_bytes(amount_scalar.to_bytes()[0 .. 8].try_into().unwrap()); Commitment::new(mask, amount) } EncryptedAmount::Compact { amount } => Commitment::new( self.commitment_mask(), u64::from_le_bytes(self.compact_amount_encryption(u64::from_le_bytes(*amount))), ), } } }