use super::*;
use crate::{
service::test_helpers::{
MockP2P,
TestContextBuilder,
},
test_helpers::TEST_COIN_AMOUNT,
};
use fuel_core_services::Service;
use fuel_core_storage::rand::{
prelude::StdRng,
SeedableRng,
};
use fuel_core_types::fuel_tx::{
field::Inputs,
AssetId,
Transaction,
TransactionBuilder,
UniqueIdentifier,
};
use std::{
ops::Deref,
time::Duration,
};
#[tokio::test]
async fn can_insert_from_p2p() {
let mut ctx_builder = TestContextBuilder::new();
let tx1 = ctx_builder.setup_script_tx(10);
let p2p = MockP2P::new_with_txs(vec![tx1.clone()]);
ctx_builder.with_p2p(p2p);
let ctx = ctx_builder.build();
let service = ctx.service();
let mut receiver = service
.shared
.tx_update_subscribe(tx1.id(&Default::default()))
.unwrap();
service.start_and_await().await.unwrap();
let res = receiver.next().await;
assert!(matches!(
res,
Some(TxStatusMessage::Status(TransactionStatus::Submitted { .. }))
));
let out = service.shared.find(vec![tx1.id(&Default::default())]);
let got_tx: Transaction = out[0].as_ref().unwrap().tx().clone().deref().into();
assert_eq!(tx1, got_tx);
}
#[tokio::test]
async fn insert_from_local_broadcasts_to_p2p() {
let mut ctx_builder = TestContextBuilder::new();
let tx1 = ctx_builder.setup_script_tx(10);
let mut p2p = MockP2P::new_with_txs(vec![]);
let mock_tx1 = tx1.clone();
p2p.expect_broadcast_transaction()
.withf(move |receive: &Arc<Transaction>| **receive == mock_tx1)
.times(1)
.returning(|_| Ok(()));
ctx_builder.with_p2p(p2p);
let ctx = ctx_builder.build_and_start().await;
let service = ctx.service();
let mut new_tx_notification = service.shared.new_tx_notification_subscribe();
let mut subscribe_update = service
.shared
.tx_update_subscribe(tx1.cached_id().unwrap())
.unwrap();
let out = service.shared.insert(vec![Arc::new(tx1.clone())]).await;
if out[0].is_ok() {
assert_eq!(
new_tx_notification.try_recv(),
Ok(tx1.cached_id().unwrap()),
"First added should be tx1"
);
let update = subscribe_update.next().await;
assert!(
matches!(
update,
Some(TxStatusMessage::Status(TransactionStatus::Submitted { .. }))
),
"Got {:?}",
update
);
} else {
panic!("Tx1 should be OK, got err");
}
}
#[tokio::test]
async fn test_insert_from_p2p_does_not_broadcast_to_p2p() {
let mut ctx_builder = TestContextBuilder::new();
let tx1 = ctx_builder.setup_script_tx(10);
let txs = vec![tx1.clone()];
let mut p2p = MockP2P::new_with_txs(txs);
let (send, mut receive) = broadcast::channel::<()>(1);
p2p.expect_broadcast_transaction().returning(move |_| {
send.send(()).unwrap();
Ok(())
});
ctx_builder.with_p2p(p2p);
let ctx = ctx_builder.build();
let service = ctx.service();
let mut receiver = service
.shared
.tx_update_subscribe(tx1.id(&Default::default()))
.unwrap();
service.start_and_await().await.unwrap();
let res = receiver.next().await;
assert!(matches!(
res,
Some(TxStatusMessage::Status(TransactionStatus::Submitted { .. }))
));
let not_broadcast =
tokio::time::timeout(Duration::from_millis(100), receive.recv()).await;
assert!(
not_broadcast.is_err(),
"expected a timeout because no broadcast should have occurred"
)
}
#[tokio::test]
async fn test_gossipped_transaction_with_check_error_rejected() {
let mut ctx_builder = TestContextBuilder::new();
let mut tx1 = ctx_builder.setup_script_tx(10);
let script = tx1.as_script_mut().unwrap();
let input = script.inputs()[0].clone();
script.inputs_mut().push(input);
let txs = vec![tx1.clone()];
let mut p2p = MockP2P::new_with_txs(txs);
let (send, mut receive) = broadcast::channel::<()>(1);
p2p.expect_notify_gossip_transaction_validity()
.returning(move |_, validity| {
assert_eq!(validity, GossipsubMessageAcceptance::Reject);
send.send(()).unwrap();
Ok(())
});
ctx_builder.with_p2p(p2p);
let ctx = ctx_builder.build();
let service = ctx.service();
service.start_and_await().await.unwrap();
let gossip_validity_notified =
tokio::time::timeout(Duration::from_millis(100), receive.recv()).await;
assert!(
gossip_validity_notified.is_ok(),
"expected to receive gossip validity notification"
)
}
#[tokio::test]
async fn test_gossipped_mint_rejected() {
let tx1 = TransactionBuilder::mint(
0u32.into(),
0,
Default::default(),
Default::default(),
1,
AssetId::BASE,
Default::default(),
)
.finalize_as_transaction();
let txs = vec![tx1.clone()];
let mut p2p = MockP2P::new_with_txs(txs);
let (send, mut receive) = broadcast::channel::<()>(1);
p2p.expect_notify_gossip_transaction_validity()
.returning(move |_, validity| {
assert_eq!(validity, GossipsubMessageAcceptance::Reject);
send.send(()).unwrap();
Ok(())
});
let mut ctx_builder = TestContextBuilder::new();
ctx_builder.with_p2p(p2p);
let ctx = ctx_builder.build();
let service = ctx.service();
service.start_and_await().await.unwrap();
let gossip_validity_notified =
tokio::time::timeout(Duration::from_millis(100), receive.recv()).await;
assert!(
gossip_validity_notified.is_ok(),
"expected to receive gossip validity notification"
)
}
#[tokio::test]
async fn test_gossipped_transaction_with_transient_error_ignored() {
let mut rng = StdRng::seed_from_u64(100);
let mut ctx_builder = TestContextBuilder::new();
let mut tx1 = ctx_builder.setup_script_tx(10);
let script = tx1.as_script_mut().unwrap();
script.inputs_mut()[0] = crate::test_helpers::random_predicate(
&mut rng,
AssetId::BASE,
TEST_COIN_AMOUNT,
None,
);
let txs = vec![tx1.clone()];
let mut p2p = MockP2P::new_with_txs(txs);
let (send, mut receive) = broadcast::channel::<()>(1);
p2p.expect_notify_gossip_transaction_validity()
.returning(move |_, validity| {
assert_eq!(validity, GossipsubMessageAcceptance::Ignore);
send.send(()).unwrap();
Ok(())
});
ctx_builder.with_p2p(p2p);
let ctx = ctx_builder.build();
let service = ctx.service();
service.start_and_await().await.unwrap();
let gossip_validity_notified =
tokio::time::timeout(Duration::from_millis(100), receive.recv()).await;
assert!(
gossip_validity_notified.is_ok(),
"expected to receive gossip validity notification"
)
}