|
@@ -4,13 +4,13 @@ use crate::{
|
|
|
config::Config,
|
|
|
status::{InternalStatus, StatusManager},
|
|
|
storage::{AccountTransactionType, Batch, ReceivedPaymentStatus, Storage},
|
|
|
- transaction::Error as TxError,
|
|
|
- transaction::Type,
|
|
|
+ token::{Token, TokenManager},
|
|
|
+ transaction::{Error as TxError, Type},
|
|
|
worker::WorkerManager,
|
|
|
AccountId, Amount, Error, Filter, PaymentFrom, PaymentId, RevId, Status, Tag, Transaction,
|
|
|
TxId,
|
|
|
};
|
|
|
-use std::{cmp::Ordering, collections::HashMap, sync::Arc};
|
|
|
+use std::{cmp::Ordering, collections::HashMap, sync::Arc, time::Duration};
|
|
|
use tokio::sync::mpsc::{self, Receiver, Sender};
|
|
|
|
|
|
/// The Verax ledger
|
|
@@ -20,6 +20,7 @@ where
|
|
|
S: Storage + Sync + Send,
|
|
|
{
|
|
|
config: Config<S>,
|
|
|
+ token_manager: TokenManager,
|
|
|
broadcaster: WorkerManager<Broadcaster>,
|
|
|
}
|
|
|
|
|
@@ -31,6 +32,7 @@ where
|
|
|
pub fn new(config: Config<S>) -> Arc<Self> {
|
|
|
Arc::new(Self {
|
|
|
config,
|
|
|
+ token_manager: TokenManager(b"test".to_vec()),
|
|
|
broadcaster: WorkerManager::new(Broadcaster::default()),
|
|
|
})
|
|
|
}
|
|
@@ -248,6 +250,15 @@ where
|
|
|
Ok(())
|
|
|
}
|
|
|
|
|
|
+ /// TODO
|
|
|
+ pub async fn lock_transaction(&self, transaction_id: &TxId) -> Result<Token, Error> {
|
|
|
+ let token = self
|
|
|
+ .token_manager
|
|
|
+ .new_token(transaction_id, Duration::from_secs(60));
|
|
|
+ self.config.storage.lock_transaction(&token).await?;
|
|
|
+ Ok(token)
|
|
|
+ }
|
|
|
+
|
|
|
/// Stores the current transaction object to the storage layer.
|
|
|
///
|
|
|
/// This method is not idempotent, and it will fail if the transaction if the requested update
|
|
@@ -255,10 +266,24 @@ where
|
|
|
///
|
|
|
/// This function will store the base transaction if it is the first revision, and will create a
|
|
|
/// new revision otherwise.
|
|
|
- pub async fn store(&self, transaction: Transaction) -> Result<Transaction, Error> {
|
|
|
+ pub async fn store(
|
|
|
+ &self,
|
|
|
+ transaction: Transaction,
|
|
|
+ update_token: Option<Vec<u8>>,
|
|
|
+ ) -> Result<Transaction, Error> {
|
|
|
transaction.validate()?;
|
|
|
|
|
|
+ let update_token = if let Some(update_token) = update_token {
|
|
|
+ Some(self.token_manager.verify(&transaction.id, &update_token)?)
|
|
|
+ } else {
|
|
|
+ None
|
|
|
+ };
|
|
|
+
|
|
|
let mut batch = self.config.storage.begin().await?;
|
|
|
+ let stored_update_token = batch.consume_update_token(&transaction.id).await?;
|
|
|
+ if stored_update_token != update_token {
|
|
|
+ return Err(Error::ValidUpdateTokenRequired);
|
|
|
+ }
|
|
|
if transaction.revision.previous.is_none() {
|
|
|
Self::store_base_transaction(&transaction, &mut batch).await?;
|
|
|
}
|
|
@@ -365,10 +390,13 @@ where
|
|
|
) -> Result<Transaction, Error> {
|
|
|
let (change_transaction, payments) = self.select_payments_from_accounts(from).await?;
|
|
|
if let Some(change_tx) = change_transaction {
|
|
|
- self.store(change_tx).await?;
|
|
|
+ self.store(change_tx, None).await?;
|
|
|
}
|
|
|
- self.store(Transaction::new(reference, status, Type::Transaction, payments, to).await?)
|
|
|
- .await
|
|
|
+ self.store(
|
|
|
+ Transaction::new(reference, status, Type::Transaction, payments, to).await?,
|
|
|
+ None,
|
|
|
+ )
|
|
|
+ .await
|
|
|
}
|
|
|
|
|
|
/// Return the balances from a given account
|
|
@@ -395,12 +423,15 @@ where
|
|
|
tags: Vec<Tag>,
|
|
|
reference: String,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
- self.store(Transaction::new_external_deposit(
|
|
|
- reference,
|
|
|
- status,
|
|
|
- tags,
|
|
|
- vec![(account.clone(), amount)],
|
|
|
- )?)
|
|
|
+ self.store(
|
|
|
+ Transaction::new_external_deposit(
|
|
|
+ reference,
|
|
|
+ status,
|
|
|
+ tags,
|
|
|
+ vec![(account.clone(), amount)],
|
|
|
+ )?,
|
|
|
+ None,
|
|
|
+ )
|
|
|
.await
|
|
|
}
|
|
|
|
|
@@ -421,11 +452,12 @@ where
|
|
|
.select_payments_from_accounts(vec![(account.clone(), amount)])
|
|
|
.await?;
|
|
|
for change_tx in change_transactions.into_iter() {
|
|
|
- self.store(change_tx).await?;
|
|
|
+ self.store(change_tx, None).await?;
|
|
|
}
|
|
|
- self.store(Transaction::new_external_withdrawal(
|
|
|
- reference, status, payments,
|
|
|
- )?)
|
|
|
+ self.store(
|
|
|
+ Transaction::new_external_withdrawal(reference, status, payments)?,
|
|
|
+ None,
|
|
|
+ )
|
|
|
.await
|
|
|
}
|
|
|
|
|
@@ -481,6 +513,7 @@ where
|
|
|
.pop()
|
|
|
.ok_or(Error::TxNotFound)?
|
|
|
.set_tags(tags, reason)?,
|
|
|
+ None,
|
|
|
)
|
|
|
.await
|
|
|
}
|
|
@@ -506,6 +539,7 @@ where
|
|
|
.pop()
|
|
|
.ok_or(Error::TxNotFound)?
|
|
|
.change_status(&self.config, new_status, reason)?,
|
|
|
+ None,
|
|
|
)
|
|
|
.await
|
|
|
}
|