mod tests_utils;
use crate::tests_utils::DURATION_10_MS;
use crate::tests_utils::checker::State;
use crate::tests_utils::test_channel::ChannelState;
use crate::tests_utils::test_runtime::block_on;
use rx_rust::disposable::Disposable;
use rx_rust::disposable::subscription::Subscription;
use rx_rust::operators::connectable::ref_count::RefCount;
use rx_rust::operators::creating::empty::Empty;
use rx_rust::operators::creating::throw::Throw;
use rx_rust::safe_lock_option;
use rx_rust::safe_lock_option_disposable;
use rx_rust::scheduler::Scheduler;
use rx_rust::subject::Subject;
use rx_rust::subject::behavior_subject::BehaviorSubject;
use rx_rust::utils::types::{Mutable, Shared};
use rx_rust::{
observable::{Observable, observable_ext::ObservableExt},
observer::{Observer, Termination, boxed_observer::BoxedObserver},
operators::creating::create::Create,
subject::publish_subject::PublishSubject,
};
use std::convert::Infallible;
use tests_utils::{checker::Checker, test_channel::test_channel, test_struct::TestStruct};
#[test]
fn test_completed() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Completed);
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Completed);
assert_eq!(channel_checker.state(), ChannelState::Completed);
}
#[test]
fn test_error() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error("error"));
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
}
#[test]
fn test_unsubscribe() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2, 3]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_2.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2, 3]);
assert_eq!(checker_2.state(), State::Dropped);
assert_eq!(channel_checker.state(), ChannelState::Unsubscribed);
}
#[test]
fn test_ref() {
let value_1 = 111;
let value_2 = 222;
let error = -1;
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
match counter {
1 => &value_1,
2 => &value_2,
_ => panic!(),
}
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [&value_1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [&value_1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [&value_1, &value_2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [&value_2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [&value_1, &value_2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [&value_2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error(&error));
assert_eq!(checker_1.values(), [&value_1, &value_2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [&value_2]);
assert_eq!(checker_2.state(), State::Error(&error));
assert_eq!(channel_checker.state(), ChannelState::Error(&error));
}
#[test]
fn test_async() {
block_on(|runtime| async move {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(move |_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = runtime
.spawn(async move { observable_1.subscribe(observer_1) })
.await
.unwrap();
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let mut sender = runtime
.spawn(async move {
sender.on_next(());
sender
})
.await
.unwrap();
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let _subscription_2 = runtime
.spawn(async move { observable_2.subscribe(observer_2) })
.await
.unwrap();
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let sender = runtime
.spawn(async move {
sender.on_next(());
sender
})
.await
.unwrap();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
runtime
.spawn(async move { subscription_1.dispose() })
.await
.unwrap();
runtime.sleep(DURATION_10_MS).await;
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
runtime
.spawn(async move { sender.on_termination(Termination::Error("error")) })
.await
.unwrap();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
});
}
#[test]
fn test_subscribe_by_different_observer() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let (on_next, on_termination) = observer_2.into_callbacks();
let _subscription_2 = observable_2.subscribe_with_callback(on_next, on_termination);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error("error"));
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
}
#[test]
fn test_unsub_on_next_by_take() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker, observer) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count()
.take(1);
let observable_1 = observable.clone();
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let _subscription = observable_1.subscribe(observer);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker.values(), [1]);
assert_eq!(checker.state(), State::Completed);
assert_eq!(channel_checker.state(), ChannelState::Unsubscribed);
}
#[test]
fn test_multiple_operation() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count()
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error("error"));
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
}
#[test]
fn test_without_convenient_api() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish();
let observable = RefCount::new(observable);
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription_1 = observable_1.subscribe(observer_1);
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);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error("error"));
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
}
#[test]
fn test_complete_on_next() {
let mut counter = 0;
let mut subject = PublishSubject::default();
let (checker, observer) = Checker::new();
let observable = subject.clone();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let _subscription = observable.clone().subscribe(observer);
let subject_cloned = subject.clone();
let _subscription = observable.subscribe_with_callback(
move |_| {
subject_cloned
.clone()
.on_termination(Termination::Completed);
},
move |_| {},
);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
subject.on_next(());
assert_eq!(checker.values(), [1]);
assert_eq!(checker.state(), State::Completed);
subject
.clone()
.on_termination(Termination::<Infallible>::Completed);
assert_eq!(checker.values(), [1]);
assert_eq!(checker.state(), State::Completed);
}
#[test]
fn test_error_on_next() {
let mut counter = 0;
let mut subject = PublishSubject::default();
let (checker, observer) = Checker::new();
let observable = subject.clone();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let _subscription = observable.clone().subscribe(observer);
let subject_cloned = subject.clone();
let _subscription = observable.subscribe_with_callback(
move |_| {
subject_cloned
.clone()
.on_termination(Termination::Error("error"));
},
move |_| {},
);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
subject.on_next(());
assert_eq!(checker.values(), [1]);
assert_eq!(checker.state(), State::Error("error"));
subject.clone().on_termination(Termination::Completed);
assert_eq!(checker.values(), [1]);
assert_eq!(checker.state(), State::Error("error"));
}
#[test]
fn test_unsub_on_next() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let (checker_3, observer_3) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
let observable_3 = observable;
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);
assert!(checker_3.values().is_empty());
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let subscription = observable_1.subscribe(observer_1);
let sub = Shared::new(Mutable::new(None::<Subscription<'static>>));
let sub_cloned = sub.clone();
safe_lock_option!(replace: sub,
observable_2
.hook_on_next(move |observer, value| {
safe_lock_option_disposable!(dispose: sub_cloned);
observer.on_next(value);
})
.subscribe(observer_2)
);
let sub = Shared::new(Mutable::new(None::<Subscription<'static>>));
let sub_cloned = sub.clone();
safe_lock_option!(replace: sub,
observable_3
.hook_on_next(move |observer, value| {
observer.on_next(value);
safe_lock_option_disposable!(dispose: sub_cloned);
})
.subscribe(observer_3)
);
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);
assert!(checker_3.values().is_empty());
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [1]);
assert_eq!(checker_2.state(), State::Dropped);
assert_eq!(checker_3.values(), [1]);
assert_eq!(checker_3.state(), State::Dropped);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
subscription.dispose();
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [1]);
assert_eq!(checker_2.state(), State::Dropped);
assert_eq!(checker_3.values(), [1]);
assert_eq!(checker_3.state(), State::Dropped);
assert_eq!(channel_checker.state(), ChannelState::Unsubscribed);
}
#[test]
fn test_sub_on_next() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let (checker_3, observer_3) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let mut observer_2 = Some(observer_2);
let mut observer_3 = Some(observer_3);
let subscription_2 = Shared::new(Mutable::new(None));
let subscription_3 = Shared::new(Mutable::new(None));
let subscription_2_cloned = subscription_2.clone();
let subscription_3_cloned = subscription_3.clone();
let _subscription = Some(
observable
.clone()
.hook_on_next(move |observer, value| {
if let Some(observer) = observer_2.take() {
safe_lock_option!(replace: subscription_2_cloned, observable.clone().subscribe(observer));
}
observer.on_next(value);
if let Some(observer) = observer_3.take() {
safe_lock_option!(replace: subscription_3_cloned, observable.clone().subscribe(observer));
}
})
.subscribe(observer_1),
);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [2]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), [2]);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
}
#[test]
fn test_next_on_next() {
let mut counter = 0;
let mut subject = PublishSubject::default();
let (checker, observer) = Checker::new();
let observable = subject.clone();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let _subscription = observable.clone().subscribe(observer);
let mut subject_cloned = subject.clone();
let _subscription = observable.subscribe_with_callback(
move |value| {
assert!(subject_cloned.terminated().is_none());
if value < 3 {
subject_cloned.on_next(());
}
subject_cloned
.clone()
.on_termination(Termination::Completed);
},
move |_| {},
);
assert!(checker.values().is_empty());
assert_eq!(checker.state(), State::Active);
subject.on_next(());
assert_eq!(checker.values(), [1, 2, 3]);
assert_eq!(checker.state(), State::Completed);
subject
.clone()
.on_termination(Termination::<Infallible>::Completed);
assert_eq!(checker.values(), [1, 2, 3]);
assert_eq!(checker.state(), State::Completed);
}
#[test]
fn test_unsub_on_completed() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let (checker_3, observer_3) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
let observable_3 = observable;
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);
assert!(checker_3.values().is_empty());
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let _subscription = observable_1.subscribe(observer_1);
let sub = Shared::new(Mutable::new(None::<Subscription<'static>>));
let sub_cloned = sub.clone();
safe_lock_option!(replace: sub,
observable_2
.hook_on_termination(move |observer, value| {
safe_lock_option_disposable!(dispose: sub_cloned);
observer.on_termination(value);
})
.subscribe(observer_2)
);
let sub = Shared::new(Mutable::new(None::<Subscription<'static>>));
let sub_cloned = sub.clone();
safe_lock_option!(replace: sub,
observable_3
.hook_on_termination(move |observer, value| {
observer.on_termination(value);
safe_lock_option_disposable!(dispose: sub_cloned);
})
.subscribe(observer_3)
);
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);
assert!(checker_3.values().is_empty());
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [1]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), [1]);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Completed);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Completed);
assert_eq!(checker_2.values(), [1]);
assert_eq!(checker_2.state(), State::Completed);
assert_eq!(checker_3.values(), [1]);
assert_eq!(checker_3.state(), State::Completed);
assert_eq!(channel_checker.state(), ChannelState::Completed);
}
#[test]
fn test_sub_on_completed() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel::<'_, _, Infallible>();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let (checker_3, observer_3) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let mut observer_2 = Some(observer_2);
let mut observer_3 = Some(observer_3);
let subscription_2 = Shared::new(Mutable::new(None));
let subscription_3 = Shared::new(Mutable::new(None));
let subscription_2_cloned = subscription_2.clone();
let subscription_3_cloned = subscription_3.clone();
let _subscription = Some(
observable
.clone()
.hook_on_termination(move |observer, value| {
if let Some(observer) = observer_2.take() {
safe_lock_option!(replace: subscription_2_cloned, observable.clone().subscribe(observer));
}
observer.on_termination(value);
if let Some(observer) = observer_3.take() {
safe_lock_option!(replace: subscription_3_cloned, observable.clone().subscribe(observer));
}
})
.subscribe(observer_1),
);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Completed);
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Completed);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Completed);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Completed);
assert_eq!(channel_checker.state(), ChannelState::Completed);
}
#[test]
fn test_unsub_on_error() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let (checker_3, observer_3) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
let observable_3 = observable;
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);
assert!(checker_3.values().is_empty());
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Initialized);
let _subscription = observable_1.subscribe(observer_1);
let sub = Shared::new(Mutable::new(None::<Subscription<'static>>));
let sub_cloned = sub.clone();
safe_lock_option!(replace: sub,
observable_2
.hook_on_termination(move |observer, value| {
safe_lock_option_disposable!(dispose: sub_cloned);
observer.on_termination(value);
})
.subscribe(observer_2)
);
let sub = Shared::new(Mutable::new(None::<Subscription<'static>>));
let sub_cloned = sub.clone();
safe_lock_option!(replace: sub,
observable_3
.hook_on_termination(move |observer, value| {
observer.on_termination(value);
safe_lock_option_disposable!(dispose: sub_cloned);
})
.subscribe(observer_3)
);
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);
assert!(checker_3.values().is_empty());
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [1]);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), [1]);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error("error"));
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Error("error"));
assert_eq!(checker_2.values(), [1]);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(checker_3.values(), [1]);
assert_eq!(checker_3.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
}
#[test]
fn test_sub_on_error() {
let mut counter = 0;
let (mut sender, observable, channel_checker) = test_channel();
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let (checker_3, observer_3) = Checker::new();
let observable = observable
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let mut observer_2 = Some(observer_2);
let mut observer_3 = Some(observer_3);
let subscription_2 = Shared::new(Mutable::new(None));
let subscription_3 = Shared::new(Mutable::new(None));
let subscription_2_cloned = subscription_2.clone();
let subscription_3_cloned = subscription_3.clone();
let _subscription = Some(
observable
.clone()
.hook_on_termination(move |observer, value| {
if let Some(observer) = observer_2.take() {
safe_lock_option!(replace: subscription_2_cloned, observable.clone().subscribe(observer));
}
observer.on_termination(value);
if let Some(observer) = observer_3.take() {
safe_lock_option!(replace: subscription_3_cloned, observable.clone().subscribe(observer));
}
})
.subscribe(observer_1),
);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Active);
assert_eq!(channel_checker.state(), ChannelState::Subscribed);
sender.on_termination(Termination::Error("error"));
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Error("error"));
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Error("error"));
assert_eq!(checker_3.values(), []);
assert_eq!(checker_3.state(), State::Error("error"));
assert_eq!(channel_checker.state(), ChannelState::Error("error"));
}
#[test]
fn test_immediate_next() {
let mut counter = 0;
let mut subject = BehaviorSubject::new(());
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = subject
.clone()
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
let subscription_1 = observable_1.subscribe(observer_1);
assert_eq!(checker_1.values(), [1]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
subject.on_next(());
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), [1, 2]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Active);
subject.on_next(());
assert_eq!(checker_1.values(), [1, 2, 3]);
assert_eq!(checker_1.state(), State::Active);
assert_eq!(checker_2.values(), [3]);
assert_eq!(checker_2.state(), State::Active);
subscription_1.dispose();
assert_eq!(checker_1.values(), [1, 2, 3]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [3]);
assert_eq!(checker_2.state(), State::Active);
subject.on_termination(Termination::<Infallible>::Completed);
assert_eq!(checker_1.values(), [1, 2, 3]);
assert_eq!(checker_1.state(), State::Dropped);
assert_eq!(checker_2.values(), [3]);
assert_eq!(checker_2.state(), State::Completed);
}
#[test]
fn test_immediate_completed() {
let mut counter = 0;
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = Empty
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
let _subscription_1 = observable_1.subscribe(observer_1);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Completed);
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Completed);
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Completed);
}
#[test]
fn test_immediate_error() {
let mut counter = 0;
let (checker_1, observer_1) = Checker::new();
let (checker_2, observer_2) = Checker::new();
let observable = Throw::new("error")
.map(|_| {
counter += 1;
counter
})
.publish()
.ref_count();
let observable_1 = observable.clone();
let observable_2 = observable.clone();
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);
let _subscription_1 = observable_1.subscribe(observer_1);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Error("error"));
assert!(checker_2.values().is_empty());
assert_eq!(checker_2.state(), State::Active);
let _subscription_2 = observable_2.subscribe(observer_2);
assert_eq!(checker_1.values(), []);
assert_eq!(checker_1.state(), State::Error("error"));
assert_eq!(checker_2.values(), []);
assert_eq!(checker_2.state(), State::Error("error"));
}
#[test]
fn test_lifetime_sub() {
let life_marker = TestStruct;
let _subscription;
{
let observable = Create::new(|mut observer| {
observer.on_next(111);
Subscription::new_with_disposal_callback(|| {
life_marker.consume_ref();
})
});
let observable = observable.publish().ref_count();
let (_, observer) = Checker::<_, ()>::new();
_subscription = observable.subscribe(observer);
}
}
#[test]
fn test_lifetime_or() {
let life_marker_2 = TestStruct;
let mut life_marker = None;
{
let observable = Create::new(|observer| {
life_marker = Some(observer);
Subscription::default()
});
let observable = observable.publish().ref_count();
let (_, mut observer) = Checker::<_, Infallible>::new();
observer.on_next(vec![&life_marker_2]);
let _subscription = observable.subscribe(observer);
}
}
#[test]
fn test_lifetime_or_sub() {
let life_marker = TestStruct;
let _subscription;
{
let (_, mut observer) = Checker::<_, Infallible>::new();
observer.on_next(&life_marker);
let observable =
Create::new(|_: BoxedObserver<'_, &TestStruct, Infallible>| Subscription::default());
let observable = observable.publish().ref_count();
_subscription = observable.subscribe(observer);
}
}
#[test]
fn test_clone() {
let observable = Create::new(|mut observer| {
observer.on_next(TestStruct);
observer.on_termination(Termination::Error(TestStruct));
Subscription::default()
});
let observable = observable.publish().ref_count();
_ = observable.clone(); }
#[test]
fn test_type_inference_with_subscribe() {
let (_, observable, _) = test_channel::<'_, i32, String>();
let observable = observable.publish().ref_count();
let observable = observable.filter(|_| true);
let (_, observer) = Checker::new();
observable.subscribe(observer);
}
#[test]
fn test_type_inference_without_subscribe() {
let (_, observable, _) = test_channel::<'_, i32, String>();
let observable = observable.publish().ref_count();
observable.filter(|_| true);
}