mod tests_utils;
use crate::tests_utils::DURATION_10_MS;
use crate::tests_utils::DURATION_30_MS;
use crate::tests_utils::DURATION_100_MS;
use crate::tests_utils::checker::State;
use crate::tests_utils::test_runtime::block_on;
use rx_rust::disposable::Disposable;
use rx_rust::disposable::subscription::Subscription;
use rx_rust::safe_lock_option;
use rx_rust::safe_lock_option_disposable;
use rx_rust::scheduler::Scheduler;
use rx_rust::utils::types::{Mutable, Shared};
use rx_rust::{
observable::{Observable, observable_ext::ObservableExt},
operators::creating::interval::Interval,
};
use tests_utils::checker::Checker;
#[test]
fn test_completed_no_delay() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), None);
let (checker, observer) = Checker::new();
let subscription = observable.subscribe(observer);
check_with_spawned_late!(
runtime,
{
assert_eq!(checker.values(), []);
assert_eq!(checker.state(), State::Active);
},
{
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Active);
}
);
runtime.sleep(DURATION_30_MS).await;
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1]);
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Active);
subscription.dispose();
check_with_abort_late!(
runtime,
{
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Active);
},
{
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
}
);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
});
}
#[test]
fn test_completed_with_delay() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let (checker, observer) = Checker::new();
let subscription = observable.subscribe(observer);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_30_MS).await;
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1]);
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Active);
subscription.dispose();
check_with_abort_late!(
runtime,
{
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Active);
},
{
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
}
);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
});
}
#[test]
fn test_unsubscribe() {
block_on(|runtime| async move {
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let observable_1 = observable;
let observable_2 = observable_1.clone();
let subscription_1 = observable_1.subscribe(observer_1);
let _subscription_2 = observable_2.subscribe(observer_2);
assert!(checker_1.values().is_empty());
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_30_MS).await;
assert!(checker_1.values().is_empty());
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0]);
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0, 1]);
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Active);
subscription_1.dispose();
check_with_abort_late!(
runtime,
{
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Active);
},
{
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Active);
}
);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [0, 1, 2, 3]);
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [0, 1, 2, 3, 4]);
assert_eq!(checker_2.state(), State::Active);
});
}
#[test]
fn test_async() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let (checker, observer) = Checker::new();
let subscription = runtime
.spawn(async move { observable.subscribe(observer) })
.await
.unwrap();
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_30_MS).await;
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1]);
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Active);
runtime
.spawn(async { subscription.dispose() })
.await
.unwrap();
runtime.sleep(DURATION_10_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0, 1, 2]);
assert_eq!(checker.state(), State::Dropped);
});
}
#[test]
fn test_subscribe_by_different_observer() {
block_on(|runtime| async move {
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let observable_1 = observable;
let observable_2 = observable_1.clone();
let subscription_1 = observable_1.subscribe(observer_1);
let (on_next, on_termination) = observer_2.into_callbacks();
let subscription_2 = observable_2.subscribe_with_callback(on_next, on_termination);
assert!(checker_1.values().is_empty());
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_30_MS).await;
assert!(checker_1.values().is_empty());
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0]);
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0, 1]);
assert_eq!(checker_2.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Active);
subscription_1.dispose();
subscription_2.dispose();
check_with_abort_late!(
runtime,
{
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Active);
},
{
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Dropped);
}
);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Dropped);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker_1.values(), [0, 1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [0, 1, 2]);
assert_eq!(checker_2.state(), State::Dropped);
});
}
#[test]
fn test_unsub_on_next_by_take() {
block_on(|runtime| async move {
let observable =
Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS)).take(1);
let (checker, observer) = Checker::new();
let _subscription = observable.subscribe(observer);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_30_MS).await;
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Completed);
});
}
#[test]
fn test_unsub_after_next() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let (checker, observer) = Checker::new();
let subscription = Shared::new(Mutable::new(None::<Subscription<'_>>));
let subscription_cloned = subscription.clone();
let (mut on_next, on_termination) = observer.into_callbacks();
safe_lock_option!(replace: subscription, observable.subscribe_with_callback(
move |value| {
on_next(value);
safe_lock_option_disposable!(dispose: subscription_cloned);
},
|termination| {
on_termination(termination);
},
));
assert_eq!(checker.values(), []);
assert_eq!(checker.state(), State::Active);
assert!(safe_lock_option!(is_some: subscription));
runtime.sleep(DURATION_100_MS - DURATION_30_MS).await;
assert_eq!(checker.values(), []);
assert_eq!(checker.state(), State::Active);
assert!(safe_lock_option!(is_some: subscription));
runtime.sleep(DURATION_30_MS * 2).await;
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Dropped);
assert!(safe_lock_option!(is_none: subscription));
});
}
#[test]
fn test_undisposed_scheduler() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let (checker, observer) = Checker::new();
let _subscription = observable.subscribe(observer);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
});
}
#[test]
fn test_scheduler_should_be_disposed_after_completed() {
block_on(|runtime| async move {
let observable =
Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS)).take(1);
let (checker, observer) = Checker::new();
assert_eq!(runtime.get_alive_tasks_count(), 0);
let _subscription = observable.subscribe(observer);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
assert_eq!(runtime.get_alive_tasks_count(), 1);
runtime.sleep(DURATION_30_MS).await;
assert_eq!(checker.values(), []);
assert_eq!(checker.state(), State::Active);
assert_eq!(runtime.get_alive_tasks_count(), 1);
runtime.sleep(DURATION_100_MS).await;
assert_eq!(checker.values(), [0]);
assert_eq!(checker.state(), State::Completed);
assert_eq!(runtime.get_alive_tasks_count(), 0);
});
}
#[test]
fn test_scheduler_should_be_disposed_after_unsub() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let (checker, observer) = Checker::new();
assert_eq!(runtime.get_alive_tasks_count(), 0);
let subscription = observable.subscribe(observer);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
assert_eq!(runtime.get_alive_tasks_count(), 1);
subscription.dispose();
assert_eq!(runtime.get_alive_tasks_count(), 0);
runtime.sleep(DURATION_10_MS).await;
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Dropped);
assert_eq!(runtime.get_alive_tasks_count(), 0);
});
}
#[test]
fn test_clone() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
_ = observable.clone();
});
}
#[test]
fn test_type_inference_with_subscribe() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
let observable = observable.filter(|_| true);
let (_, observer) = Checker::new();
observable.subscribe(observer);
});
}
#[test]
fn test_type_inference_without_subscribe() {
block_on(|runtime| async move {
let observable = Interval::new(DURATION_100_MS, runtime.clone(), Some(DURATION_100_MS));
observable.filter(|_| true);
});
}