1a19613cf3ed8708b707caab09c8d3b2b55edc9f
[folly.git] / folly / futures / test / TimekeeperTest.cpp
1 /*
2  * Copyright 2014 Facebook, Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *   http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 #include <gtest/gtest.h>
17
18 #include <folly/futures/Timekeeper.h>
19 #include <unistd.h>
20
21 using namespace folly;
22 using namespace std::chrono;
23 using folly::Timekeeper;
24 using Duration = folly::Duration;
25
26 std::chrono::milliseconds const one_ms(1);
27 std::chrono::milliseconds const awhile(10);
28
29 std::chrono::steady_clock::time_point now() {
30   return std::chrono::steady_clock::now();
31 }
32
33 struct TimekeeperFixture : public testing::Test {
34   TimekeeperFixture() :
35     timeLord_(folly::detail::getTimekeeperSingleton())
36   {}
37
38   Timekeeper* timeLord_;
39 };
40
41 TEST_F(TimekeeperFixture, after) {
42   Duration waited(0);
43
44   auto t1 = now();
45   auto f = timeLord_->after(awhile);
46   EXPECT_FALSE(f.isReady());
47   f.get();
48   auto t2 = now();
49
50   EXPECT_GE(t2 - t1, awhile);
51 }
52
53 TEST(Timekeeper, futureGet) {
54   Promise<int> p;
55   std::thread([&]{ p.setValue(42); }).detach();
56   EXPECT_EQ(42, p.getFuture().get());
57 }
58
59 TEST(Timekeeper, futureGetBeforeTimeout) {
60   Promise<int> p;
61   auto t = std::thread([&]{ p.setValue(42); });
62   // Technically this is a race and if the test server is REALLY overloaded
63   // and it takes more than a second to do that thread it could be flaky. But
64   // I want a low timeout (in human terms) so if this regresses and someone
65   // runs it by hand they're not sitting there forever wondering why it's
66   // blocked, and get a useful error message instead. If it does get flaky,
67   // empirically increase the timeout to the point where it's very improbable.
68   EXPECT_EQ(42, p.getFuture().get(seconds(2)));
69   t.join();
70 }
71
72 TEST(Timekeeper, futureGetTimeout) {
73   Promise<int> p;
74   EXPECT_THROW(p.getFuture().get(Duration(1)), folly::TimedOut);
75 }
76
77 TEST(Timekeeper, futureSleep) {
78   auto t1 = now();
79   futures::sleep(one_ms).get();
80   EXPECT_GE(now() - t1, one_ms);
81 }
82
83 TEST(Timekeeper, futureDelayed) {
84   auto t1 = now();
85   auto dur = makeFuture()
86     .delayed(one_ms)
87     .then([=]{ return now() - t1; })
88     .get();
89
90   EXPECT_GE(dur, one_ms);
91 }
92
93 TEST(Timekeeper, futureWithinThrows) {
94   Promise<int> p;
95   auto f = p.getFuture()
96     .within(one_ms)
97     .onError([](TimedOut&) { return -1; });
98
99   EXPECT_EQ(-1, f.get());
100 }
101
102 TEST(Timekeeper, futureWithinAlreadyComplete) {
103   auto f = makeFuture(42)
104     .within(one_ms)
105     .onError([&](TimedOut&){ return -1; });
106
107   EXPECT_EQ(42, f.get());
108 }
109
110 TEST(Timekeeper, futureWithinFinishesInTime) {
111   Promise<int> p;
112   auto f = p.getFuture()
113     .within(std::chrono::minutes(1))
114     .onError([&](TimedOut&){ return -1; });
115   p.setValue(42);
116
117   EXPECT_EQ(42, f.get());
118 }
119
120 TEST(Timekeeper, futureWithinVoidSpecialization) {
121   makeFuture().within(one_ms);
122 }
123
124 TEST(Timekeeper, futureWithinException) {
125   Promise<void> p;
126   auto f = p.getFuture().within(awhile, std::runtime_error("expected"));
127   EXPECT_THROW(f.get(), std::runtime_error);
128 }
129
130 TEST(Timekeeper, onTimeout) {
131   bool flag = false;
132   makeFuture(42).delayed(one_ms)
133     .onTimeout(Duration(0), [&]{ flag = true; return -1; })
134     .get();
135   EXPECT_TRUE(flag);
136 }
137
138 TEST(Timekeeper, onTimeoutReturnsFuture) {
139   bool flag = false;
140   makeFuture(42).delayed(one_ms)
141     .onTimeout(Duration(0), [&]{ flag = true; return makeFuture(-1); })
142     .get();
143   EXPECT_TRUE(flag);
144 }
145
146 TEST(Timekeeper, onTimeoutVoid) {
147   makeFuture().delayed(one_ms)
148     .onTimeout(Duration(0), [&]{
149      });
150   makeFuture().delayed(one_ms)
151     .onTimeout(Duration(0), [&]{
152        return makeFuture<void>(std::runtime_error("expected"));
153      });
154   // just testing compilation here
155 }
156
157 // TODO(5921764)
158 /*
159 TEST(Timekeeper, onTimeoutPropagates) {
160   bool flag = false;
161   EXPECT_THROW(
162     makeFuture(42).delayed(one_ms)
163       .onTimeout(Duration(0), [&]{ flag = true; })
164       .get(),
165     TimedOut);
166   EXPECT_TRUE(flag);
167 }
168 */