|
@@ -135,6 +135,10 @@ pub enum Error {
|
|
/// Error with the encoding/decoding
|
|
/// Error with the encoding/decoding
|
|
Encoding(String),
|
|
Encoding(String),
|
|
|
|
|
|
|
|
+ #[error("Transaction already exists: {0}")]
|
|
|
|
+ /// The transaction already exists. Error thrown when the replay protection is already stored
|
|
|
|
+ AlreadyExists(TxId),
|
|
|
|
+
|
|
#[error("Not enough unspent payments (missing {0} cents)")]
|
|
#[error("Not enough unspent payments (missing {0} cents)")]
|
|
/// TODO: Convert the AmountCents to Amount for better error reporting upstream
|
|
/// TODO: Convert the AmountCents to Amount for better error reporting upstream
|
|
NotEnoughUnspentPayments(AmountCents),
|
|
NotEnoughUnspentPayments(AmountCents),
|
|
@@ -160,6 +164,13 @@ pub trait Batch<'a> {
|
|
status: ReceivedPaymentStatus,
|
|
status: ReceivedPaymentStatus,
|
|
) -> Result<(), Error>;
|
|
) -> Result<(), Error>;
|
|
|
|
|
|
|
|
+ /// Stores the replay protection. It fails if the protection is already stored.
|
|
|
|
+ async fn store_replay_protection(
|
|
|
|
+ &mut self,
|
|
|
|
+ protection: &str,
|
|
|
|
+ transaction_id: &TxId,
|
|
|
|
+ ) -> Result<(), Error>;
|
|
|
|
+
|
|
/// Create a new list of payments
|
|
/// Create a new list of payments
|
|
async fn create_payments(
|
|
async fn create_payments(
|
|
&mut self,
|
|
&mut self,
|
|
@@ -375,7 +386,55 @@ pub mod test {
|
|
$crate::storage_unit_test!(not_spendable_new_payments_not_spendable);
|
|
$crate::storage_unit_test!(not_spendable_new_payments_not_spendable);
|
|
$crate::storage_unit_test!(subscribe_realtime);
|
|
$crate::storage_unit_test!(subscribe_realtime);
|
|
$crate::storage_unit_test!(transaction_locking);
|
|
$crate::storage_unit_test!(transaction_locking);
|
|
|
|
+ $crate::storage_unit_test!(transaction_replay_protection);
|
|
|
|
+ };
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ pub async fn transaction_replay_protection<T>(storage: T)
|
|
|
|
+ where
|
|
|
|
+ T: Storage + Send + Sync,
|
|
|
|
+ {
|
|
|
|
+ let config = Config {
|
|
|
|
+ storage,
|
|
|
|
+ token_manager: Default::default(),
|
|
|
|
+ status: StatusManager::default(),
|
|
};
|
|
};
|
|
|
|
+
|
|
|
|
+ let ledger = Ledger::new(config);
|
|
|
|
+
|
|
|
|
+ let asset: Asset = "USD/2".parse().expect("valid asset");
|
|
|
|
+ let deposit = Transaction::new_external_deposit(
|
|
|
|
+ "test reference".to_owned(),
|
|
|
|
+ "pending".into(),
|
|
|
|
+ vec![],
|
|
|
|
+ vec![(
|
|
|
|
+ "alice".parse().expect("account"),
|
|
|
|
+ asset.from_human("100.99").expect("valid amount"),
|
|
|
|
+ )],
|
|
|
|
+ )
|
|
|
|
+ .expect("valid tx")
|
|
|
|
+ .set_replay_protection("test".to_owned())
|
|
|
|
+ .expect("valid tx");
|
|
|
|
+
|
|
|
|
+ assert!(ledger.store(deposit).await.is_ok());
|
|
|
|
+
|
|
|
|
+ let deposit = Transaction::new_external_deposit(
|
|
|
|
+ "test reference".to_owned(),
|
|
|
|
+ "pending".into(),
|
|
|
|
+ vec![],
|
|
|
|
+ vec![(
|
|
|
|
+ "alice".parse().expect("account"),
|
|
|
|
+ asset.from_human("100.99").expect("valid amount"),
|
|
|
|
+ )],
|
|
|
|
+ )
|
|
|
|
+ .expect("valid tx")
|
|
|
|
+ .set_replay_protection("test".to_owned())
|
|
|
|
+ .expect("valid tx");
|
|
|
|
+
|
|
|
|
+ let result = ledger.store(deposit).await;
|
|
|
|
+
|
|
|
|
+ assert!(result.is_err());
|
|
|
|
+ assert!(result.unwrap_err().to_string().contains("already exists"));
|
|
}
|
|
}
|
|
|
|
|
|
pub async fn transaction_locking<T>(storage: T)
|
|
pub async fn transaction_locking<T>(storage: T)
|