blob: 34e5bb049cdaadabf5459655df68390967736e33 [file] [log] [blame]
Patrick Williams73e278b2022-09-16 08:31:36 -05001#include <sdbusplus/async.hpp>
2
3#include <gtest/gtest.h>
4
Patrick Williams78e436f2022-09-21 10:06:20 -05005struct Context : public testing::Test
Patrick Williams73e278b2022-09-16 08:31:36 -05006{
Patrick Williams78e436f2022-09-21 10:06:20 -05007 ~Context() noexcept = default;
8
9 void TearDown() override
10 {
11 // Destructing the context can throw, so we have to do it in
12 // the TearDown in order to make our destructor noexcept.
13 ctx.reset();
14 }
15
Patrick Williams1b7b54c2022-09-21 10:49:45 -050016 void runToStop()
17 {
Patrick Williams3c242ba2022-09-23 09:51:55 -050018 ctx->spawn(std::execution::just() |
19 std::execution::then([this]() { ctx->request_stop(); }));
20 ctx->run();
Patrick Williams1b7b54c2022-09-21 10:49:45 -050021 }
22
Patrick Williams78e436f2022-09-21 10:06:20 -050023 std::optional<sdbusplus::async::context> ctx{std::in_place};
24};
25
26TEST_F(Context, RunSimple)
27{
Patrick Williams1b7b54c2022-09-21 10:49:45 -050028 runToStop();
Patrick Williams73e278b2022-09-16 08:31:36 -050029}
Patrick Williams0139cac2022-09-20 17:04:17 -050030
Patrick Williams78e436f2022-09-21 10:06:20 -050031TEST_F(Context, SpawnedTask)
Patrick Williams0139cac2022-09-20 17:04:17 -050032{
Patrick Williams78e436f2022-09-21 10:06:20 -050033 ctx->spawn(std::execution::just());
Patrick Williams1b7b54c2022-09-21 10:49:45 -050034 runToStop();
Patrick Williams78e436f2022-09-21 10:06:20 -050035}
Patrick Williams0139cac2022-09-20 17:04:17 -050036
Patrick Williams78e436f2022-09-21 10:06:20 -050037TEST_F(Context, SpawnDelayedTask)
38{
39 using namespace std::literals;
40 static constexpr auto timeout = 500ms;
41
42 auto start = std::chrono::steady_clock::now();
43
44 bool ran = false;
45 ctx->spawn(sdbusplus::async::sleep_for(*ctx, timeout) |
46 std::execution::then([&ran]() { ran = true; }));
47
Patrick Williams1b7b54c2022-09-21 10:49:45 -050048 runToStop();
Patrick Williams78e436f2022-09-21 10:06:20 -050049
50 auto stop = std::chrono::steady_clock::now();
51
52 EXPECT_TRUE(ran);
53 EXPECT_GT(stop - start, timeout);
54 EXPECT_LT(stop - start, timeout * 2);
Patrick Williams0139cac2022-09-20 17:04:17 -050055}
Patrick Williams1b7b54c2022-09-21 10:49:45 -050056
Patrick Williamsd6b05cc2022-09-23 09:38:38 -050057TEST_F(Context, SpawnRecursiveTask)
58{
59 struct _
60 {
61 static auto one(size_t count, size_t& executed)
62 -> sdbusplus::async::task<size_t>
63 {
64 if (count)
65 {
66 ++executed;
67 co_return (co_await one(count - 1, executed)) + 1;
68 }
69 co_return co_await std::execution::just(0);
70 }
71 };
72
73 static constexpr size_t count = 100;
74 size_t executed = 0;
75
76 ctx->spawn(_::one(count, executed) | std::execution::then([=](auto result) {
77 EXPECT_EQ(result, count);
78 }));
79
80 runToStop();
81
82 EXPECT_EQ(executed, count);
83}
84
Patrick Williams1b7b54c2022-09-21 10:49:45 -050085TEST_F(Context, DestructMatcherWithPendingAwait)
86{
87 using namespace std::literals;
88
89 bool ran = false;
90 auto m = std::make_optional<sdbusplus::async::match>(
91 *ctx, sdbusplus::bus::match::rules::interfacesAdded(
92 "/this/is/a/bogus/path/for/SpawnMatcher"));
93
94 // Await the match completion (which will never happen).
95 ctx->spawn(m->next() | std::execution::then([&ran](...) { ran = true; }));
96
97 // Destruct the match.
98 ctx->spawn(sdbusplus::async::sleep_for(*ctx, 1ms) |
99 std::execution::then([&m](...) { m.reset(); }));
100
Patrick Williams4cfc2842022-09-22 09:53:33 -0500101 EXPECT_THROW(runToStop(), sdbusplus::exception::UnhandledStop);
Patrick Williams1b7b54c2022-09-21 10:49:45 -0500102 EXPECT_FALSE(ran);
103}
104
105TEST_F(Context, DestructMatcherWithPendingAwaitAsTask)
106{
107 using namespace std::literals;
108
109 auto m = std::make_optional<sdbusplus::async::match>(
110 *ctx, sdbusplus::bus::match::rules::interfacesAdded(
111 "/this/is/a/bogus/path/for/SpawnMatcher"));
112
113 struct _
114 {
115 static auto fn(decltype(m->next()) snd, bool& ran)
116 -> sdbusplus::async::task<>
117 {
118 co_await std::move(snd);
119 ran = true;
120 co_return;
121 }
122 };
123
124 bool ran = false;
125 ctx->spawn(_::fn(m->next(), ran));
126 ctx->spawn(sdbusplus::async::sleep_for(*ctx, 1ms) |
127 std::execution::then([&]() { m.reset(); }));
128
Patrick Williams4cfc2842022-09-22 09:53:33 -0500129 EXPECT_THROW(runToStop(), sdbusplus::exception::UnhandledStop);
Patrick Williams1b7b54c2022-09-21 10:49:45 -0500130 EXPECT_FALSE(ran);
131}