Return Timelocked instead of (Timelock, Vec<SpendableOutput>)

Timelocked doesn't expose the Vec yet requires calling not_locked(), 
unlocked(Timelock), or ignore_timelock().
This commit is contained in:
Luke Parker 2022-07-09 18:53:52 -04:00
parent a4cd1755a5
commit 53267a46c8
No known key found for this signature in database
GPG key ID: F9F1386DB1E119B6
5 changed files with 48 additions and 15 deletions

View file

@ -1,3 +1,5 @@
use core::cmp::Ordering;
use curve25519_dalek::edwards::EdwardsPoint; use curve25519_dalek::edwards::EdwardsPoint;
use crate::{hash, serialize::*, ringct::{RctPrunable, RctSignatures}}; use crate::{hash, serialize::*, ringct::{RctPrunable, RctSignatures}};
@ -131,6 +133,17 @@ impl Timelock {
} }
} }
impl PartialOrd for Timelock {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
match (self, other) {
(Timelock::None, _) => Some(Ordering::Less),
(Timelock::Block(a), Timelock::Block(b)) => a.partial_cmp(b),
(Timelock::Time(a), Timelock::Time(b)) => a.partial_cmp(b),
_ => None
}
}
}
#[derive(Clone, PartialEq, Debug)] #[derive(Clone, PartialEq, Debug)]
pub struct TransactionPrefix { pub struct TransactionPrefix {
pub version: u64, pub version: u64,

View file

@ -24,6 +24,30 @@ pub struct SpendableOutput {
pub commitment: Commitment pub commitment: Commitment
} }
pub struct Timelocked(Timelock, Vec<SpendableOutput>);
impl Timelocked {
pub fn timelock(&self) -> Timelock {
self.0
}
pub fn not_locked(&self) -> Vec<SpendableOutput> {
if self.0 == Timelock::None {
return self.1.clone();
}
vec![]
}
/// Returns None if the Timelocks aren't comparable. Returns Some(vec![]) if none are unlocked
pub fn unlocked(&self, timelock: Timelock) -> Option<Vec<SpendableOutput>> {
// If the Timelocks are comparable, return the outputs if they're now unlocked
self.0.partial_cmp(&timelock).filter(|_| self.0 <= timelock).map(|_| self.1.clone())
}
pub fn ignore_timelock(&self) -> Vec<SpendableOutput> {
self.1.clone()
}
}
impl SpendableOutput { impl SpendableOutput {
pub fn serialize(&self) -> Vec<u8> { pub fn serialize(&self) -> Vec<u8> {
let mut res = Vec::with_capacity(32 + 1 + 32 + 32 + 40); let mut res = Vec::with_capacity(32 + 1 + 32 + 32 + 40);
@ -57,7 +81,7 @@ impl Transaction {
&self, &self,
view: ViewPair, view: ViewPair,
guaranteed: bool guaranteed: bool
) -> (Vec<SpendableOutput>, Timelock) { ) -> Timelocked {
let mut extra = vec![]; let mut extra = vec![];
write_varint(&u64::try_from(self.prefix.extra.len()).unwrap(), &mut extra).unwrap(); write_varint(&u64::try_from(self.prefix.extra.len()).unwrap(), &mut extra).unwrap();
extra.extend(&self.prefix.extra); extra.extend(&self.prefix.extra);
@ -75,7 +99,7 @@ impl Transaction {
pubkeys = m_pubkeys.iter().map(|key| key.point.decompress()).filter_map(|key| key).collect(); pubkeys = m_pubkeys.iter().map(|key| key.point.decompress()).filter_map(|key| key).collect();
} else { } else {
return (vec![], self.prefix.timelock); return Timelocked(self.prefix.timelock, vec![]);
}; };
let mut res = vec![]; let mut res = vec![];
@ -132,6 +156,6 @@ impl Transaction {
} }
} }
(res, self.prefix.timelock) Timelocked(self.prefix.timelock, res)
} }
} }

View file

@ -91,7 +91,7 @@ async fn send_core(test: usize, multisig: bool) {
// Grab the largest output available // Grab the largest output available
let output = { let output = {
let mut outputs = tx.as_ref().unwrap().scan(view_pair, false).0; let mut outputs = tx.as_ref().unwrap().scan(view_pair, false).ignore_timelock();
outputs.sort_by(|x, y| x.commitment.amount.cmp(&y.commitment.amount).reverse()); outputs.sort_by(|x, y| x.commitment.amount.cmp(&y.commitment.amount).reverse());
outputs.swap_remove(0) outputs.swap_remove(0)
}; };
@ -116,7 +116,7 @@ async fn send_core(test: usize, multisig: bool) {
for i in (start + 1) .. (start + 9) { for i in (start + 1) .. (start + 9) {
let tx = rpc.get_block_transactions(i).await.unwrap().swap_remove(0); let tx = rpc.get_block_transactions(i).await.unwrap().swap_remove(0);
let output = tx.scan(view_pair, false).0.swap_remove(0); let output = tx.scan(view_pair, false).ignore_timelock().swap_remove(0);
amount += output.commitment.amount; amount += output.commitment.amount;
outputs.push(output); outputs.push(output);
} }

3
processor/build.rs Normal file
View file

@ -0,0 +1,3 @@
fn main() {
println!("cargo:rustc-link-arg=-zmuldefs");
}

View file

@ -9,7 +9,7 @@ use transcript::RecommendedTranscript;
use frost::{curve::Ed25519, FrostKeys}; use frost::{curve::Ed25519, FrostKeys};
use monero_serai::{ use monero_serai::{
transaction::{Timelock, Transaction}, transaction::Transaction,
rpc::Rpc, rpc::Rpc,
wallet::{ wallet::{
ViewPair, address::{Network, AddressType, Address}, ViewPair, address::{Network, AddressType, Address},
@ -126,14 +126,7 @@ impl Coin for Monero {
async fn get_outputs(&self, block: &Self::Block, key: dfg::EdwardsPoint) -> Vec<Self::Output> { async fn get_outputs(&self, block: &Self::Block, key: dfg::EdwardsPoint) -> Vec<Self::Output> {
block block
.iter() .iter()
.flat_map(|tx| { .flat_map(|tx| tx.scan(self.view_pair(key), true).not_locked())
let (outputs, timelock) = tx.scan(self.view_pair(key), true);
if timelock == Timelock::None {
outputs
} else {
vec![]
}
})
.map(Output::from) .map(Output::from)
.collect() .collect()
} }
@ -215,7 +208,7 @@ impl Coin for Monero {
let outputs = self.rpc let outputs = self.rpc
.get_block_transactions_possible(height).await.unwrap() .get_block_transactions_possible(height).await.unwrap()
.swap_remove(0).scan(self.empty_view_pair(), false).0; .swap_remove(0).scan(self.empty_view_pair(), false).ignore_timelock();
let amount = outputs[0].commitment.amount; let amount = outputs[0].commitment.amount;
let fee = 1000000000; // TODO let fee = 1000000000; // TODO