|
@@ -4,8 +4,8 @@ use crate::{
|
|
|
config::Config,
|
|
|
status::{InternalStatus, StatusManager},
|
|
|
storage::{AccountTransactionType, Batch, ReceivedPaymentStatus, Storage},
|
|
|
- transaction::Error as TxError,
|
|
|
- transaction::Type,
|
|
|
+ token::TokenPayload,
|
|
|
+ transaction::{Error as TxError, Type},
|
|
|
worker::WorkerManager,
|
|
|
AccountId, Amount, Error, Filter, PaymentFrom, PaymentId, RevId, Status, Tag, Transaction,
|
|
|
TxId,
|
|
@@ -258,54 +258,92 @@ where
|
|
|
pub async fn store(&self, transaction: Transaction) -> Result<Transaction, Error> {
|
|
|
transaction.validate()?;
|
|
|
|
|
|
+ let is_updating_status = 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
|
|
|
+ let previous = self
|
|
|
+ .config
|
|
|
+ .storage
|
|
|
+ .find(Filter {
|
|
|
+ revisions: vec![previous.clone()],
|
|
|
+ ..Default::default()
|
|
|
+ })
|
|
|
+ .await?
|
|
|
+ .pop()
|
|
|
+ .ok_or(Error::TxNotFound)?;
|
|
|
+
|
|
|
+ if let Some(lock_token) = previous.revision.locked.as_ref() {
|
|
|
+ self.config
|
|
|
+ .token_manager
|
|
|
+ .verify(lock_token.to_owned(), &transaction.revision.update_token)?
|
|
|
+ }
|
|
|
+
|
|
|
+ if previous.revision.status != transaction.revision.status {
|
|
|
+ self.config
|
|
|
+ .status
|
|
|
+ .is_valid_transition(&previous.revision.status, &transaction.revision.status)?;
|
|
|
+ true
|
|
|
+ } else {
|
|
|
+ false
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ true
|
|
|
+ };
|
|
|
+
|
|
|
let mut batch = self.config.storage.begin().await?;
|
|
|
if transaction.revision.previous.is_none() {
|
|
|
Self::store_base_transaction(&transaction, &mut batch).await?;
|
|
|
}
|
|
|
|
|
|
- let (created_updated, spent_updated) = match self
|
|
|
- .config
|
|
|
- .status
|
|
|
- .internal_type(&transaction.revision.status)
|
|
|
- {
|
|
|
- InternalStatus::Reverted => {
|
|
|
- batch
|
|
|
- .update_transaction_payments(
|
|
|
- &transaction.id,
|
|
|
- ReceivedPaymentStatus::Failed,
|
|
|
- ReceivedPaymentStatus::Spendable,
|
|
|
- )
|
|
|
- .await?
|
|
|
+ if is_updating_status {
|
|
|
+ let (created_updated, spent_updated) = match self
|
|
|
+ .config
|
|
|
+ .status
|
|
|
+ .internal_type(&transaction.revision.status)
|
|
|
+ {
|
|
|
+ InternalStatus::Reverted => {
|
|
|
+ batch
|
|
|
+ .update_transaction_payments(
|
|
|
+ &transaction.id,
|
|
|
+ ReceivedPaymentStatus::Failed,
|
|
|
+ ReceivedPaymentStatus::Spendable,
|
|
|
+ )
|
|
|
+ .await?
|
|
|
+ }
|
|
|
+ InternalStatus::Spendable => {
|
|
|
+ batch
|
|
|
+ .update_transaction_payments(
|
|
|
+ &transaction.id,
|
|
|
+ ReceivedPaymentStatus::Spendable,
|
|
|
+ ReceivedPaymentStatus::Spent,
|
|
|
+ )
|
|
|
+ .await?
|
|
|
+ }
|
|
|
+ _ => (transaction.creates.len(), transaction.spends.len()),
|
|
|
+ };
|
|
|
+
|
|
|
+ if transaction.creates.len() != created_updated
|
|
|
+ || transaction.spends.len() != spent_updated
|
|
|
+ {
|
|
|
+ return Err(Error::Transaction(TxError::NoUpdate));
|
|
|
}
|
|
|
- InternalStatus::Spendable => {
|
|
|
+
|
|
|
+ if self
|
|
|
+ .config
|
|
|
+ .status
|
|
|
+ .is_spendable(&transaction.revision.status)
|
|
|
+ {
|
|
|
batch
|
|
|
.update_transaction_payments(
|
|
|
&transaction.id,
|
|
|
ReceivedPaymentStatus::Spendable,
|
|
|
ReceivedPaymentStatus::Spent,
|
|
|
)
|
|
|
- .await?
|
|
|
+ .await?;
|
|
|
}
|
|
|
- _ => (transaction.creates.len(), transaction.spends.len()),
|
|
|
- };
|
|
|
-
|
|
|
- if transaction.creates.len() != created_updated || transaction.spends.len() != spent_updated
|
|
|
- {
|
|
|
- return Err(Error::Transaction(TxError::NoUpdate));
|
|
|
- }
|
|
|
-
|
|
|
- if self
|
|
|
- .config
|
|
|
- .status
|
|
|
- .is_spendable(&transaction.revision.status)
|
|
|
- {
|
|
|
- batch
|
|
|
- .update_transaction_payments(
|
|
|
- &transaction.id,
|
|
|
- ReceivedPaymentStatus::Spendable,
|
|
|
- ReceivedPaymentStatus::Spent,
|
|
|
- )
|
|
|
- .await?;
|
|
|
}
|
|
|
|
|
|
batch
|
|
@@ -461,12 +499,38 @@ where
|
|
|
&self.config.status
|
|
|
}
|
|
|
|
|
|
+ /// Locks the transaction and returns a token that can be used to unlock it.
|
|
|
+ ///
|
|
|
+ /// Locked transactions cannot be updated without the TokenPayload or until it expires.
|
|
|
+ pub async fn lock_transaction(
|
|
|
+ &self,
|
|
|
+ transaction_id: TxId,
|
|
|
+ owner: String,
|
|
|
+ ) -> Result<(Transaction, TokenPayload), 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(owner, &self.config.token_manager)?;
|
|
|
+ Ok((self.store(new_revision).await?, 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<TokenPayload>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
let filter = Filter {
|
|
|
revisions: vec![revision_id],
|
|
@@ -480,7 +544,7 @@ where
|
|
|
.await?
|
|
|
.pop()
|
|
|
.ok_or(Error::TxNotFound)?
|
|
|
- .set_tags(tags, reason)?,
|
|
|
+ .set_tags(tags, reason, update_token)?,
|
|
|
)
|
|
|
.await
|
|
|
}
|
|
@@ -492,6 +556,7 @@ where
|
|
|
revision_id: RevId,
|
|
|
new_status: Status,
|
|
|
reason: String,
|
|
|
+ update_token: Option<TokenPayload>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
let filter = Filter {
|
|
|
revisions: vec![revision_id],
|
|
@@ -505,7 +570,7 @@ where
|
|
|
.await?
|
|
|
.pop()
|
|
|
.ok_or(Error::TxNotFound)?
|
|
|
- .change_status(&self.config, new_status, reason)?,
|
|
|
+ .change_status(&self.config, new_status, reason, update_token)?,
|
|
|
)
|
|
|
.await
|
|
|
}
|