blob: c1dd33033f662fbb8b458de5b58cdcc94cb223c8 [file] [log] [blame]
Darin Petkov1023a602010-08-30 13:47:51 -07001// Copyright (c) 2010 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 <gtest/gtest.h>
6
7#include "update_engine/update_attempter_mock.h"
8#include "update_engine/update_check_scheduler.h"
9
10using std::string;
11using testing::_;
12using testing::AllOf;
Darin Petkov2a0e6332010-09-24 14:43:41 -070013using testing::Assign;
Darin Petkov1023a602010-08-30 13:47:51 -070014using testing::Ge;
15using testing::Le;
16using testing::MockFunction;
17using testing::Return;
18
19namespace chromeos_update_engine {
20
21namespace {
22void FuzzRange(int interval, int fuzz, int* interval_min, int* interval_max) {
23 *interval_min = interval - fuzz / 2;
24 *interval_max = interval + fuzz - fuzz / 2;
25}
26} // namespace {}
27
28// Test a subclass rather than the main class directly so that we can mock out
29// GLib and utils in tests. There're explicit unit test for the wrapper methods.
30class UpdateCheckSchedulerUnderTest : public UpdateCheckScheduler {
31 public:
32 UpdateCheckSchedulerUnderTest(UpdateAttempter* update_attempter)
33 : UpdateCheckScheduler(update_attempter) {}
34
35 MOCK_METHOD2(GTimeoutAddSeconds, guint(guint seconds, GSourceFunc function));
36 MOCK_METHOD0(IsBootDeviceRemovable, bool());
37 MOCK_METHOD0(IsOfficialBuild, bool());
Darin Petkov2a0e6332010-09-24 14:43:41 -070038 MOCK_METHOD0(IsOOBEComplete, bool());
Darin Petkov1023a602010-08-30 13:47:51 -070039};
40
41class UpdateCheckSchedulerTest : public ::testing::Test {
Darin Petkovf42cc1c2010-09-01 09:03:02 -070042 public:
Andrew de los Reyes000d8952011-03-02 15:21:14 -080043 UpdateCheckSchedulerTest() : scheduler_(&attempter_), attempter_(&dbus_) {}
Darin Petkovf42cc1c2010-09-01 09:03:02 -070044
Darin Petkov1023a602010-08-30 13:47:51 -070045 protected:
46 virtual void SetUp() {
47 test_ = this;
48 loop_ = NULL;
Darin Petkovf42cc1c2010-09-01 09:03:02 -070049 EXPECT_EQ(&attempter_, scheduler_.update_attempter_);
50 EXPECT_FALSE(scheduler_.enabled_);
51 EXPECT_FALSE(scheduler_.scheduled_);
52 EXPECT_EQ(0, scheduler_.last_interval_);
Darin Petkov85ced132010-09-01 10:20:56 -070053 EXPECT_EQ(0, scheduler_.poll_interval_);
Darin Petkov1023a602010-08-30 13:47:51 -070054 }
55
56 virtual void TearDown() {
57 test_ = NULL;
58 loop_ = NULL;
Darin Petkov1023a602010-08-30 13:47:51 -070059 }
60
61 static gboolean SourceCallback(gpointer data) {
62 g_main_loop_quit(test_->loop_);
63 // Forwards the call to the function mock so that expectations can be set.
64 return test_->source_callback_.Call(data);
65 }
66
Darin Petkovf42cc1c2010-09-01 09:03:02 -070067 UpdateCheckSchedulerUnderTest scheduler_;
Andrew de los Reyes000d8952011-03-02 15:21:14 -080068 MockDbusGlib dbus_;
Darin Petkov1023a602010-08-30 13:47:51 -070069 UpdateAttempterMock attempter_;
70 MockFunction<gboolean(gpointer data)> source_callback_;
71 GMainLoop* loop_;
72 static UpdateCheckSchedulerTest* test_;
73};
74
75UpdateCheckSchedulerTest* UpdateCheckSchedulerTest::test_ = NULL;
76
77TEST_F(UpdateCheckSchedulerTest, CanScheduleTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -070078 EXPECT_FALSE(scheduler_.CanSchedule());
79 scheduler_.enabled_ = true;
80 EXPECT_TRUE(scheduler_.CanSchedule());
81 scheduler_.scheduled_ = true;
82 EXPECT_FALSE(scheduler_.CanSchedule());
83 scheduler_.enabled_ = false;
84 EXPECT_FALSE(scheduler_.CanSchedule());
Darin Petkov1023a602010-08-30 13:47:51 -070085}
86
87TEST_F(UpdateCheckSchedulerTest, ComputeNextIntervalAndFuzzBackoffTest) {
88 int interval, fuzz;
89 attempter_.set_http_response_code(500);
90 int last_interval = UpdateCheckScheduler::kTimeoutPeriodic + 50;
Darin Petkovf42cc1c2010-09-01 09:03:02 -070091 scheduler_.last_interval_ = last_interval;
92 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
Darin Petkov1023a602010-08-30 13:47:51 -070093 EXPECT_EQ(2 * last_interval, interval);
94 EXPECT_EQ(2 * last_interval, fuzz);
95
96 attempter_.set_http_response_code(503);
Darin Petkov85ced132010-09-01 10:20:56 -070097 scheduler_.last_interval_ = UpdateCheckScheduler::kTimeoutMaxBackoff / 2 + 1;
Darin Petkovf42cc1c2010-09-01 09:03:02 -070098 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
Darin Petkov1023a602010-08-30 13:47:51 -070099 EXPECT_EQ(UpdateCheckScheduler::kTimeoutMaxBackoff, interval);
100 EXPECT_EQ(UpdateCheckScheduler::kTimeoutMaxBackoff, fuzz);
101}
102
Darin Petkov85ced132010-09-01 10:20:56 -0700103TEST_F(UpdateCheckSchedulerTest, ComputeNextIntervalAndFuzzPollTest) {
104 int interval, fuzz;
105 int poll_interval = UpdateCheckScheduler::kTimeoutPeriodic + 50;
106 scheduler_.set_poll_interval(poll_interval);
107 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
108 EXPECT_EQ(poll_interval, interval);
109 EXPECT_EQ(poll_interval, fuzz);
110
111 scheduler_.set_poll_interval(UpdateCheckScheduler::kTimeoutMaxBackoff + 1);
112 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
113 EXPECT_EQ(UpdateCheckScheduler::kTimeoutMaxBackoff, interval);
114 EXPECT_EQ(UpdateCheckScheduler::kTimeoutMaxBackoff, fuzz);
115
116 scheduler_.set_poll_interval(UpdateCheckScheduler::kTimeoutPeriodic - 1);
117 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
118 EXPECT_EQ(UpdateCheckScheduler::kTimeoutPeriodic, interval);
119 EXPECT_EQ(UpdateCheckScheduler::kTimeoutRegularFuzz, fuzz);
120}
121
122TEST_F(UpdateCheckSchedulerTest, ComputeNextIntervalAndFuzzPriorityTest) {
123 int interval, fuzz;
124 attempter_.set_http_response_code(500);
125 scheduler_.last_interval_ = UpdateCheckScheduler::kTimeoutPeriodic + 50;
126 int poll_interval = UpdateCheckScheduler::kTimeoutPeriodic + 100;
127 scheduler_.set_poll_interval(poll_interval);
128 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
129 EXPECT_EQ(poll_interval, interval);
130 EXPECT_EQ(poll_interval, fuzz);
131}
132
Darin Petkov1023a602010-08-30 13:47:51 -0700133TEST_F(UpdateCheckSchedulerTest, ComputeNextIntervalAndFuzzTest) {
134 int interval, fuzz;
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700135 scheduler_.ComputeNextIntervalAndFuzz(&interval, &fuzz);
Darin Petkov1023a602010-08-30 13:47:51 -0700136 EXPECT_EQ(UpdateCheckScheduler::kTimeoutPeriodic, interval);
137 EXPECT_EQ(UpdateCheckScheduler::kTimeoutRegularFuzz, fuzz);
138}
139
140TEST_F(UpdateCheckSchedulerTest, GTimeoutAddSecondsTest) {
141 loop_ = g_main_loop_new(g_main_context_default(), FALSE);
142 // Invokes the actual GLib wrapper method rather than the subclass mock.
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700143 scheduler_.UpdateCheckScheduler::GTimeoutAddSeconds(0, SourceCallback);
144 EXPECT_CALL(source_callback_, Call(&scheduler_)).Times(1);
Darin Petkov1023a602010-08-30 13:47:51 -0700145 g_main_loop_run(loop_);
146 g_main_loop_unref(loop_);
147}
148
149TEST_F(UpdateCheckSchedulerTest, IsBootDeviceRemovableTest) {
150 // Invokes the actual utils wrapper method rather than the subclass mock.
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700151 EXPECT_FALSE(scheduler_.UpdateCheckScheduler::IsBootDeviceRemovable());
Darin Petkov1023a602010-08-30 13:47:51 -0700152}
153
Darin Petkov2a0e6332010-09-24 14:43:41 -0700154TEST_F(UpdateCheckSchedulerTest, IsOOBECompleteTest) {
155 // Invokes the actual utils wrapper method rather than the subclass mock.
156 EXPECT_FALSE(scheduler_.UpdateCheckScheduler::IsOOBEComplete());
157}
158
Darin Petkov1023a602010-08-30 13:47:51 -0700159TEST_F(UpdateCheckSchedulerTest, IsOfficialBuildTest) {
160 // Invokes the actual utils wrapper method rather than the subclass mock.
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700161 EXPECT_TRUE(scheduler_.UpdateCheckScheduler::IsOfficialBuild());
Darin Petkov1023a602010-08-30 13:47:51 -0700162}
163
164TEST_F(UpdateCheckSchedulerTest, RunBootDeviceRemovableTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700165 scheduler_.enabled_ = true;
166 EXPECT_CALL(scheduler_, IsOfficialBuild()).Times(1).WillOnce(Return(true));
167 EXPECT_CALL(scheduler_, IsBootDeviceRemovable())
Darin Petkov1023a602010-08-30 13:47:51 -0700168 .Times(1)
169 .WillOnce(Return(true));
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700170 scheduler_.Run();
171 EXPECT_FALSE(scheduler_.enabled_);
Darin Petkov1023a602010-08-30 13:47:51 -0700172 EXPECT_EQ(NULL, attempter_.update_check_scheduler());
173}
174
175TEST_F(UpdateCheckSchedulerTest, RunNonOfficialBuildTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700176 scheduler_.enabled_ = true;
177 EXPECT_CALL(scheduler_, IsOfficialBuild()).Times(1).WillOnce(Return(false));
178 scheduler_.Run();
179 EXPECT_FALSE(scheduler_.enabled_);
Darin Petkov1023a602010-08-30 13:47:51 -0700180 EXPECT_EQ(NULL, attempter_.update_check_scheduler());
181}
182
183TEST_F(UpdateCheckSchedulerTest, RunTest) {
184 int interval_min, interval_max;
185 FuzzRange(UpdateCheckScheduler::kTimeoutOnce,
186 UpdateCheckScheduler::kTimeoutRegularFuzz,
187 &interval_min,
188 &interval_max);
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700189 EXPECT_CALL(scheduler_, IsOfficialBuild()).Times(1).WillOnce(Return(true));
190 EXPECT_CALL(scheduler_, IsBootDeviceRemovable())
Darin Petkov1023a602010-08-30 13:47:51 -0700191 .Times(1)
192 .WillOnce(Return(false));
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700193 EXPECT_CALL(scheduler_,
Darin Petkov1023a602010-08-30 13:47:51 -0700194 GTimeoutAddSeconds(AllOf(Ge(interval_min), Le(interval_max)),
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700195 scheduler_.StaticCheck)).Times(1);
196 scheduler_.Run();
197 EXPECT_TRUE(scheduler_.enabled_);
198 EXPECT_EQ(&scheduler_, attempter_.update_check_scheduler());
Darin Petkov1023a602010-08-30 13:47:51 -0700199}
200
201TEST_F(UpdateCheckSchedulerTest, ScheduleCheckDisabledTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700202 EXPECT_CALL(scheduler_, GTimeoutAddSeconds(_, _)).Times(0);
203 scheduler_.ScheduleCheck(250, 30);
204 EXPECT_EQ(0, scheduler_.last_interval_);
205 EXPECT_FALSE(scheduler_.scheduled_);
Darin Petkov1023a602010-08-30 13:47:51 -0700206}
207
208TEST_F(UpdateCheckSchedulerTest, ScheduleCheckEnabledTest) {
209 int interval_min, interval_max;
210 FuzzRange(100, 10, &interval_min,&interval_max);
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700211 EXPECT_CALL(scheduler_,
Darin Petkov1023a602010-08-30 13:47:51 -0700212 GTimeoutAddSeconds(AllOf(Ge(interval_min), Le(interval_max)),
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700213 scheduler_.StaticCheck)).Times(1);
214 scheduler_.enabled_ = true;
215 scheduler_.ScheduleCheck(100, 10);
216 EXPECT_EQ(100, scheduler_.last_interval_);
217 EXPECT_TRUE(scheduler_.scheduled_);
Darin Petkov1023a602010-08-30 13:47:51 -0700218}
219
220TEST_F(UpdateCheckSchedulerTest, ScheduleCheckNegativeIntervalTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700221 EXPECT_CALL(scheduler_, GTimeoutAddSeconds(0, scheduler_.StaticCheck))
Darin Petkov1023a602010-08-30 13:47:51 -0700222 .Times(1);
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700223 scheduler_.enabled_ = true;
224 scheduler_.ScheduleCheck(-50, 20);
225 EXPECT_TRUE(scheduler_.scheduled_);
Darin Petkov1023a602010-08-30 13:47:51 -0700226}
227
228TEST_F(UpdateCheckSchedulerTest, ScheduleNextCheckDisabledTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700229 EXPECT_CALL(scheduler_, GTimeoutAddSeconds(_, _)).Times(0);
230 scheduler_.ScheduleNextCheck();
Darin Petkov1023a602010-08-30 13:47:51 -0700231}
232
233TEST_F(UpdateCheckSchedulerTest, ScheduleNextCheckEnabledTest) {
234 int interval_min, interval_max;
235 FuzzRange(UpdateCheckScheduler::kTimeoutPeriodic,
236 UpdateCheckScheduler::kTimeoutRegularFuzz,
237 &interval_min,
238 &interval_max);
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700239 EXPECT_CALL(scheduler_,
Darin Petkov1023a602010-08-30 13:47:51 -0700240 GTimeoutAddSeconds(AllOf(Ge(interval_min), Le(interval_max)),
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700241 scheduler_.StaticCheck)).Times(1);
242 scheduler_.enabled_ = true;
243 scheduler_.ScheduleNextCheck();
Darin Petkov1023a602010-08-30 13:47:51 -0700244}
245
246TEST_F(UpdateCheckSchedulerTest, SetUpdateStatusIdleDisabledTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700247 EXPECT_CALL(scheduler_, GTimeoutAddSeconds(_, _)).Times(0);
248 scheduler_.SetUpdateStatus(UPDATE_STATUS_IDLE);
Darin Petkov1023a602010-08-30 13:47:51 -0700249}
250
251TEST_F(UpdateCheckSchedulerTest, SetUpdateStatusIdleEnabledTest) {
252 int interval_min, interval_max;
253 FuzzRange(UpdateCheckScheduler::kTimeoutPeriodic,
254 UpdateCheckScheduler::kTimeoutRegularFuzz,
255 &interval_min,
256 &interval_max);
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700257 EXPECT_CALL(scheduler_,
Darin Petkov1023a602010-08-30 13:47:51 -0700258 GTimeoutAddSeconds(AllOf(Ge(interval_min), Le(interval_max)),
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700259 scheduler_.StaticCheck)).Times(1);
260 scheduler_.enabled_ = true;
261 scheduler_.SetUpdateStatus(UPDATE_STATUS_IDLE);
Darin Petkov1023a602010-08-30 13:47:51 -0700262}
263
264TEST_F(UpdateCheckSchedulerTest, SetUpdateStatusNonIdleTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700265 EXPECT_CALL(scheduler_, GTimeoutAddSeconds(_, _)).Times(0);
266 scheduler_.SetUpdateStatus(UPDATE_STATUS_DOWNLOADING);
267 scheduler_.enabled_ = true;
268 scheduler_.SetUpdateStatus(UPDATE_STATUS_DOWNLOADING);
Darin Petkov1023a602010-08-30 13:47:51 -0700269}
270
Darin Petkov2a0e6332010-09-24 14:43:41 -0700271TEST_F(UpdateCheckSchedulerTest, StaticCheckOOBECompleteTest) {
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700272 scheduler_.scheduled_ = true;
Darin Petkov2a0e6332010-09-24 14:43:41 -0700273 EXPECT_CALL(scheduler_, IsOOBEComplete()).Times(1).WillOnce(Return(true));
Andrew de los Reyes45168102010-11-22 11:13:50 -0800274 EXPECT_CALL(attempter_, Update("", "", false))
Darin Petkov2a0e6332010-09-24 14:43:41 -0700275 .Times(1)
276 .WillOnce(Assign(&scheduler_.scheduled_, true));
277 scheduler_.enabled_ = true;
278 EXPECT_CALL(scheduler_, GTimeoutAddSeconds(_, _)).Times(0);
279 UpdateCheckSchedulerUnderTest::StaticCheck(&scheduler_);
280}
281
282TEST_F(UpdateCheckSchedulerTest, StaticCheckOOBENotCompleteTest) {
283 scheduler_.scheduled_ = true;
284 EXPECT_CALL(scheduler_, IsOOBEComplete()).Times(1).WillOnce(Return(false));
Andrew de los Reyes45168102010-11-22 11:13:50 -0800285 EXPECT_CALL(attempter_, Update("", "", _)).Times(0);
Darin Petkov2a0e6332010-09-24 14:43:41 -0700286 int interval_min, interval_max;
287 FuzzRange(UpdateCheckScheduler::kTimeoutOnce,
288 UpdateCheckScheduler::kTimeoutRegularFuzz,
289 &interval_min,
290 &interval_max);
291 scheduler_.enabled_ = true;
292 EXPECT_CALL(scheduler_,
293 GTimeoutAddSeconds(AllOf(Ge(interval_min), Le(interval_max)),
294 scheduler_.StaticCheck)).Times(1);
Darin Petkovf42cc1c2010-09-01 09:03:02 -0700295 UpdateCheckSchedulerUnderTest::StaticCheck(&scheduler_);
Darin Petkov1023a602010-08-30 13:47:51 -0700296}
297
298} // namespace chromeos_update_engine