|
3 | 3 | #include <folly/Baton.h>
|
4 | 4 | #include <gtest/gtest.h>
|
5 | 5 | #include <atomic>
|
6 |
| -#include "Tuple.h" |
| 6 | + |
7 | 7 | #include "yarpl/Observable.h"
|
| 8 | +#include "yarpl/Observables.h" |
8 | 9 | #include "yarpl/ThreadScheduler.h"
|
9 | 10 | #include "yarpl/flowable/Subscriber.h"
|
10 | 11 | #include "yarpl/flowable/Subscribers.h"
|
11 | 12 | #include "yarpl/observable/Observers.h"
|
12 | 13 | #include "yarpl/observable/Subscriptions.h"
|
13 | 14 |
|
| 15 | +#include "Tuple.h" |
| 16 | + |
14 | 17 | // TODO can we eliminate need to import both of these?
|
15 | 18 | using namespace yarpl;
|
16 | 19 | using namespace yarpl::observable;
|
17 | 20 |
|
| 21 | +namespace { |
| 22 | + |
| 23 | +void unreachable() { |
| 24 | + EXPECT_TRUE(false); |
| 25 | +} |
| 26 | + |
| 27 | +template <typename T> |
| 28 | +class CollectingObserver : public Observer<T> { |
| 29 | + public: |
| 30 | + static_assert( |
| 31 | + std::is_copy_constructible<T>::value, |
| 32 | + "CollectingSubscriber needs to copy the value in order to collect it"); |
| 33 | + |
| 34 | + void onNext(T next) override { |
| 35 | + Observer<T>::onNext(next); |
| 36 | + values_.push_back(std::move(next)); |
| 37 | + } |
| 38 | + |
| 39 | + void onComplete() override { |
| 40 | + Observer<T>::onComplete(); |
| 41 | + complete_ = true; |
| 42 | + } |
| 43 | + |
| 44 | + void onError(std::exception_ptr ex) override { |
| 45 | + Observer<T>::onError(ex); |
| 46 | + error_ = true; |
| 47 | + |
| 48 | + try { |
| 49 | + std::rethrow_exception(ex); |
| 50 | + } catch (const std::exception& e) { |
| 51 | + errorMsg_ = e.what(); |
| 52 | + } |
| 53 | + } |
| 54 | + |
| 55 | + const std::vector<T>& values() const { |
| 56 | + return values_; |
| 57 | + } |
| 58 | + |
| 59 | + bool complete() const { |
| 60 | + return complete_; |
| 61 | + } |
| 62 | + |
| 63 | + bool error() const { |
| 64 | + return error_; |
| 65 | + } |
| 66 | + |
| 67 | + const std::string& errorMsg() const { |
| 68 | + return errorMsg_; |
| 69 | + } |
| 70 | + |
| 71 | + private: |
| 72 | + std::vector<T> values_; |
| 73 | + std::string errorMsg_; |
| 74 | + bool complete_{false}; |
| 75 | + bool error_{false}; |
| 76 | +}; |
| 77 | + |
| 78 | +/// Construct a pipeline with a collecting observer against the supplied |
| 79 | +/// observable. Return the items that were sent to the observer. If some |
| 80 | +/// exception was sent, the exception is thrown. |
| 81 | +template <typename T> |
| 82 | +std::vector<T> run(Reference<Observable<T>> observable) { |
| 83 | + auto collector = make_ref<CollectingObserver<T>>(); |
| 84 | + observable->subscribe(collector); |
| 85 | + return collector->values(); |
| 86 | +} |
| 87 | + |
| 88 | +} // namespace |
| 89 | + |
18 | 90 | TEST(Observable, SingleOnNext) {
|
19 | 91 | {
|
20 | 92 | ASSERT_EQ(std::size_t{0}, Refcounted::objects());
|
@@ -262,3 +334,124 @@ TEST(Observable, toFlowableWithCancel) {
|
262 | 334 | }
|
263 | 335 | ASSERT_EQ(std::size_t{0}, Refcounted::objects());
|
264 | 336 | }
|
| 337 | + |
| 338 | +TEST(Observable, Just) { |
| 339 | + ASSERT_EQ(0u, Refcounted::objects()); |
| 340 | + |
| 341 | + EXPECT_EQ(run(Observables::just(22)), std::vector<int>{22}); |
| 342 | + EXPECT_EQ( |
| 343 | + run(Observables::justN({12, 34, 56, 98})), |
| 344 | + std::vector<int>({12, 34, 56, 98})); |
| 345 | + EXPECT_EQ( |
| 346 | + run(Observables::justN({"ab", "pq", "yz"})), |
| 347 | + std::vector<const char*>({"ab", "pq", "yz"})); |
| 348 | + |
| 349 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 350 | +} |
| 351 | + |
| 352 | +TEST(Observable, Range) { |
| 353 | + ASSERT_EQ(0u, Refcounted::objects()); |
| 354 | + |
| 355 | + auto observable = Observables::range(10, 14); |
| 356 | + EXPECT_EQ(run(std::move(observable)), std::vector<int64_t>({10, 11, 12, 13})); |
| 357 | + |
| 358 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 359 | +} |
| 360 | + |
| 361 | +TEST(Observable, RangeWithMap) { |
| 362 | + ASSERT_EQ(0u, Refcounted::objects()); |
| 363 | + |
| 364 | + auto observable = Observables::range(1, 4) |
| 365 | + ->map([](int64_t v) { return v * v; }) |
| 366 | + ->map([](int64_t v) { return v * v; }) |
| 367 | + ->map([](int64_t v) { return std::to_string(v); }); |
| 368 | + EXPECT_EQ( |
| 369 | + run(std::move(observable)), std::vector<std::string>({"1", "16", "81"})); |
| 370 | + |
| 371 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 372 | +} |
| 373 | + |
| 374 | +// TODO: Hits ASAN errors. |
| 375 | +TEST(Observable, DISABLED_SimpleTake) { |
| 376 | + ASSERT_EQ(0u, Refcounted::objects()); |
| 377 | + |
| 378 | + EXPECT_EQ( |
| 379 | + run(Observables::range(0, 100)->take(3)), |
| 380 | + std::vector<int64_t>({0, 1, 2})); |
| 381 | + |
| 382 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 383 | +} |
| 384 | + |
| 385 | +TEST(Observable, Error) { |
| 386 | + auto observable = Observables::error<int>(std::runtime_error("something broke!")); |
| 387 | + auto collector = make_ref<CollectingObserver<int>>(); |
| 388 | + observable->subscribe(collector); |
| 389 | + |
| 390 | + EXPECT_EQ(collector->complete(), false); |
| 391 | + EXPECT_EQ(collector->error(), true); |
| 392 | + EXPECT_EQ(collector->errorMsg(), "something broke!"); |
| 393 | +} |
| 394 | + |
| 395 | +TEST(Observable, ErrorPtr) { |
| 396 | + auto observable = Observables::error<int>( |
| 397 | + std::make_exception_ptr(std::runtime_error("something broke!"))); |
| 398 | + auto collector = make_ref<CollectingObserver<int>>(); |
| 399 | + observable->subscribe(collector); |
| 400 | + |
| 401 | + EXPECT_EQ(collector->complete(), false); |
| 402 | + EXPECT_EQ(collector->error(), true); |
| 403 | + EXPECT_EQ(collector->errorMsg(), "something broke!"); |
| 404 | +} |
| 405 | + |
| 406 | +TEST(Observable, Empty) { |
| 407 | + auto observable = Observables::empty<int>(); |
| 408 | + auto collector = make_ref<CollectingObserver<int>>(); |
| 409 | + observable->subscribe(collector); |
| 410 | + |
| 411 | + EXPECT_EQ(collector->complete(), true); |
| 412 | + EXPECT_EQ(collector->error(), false); |
| 413 | +} |
| 414 | + |
| 415 | +TEST(Observable, ObserversComplete) { |
| 416 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 417 | + |
| 418 | + auto observable = Observables::empty<int>(); |
| 419 | + EXPECT_EQ(1u, Refcounted::objects()); |
| 420 | + |
| 421 | + bool completed = false; |
| 422 | + |
| 423 | + auto observer = Observers::create<int>( |
| 424 | + [](int) { unreachable(); }, |
| 425 | + [](std::exception_ptr) { unreachable(); }, |
| 426 | + [&] { completed = true; } |
| 427 | + ); |
| 428 | + |
| 429 | + observable->subscribe(std::move(observer)); |
| 430 | + observable.reset(); |
| 431 | + |
| 432 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 433 | + |
| 434 | + EXPECT_TRUE(completed); |
| 435 | +} |
| 436 | + |
| 437 | +TEST(Observable, ObserversError) { |
| 438 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 439 | + |
| 440 | + auto observable = Observables::error<int>(std::runtime_error("Whoops")); |
| 441 | + EXPECT_EQ(1u, Refcounted::objects()); |
| 442 | + |
| 443 | + bool errored = false; |
| 444 | + |
| 445 | + auto observer = Observers::create<int>( |
| 446 | + [](int) { unreachable(); }, |
| 447 | + [&](std::exception_ptr) { errored = true; }, |
| 448 | + [] { unreachable(); } |
| 449 | + ); |
| 450 | + |
| 451 | + observable->subscribe(std::move(observer)); |
| 452 | + observable.reset(); |
| 453 | + |
| 454 | + EXPECT_EQ(0u, Refcounted::objects()); |
| 455 | + |
| 456 | + EXPECT_TRUE(errored); |
| 457 | +} |
0 commit comments