use std::future; use std::iter::Take; use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::Arc; use tokio_retry::{Retry, RetryIf}; #[tokio::test] async fn attempts_just_once() { use std::iter::empty; let counter = Arc::new(AtomicUsize::new(0)); let cloned_counter = counter.clone(); let future = Retry::spawn(empty(), move || { cloned_counter.fetch_add(1, Ordering::SeqCst); future::ready(Err::<(), u64>(42)) }); let res = future.await; assert_eq!(res, Err(42)); assert_eq!(counter.load(Ordering::SeqCst), 1); } #[tokio::test] async fn attempts_until_max_retries_exceeded() { use tokio_retry::strategy::FixedInterval; let s = FixedInterval::from_millis(100).take(2); let counter = Arc::new(AtomicUsize::new(0)); let cloned_counter = counter.clone(); let future = Retry::spawn(s, move || { cloned_counter.fetch_add(1, Ordering::SeqCst); future::ready(Err::<(), u64>(42)) }); let res = future.await; assert_eq!(res, Err(42)); assert_eq!(counter.load(Ordering::SeqCst), 3); } #[tokio::test] async fn attempts_until_success() { use tokio_retry::strategy::FixedInterval; let s = FixedInterval::from_millis(100); let counter = Arc::new(AtomicUsize::new(0)); let cloned_counter = counter.clone(); let future = Retry::spawn(s, move || { let previous = cloned_counter.fetch_add(1, Ordering::SeqCst); if previous < 3 { future::ready(Err::<(), u64>(42)) } else { future::ready(Ok::<(), u64>(())) } }); let res = future.await; assert_eq!(res, Ok(())); assert_eq!(counter.load(Ordering::SeqCst), 4); } #[tokio::test] async fn compatible_with_tokio_core() { use tokio_retry::strategy::FixedInterval; let s = FixedInterval::from_millis(100); let counter = Arc::new(AtomicUsize::new(0)); let cloned_counter = counter.clone(); let future = Retry::spawn(s, move || { let previous = cloned_counter.fetch_add(1, Ordering::SeqCst); if previous < 3 { future::ready(Err::<(), u64>(42)) } else { future::ready(Ok::<(), u64>(())) } }); let res = future.await; assert_eq!(res, Ok(())); assert_eq!(counter.load(Ordering::SeqCst), 4); } #[tokio::test] async fn attempts_retry_only_if_given_condition_is_true() { use tokio_retry::strategy::FixedInterval; let s = FixedInterval::from_millis(100).take(5); let counter = Arc::new(AtomicUsize::new(0)); let cloned_counter = counter.clone(); let future: RetryIf, _, fn(&usize) -> _> = RetryIf::spawn( s, move || { let previous = cloned_counter.fetch_add(1, Ordering::SeqCst); future::ready(Err::<(), usize>(previous + 1)) }, |e: &usize| *e < 3, ); let res = future.await; assert_eq!(res, Err(3)); assert_eq!(counter.load(Ordering::SeqCst), 3); }