|
@@ -23,8 +23,6 @@ use std::collections::HashMap;
|
|
|
/// security that its content was not altered.
|
|
|
#[derive(Debug, Clone, Deserialize, Serialize)]
|
|
|
pub struct Revision {
|
|
|
- /// A pointer to the first revision of the transaction.
|
|
|
- first: Option<RevisionId>,
|
|
|
/// Any previous transaction that this transaction is replacing.
|
|
|
previous: Option<RevisionId>,
|
|
|
/// A human-readable description of the transaction changes.
|
|
@@ -161,26 +159,18 @@ impl Revision {
|
|
|
pub struct Transaction {
|
|
|
/// The RevisionId is the RevisionID of the first revision of the transaction.
|
|
|
pub id: RevisionId,
|
|
|
+
|
|
|
/// Current Revision ID.
|
|
|
- pub current: RevisionId,
|
|
|
+ pub revision_id: RevisionId,
|
|
|
+
|
|
|
+ /// Latest revision of this transaction
|
|
|
+ pub lastest_revision: RevisionId,
|
|
|
+
|
|
|
/// The transaction inner details
|
|
|
#[serde(flatten)]
|
|
|
inner: Revision,
|
|
|
}
|
|
|
|
|
|
-impl TryFrom<Revision> for Transaction {
|
|
|
- type Error = Error;
|
|
|
-
|
|
|
- fn try_from(inner: Revision) -> Result<Self, Self::Error> {
|
|
|
- let id = inner.calculate_id()?;
|
|
|
- Ok(Transaction {
|
|
|
- id: inner.first.clone().unwrap_or_else(|| id.clone()),
|
|
|
- current: id,
|
|
|
- inner,
|
|
|
- })
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
impl Transaction {
|
|
|
/// Creates a new external deposit transaction
|
|
|
///
|
|
@@ -192,52 +182,25 @@ impl Transaction {
|
|
|
status: Status,
|
|
|
pay_to: Vec<(AccountId, Amount)>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
- Revision {
|
|
|
- first: None,
|
|
|
- changelog: "".to_owned(),
|
|
|
- previous: None,
|
|
|
- spends: vec![],
|
|
|
- creates: pay_to
|
|
|
- .into_iter()
|
|
|
- .map(|(to, amount)| PaymentTo { to, amount })
|
|
|
- .collect(),
|
|
|
- reference,
|
|
|
- typ: Type::Deposit,
|
|
|
- tags: Vec::new(),
|
|
|
- status,
|
|
|
- created_at: Utc::now(),
|
|
|
- updated_at: Utc::now(),
|
|
|
- }
|
|
|
- .try_into()
|
|
|
- }
|
|
|
-
|
|
|
- /// Returns the previous revision of this transaction, if any
|
|
|
- pub fn previous(&self) -> Option<RevisionId> {
|
|
|
- self.inner.previous.clone()
|
|
|
- }
|
|
|
-
|
|
|
- /// Returns a unique list of accounts involved in this transaction.
|
|
|
- ///
|
|
|
- /// Accounts are sorted and unique, and they include the accounts that spent and that receives
|
|
|
- pub fn accounts(&self) -> Vec<AccountId> {
|
|
|
- let mut accounts = self
|
|
|
- .inner
|
|
|
- .creates
|
|
|
- .iter()
|
|
|
- .map(|x| x.to.clone())
|
|
|
- .collect::<Vec<_>>();
|
|
|
-
|
|
|
- accounts.extend(
|
|
|
- self.inner
|
|
|
- .spends
|
|
|
- .iter()
|
|
|
- .map(|x| x.from.clone())
|
|
|
- .collect::<Vec<_>>(),
|
|
|
- );
|
|
|
-
|
|
|
- accounts.sort();
|
|
|
- accounts.dedup();
|
|
|
- accounts
|
|
|
+ Self::from_revision(
|
|
|
+ Revision {
|
|
|
+ changelog: "".to_owned(),
|
|
|
+ previous: None,
|
|
|
+ spends: vec![],
|
|
|
+ creates: pay_to
|
|
|
+ .into_iter()
|
|
|
+ .map(|(to, amount)| PaymentTo { to, amount })
|
|
|
+ .collect(),
|
|
|
+ reference,
|
|
|
+ typ: Type::Deposit,
|
|
|
+ tags: Vec::new(),
|
|
|
+ status,
|
|
|
+ created_at: Utc::now(),
|
|
|
+ updated_at: Utc::now(),
|
|
|
+ },
|
|
|
+ None,
|
|
|
+ None,
|
|
|
+ )
|
|
|
}
|
|
|
|
|
|
/// Creates a new external withdrawal transaction
|
|
@@ -248,25 +211,22 @@ impl Transaction {
|
|
|
status: Status,
|
|
|
spend: Vec<PaymentFrom>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
- Revision {
|
|
|
- first: None,
|
|
|
- changelog: "".to_owned(),
|
|
|
- previous: None,
|
|
|
- spends: spend,
|
|
|
- creates: vec![],
|
|
|
- reference,
|
|
|
- typ: Type::Withdrawal,
|
|
|
- tags: Vec::new(),
|
|
|
- status,
|
|
|
- created_at: Utc::now(),
|
|
|
- updated_at: Utc::now(),
|
|
|
- }
|
|
|
- .try_into()
|
|
|
- }
|
|
|
-
|
|
|
- /// Gets the inner transaction
|
|
|
- pub fn inner(&self) -> &Revision {
|
|
|
- &self.inner
|
|
|
+ Self::from_revision(
|
|
|
+ Revision {
|
|
|
+ changelog: "".to_owned(),
|
|
|
+ previous: None,
|
|
|
+ spends: spend,
|
|
|
+ creates: vec![],
|
|
|
+ reference,
|
|
|
+ typ: Type::Withdrawal,
|
|
|
+ tags: Vec::new(),
|
|
|
+ status,
|
|
|
+ created_at: Utc::now(),
|
|
|
+ updated_at: Utc::now(),
|
|
|
+ },
|
|
|
+ None,
|
|
|
+ None,
|
|
|
+ )
|
|
|
}
|
|
|
|
|
|
/// Creates a new transaction
|
|
@@ -287,20 +247,74 @@ impl Transaction {
|
|
|
.map(|(to, amount)| PaymentTo { to, amount })
|
|
|
.collect();
|
|
|
|
|
|
- Revision {
|
|
|
- first: None,
|
|
|
- changelog: "".to_owned(),
|
|
|
- previous: None,
|
|
|
- spends,
|
|
|
- creates: create,
|
|
|
- reference,
|
|
|
- typ,
|
|
|
- tags: Vec::new(),
|
|
|
- status,
|
|
|
- created_at: Utc::now(),
|
|
|
- updated_at: Utc::now(),
|
|
|
- }
|
|
|
- .try_into()
|
|
|
+ Self::from_revision(
|
|
|
+ Revision {
|
|
|
+ changelog: "".to_owned(),
|
|
|
+ previous: None,
|
|
|
+ spends,
|
|
|
+ creates: create,
|
|
|
+ reference,
|
|
|
+ typ,
|
|
|
+ tags: Vec::new(),
|
|
|
+ status,
|
|
|
+ created_at: Utc::now(),
|
|
|
+ updated_at: Utc::now(),
|
|
|
+ },
|
|
|
+ None,
|
|
|
+ None,
|
|
|
+ )
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Creates a new transaction object from a given revision
|
|
|
+ pub fn from_revision(
|
|
|
+ revision: Revision,
|
|
|
+ first_revision: Option<RevisionId>,
|
|
|
+ lastest_revision: Option<RevisionId>,
|
|
|
+ ) -> Result<Self, Error> {
|
|
|
+ let revision_id = revision.calculate_id()?;
|
|
|
+ let first_revision = first_revision.unwrap_or_else(|| revision_id.clone());
|
|
|
+ let lastest_revision = lastest_revision.unwrap_or_else(|| revision_id.clone());
|
|
|
+
|
|
|
+ Ok(Transaction {
|
|
|
+ id: first_revision,
|
|
|
+ revision_id,
|
|
|
+ lastest_revision,
|
|
|
+ inner: revision,
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Gets the inner transaction
|
|
|
+ pub fn inner(&self) -> &Revision {
|
|
|
+ &self.inner
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Returns the previous revision of this transaction, if any
|
|
|
+ pub fn previous(&self) -> Option<RevisionId> {
|
|
|
+ self.inner.previous.clone()
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Returns a unique list of accounts involved in this transaction.
|
|
|
+ ///
|
|
|
+ /// Accounts are sorted and unique, and they include the accounts that spent and that receives
|
|
|
+ pub fn accounts(&self) -> Vec<AccountId> {
|
|
|
+ let mut accounts = self
|
|
|
+ .inner
|
|
|
+ .creates
|
|
|
+ .iter()
|
|
|
+ .map(|x| x.to.clone())
|
|
|
+ .collect::<Vec<_>>();
|
|
|
+
|
|
|
+ accounts.extend(
|
|
|
+ self.inner
|
|
|
+ .spends
|
|
|
+ .iter()
|
|
|
+ .map(|x| x.from.clone())
|
|
|
+ .collect::<Vec<_>>(),
|
|
|
+ );
|
|
|
+
|
|
|
+ accounts.sort();
|
|
|
+ accounts.dedup();
|
|
|
+ accounts
|
|
|
}
|
|
|
|
|
|
/// Prepares a transaction ammend to update its status.
|
|
@@ -324,13 +338,10 @@ impl Transaction {
|
|
|
.is_valid_transition(&self.inner.status, &new_status)?;
|
|
|
let mut inner = self.inner;
|
|
|
inner.changelog = reason;
|
|
|
- if inner.first.is_none() {
|
|
|
- inner.first = Some(self.id);
|
|
|
- }
|
|
|
inner.updated_at = Utc::now();
|
|
|
- inner.previous = Some(self.current);
|
|
|
+ inner.previous = Some(self.revision_id);
|
|
|
inner.status = new_status;
|
|
|
- let mut x: Transaction = inner.try_into()?;
|
|
|
+ let mut x: Transaction = Transaction::from_revision(inner, Some(self.id), None)?;
|
|
|
x.persist(config).await?;
|
|
|
Ok(x)
|
|
|
}
|
|
@@ -339,7 +350,7 @@ impl Transaction {
|
|
|
pub(crate) fn validate(&self) -> Result<(), Error> {
|
|
|
let calculated_revision_id = self.inner.calculate_id()?;
|
|
|
|
|
|
- if self.current != calculated_revision_id
|
|
|
+ if self.revision_id != calculated_revision_id
|
|
|
|| (self.inner.previous.is_none() && self.id != calculated_revision_id)
|
|
|
{
|
|
|
return Err(Error::InvalidRevisionId(
|
|
@@ -404,13 +415,12 @@ impl Transaction {
|
|
|
|
|
|
if let Some(previous_id) = &self.inner.previous {
|
|
|
// Make sure this update is updating the last revision and the status is not final
|
|
|
- let current_transaction = batch
|
|
|
- .get_transaction_with_revision(&self.id, previous_id)
|
|
|
- .await?;
|
|
|
+ let current_transaction = batch.get_and_lock_transaction(&self.id).await?;
|
|
|
|
|
|
if config.status.is_final(¤t_transaction.inner.status)
|
|
|
|| self.inner.transaction_fingerprint()?
|
|
|
!= current_transaction.inner.transaction_fingerprint()?
|
|
|
+ || *previous_id != current_transaction.lastest_revision
|
|
|
{
|
|
|
return Err(Error::TransactionUpdatesNotAllowed);
|
|
|
}
|