|
@@ -1,6 +1,6 @@
|
|
|
use crate::{
|
|
|
- config::Config, storage::Storage, transaction::Type, AccountId, Amount, Error, PaymentFrom,
|
|
|
- PaymentId, Status, Transaction, TransactionId,
|
|
|
+ amount::AmountCents, config::Config, storage::Storage, transaction::Type, AccountId, Amount,
|
|
|
+ Error, PaymentFrom, PaymentId, Status, Transaction, TransactionId,
|
|
|
};
|
|
|
use std::{cmp::Ordering, collections::HashMap, sync::Arc};
|
|
|
|
|
@@ -44,12 +44,14 @@ where
|
|
|
&self,
|
|
|
payments: Vec<(AccountId, Amount)>,
|
|
|
) -> Result<(Option<Transaction>, Vec<PaymentFrom>), Error> {
|
|
|
- let mut to_spend = HashMap::new();
|
|
|
+ let mut to_spend = HashMap::<_, AmountCents>::new();
|
|
|
|
|
|
for (account_id, amount) in payments.into_iter() {
|
|
|
let id = (account_id, amount.asset().clone());
|
|
|
if let Some(value) = to_spend.get_mut(&id) {
|
|
|
- *value += amount.cents();
|
|
|
+ *value = value
|
|
|
+ .checked_add(amount.cents())
|
|
|
+ .ok_or(Error::Overflow("cents".to_owned()))?;
|
|
|
} else {
|
|
|
to_spend.insert(id, amount.cents());
|
|
|
}
|
|
@@ -67,7 +69,9 @@ where
|
|
|
.await?;
|
|
|
for payment in iterator.into_iter() {
|
|
|
let cents = payment.amount.cents();
|
|
|
- to_spend_cents -= cents;
|
|
|
+ to_spend_cents = to_spend_cents
|
|
|
+ .checked_sub(cents)
|
|
|
+ .ok_or(Error::Underflow("cents".to_owned()))?;
|
|
|
payments.push(payment);
|
|
|
match to_spend_cents.cmp(&0) {
|
|
|
Ordering::Equal => {
|
|
@@ -89,8 +93,14 @@ where
|
|
|
.ok_or(Error::InsufficientBalance(account.clone(), asset.clone()))?;
|
|
|
|
|
|
change_input.push(input);
|
|
|
- change_output
|
|
|
- .push((account.clone(), asset.new_amount(cents - to_spend_cents)));
|
|
|
+ change_output.push((
|
|
|
+ account.clone(),
|
|
|
+ asset.new_amount(
|
|
|
+ cents
|
|
|
+ .checked_sub(to_spend_cents)
|
|
|
+ .ok_or(Error::Underflow("change cents".to_owned()))?,
|
|
|
+ ),
|
|
|
+ ));
|
|
|
change_output.push((account.clone(), asset.new_amount(to_spend_cents)));
|
|
|
|
|
|
// Go to the next payment
|
|
@@ -131,13 +141,17 @@ where
|
|
|
let creates = split_input.creates();
|
|
|
for i in 0..total {
|
|
|
// Spend the new payment
|
|
|
+ let index = i
|
|
|
+ .checked_mul(2)
|
|
|
+ .ok_or(Error::Overflow("index overflow".to_owned()))?;
|
|
|
+ let uindex: usize = index.into();
|
|
|
payments.push(PaymentFrom {
|
|
|
id: PaymentId {
|
|
|
transaction: split_input.id.clone(),
|
|
|
- position: i * 2,
|
|
|
+ position: index,
|
|
|
},
|
|
|
- from: creates[(i * 2) as usize].to.clone(),
|
|
|
- amount: creates[(i * 2) as usize].amount.clone(),
|
|
|
+ from: creates[uindex].to.clone(),
|
|
|
+ amount: creates[uindex].amount.clone(),
|
|
|
});
|
|
|
}
|
|
|
Some(split_input)
|