016-pub_sub.cpp 2.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. //
  2. // Copyright (c) 2019-2020 Ivan Baidakou (basiliscos) (the dot dmol at gmail dot com)
  3. //
  4. // Distributed under the MIT Software License
  5. //
  6. #include "rotor.hpp"
  7. #include "supervisor_test.h"
  8. #include "access.h"
  9. namespace r = rotor;
  10. namespace rt = r::test;
  11. struct payload_t {};
  12. struct pub_config_t : r::actor_config_t {
  13. r::address_ptr_t pub_addr;
  14. using r::actor_config_t::actor_config_t;
  15. };
  16. template <typename Actor> struct pub_config_builder_t : r::actor_config_builder_t<Actor> {
  17. using builder_t = typename Actor::template config_builder_t<Actor>;
  18. using parent_t = r::actor_config_builder_t<Actor>;
  19. using parent_t::parent_t;
  20. builder_t &&pub_addr(const r::address_ptr_t &addr) {
  21. parent_t::config.pub_addr = addr;
  22. return std::move(*static_cast<builder_t *>(this));
  23. }
  24. bool validate() noexcept override { return parent_t::config.pub_addr && parent_t::validate(); }
  25. };
  26. struct pub_t : public r::actor_base_t {
  27. using config_t = pub_config_t;
  28. template <typename Actor> using config_builder_t = pub_config_builder_t<Actor>;
  29. explicit pub_t(config_t &cfg) : r::actor_base_t(cfg), pub_addr{cfg.pub_addr} {}
  30. void on_start() noexcept override {
  31. r::actor_base_t::on_start();
  32. send<payload_t>(pub_addr);
  33. }
  34. r::address_ptr_t pub_addr;
  35. };
  36. struct sub_t : public r::actor_base_t {
  37. using config_t = pub_config_t;
  38. template <typename Actor> using config_builder_t = pub_config_builder_t<Actor>;
  39. explicit sub_t(config_t &cfg) : r::actor_base_t(cfg), pub_addr{cfg.pub_addr} {}
  40. void configure(r::plugin::plugin_base_t &plugin) noexcept override {
  41. plugin.with_casted<r::plugin::starter_plugin_t>(
  42. [this](auto &p) { p.subscribe_actor(&sub_t::on_payload, pub_addr); });
  43. }
  44. void on_payload(r::message_t<payload_t> &) noexcept { ++received; }
  45. std::uint16_t received = 0;
  46. r::address_ptr_t pub_addr;
  47. };
  48. TEST_CASE("ping-pong-pub-sub", "[supervisor]") {
  49. r::system_context_t system_context;
  50. auto sup = system_context.create_supervisor<rt::supervisor_test_t>().timeout(rt::default_timeout).finish();
  51. auto pub_addr = sup->create_address();
  52. sup->create_actor<pub_t>().pub_addr(pub_addr).timeout(rt::default_timeout).finish();
  53. auto sub1 = sup->create_actor<sub_t>().pub_addr(pub_addr).timeout(rt::default_timeout).finish();
  54. auto sub2 = sup->create_actor<sub_t>().pub_addr(pub_addr).timeout(rt::default_timeout).finish();
  55. sup->do_process();
  56. REQUIRE(sub1->received == 1);
  57. REQUIRE(sub2->received == 1);
  58. sup->do_shutdown();
  59. sup->do_process();
  60. REQUIRE(sup->get_state() == r::state_t::SHUT_DOWN);
  61. REQUIRE(sup->get_leader_queue().size() == 0);
  62. REQUIRE(sup->get_points().size() == 0);
  63. CHECK(rt::empty(sup->get_subscription()));
  64. }