blob: 8135841a9011ba49bddb644f83ecf6a1a9d1f8ab [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));
Darin Petkov9ce452b2010-11-17 14:33:28 -080052 if (expected_code_ != kActionCodeDownloadWriteError) {
53 ASSERT_EQ(expected_data_.size(), found_data.size());
54 for (unsigned i = 0; i < expected_data_.size(); i++) {
55 EXPECT_EQ(expected_data_[i], found_data[i]);
56 }
rspangler@google.com49fdf182009-10-10 00:57:34 +000057 }
58 processing_done_called_ = true;
59 }
60
adlr@google.comc98a7ed2009-12-04 18:54:03 +000061 virtual void ActionCompleted(ActionProcessor* processor,
62 AbstractAction* action,
Darin Petkovc1a8b422010-07-19 11:34:49 -070063 ActionExitCode code) {
Darin Petkovc97435c2010-07-20 12:37:43 -070064 const string type = action->Type();
65 if (type == DownloadAction::StaticType()) {
66 EXPECT_EQ(expected_code_, code);
67 } else {
68 EXPECT_EQ(kActionCodeSuccess, code);
69 }
rspangler@google.com49fdf182009-10-10 00:57:34 +000070 }
71
72 GMainLoop *loop_;
73 string path_;
74 vector<char> expected_data_;
75 bool processing_done_called_;
Darin Petkovc97435c2010-07-20 12:37:43 -070076 ActionExitCode expected_code_;
rspangler@google.com49fdf182009-10-10 00:57:34 +000077};
78
Darin Petkov9ce452b2010-11-17 14:33:28 -080079class TestDirectFileWriter : public DirectFileWriter {
80 public:
81 TestDirectFileWriter() : fail_write_(0), current_write_(0) {}
82 void set_fail_write(int fail_write) { fail_write_ = fail_write; }
83
84 virtual ssize_t Write(const void* bytes, size_t count) {
85 if (++current_write_ == fail_write_) {
86 return -EINVAL;
87 }
88 return DirectFileWriter::Write(bytes, count);
89 }
90
91 private:
92 // If positive, fail on the |fail_write_| call to Write.
93 int fail_write_;
94 int current_write_;
95};
96
rspangler@google.com49fdf182009-10-10 00:57:34 +000097struct EntryPointArgs {
98 const vector<char> *data;
99 GMainLoop *loop;
100 ActionProcessor *processor;
101};
102
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700103struct StartProcessorInRunLoopArgs {
104 ActionProcessor* processor;
105 MockHttpFetcher* http_fetcher;
106};
107
rspangler@google.com49fdf182009-10-10 00:57:34 +0000108gboolean StartProcessorInRunLoop(gpointer data) {
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700109 ActionProcessor* processor =
110 reinterpret_cast<StartProcessorInRunLoopArgs*>(data)->processor;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000111 processor->StartProcessing();
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700112 MockHttpFetcher* http_fetcher =
113 reinterpret_cast<StartProcessorInRunLoopArgs*>(data)->http_fetcher;
114 http_fetcher->SetOffset(1);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000115 return FALSE;
116}
117
Darin Petkov9d911fa2010-08-19 09:36:08 -0700118void TestWithData(const vector<char>& data,
119 bool hash_test,
Darin Petkov50332f12010-09-24 11:44:47 -0700120 bool size_test,
Darin Petkov9ce452b2010-11-17 14:33:28 -0800121 int fail_write,
Darin Petkov9d911fa2010-08-19 09:36:08 -0700122 bool use_download_delegate) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000123 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
124
125 // TODO(adlr): see if we need a different file for build bots
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700126 ScopedTempFile output_temp_file;
Darin Petkov9ce452b2010-11-17 14:33:28 -0800127 TestDirectFileWriter writer;
128 writer.set_fail_write(fail_write);
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700129
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700130 // We pull off the first byte from data and seek past it.
131
Darin Petkovc97435c2010-07-20 12:37:43 -0700132 string hash = hash_test ?
133 OmahaHashCalculator::OmahaHashOfString("random string") :
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700134 OmahaHashCalculator::OmahaHashOfBytes(&data[1], data.size() - 1);
Darin Petkov50332f12010-09-24 11:44:47 -0700135 uint64_t size = data.size() + (size_test ? 1 : 0);
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700136 InstallPlan install_plan(true,
Darin Petkov0406e402010-10-06 21:33:11 -0700137 false,
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700138 "",
Darin Petkov50332f12010-09-24 11:44:47 -0700139 size,
Darin Petkovc97435c2010-07-20 12:37:43 -0700140 hash,
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700141 output_temp_file.GetPath(),
142 "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000143 ObjectFeederAction<InstallPlan> feeder_action;
144 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700145 PrefsMock prefs;
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700146 MockHttpFetcher* http_fetcher = new MockHttpFetcher(&data[0], data.size());
147 // takes ownership of passed in HttpFetcher
148 DownloadAction download_action(&prefs, http_fetcher);
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700149 download_action.SetTestFileWriter(&writer);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000150 BondActions(&feeder_action, &download_action);
Darin Petkov9d911fa2010-08-19 09:36:08 -0700151 DownloadActionDelegateMock download_delegate;
152 if (use_download_delegate) {
153 InSequence s;
154 download_action.set_delegate(&download_delegate);
155 EXPECT_CALL(download_delegate, SetDownloadStatus(true)).Times(1);
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700156 if (data.size() > kMockHttpFetcherChunkSize)
157 EXPECT_CALL(download_delegate,
158 BytesReceived(1 + kMockHttpFetcherChunkSize, _));
Darin Petkov9ce452b2010-11-17 14:33:28 -0800159 EXPECT_CALL(download_delegate, BytesReceived(_, _)).Times(AtLeast(1));
Darin Petkov9d911fa2010-08-19 09:36:08 -0700160 EXPECT_CALL(download_delegate, SetDownloadStatus(false)).Times(1);
161 }
Darin Petkov50332f12010-09-24 11:44:47 -0700162 ActionExitCode expected_code = kActionCodeSuccess;
163 if (hash_test)
164 expected_code = kActionCodeDownloadHashMismatchError;
165 else if (size_test)
166 expected_code = kActionCodeDownloadSizeMismatchError;
Darin Petkov9ce452b2010-11-17 14:33:28 -0800167 else if (fail_write > 0)
168 expected_code = kActionCodeDownloadWriteError;
Darin Petkov50332f12010-09-24 11:44:47 -0700169 DownloadActionTestProcessorDelegate delegate(expected_code);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000170 delegate.loop_ = loop;
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700171 delegate.expected_data_ = vector<char>(data.begin() + 1, data.end());
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700172 delegate.path_ = output_temp_file.GetPath();
rspangler@google.com49fdf182009-10-10 00:57:34 +0000173 ActionProcessor processor;
174 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000175 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000176 processor.EnqueueAction(&download_action);
177
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700178 StartProcessorInRunLoopArgs args;
179 args.processor = &processor;
180 args.http_fetcher = http_fetcher;
181 g_timeout_add(0, &StartProcessorInRunLoop, &args);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000182 g_main_loop_run(loop);
183 g_main_loop_unref(loop);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000184}
185} // namespace {}
186
187TEST(DownloadActionTest, SimpleTest) {
188 vector<char> small;
189 const char* foo = "foo";
190 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700191 TestWithData(small,
192 false, // hash_test
193 false, // size_test
Darin Petkov9ce452b2010-11-17 14:33:28 -0800194 0, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700195 true); // use_download_delegate
rspangler@google.com49fdf182009-10-10 00:57:34 +0000196}
197
198TEST(DownloadActionTest, LargeTest) {
199 vector<char> big(5 * kMockHttpFetcherChunkSize);
200 char c = '0';
201 for (unsigned int i = 0; i < big.size(); i++) {
202 big[i] = c;
Darin Petkov9ce452b2010-11-17 14:33:28 -0800203 c = ('9' == c) ? '0' : c + 1;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000204 }
Darin Petkov50332f12010-09-24 11:44:47 -0700205 TestWithData(big,
206 false, // hash_test
207 false, // size_test
Darin Petkov9ce452b2010-11-17 14:33:28 -0800208 0, // fail_write
209 true); // use_download_delegate
210}
211
212TEST(DownloadActionTest, FailWriteTest) {
213 vector<char> big(5 * kMockHttpFetcherChunkSize);
214 char c = '0';
215 for (unsigned int i = 0; i < big.size(); i++) {
216 big[i] = c;
217 c = ('9' == c) ? '0' : c + 1;
218 }
219 TestWithData(big,
220 false, // hash_test
221 false, // size_test
222 2, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700223 true); // use_download_delegate
Darin Petkovc97435c2010-07-20 12:37:43 -0700224}
225
226TEST(DownloadActionTest, BadHashTest) {
227 vector<char> small;
228 const char* foo = "foo";
229 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700230 TestWithData(small,
231 true, // hash_test
232 false, // size_test
Darin Petkov9ce452b2010-11-17 14:33:28 -0800233 0, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700234 true); // use_download_delegate
235}
236
237TEST(DownloadActionTest, BadSizeTest) {
238 const char* something = "something";
239 vector<char> small(something, something + strlen(something));
240 TestWithData(small,
241 false, // hash_test
242 true, // size_test
Darin Petkov9ce452b2010-11-17 14:33:28 -0800243 0, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700244 true); // use_download_delegate
Darin Petkov9d911fa2010-08-19 09:36:08 -0700245}
246
247TEST(DownloadActionTest, NoDownloadDelegateTest) {
248 vector<char> small;
249 const char* foo = "foofoo";
250 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700251 TestWithData(small,
252 false, // hash_test
253 false, // size_test
Darin Petkov9ce452b2010-11-17 14:33:28 -0800254 0, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700255 false); // use_download_delegate
rspangler@google.com49fdf182009-10-10 00:57:34 +0000256}
257
258namespace {
259class TerminateEarlyTestProcessorDelegate : public ActionProcessorDelegate {
260 public:
261 void ProcessingStopped(const ActionProcessor* processor) {
262 ASSERT_TRUE(loop_);
263 g_main_loop_quit(loop_);
264 }
265 GMainLoop *loop_;
266};
267
268gboolean TerminateEarlyTestStarter(gpointer data) {
269 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
270 processor->StartProcessing();
271 CHECK(processor->IsRunning());
272 processor->StopProcessing();
273 return FALSE;
274}
275
Darin Petkov9d911fa2010-08-19 09:36:08 -0700276void TestTerminateEarly(bool use_download_delegate) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000277 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
278
279 vector<char> data(kMockHttpFetcherChunkSize + kMockHttpFetcherChunkSize / 2);
280 memset(&data[0], 0, data.size());
281
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700282 ScopedTempFile temp_file;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000283 {
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700284 DirectFileWriter writer;
285
rspangler@google.com49fdf182009-10-10 00:57:34 +0000286 // takes ownership of passed in HttpFetcher
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000287 ObjectFeederAction<InstallPlan> feeder_action;
Darin Petkov0406e402010-10-06 21:33:11 -0700288 InstallPlan install_plan(true, false, "", 0, "", temp_file.GetPath(), "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000289 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700290 PrefsMock prefs;
291 DownloadAction download_action(&prefs,
292 new MockHttpFetcher(&data[0], data.size()));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700293 download_action.SetTestFileWriter(&writer);
Darin Petkov9d911fa2010-08-19 09:36:08 -0700294 DownloadActionDelegateMock download_delegate;
295 if (use_download_delegate) {
296 InSequence s;
297 download_action.set_delegate(&download_delegate);
298 EXPECT_CALL(download_delegate, SetDownloadStatus(true)).Times(1);
299 EXPECT_CALL(download_delegate, SetDownloadStatus(false)).Times(1);
300 }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000301 TerminateEarlyTestProcessorDelegate delegate;
302 delegate.loop_ = loop;
303 ActionProcessor processor;
304 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000305 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000306 processor.EnqueueAction(&download_action);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000307 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000308
309 g_timeout_add(0, &TerminateEarlyTestStarter, &processor);
310 g_main_loop_run(loop);
311 g_main_loop_unref(loop);
312 }
313
314 // 1 or 0 chunks should have come through
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700315 const off_t resulting_file_size(utils::FileSize(temp_file.GetPath()));
316 EXPECT_GE(resulting_file_size, 0);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000317 if (resulting_file_size != 0)
318 EXPECT_EQ(kMockHttpFetcherChunkSize, resulting_file_size);
319}
320
Darin Petkov9d911fa2010-08-19 09:36:08 -0700321} // namespace {}
322
323TEST(DownloadActionTest, TerminateEarlyTest) {
324 TestTerminateEarly(true);
325}
326
327TEST(DownloadActionTest, TerminateEarlyNoDownloadDelegateTest) {
328 TestTerminateEarly(false);
329}
330
rspangler@google.com49fdf182009-10-10 00:57:34 +0000331class DownloadActionTestAction;
332
333template<>
334class ActionTraits<DownloadActionTestAction> {
335 public:
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000336 typedef InstallPlan OutputObjectType;
337 typedef InstallPlan InputObjectType;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000338};
339
340// This is a simple Action class for testing.
341struct DownloadActionTestAction : public Action<DownloadActionTestAction> {
342 DownloadActionTestAction() : did_run_(false) {}
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000343 typedef InstallPlan InputObjectType;
344 typedef InstallPlan OutputObjectType;
345 ActionPipe<InstallPlan>* in_pipe() { return in_pipe_.get(); }
346 ActionPipe<InstallPlan>* out_pipe() { return out_pipe_.get(); }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000347 ActionProcessor* processor() { return processor_; }
348 void PerformAction() {
349 did_run_ = true;
350 ASSERT_TRUE(HasInputObject());
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000351 EXPECT_TRUE(expected_input_object_ == GetInputObject());
rspangler@google.com49fdf182009-10-10 00:57:34 +0000352 ASSERT_TRUE(processor());
Darin Petkovc1a8b422010-07-19 11:34:49 -0700353 processor()->ActionComplete(this, kActionCodeSuccess);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000354 }
355 string Type() const { return "DownloadActionTestAction"; }
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000356 InstallPlan expected_input_object_;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000357 bool did_run_;
358};
359
360namespace {
361// This class is an ActionProcessorDelegate that simply terminates the
362// run loop when the ActionProcessor has completed processing. It's used
363// only by the test PassObjectOutTest.
364class PassObjectOutTestProcessorDelegate : public ActionProcessorDelegate {
365 public:
Darin Petkovc1a8b422010-07-19 11:34:49 -0700366 void ProcessingDone(const ActionProcessor* processor, ActionExitCode code) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000367 ASSERT_TRUE(loop_);
368 g_main_loop_quit(loop_);
369 }
370 GMainLoop *loop_;
371};
372
373gboolean PassObjectOutTestStarter(gpointer data) {
374 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
375 processor->StartProcessing();
376 return FALSE;
377}
378}
379
380TEST(DownloadActionTest, PassObjectOutTest) {
381 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
382
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700383 DirectFileWriter writer;
384
rspangler@google.com49fdf182009-10-10 00:57:34 +0000385 // takes ownership of passed in HttpFetcher
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700386 InstallPlan install_plan(true,
Darin Petkov0406e402010-10-06 21:33:11 -0700387 false,
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700388 "",
Darin Petkov50332f12010-09-24 11:44:47 -0700389 1,
Andrew de los Reyes1e338b82010-01-22 14:57:27 -0800390 OmahaHashCalculator::OmahaHashOfString("x"),
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700391 "/dev/null",
Andrew de los Reyes1e338b82010-01-22 14:57:27 -0800392 "/dev/null");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000393 ObjectFeederAction<InstallPlan> feeder_action;
394 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700395 PrefsMock prefs;
396 DownloadAction download_action(&prefs, new MockHttpFetcher("x", 1));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700397 download_action.SetTestFileWriter(&writer);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000398
399 DownloadActionTestAction test_action;
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000400 test_action.expected_input_object_ = install_plan;
401 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000402 BondActions(&download_action, &test_action);
403
404 ActionProcessor processor;
405 PassObjectOutTestProcessorDelegate delegate;
406 delegate.loop_ = loop;
407 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000408 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000409 processor.EnqueueAction(&download_action);
410 processor.EnqueueAction(&test_action);
411
412 g_timeout_add(0, &PassObjectOutTestStarter, &processor);
413 g_main_loop_run(loop);
414 g_main_loop_unref(loop);
415
416 EXPECT_EQ(true, test_action.did_run_);
417}
418
419TEST(DownloadActionTest, BadOutFileTest) {
420 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
421
422 const string path("/fake/path/that/cant/be/created/because/of/missing/dirs");
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700423 DirectFileWriter writer;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000424
425 // takes ownership of passed in HttpFetcher
Darin Petkov0406e402010-10-06 21:33:11 -0700426 InstallPlan install_plan(true, false, "", 0, "", path, "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000427 ObjectFeederAction<InstallPlan> feeder_action;
428 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700429 PrefsMock prefs;
430 DownloadAction download_action(&prefs, new MockHttpFetcher("x", 1));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700431 download_action.SetTestFileWriter(&writer);
Darin Petkovc1a8b422010-07-19 11:34:49 -0700432
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000433 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000434
435 ActionProcessor processor;
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000436 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000437 processor.EnqueueAction(&download_action);
438 processor.StartProcessing();
439 ASSERT_FALSE(processor.IsRunning());
440
441 g_main_loop_unref(loop);
442}
443
444} // namespace chromeos_update_engine