|
@@ -4,6 +4,10 @@ use crate::{
|
|
|
Tag, Transaction, TxId,
|
|
|
};
|
|
|
use std::{cmp::Ordering, collections::HashMap, sync::Arc};
|
|
|
+use tokio::sync::{
|
|
|
+ mpsc::{self, Receiver, Sender},
|
|
|
+ RwLock,
|
|
|
+};
|
|
|
|
|
|
/// The Verax ledger
|
|
|
#[derive(Debug)]
|
|
@@ -11,18 +15,8 @@ pub struct Ledger<S>
|
|
|
where
|
|
|
S: Storage + Sync + Send,
|
|
|
{
|
|
|
- config: Arc<Config<S>>,
|
|
|
-}
|
|
|
-
|
|
|
-impl<S> Clone for Ledger<S>
|
|
|
-where
|
|
|
- S: Storage + Sync + Send,
|
|
|
-{
|
|
|
- fn clone(&self) -> Self {
|
|
|
- Self {
|
|
|
- config: self.config.clone(),
|
|
|
- }
|
|
|
- }
|
|
|
+ config: Config<S>,
|
|
|
+ listeners: RwLock<HashMap<Filter, Vec<Sender<Transaction>>>>,
|
|
|
}
|
|
|
|
|
|
impl<S> Ledger<S>
|
|
@@ -30,10 +24,26 @@ where
|
|
|
S: Storage + Sync + Send,
|
|
|
{
|
|
|
/// Creates a new ledger instance
|
|
|
- pub fn new(config: Config<S>) -> Self {
|
|
|
- Self {
|
|
|
- config: Arc::new(config),
|
|
|
+ pub fn new(config: Config<S>) -> Arc<Self> {
|
|
|
+ Arc::new(Self {
|
|
|
+ config,
|
|
|
+ listeners: RwLock::new(HashMap::new()),
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Creates a new subscription for future transactions
|
|
|
+ pub async fn subscribe(&self, filter: Filter) -> Receiver<Transaction> {
|
|
|
+ let (sender, receiver) = mpsc::channel(100);
|
|
|
+ let mut listeners = self.listeners.write().await;
|
|
|
+ let filter = filter.prepare();
|
|
|
+
|
|
|
+ if let Some(previous_values) = listeners.get_mut(&filter) {
|
|
|
+ previous_values.push(sender);
|
|
|
+ } else {
|
|
|
+ listeners.insert(filter, vec![sender]);
|
|
|
}
|
|
|
+
|
|
|
+ receiver
|
|
|
}
|
|
|
|
|
|
/// The internal usage is to select unspent payments for each account to
|
|
@@ -48,9 +58,9 @@ where
|
|
|
///
|
|
|
/// This function returns a vector of payments to be used as inputs and
|
|
|
/// optionally a dependent transaction to be executed first. This
|
|
|
- /// transaction is an internal transaction and it settles immediately. It is
|
|
|
- /// used to split an existing payment into two payments, one to be used as
|
|
|
- /// input and the other to be used as change. This is done to avoid locking
|
|
|
+ /// transaction is an internal transaction and it settles immediately. It
|
|
|
+ /// isutxo/src/storage/cache/mod.rs used to split an existing payment into two payments, one
|
|
|
+ /// to be used as input and the other to be used as change. This is done to avoid locking
|
|
|
/// any change amount until the main transaction settles.
|
|
|
async fn select_payments_from_accounts(
|
|
|
&self,
|
|
@@ -172,6 +182,32 @@ where
|
|
|
Ok((exchange_tx, payments))
|
|
|
}
|
|
|
|
|
|
+ /// TODO: Move the whole logic of persisting the transaction and the revision into this layer,
|
|
|
+ /// instead of having them at the transaction layer
|
|
|
+ async fn persist(&self, mut transaction: Transaction) -> Result<Transaction, Error> {
|
|
|
+ transaction.persist(&self.config).await?;
|
|
|
+
|
|
|
+ let listeners = self.listeners.read().await;
|
|
|
+ for (filter, listeners) in listeners.iter() {
|
|
|
+ println!(
|
|
|
+ "filter {:#?} -> {:#?} -> {}",
|
|
|
+ filter,
|
|
|
+ transaction.revision.tags,
|
|
|
+ filter.matches(&transaction.transaction, &transaction.revision)
|
|
|
+ );
|
|
|
+ if filter.matches(&transaction.transaction, &transaction.revision) {
|
|
|
+ for listener in listeners.iter() {
|
|
|
+ listener
|
|
|
+ .send(transaction.clone())
|
|
|
+ .await
|
|
|
+ .expect("listener failed");
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ Ok(transaction)
|
|
|
+ }
|
|
|
+
|
|
|
/// Creates a new transaction and returns it.
|
|
|
///
|
|
|
/// The input is pretty simple, take this amounts from these given accounts
|
|
@@ -200,13 +236,11 @@ where
|
|
|
to: Vec<(AccountId, Amount)>,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
let (change_transaction, payments) = self.select_payments_from_accounts(from).await?;
|
|
|
- if let Some(mut change_tx) = change_transaction {
|
|
|
- change_tx.persist(&self.config).await?;
|
|
|
+ if let Some(change_tx) = change_transaction {
|
|
|
+ self.persist(change_tx).await?;
|
|
|
}
|
|
|
- let mut transaction =
|
|
|
- Transaction::new(reference, status, Type::Transaction, payments, to).await?;
|
|
|
- transaction.persist(&self.config).await?;
|
|
|
- Ok(transaction)
|
|
|
+ self.persist(Transaction::new(reference, status, Type::Transaction, payments, to).await?)
|
|
|
+ .await
|
|
|
}
|
|
|
|
|
|
/// Return the balances from a given account
|
|
@@ -232,10 +266,12 @@ where
|
|
|
status: Status,
|
|
|
reference: String,
|
|
|
) -> Result<Transaction, Error> {
|
|
|
- let mut transaction =
|
|
|
- Transaction::new_external_deposit(reference, status, vec![(account.clone(), amount)])?;
|
|
|
- transaction.persist(&self.config).await?;
|
|
|
- Ok(transaction)
|
|
|
+ self.persist(Transaction::new_external_deposit(
|
|
|
+ reference,
|
|
|
+ status,
|
|
|
+ vec![(account.clone(), amount)],
|
|
|
+ )?)
|
|
|
+ .await
|
|
|
}
|
|
|
|
|
|
/// Creates a new withdrawal transaction and returns it.
|
|
@@ -254,12 +290,13 @@ where
|
|
|
let (change_transactions, payments) = self
|
|
|
.select_payments_from_accounts(vec![(account.clone(), amount)])
|
|
|
.await?;
|
|
|
- for mut change_tx in change_transactions.into_iter() {
|
|
|
- change_tx.persist(&self.config).await?;
|
|
|
+ for change_tx in change_transactions.into_iter() {
|
|
|
+ self.persist(change_tx).await?;
|
|
|
}
|
|
|
- let mut transaction = Transaction::new_external_withdrawal(reference, status, payments)?;
|
|
|
- transaction.persist(&self.config).await?;
|
|
|
- Ok(transaction)
|
|
|
+ self.persist(Transaction::new_external_withdrawal(
|
|
|
+ reference, status, payments,
|
|
|
+ )?)
|
|
|
+ .await
|
|
|
}
|
|
|
|
|
|
/// Returns the payment object by a given payment id
|
|
@@ -306,15 +343,16 @@ where
|
|
|
limit: 1,
|
|
|
..Default::default()
|
|
|
};
|
|
|
- Ok(self
|
|
|
- .config
|
|
|
- .storage
|
|
|
- .find(filter)
|
|
|
- .await?
|
|
|
- .pop()
|
|
|
- .ok_or(Error::TxNotFound)?
|
|
|
- .set_tags(&self.config, tags, reason)
|
|
|
- .await?)
|
|
|
+ self.persist(
|
|
|
+ self.config
|
|
|
+ .storage
|
|
|
+ .find(filter)
|
|
|
+ .await?
|
|
|
+ .pop()
|
|
|
+ .ok_or(Error::TxNotFound)?
|
|
|
+ .set_tags(tags, reason)?,
|
|
|
+ )
|
|
|
+ .await
|
|
|
}
|
|
|
|
|
|
/// Attempts to change the status of a given transaction id. On success the
|