|
@@ -4,13 +4,13 @@ use crate::{
|
|
|
config::Config,
|
|
|
status::{InternalStatus, StatusManager},
|
|
|
storage::{AccountTransactionType, Batch, ReceivedPaymentStatus, Storage},
|
|
|
- token::{Token, TokenManager},
|
|
|
+ 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, time::Duration};
|
|
|
+use std::{cmp::Ordering, collections::HashMap, sync::Arc};
|
|
|
use tokio::sync::mpsc::{self, Receiver, Sender};
|
|
|
|
|
|
/// The Verax ledger
|
|
@@ -250,15 +250,6 @@ 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
|
|
@@ -266,24 +257,34 @@ 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,
|
|
|
- update_token: Option<Vec<u8>>,
|
|
|
- ) -> Result<Transaction, Error> {
|
|
|
+ pub async fn store(&self, transaction: Transaction) -> 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
|
|
|
- };
|
|
|
+ if let Some(previous) = &transaction.revision.previous {
|
|
|
+ // Although this operation to check the previous version is being performed outside of
|
|
|
+ // the writer batch, it is safe to do so because the previous version is
|
|
|
+ // already stored in the storage layer, and the batch will make sure
|
|
|
+ // the previous version is the current revision, or else the entire operation will be
|
|
|
+ // revered
|
|
|
+ if let Some(lock_token) = self
|
|
|
+ .config
|
|
|
+ .storage
|
|
|
+ .find(Filter {
|
|
|
+ revisions: vec![previous.clone()],
|
|
|
+ ..Default::default()
|
|
|
+ })
|
|
|
+ .await?
|
|
|
+ .pop()
|
|
|
+ .ok_or(Error::TxNotFound)?
|
|
|
+ .revision
|
|
|
+ .locked
|
|
|
+ {
|
|
|
+ self.token_manager
|
|
|
+ .verify(lock_token, &transaction.revision.update_token)?
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
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?;
|
|
|
}
|
|
@@ -390,13 +391,10 @@ 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, None).await?;
|
|
|
+ self.store(change_tx).await?;
|
|
|
}
|
|
|
- self.store(
|
|
|
- Transaction::new(reference, status, Type::Transaction, payments, to).await?,
|
|
|
- None,
|
|
|
- )
|
|
|
- .await
|
|
|
+ self.store(Transaction::new(reference, status, Type::Transaction, payments, to).await?)
|
|
|
+ .await
|
|
|
}
|
|
|
|
|
|
/// Return the balances from a given account
|
|
@@ -423,15 +421,12 @@ where
|
|
|
tags: Vec<Tag>,
|
|
|
reference: String,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
- self.store(
|
|
|
- Transaction::new_external_deposit(
|
|
|
- reference,
|
|
|
- status,
|
|
|
- tags,
|
|
|
- vec![(account.clone(), amount)],
|
|
|
- )?,
|
|
|
- None,
|
|
|
- )
|
|
|
+ self.store(Transaction::new_external_deposit(
|
|
|
+ reference,
|
|
|
+ status,
|
|
|
+ tags,
|
|
|
+ vec![(account.clone(), amount)],
|
|
|
+ )?)
|
|
|
.await
|
|
|
}
|
|
|
|
|
@@ -452,12 +447,11 @@ where
|
|
|
.select_payments_from_accounts(vec![(account.clone(), amount)])
|
|
|
.await?;
|
|
|
for change_tx in change_transactions.into_iter() {
|
|
|
- self.store(change_tx, None).await?;
|
|
|
+ self.store(change_tx).await?;
|
|
|
}
|
|
|
- self.store(
|
|
|
- Transaction::new_external_withdrawal(reference, status, payments)?,
|
|
|
- None,
|
|
|
- )
|
|
|
+ self.store(Transaction::new_external_withdrawal(
|
|
|
+ reference, status, payments,
|
|
|
+ )?)
|
|
|
.await
|
|
|
}
|
|
|
|
|
@@ -493,12 +487,33 @@ where
|
|
|
&self.config.status
|
|
|
}
|
|
|
|
|
|
+ /// TODO
|
|
|
+ pub async fn lock_transaction(&self, transaction_id: TxId) -> Result<Vec<u8>, Error> {
|
|
|
+ let filter = Filter {
|
|
|
+ ids: vec![transaction_id.clone()],
|
|
|
+ limit: 1,
|
|
|
+ ..Default::default()
|
|
|
+ };
|
|
|
+
|
|
|
+ let (new_revision, secret) = self
|
|
|
+ .config
|
|
|
+ .storage
|
|
|
+ .find(filter)
|
|
|
+ .await?
|
|
|
+ .pop()
|
|
|
+ .ok_or(Error::TxNotFound)?
|
|
|
+ .lock_transaction(&self.token_manager)?;
|
|
|
+ self.store(new_revision).await?;
|
|
|
+ Ok(secret)
|
|
|
+ }
|
|
|
+
|
|
|
/// Updates a transaction and updates their tags to this given set
|
|
|
pub async fn set_tags(
|
|
|
&self,
|
|
|
revision_id: RevId,
|
|
|
tags: Vec<Tag>,
|
|
|
reason: String,
|
|
|
+ update_token: Option<Vec<u8>>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
let filter = Filter {
|
|
|
revisions: vec![revision_id],
|
|
@@ -512,8 +527,7 @@ where
|
|
|
.await?
|
|
|
.pop()
|
|
|
.ok_or(Error::TxNotFound)?
|
|
|
- .set_tags(tags, reason)?,
|
|
|
- None,
|
|
|
+ .set_tags(tags, reason, update_token)?,
|
|
|
)
|
|
|
.await
|
|
|
}
|
|
@@ -525,6 +539,7 @@ where
|
|
|
revision_id: RevId,
|
|
|
new_status: Status,
|
|
|
reason: String,
|
|
|
+ update_token: Option<Vec<u8>>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
let filter = Filter {
|
|
|
revisions: vec![revision_id],
|
|
@@ -538,8 +553,7 @@ where
|
|
|
.await?
|
|
|
.pop()
|
|
|
.ok_or(Error::TxNotFound)?
|
|
|
- .change_status(&self.config, new_status, reason)?,
|
|
|
- None,
|
|
|
+ .change_status(&self.config, new_status, reason, update_token)?,
|
|
|
)
|
|
|
.await
|
|
|
}
|