1 // Copyright 2015 The Chromium OS Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <brillo/backoff_entry.h>
6 #include <gtest/gtest.h>
7 
8 using base::TimeDelta;
9 using base::TimeTicks;
10 
11 namespace brillo {
12 
13 BackoffEntry::Policy base_policy = { 0, 1000, 2.0, 0.0, 20000, 2000, false };
14 
15 class TestBackoffEntry : public BackoffEntry {
16  public:
TestBackoffEntry(const Policy * const policy)17   explicit TestBackoffEntry(const Policy* const policy)
18       : BackoffEntry(policy),
19         now_(TimeTicks()) {
20     // Work around initialization in constructor not picking up
21     // fake time.
22     SetCustomReleaseTime(TimeTicks());
23   }
24 
~TestBackoffEntry()25   ~TestBackoffEntry() override {}
26 
ImplGetTimeNow() const27   TimeTicks ImplGetTimeNow() const override { return now_; }
28 
set_now(const TimeTicks & now)29   void set_now(const TimeTicks& now) {
30     now_ = now;
31   }
32 
33  private:
34   TimeTicks now_;
35 
36   DISALLOW_COPY_AND_ASSIGN(TestBackoffEntry);
37 };
38 
TEST(BackoffEntryTest,BaseTest)39 TEST(BackoffEntryTest, BaseTest) {
40   TestBackoffEntry entry(&base_policy);
41   EXPECT_FALSE(entry.ShouldRejectRequest());
42   EXPECT_EQ(TimeDelta(), entry.GetTimeUntilRelease());
43 
44   entry.InformOfRequest(false);
45   EXPECT_TRUE(entry.ShouldRejectRequest());
46   EXPECT_EQ(TimeDelta::FromMilliseconds(1000), entry.GetTimeUntilRelease());
47 }
48 
TEST(BackoffEntryTest,CanDiscardNeverExpires)49 TEST(BackoffEntryTest, CanDiscardNeverExpires) {
50   BackoffEntry::Policy never_expires_policy = base_policy;
51   never_expires_policy.entry_lifetime_ms = -1;
52   TestBackoffEntry never_expires(&never_expires_policy);
53   EXPECT_FALSE(never_expires.CanDiscard());
54   never_expires.set_now(TimeTicks() + TimeDelta::FromDays(100));
55   EXPECT_FALSE(never_expires.CanDiscard());
56 }
57 
TEST(BackoffEntryTest,CanDiscard)58 TEST(BackoffEntryTest, CanDiscard) {
59   TestBackoffEntry entry(&base_policy);
60   // Because lifetime is non-zero, we shouldn't be able to discard yet.
61   EXPECT_FALSE(entry.CanDiscard());
62 
63   // Test the "being used" case.
64   entry.InformOfRequest(false);
65   EXPECT_FALSE(entry.CanDiscard());
66 
67   // Test the case where there are errors but we can time out.
68   entry.set_now(
69       entry.GetReleaseTime() + TimeDelta::FromMilliseconds(1));
70   EXPECT_FALSE(entry.CanDiscard());
71   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(
72       base_policy.maximum_backoff_ms + 1));
73   EXPECT_TRUE(entry.CanDiscard());
74 
75   // Test the final case (no errors, dependent only on specified lifetime).
76   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(
77       base_policy.entry_lifetime_ms - 1));
78   entry.InformOfRequest(true);
79   EXPECT_FALSE(entry.CanDiscard());
80   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(
81       base_policy.entry_lifetime_ms));
82   EXPECT_TRUE(entry.CanDiscard());
83 }
84 
TEST(BackoffEntryTest,CanDiscardAlwaysDelay)85 TEST(BackoffEntryTest, CanDiscardAlwaysDelay) {
86   BackoffEntry::Policy always_delay_policy = base_policy;
87   always_delay_policy.always_use_initial_delay = true;
88   always_delay_policy.entry_lifetime_ms = 0;
89 
90   TestBackoffEntry entry(&always_delay_policy);
91 
92   // Because lifetime is non-zero, we shouldn't be able to discard yet.
93   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(2000));
94   EXPECT_TRUE(entry.CanDiscard());
95 
96   // Even with no failures, we wait until the delay before we allow discard.
97   entry.InformOfRequest(true);
98   EXPECT_FALSE(entry.CanDiscard());
99 
100   // Wait until the delay expires, and we can discard the entry again.
101   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(1000));
102   EXPECT_TRUE(entry.CanDiscard());
103 }
104 
TEST(BackoffEntryTest,CanDiscardNotStored)105 TEST(BackoffEntryTest, CanDiscardNotStored) {
106   BackoffEntry::Policy no_store_policy = base_policy;
107   no_store_policy.entry_lifetime_ms = 0;
108   TestBackoffEntry not_stored(&no_store_policy);
109   EXPECT_TRUE(not_stored.CanDiscard());
110 }
111 
TEST(BackoffEntryTest,ShouldIgnoreFirstTwo)112 TEST(BackoffEntryTest, ShouldIgnoreFirstTwo) {
113   BackoffEntry::Policy lenient_policy = base_policy;
114   lenient_policy.num_errors_to_ignore = 2;
115 
116   BackoffEntry entry(&lenient_policy);
117 
118   entry.InformOfRequest(false);
119   EXPECT_FALSE(entry.ShouldRejectRequest());
120 
121   entry.InformOfRequest(false);
122   EXPECT_FALSE(entry.ShouldRejectRequest());
123 
124   entry.InformOfRequest(false);
125   EXPECT_TRUE(entry.ShouldRejectRequest());
126 }
127 
TEST(BackoffEntryTest,ReleaseTimeCalculation)128 TEST(BackoffEntryTest, ReleaseTimeCalculation) {
129   TestBackoffEntry entry(&base_policy);
130 
131   // With zero errors, should return "now".
132   TimeTicks result = entry.GetReleaseTime();
133   EXPECT_EQ(entry.ImplGetTimeNow(), result);
134 
135   // 1 error.
136   entry.InformOfRequest(false);
137   result = entry.GetReleaseTime();
138   EXPECT_EQ(entry.ImplGetTimeNow() + TimeDelta::FromMilliseconds(1000), result);
139   EXPECT_EQ(TimeDelta::FromMilliseconds(1000), entry.GetTimeUntilRelease());
140 
141   // 2 errors.
142   entry.InformOfRequest(false);
143   result = entry.GetReleaseTime();
144   EXPECT_EQ(entry.ImplGetTimeNow() + TimeDelta::FromMilliseconds(2000), result);
145   EXPECT_EQ(TimeDelta::FromMilliseconds(2000), entry.GetTimeUntilRelease());
146 
147   // 3 errors.
148   entry.InformOfRequest(false);
149   result = entry.GetReleaseTime();
150   EXPECT_EQ(entry.ImplGetTimeNow() + TimeDelta::FromMilliseconds(4000), result);
151   EXPECT_EQ(TimeDelta::FromMilliseconds(4000), entry.GetTimeUntilRelease());
152 
153   // 6 errors (to check it doesn't pass maximum).
154   entry.InformOfRequest(false);
155   entry.InformOfRequest(false);
156   entry.InformOfRequest(false);
157   result = entry.GetReleaseTime();
158   EXPECT_EQ(
159       entry.ImplGetTimeNow() + TimeDelta::FromMilliseconds(20000), result);
160 }
161 
TEST(BackoffEntryTest,ReleaseTimeCalculationAlwaysDelay)162 TEST(BackoffEntryTest, ReleaseTimeCalculationAlwaysDelay) {
163   BackoffEntry::Policy always_delay_policy = base_policy;
164   always_delay_policy.always_use_initial_delay = true;
165   always_delay_policy.num_errors_to_ignore = 2;
166 
167   TestBackoffEntry entry(&always_delay_policy);
168 
169   // With previous requests, should return "now".
170   TimeTicks result = entry.GetReleaseTime();
171   EXPECT_EQ(TimeDelta(), entry.GetTimeUntilRelease());
172 
173   // 1 error.
174   entry.InformOfRequest(false);
175   EXPECT_EQ(TimeDelta::FromMilliseconds(1000), entry.GetTimeUntilRelease());
176 
177   // 2 errors.
178   entry.InformOfRequest(false);
179   EXPECT_EQ(TimeDelta::FromMilliseconds(1000), entry.GetTimeUntilRelease());
180 
181   // 3 errors, exponential backoff starts.
182   entry.InformOfRequest(false);
183   EXPECT_EQ(TimeDelta::FromMilliseconds(2000), entry.GetTimeUntilRelease());
184 
185   // 4 errors.
186   entry.InformOfRequest(false);
187   EXPECT_EQ(TimeDelta::FromMilliseconds(4000), entry.GetTimeUntilRelease());
188 
189   // 8 errors (to check it doesn't pass maximum).
190   entry.InformOfRequest(false);
191   entry.InformOfRequest(false);
192   entry.InformOfRequest(false);
193   entry.InformOfRequest(false);
194   result = entry.GetReleaseTime();
195   EXPECT_EQ(TimeDelta::FromMilliseconds(20000), entry.GetTimeUntilRelease());
196 }
197 
TEST(BackoffEntryTest,ReleaseTimeCalculationWithJitter)198 TEST(BackoffEntryTest, ReleaseTimeCalculationWithJitter) {
199   for (int i = 0; i < 10; ++i) {
200     BackoffEntry::Policy jittery_policy = base_policy;
201     jittery_policy.jitter_factor = 0.2;
202 
203     TestBackoffEntry entry(&jittery_policy);
204 
205     entry.InformOfRequest(false);
206     entry.InformOfRequest(false);
207     entry.InformOfRequest(false);
208     TimeTicks result = entry.GetReleaseTime();
209     EXPECT_LE(
210         entry.ImplGetTimeNow() + TimeDelta::FromMilliseconds(3200), result);
211     EXPECT_GE(
212         entry.ImplGetTimeNow() + TimeDelta::FromMilliseconds(4000), result);
213   }
214 }
215 
TEST(BackoffEntryTest,FailureThenSuccess)216 TEST(BackoffEntryTest, FailureThenSuccess) {
217   TestBackoffEntry entry(&base_policy);
218 
219   // Failure count 1, establishes horizon.
220   entry.InformOfRequest(false);
221   TimeTicks release_time = entry.GetReleaseTime();
222   EXPECT_EQ(TimeTicks() + TimeDelta::FromMilliseconds(1000), release_time);
223 
224   // Success, failure count 0, should not advance past
225   // the horizon that was already set.
226   entry.set_now(release_time - TimeDelta::FromMilliseconds(200));
227   entry.InformOfRequest(true);
228   EXPECT_EQ(release_time, entry.GetReleaseTime());
229 
230   // Failure, failure count 1.
231   entry.InformOfRequest(false);
232   EXPECT_EQ(release_time + TimeDelta::FromMilliseconds(800),
233             entry.GetReleaseTime());
234 }
235 
TEST(BackoffEntryTest,FailureThenSuccessAlwaysDelay)236 TEST(BackoffEntryTest, FailureThenSuccessAlwaysDelay) {
237   BackoffEntry::Policy always_delay_policy = base_policy;
238   always_delay_policy.always_use_initial_delay = true;
239   always_delay_policy.num_errors_to_ignore = 1;
240 
241   TestBackoffEntry entry(&always_delay_policy);
242 
243   // Failure count 1.
244   entry.InformOfRequest(false);
245   EXPECT_EQ(TimeDelta::FromMilliseconds(1000), entry.GetTimeUntilRelease());
246 
247   // Failure count 2.
248   entry.InformOfRequest(false);
249   EXPECT_EQ(TimeDelta::FromMilliseconds(2000), entry.GetTimeUntilRelease());
250   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(2000));
251 
252   // Success.  We should go back to the original delay.
253   entry.InformOfRequest(true);
254   EXPECT_EQ(TimeDelta::FromMilliseconds(1000), entry.GetTimeUntilRelease());
255 
256   // Failure count reaches 2 again.  We should increase the delay once more.
257   entry.InformOfRequest(false);
258   EXPECT_EQ(TimeDelta::FromMilliseconds(2000), entry.GetTimeUntilRelease());
259   entry.set_now(entry.GetReleaseTime() + TimeDelta::FromMilliseconds(2000));
260 }
261 
TEST(BackoffEntryTest,RetainCustomHorizon)262 TEST(BackoffEntryTest, RetainCustomHorizon) {
263   TestBackoffEntry custom(&base_policy);
264   TimeTicks custom_horizon = TimeTicks() + TimeDelta::FromDays(3);
265   custom.SetCustomReleaseTime(custom_horizon);
266   custom.InformOfRequest(false);
267   custom.InformOfRequest(true);
268   custom.set_now(TimeTicks() + TimeDelta::FromDays(2));
269   custom.InformOfRequest(false);
270   custom.InformOfRequest(true);
271   EXPECT_EQ(custom_horizon, custom.GetReleaseTime());
272 
273   // Now check that once we are at or past the custom horizon,
274   // we get normal behavior.
275   custom.set_now(TimeTicks() + TimeDelta::FromDays(3));
276   custom.InformOfRequest(false);
277   EXPECT_EQ(
278       TimeTicks() + TimeDelta::FromDays(3) + TimeDelta::FromMilliseconds(1000),
279       custom.GetReleaseTime());
280 }
281 
TEST(BackoffEntryTest,RetainCustomHorizonWhenInitialErrorsIgnored)282 TEST(BackoffEntryTest, RetainCustomHorizonWhenInitialErrorsIgnored) {
283   // Regression test for a bug discovered during code review.
284   BackoffEntry::Policy lenient_policy = base_policy;
285   lenient_policy.num_errors_to_ignore = 1;
286   TestBackoffEntry custom(&lenient_policy);
287   TimeTicks custom_horizon = TimeTicks() + TimeDelta::FromDays(3);
288   custom.SetCustomReleaseTime(custom_horizon);
289   custom.InformOfRequest(false);  // This must not reset the horizon.
290   EXPECT_EQ(custom_horizon, custom.GetReleaseTime());
291 }
292 
TEST(BackoffEntryTest,OverflowProtection)293 TEST(BackoffEntryTest, OverflowProtection) {
294   BackoffEntry::Policy large_multiply_policy = base_policy;
295   large_multiply_policy.multiply_factor = 256;
296   TestBackoffEntry custom(&large_multiply_policy);
297 
298   // Trigger enough failures such that more than 11 bits of exponent are used
299   // to represent the exponential backoff intermediate values. Given a multiply
300   // factor of 256 (2^8), 129 iterations is enough: 2^(8*(129-1)) = 2^1024.
301   for (int i = 0; i < 129; ++i) {
302     custom.set_now(custom.ImplGetTimeNow() + custom.GetTimeUntilRelease());
303     custom.InformOfRequest(false);
304     ASSERT_TRUE(custom.ShouldRejectRequest());
305   }
306 
307   // Max delay should still be respected.
308   EXPECT_EQ(20000, custom.GetTimeUntilRelease().InMilliseconds());
309 }
310 
311 }  // namespace brillo
312