blob: 3e8aabd0114451bfe1ebd314d430bd9ed70e05b0 [file] [log] [blame]
rspangler@google.com49fdf182009-10-10 00:57:34 +00001// Copyright (c) 2009 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 <string>
6#include <vector>
Darin Petkov73058b42010-10-06 16:32:19 -07007
rspangler@google.com49fdf182009-10-10 00:57:34 +00008#include <glib.h>
Darin Petkov9d911fa2010-08-19 09:36:08 -07009#include <gmock/gmock.h>
rspangler@google.com49fdf182009-10-10 00:57:34 +000010#include <gtest/gtest.h>
Darin Petkov73058b42010-10-06 16:32:19 -070011
rspangler@google.com49fdf182009-10-10 00:57:34 +000012#include "update_engine/action_pipe.h"
13#include "update_engine/download_action.h"
14#include "update_engine/mock_http_fetcher.h"
15#include "update_engine/omaha_hash_calculator.h"
Darin Petkov73058b42010-10-06 16:32:19 -070016#include "update_engine/prefs_mock.h"
rspangler@google.com49fdf182009-10-10 00:57:34 +000017#include "update_engine/test_utils.h"
adlr@google.comc98a7ed2009-12-04 18:54:03 +000018#include "update_engine/utils.h"
rspangler@google.com49fdf182009-10-10 00:57:34 +000019
20namespace chromeos_update_engine {
21
22using std::string;
23using std::vector;
Darin Petkov9d911fa2010-08-19 09:36:08 -070024using testing::_;
25using testing::AtLeast;
26using testing::InSequence;
rspangler@google.com49fdf182009-10-10 00:57:34 +000027
28class DownloadActionTest : public ::testing::Test { };
29
30namespace {
Darin Petkov9d911fa2010-08-19 09:36:08 -070031class DownloadActionDelegateMock : public DownloadActionDelegate {
32 public:
33 MOCK_METHOD1(SetDownloadStatus, void(bool active));
34 MOCK_METHOD2(BytesReceived, void(uint64_t bytes_received, uint64_t total));
35};
36
rspangler@google.com49fdf182009-10-10 00:57:34 +000037class DownloadActionTestProcessorDelegate : public ActionProcessorDelegate {
38 public:
Darin Petkovc97435c2010-07-20 12:37:43 -070039 explicit DownloadActionTestProcessorDelegate(ActionExitCode expected_code)
40 : loop_(NULL),
41 processing_done_called_(false),
42 expected_code_(expected_code) {}
rspangler@google.com49fdf182009-10-10 00:57:34 +000043 virtual ~DownloadActionTestProcessorDelegate() {
44 EXPECT_TRUE(processing_done_called_);
45 }
Darin Petkovc1a8b422010-07-19 11:34:49 -070046 virtual void ProcessingDone(const ActionProcessor* processor,
47 ActionExitCode code) {
rspangler@google.com49fdf182009-10-10 00:57:34 +000048 ASSERT_TRUE(loop_);
49 g_main_loop_quit(loop_);
adlr@google.comc98a7ed2009-12-04 18:54:03 +000050 vector<char> found_data;
51 ASSERT_TRUE(utils::ReadFile(path_, &found_data));
rspangler@google.com49fdf182009-10-10 00:57:34 +000052 ASSERT_EQ(expected_data_.size(), found_data.size());
53 for (unsigned i = 0; i < expected_data_.size(); i++) {
54 EXPECT_EQ(expected_data_[i], found_data[i]);
55 }
56 processing_done_called_ = true;
57 }
58
adlr@google.comc98a7ed2009-12-04 18:54:03 +000059 virtual void ActionCompleted(ActionProcessor* processor,
60 AbstractAction* action,
Darin Petkovc1a8b422010-07-19 11:34:49 -070061 ActionExitCode code) {
Darin Petkovc97435c2010-07-20 12:37:43 -070062 const string type = action->Type();
63 if (type == DownloadAction::StaticType()) {
64 EXPECT_EQ(expected_code_, code);
65 } else {
66 EXPECT_EQ(kActionCodeSuccess, code);
67 }
rspangler@google.com49fdf182009-10-10 00:57:34 +000068 }
69
70 GMainLoop *loop_;
71 string path_;
72 vector<char> expected_data_;
73 bool processing_done_called_;
Darin Petkovc97435c2010-07-20 12:37:43 -070074 ActionExitCode expected_code_;
rspangler@google.com49fdf182009-10-10 00:57:34 +000075};
76
77struct EntryPointArgs {
78 const vector<char> *data;
79 GMainLoop *loop;
80 ActionProcessor *processor;
81};
82
83gboolean StartProcessorInRunLoop(gpointer data) {
84 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
85 processor->StartProcessing();
86 return FALSE;
87}
88
Darin Petkov9d911fa2010-08-19 09:36:08 -070089void TestWithData(const vector<char>& data,
90 bool hash_test,
Darin Petkov50332f12010-09-24 11:44:47 -070091 bool size_test,
Darin Petkov9d911fa2010-08-19 09:36:08 -070092 bool use_download_delegate) {
rspangler@google.com49fdf182009-10-10 00:57:34 +000093 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
94
95 // TODO(adlr): see if we need a different file for build bots
Andrew de los Reyesf9185172010-05-03 11:07:05 -070096 ScopedTempFile output_temp_file;
97 DirectFileWriter writer;
98
rspangler@google.com49fdf182009-10-10 00:57:34 +000099 // takes ownership of passed in HttpFetcher
Darin Petkovc97435c2010-07-20 12:37:43 -0700100 string hash = hash_test ?
101 OmahaHashCalculator::OmahaHashOfString("random string") :
102 OmahaHashCalculator::OmahaHashOfData(data);
Darin Petkov50332f12010-09-24 11:44:47 -0700103 uint64_t size = data.size() + (size_test ? 1 : 0);
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700104 InstallPlan install_plan(true,
105 "",
Darin Petkov50332f12010-09-24 11:44:47 -0700106 size,
Darin Petkovc97435c2010-07-20 12:37:43 -0700107 hash,
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700108 output_temp_file.GetPath(),
109 "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000110 ObjectFeederAction<InstallPlan> feeder_action;
111 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700112 PrefsMock prefs;
113 DownloadAction download_action(&prefs, new MockHttpFetcher(&data[0],
114 data.size()));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700115 download_action.SetTestFileWriter(&writer);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000116 BondActions(&feeder_action, &download_action);
Darin Petkov9d911fa2010-08-19 09:36:08 -0700117 DownloadActionDelegateMock download_delegate;
118 if (use_download_delegate) {
119 InSequence s;
120 download_action.set_delegate(&download_delegate);
121 EXPECT_CALL(download_delegate, SetDownloadStatus(true)).Times(1);
122 EXPECT_CALL(download_delegate, BytesReceived(_, _)).Times(AtLeast(1));
123 EXPECT_CALL(download_delegate, SetDownloadStatus(false)).Times(1);
124 }
Darin Petkov50332f12010-09-24 11:44:47 -0700125 ActionExitCode expected_code = kActionCodeSuccess;
126 if (hash_test)
127 expected_code = kActionCodeDownloadHashMismatchError;
128 else if (size_test)
129 expected_code = kActionCodeDownloadSizeMismatchError;
130 DownloadActionTestProcessorDelegate delegate(expected_code);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000131 delegate.loop_ = loop;
132 delegate.expected_data_ = data;
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700133 delegate.path_ = output_temp_file.GetPath();
rspangler@google.com49fdf182009-10-10 00:57:34 +0000134 ActionProcessor processor;
135 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000136 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000137 processor.EnqueueAction(&download_action);
138
139 g_timeout_add(0, &StartProcessorInRunLoop, &processor);
140 g_main_loop_run(loop);
141 g_main_loop_unref(loop);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000142}
143} // namespace {}
144
145TEST(DownloadActionTest, SimpleTest) {
146 vector<char> small;
147 const char* foo = "foo";
148 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700149 TestWithData(small,
150 false, // hash_test
151 false, // size_test
152 true); // use_download_delegate
rspangler@google.com49fdf182009-10-10 00:57:34 +0000153}
154
155TEST(DownloadActionTest, LargeTest) {
156 vector<char> big(5 * kMockHttpFetcherChunkSize);
157 char c = '0';
158 for (unsigned int i = 0; i < big.size(); i++) {
159 big[i] = c;
160 if ('9' == c)
161 c = '0';
162 else
163 c++;
164 }
Darin Petkov50332f12010-09-24 11:44:47 -0700165 TestWithData(big,
166 false, // hash_test
167 false, // size_test
168 true); // use_download_delegate
Darin Petkovc97435c2010-07-20 12:37:43 -0700169}
170
171TEST(DownloadActionTest, BadHashTest) {
172 vector<char> small;
173 const char* foo = "foo";
174 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700175 TestWithData(small,
176 true, // hash_test
177 false, // size_test
178 true); // use_download_delegate
179}
180
181TEST(DownloadActionTest, BadSizeTest) {
182 const char* something = "something";
183 vector<char> small(something, something + strlen(something));
184 TestWithData(small,
185 false, // hash_test
186 true, // size_test
187 true); // use_download_delegate
Darin Petkov9d911fa2010-08-19 09:36:08 -0700188}
189
190TEST(DownloadActionTest, NoDownloadDelegateTest) {
191 vector<char> small;
192 const char* foo = "foofoo";
193 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700194 TestWithData(small,
195 false, // hash_test
196 false, // size_test
197 false); // use_download_delegate
rspangler@google.com49fdf182009-10-10 00:57:34 +0000198}
199
200namespace {
201class TerminateEarlyTestProcessorDelegate : public ActionProcessorDelegate {
202 public:
203 void ProcessingStopped(const ActionProcessor* processor) {
204 ASSERT_TRUE(loop_);
205 g_main_loop_quit(loop_);
206 }
207 GMainLoop *loop_;
208};
209
210gboolean TerminateEarlyTestStarter(gpointer data) {
211 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
212 processor->StartProcessing();
213 CHECK(processor->IsRunning());
214 processor->StopProcessing();
215 return FALSE;
216}
217
Darin Petkov9d911fa2010-08-19 09:36:08 -0700218void TestTerminateEarly(bool use_download_delegate) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000219 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
220
221 vector<char> data(kMockHttpFetcherChunkSize + kMockHttpFetcherChunkSize / 2);
222 memset(&data[0], 0, data.size());
223
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700224 ScopedTempFile temp_file;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000225 {
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700226 DirectFileWriter writer;
227
rspangler@google.com49fdf182009-10-10 00:57:34 +0000228 // takes ownership of passed in HttpFetcher
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000229 ObjectFeederAction<InstallPlan> feeder_action;
Andrew de los Reyes63b96d72010-05-10 13:08:54 -0700230 InstallPlan install_plan(true, "", 0, "", temp_file.GetPath(), "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000231 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700232 PrefsMock prefs;
233 DownloadAction download_action(&prefs,
234 new MockHttpFetcher(&data[0], data.size()));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700235 download_action.SetTestFileWriter(&writer);
Darin Petkov9d911fa2010-08-19 09:36:08 -0700236 DownloadActionDelegateMock download_delegate;
237 if (use_download_delegate) {
238 InSequence s;
239 download_action.set_delegate(&download_delegate);
240 EXPECT_CALL(download_delegate, SetDownloadStatus(true)).Times(1);
241 EXPECT_CALL(download_delegate, SetDownloadStatus(false)).Times(1);
242 }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000243 TerminateEarlyTestProcessorDelegate delegate;
244 delegate.loop_ = loop;
245 ActionProcessor processor;
246 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000247 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000248 processor.EnqueueAction(&download_action);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000249 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000250
251 g_timeout_add(0, &TerminateEarlyTestStarter, &processor);
252 g_main_loop_run(loop);
253 g_main_loop_unref(loop);
254 }
255
256 // 1 or 0 chunks should have come through
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700257 const off_t resulting_file_size(utils::FileSize(temp_file.GetPath()));
258 EXPECT_GE(resulting_file_size, 0);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000259 if (resulting_file_size != 0)
260 EXPECT_EQ(kMockHttpFetcherChunkSize, resulting_file_size);
261}
262
Darin Petkov9d911fa2010-08-19 09:36:08 -0700263} // namespace {}
264
265TEST(DownloadActionTest, TerminateEarlyTest) {
266 TestTerminateEarly(true);
267}
268
269TEST(DownloadActionTest, TerminateEarlyNoDownloadDelegateTest) {
270 TestTerminateEarly(false);
271}
272
rspangler@google.com49fdf182009-10-10 00:57:34 +0000273class DownloadActionTestAction;
274
275template<>
276class ActionTraits<DownloadActionTestAction> {
277 public:
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000278 typedef InstallPlan OutputObjectType;
279 typedef InstallPlan InputObjectType;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000280};
281
282// This is a simple Action class for testing.
283struct DownloadActionTestAction : public Action<DownloadActionTestAction> {
284 DownloadActionTestAction() : did_run_(false) {}
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000285 typedef InstallPlan InputObjectType;
286 typedef InstallPlan OutputObjectType;
287 ActionPipe<InstallPlan>* in_pipe() { return in_pipe_.get(); }
288 ActionPipe<InstallPlan>* out_pipe() { return out_pipe_.get(); }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000289 ActionProcessor* processor() { return processor_; }
290 void PerformAction() {
291 did_run_ = true;
292 ASSERT_TRUE(HasInputObject());
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000293 EXPECT_TRUE(expected_input_object_ == GetInputObject());
rspangler@google.com49fdf182009-10-10 00:57:34 +0000294 ASSERT_TRUE(processor());
Darin Petkovc1a8b422010-07-19 11:34:49 -0700295 processor()->ActionComplete(this, kActionCodeSuccess);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000296 }
297 string Type() const { return "DownloadActionTestAction"; }
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000298 InstallPlan expected_input_object_;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000299 bool did_run_;
300};
301
302namespace {
303// This class is an ActionProcessorDelegate that simply terminates the
304// run loop when the ActionProcessor has completed processing. It's used
305// only by the test PassObjectOutTest.
306class PassObjectOutTestProcessorDelegate : public ActionProcessorDelegate {
307 public:
Darin Petkovc1a8b422010-07-19 11:34:49 -0700308 void ProcessingDone(const ActionProcessor* processor, ActionExitCode code) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000309 ASSERT_TRUE(loop_);
310 g_main_loop_quit(loop_);
311 }
312 GMainLoop *loop_;
313};
314
315gboolean PassObjectOutTestStarter(gpointer data) {
316 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
317 processor->StartProcessing();
318 return FALSE;
319}
320}
321
322TEST(DownloadActionTest, PassObjectOutTest) {
323 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
324
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700325 DirectFileWriter writer;
326
rspangler@google.com49fdf182009-10-10 00:57:34 +0000327 // takes ownership of passed in HttpFetcher
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700328 InstallPlan install_plan(true,
329 "",
Darin Petkov50332f12010-09-24 11:44:47 -0700330 1,
Andrew de los Reyes1e338b82010-01-22 14:57:27 -0800331 OmahaHashCalculator::OmahaHashOfString("x"),
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700332 "/dev/null",
Andrew de los Reyes1e338b82010-01-22 14:57:27 -0800333 "/dev/null");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000334 ObjectFeederAction<InstallPlan> feeder_action;
335 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700336 PrefsMock prefs;
337 DownloadAction download_action(&prefs, new MockHttpFetcher("x", 1));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700338 download_action.SetTestFileWriter(&writer);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000339
340 DownloadActionTestAction test_action;
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000341 test_action.expected_input_object_ = install_plan;
342 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000343 BondActions(&download_action, &test_action);
344
345 ActionProcessor processor;
346 PassObjectOutTestProcessorDelegate delegate;
347 delegate.loop_ = loop;
348 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000349 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000350 processor.EnqueueAction(&download_action);
351 processor.EnqueueAction(&test_action);
352
353 g_timeout_add(0, &PassObjectOutTestStarter, &processor);
354 g_main_loop_run(loop);
355 g_main_loop_unref(loop);
356
357 EXPECT_EQ(true, test_action.did_run_);
358}
359
360TEST(DownloadActionTest, BadOutFileTest) {
361 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
362
363 const string path("/fake/path/that/cant/be/created/because/of/missing/dirs");
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700364 DirectFileWriter writer;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000365
366 // takes ownership of passed in HttpFetcher
Andrew de los Reyes63b96d72010-05-10 13:08:54 -0700367 InstallPlan install_plan(true, "", 0, "", path, "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000368 ObjectFeederAction<InstallPlan> feeder_action;
369 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700370 PrefsMock prefs;
371 DownloadAction download_action(&prefs, new MockHttpFetcher("x", 1));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700372 download_action.SetTestFileWriter(&writer);
Darin Petkovc1a8b422010-07-19 11:34:49 -0700373
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000374 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000375
376 ActionProcessor processor;
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000377 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000378 processor.EnqueueAction(&download_action);
379 processor.StartProcessing();
380 ASSERT_FALSE(processor.IsRunning());
381
382 g_main_loop_unref(loop);
383}
384
385} // namespace chromeos_update_engine