#include "../test.h" #include SCENARIO("switch_on_error stops on completion", "[switch_on_error][on_error_resume_next][operators]"){ GIVEN("a test hot observable of ints"){ auto sc = rxsc::make_test(); auto w = sc.create_worker(); const rxsc::test::messages on; long invoked = 0; auto xs = sc.make_hot_observable({ on.next(180, 1), on.next(210, 2), on.next(240, 3), on.next(290, 4), on.next(350, 5), on.completed(400), on.next(410, -1), on.completed(420), on.error(430, std::runtime_error("error on unsubscribed stream")) }); auto ys = sc.make_cold_observable({ on.next(10, -1), on.completed(20), }); WHEN("passed through unchanged"){ auto res = w.start( [xs, ys, &invoked]() { return xs .switch_on_error([ys, &invoked](rxu::error_ptr) { invoked++; return ys; }) // forget type to workaround lambda deduction bug on msvc 2013 .as_dynamic(); } ); THEN("the output stops on completion"){ auto required = rxu::to_vector({ on.next(210, 2), on.next(240, 3), on.next(290, 4), on.next(350, 5), on.completed(400) }); auto actual = res.get_observer().messages(); REQUIRE(required == actual); } THEN("there was one xs subscription and one unsubscription"){ auto required = rxu::to_vector({ on.subscribe(200, 400) }); auto actual = xs.subscriptions(); REQUIRE(required == actual); } THEN("there was no ys subscription"){ auto required = std::vector(); auto actual = ys.subscriptions(); REQUIRE(required == actual); } THEN("switch_on_error selector was not called"){ REQUIRE(0 == invoked); } } } } SCENARIO("on_error_resume_next stops on completion", "[on_error_resume_next][operators]"){ GIVEN("a test hot observable of ints"){ auto sc = rxsc::make_test(); auto w = sc.create_worker(); const rxsc::test::messages on; long invoked = 0; auto xs = sc.make_hot_observable({ on.next(180, 1), on.next(210, 2), on.next(240, 3), on.next(290, 4), on.next(350, 5), on.completed(400), on.next(410, -1), on.completed(420), on.error(430, std::runtime_error("error on unsubscribed stream")) }); auto ys = sc.make_cold_observable({ on.next(10, -1), on.completed(20), }); WHEN("passed through unchanged"){ auto res = w.start( [xs, ys, &invoked]() { return xs .on_error_resume_next([ys, &invoked](rxu::error_ptr) { invoked++; return ys; }) // forget type to workaround lambda deduction bug on msvc 2013 .as_dynamic(); } ); THEN("the output stops on completion"){ auto required = rxu::to_vector({ on.next(210, 2), on.next(240, 3), on.next(290, 4), on.next(350, 5), on.completed(400) }); auto actual = res.get_observer().messages(); REQUIRE(required == actual); } THEN("there was one xs subscription and one unsubscription"){ auto required = rxu::to_vector({ on.subscribe(200, 400) }); auto actual = xs.subscriptions(); REQUIRE(required == actual); } THEN("there was no ys subscription"){ auto required = std::vector(); auto actual = ys.subscriptions(); REQUIRE(required == actual); } THEN("on_error_resume_next selector was not called"){ REQUIRE(0 == invoked); } } } } SCENARIO("on_error_resume_next stops on error", "[on_error_resume_next][operators]"){ GIVEN("a test hot observable of ints"){ auto sc = rxsc::make_test(); auto w = sc.create_worker(); const rxsc::test::messages on; std::runtime_error ex("on_error_resume_next on_error from source"); long invoked = 0; auto xs = sc.make_hot_observable({ on.next(180, 1), on.next(210, 2), on.next(240, 3), on.next(290, 4), on.error(300, ex), on.next(350, 5), on.completed(400), on.next(410, -1), on.completed(420), on.error(430, std::runtime_error("error on unsubscribed stream")) }); auto ys = sc.make_cold_observable({ on.next(10, -1), on.completed(20), }); WHEN("are resumed after an error"){ auto res = w.start( [xs, ys, &invoked]() { return xs .on_error_resume_next([ys, &invoked](rxu::error_ptr) { invoked++; return ys; }) // forget type to workaround lambda deduction bug on msvc 2013 .as_dynamic(); } ); THEN("the output stops on completion"){ auto required = rxu::to_vector({ on.next(210, 2), on.next(240, 3), on.next(290, 4), on.next(310, -1), on.completed(320) }); auto actual = res.get_observer().messages(); REQUIRE(required == actual); } THEN("there was one xs subscription and one unsubscription"){ auto required = rxu::to_vector({ on.subscribe(200, 300) }); auto actual = xs.subscriptions(); REQUIRE(required == actual); } THEN("there was one ys subscription and one unsubscription"){ auto required = rxu::to_vector({ on.subscribe(300, 320) }); auto actual = ys.subscriptions(); REQUIRE(required == actual); } THEN("on_error_resume_next selector was called once"){ REQUIRE(1 == invoked); } } } }