|
@@ -277,6 +277,16 @@ pub trait Storage {
|
|
|
/// Returns the balances for a given account
|
|
|
async fn get_balance(&self, account: &AccountId) -> Result<Vec<Amount>, Error>;
|
|
|
|
|
|
+ /// Returns all the negative payments that are unspent by any account
|
|
|
+ ///
|
|
|
+ /// If any unspent negative deposit exists with the given account and asset, it must be spend in
|
|
|
+ /// the transaction.
|
|
|
+ async fn get_negative_unspent_payments(
|
|
|
+ &self,
|
|
|
+ account: &AccountId,
|
|
|
+ asset: &Asset,
|
|
|
+ ) -> Result<Vec<PaymentFrom>, Error>;
|
|
|
+
|
|
|
/// Returns a list of unspent payments for a given account and asset.
|
|
|
///
|
|
|
/// The payments should be returned sorted by ascending amount, this bit is
|
|
@@ -284,13 +294,39 @@ pub trait Storage {
|
|
|
/// account. It will also improve the database by using many small payments
|
|
|
/// instead of a few large ones, which will make the database faster leaving
|
|
|
/// fewer unspent payments when checking for balances.
|
|
|
- async fn get_unspent_payments(
|
|
|
+ async fn get_positive_unspent_payments(
|
|
|
&self,
|
|
|
account: &AccountId,
|
|
|
asset: &Asset,
|
|
|
target_amount: AmountCents,
|
|
|
) -> Result<Vec<PaymentFrom>, Error>;
|
|
|
|
|
|
+ /// Returns a list of unspent payments for a given account and asset.
|
|
|
+ ///
|
|
|
+ /// This list includes all the negative unspent payments and the list of positive unspent
|
|
|
+ /// payments to cover the target_amount
|
|
|
+ async fn get_unspent_payments(
|
|
|
+ &self,
|
|
|
+ account: &AccountId,
|
|
|
+ asset: &Asset,
|
|
|
+ target_amount: AmountCents,
|
|
|
+ ) -> Result<Vec<PaymentFrom>, Error> {
|
|
|
+ let mut payments = self.get_negative_unspent_payments(account, asset).await?;
|
|
|
+ let target_amount = target_amount
|
|
|
+ + payments
|
|
|
+ .iter()
|
|
|
+ .map(|payment| payment.amount.cents())
|
|
|
+ .sum::<AmountCents>()
|
|
|
+ .abs();
|
|
|
+
|
|
|
+ payments.extend(
|
|
|
+ self.get_positive_unspent_payments(account, asset, target_amount)
|
|
|
+ .await?
|
|
|
+ .into_iter(),
|
|
|
+ );
|
|
|
+ Ok(payments)
|
|
|
+ }
|
|
|
+
|
|
|
/*
|
|
|
///
|
|
|
async fn get_revision(&self, revision_id: &TxId) -> Result<Transaction, Error>;
|
|
@@ -363,6 +399,8 @@ pub trait Storage {
|
|
|
|
|
|
#[cfg(test)]
|
|
|
pub mod test {
|
|
|
+ use std::collections::HashMap;
|
|
|
+
|
|
|
use super::*;
|
|
|
use crate::{config::Config, status::StatusManager, Transaction};
|
|
|
use rand::Rng;
|
|
@@ -384,7 +422,7 @@ pub mod test {
|
|
|
$crate::storage_unit_test!(transaction);
|
|
|
$crate::storage_unit_test!(transaction_does_not_update_stale_transactions);
|
|
|
$crate::storage_unit_test!(transaction_not_available_until_commit);
|
|
|
- $crate::storage_unit_test!(sorted_unspent_payments);
|
|
|
+ $crate::storage_unit_test!(payments_always_include_negative_amounts);
|
|
|
$crate::storage_unit_test!(does_not_update_spent_payments);
|
|
|
$crate::storage_unit_test!(does_not_spend_unspendable_payments);
|
|
|
$crate::storage_unit_test!(spend_spendable_payments);
|
|
@@ -713,7 +751,7 @@ pub mod test {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- pub async fn sorted_unspent_payments<T>(storage: T)
|
|
|
+ pub async fn payments_always_include_negative_amounts<T>(storage: T)
|
|
|
where
|
|
|
T: Storage + Send + Sync,
|
|
|
{
|
|
@@ -726,13 +764,21 @@ pub mod test {
|
|
|
let target_inputs_per_account = 20;
|
|
|
let usd: Asset = "USD/2".parse().expect("valid asset");
|
|
|
|
|
|
+ let mut negative_payments_per_account = HashMap::new();
|
|
|
+
|
|
|
for (index, account) in accounts.iter().enumerate() {
|
|
|
let transaction_id: TxId = vec![index as u8; 32].try_into().expect("valid tx id");
|
|
|
+ let mut negative_payments: usize = 0;
|
|
|
let recipients = (0..target_inputs_per_account)
|
|
|
.map(|_| {
|
|
|
let amount = usd
|
|
|
.from_human(&format!("{}", rng.gen_range(-1000.0..1000.0)))
|
|
|
.expect("valid amount");
|
|
|
+
|
|
|
+ if amount.cents().is_negative() {
|
|
|
+ negative_payments += 1;
|
|
|
+ }
|
|
|
+
|
|
|
PaymentTo {
|
|
|
to: account.clone(),
|
|
|
amount,
|
|
@@ -740,6 +786,8 @@ pub mod test {
|
|
|
})
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
+ negative_payments_per_account.insert(account.clone(), negative_payments);
|
|
|
+
|
|
|
writer
|
|
|
.create_payments(
|
|
|
&transaction_id,
|
|
@@ -765,13 +813,19 @@ pub mod test {
|
|
|
.map(|x| x.amount.cents())
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
- let mut sorted = all_unspent_amounts.clone();
|
|
|
- sorted.sort();
|
|
|
+ let expected_negative_payments = all_unspent_amounts
|
|
|
+ .iter()
|
|
|
+ .filter(|x| x.is_negative())
|
|
|
+ .count();
|
|
|
|
|
|
+ assert_eq!(
|
|
|
+ Some(expected_negative_payments),
|
|
|
+ negative_payments_per_account.get(account).cloned()
|
|
|
+ );
|
|
|
assert_eq!(target_inputs_per_account, all_unspent_amounts.len());
|
|
|
- assert_eq!(sorted, all_unspent_amounts);
|
|
|
+
|
|
|
if !at_least_one_negative_amount {
|
|
|
- at_least_one_negative_amount = sorted[0] < 0;
|
|
|
+ at_least_one_negative_amount = expected_negative_payments > 0;
|
|
|
}
|
|
|
}
|
|
|
assert!(at_least_one_negative_amount);
|