blob: 3911d55365d00948c174eee3ca3a79ec7fe67261 [file] [log] [blame]
Darin Petkov7ed561b2011-10-04 02:59:03 -07001// Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
rspangler@google.com49fdf182009-10-10 00:57:34 +00002// 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:
David Zeuthena99981f2013-04-29 13:42:47 -070039 explicit DownloadActionTestProcessorDelegate(ErrorCode expected_code)
Darin Petkovc97435c2010-07-20 12:37:43 -070040 : 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,
David Zeuthena99981f2013-04-29 13:42:47 -070047 ErrorCode 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));
David Zeuthena99981f2013-04-29 13:42:47 -070052 if (expected_code_ != kErrorCodeDownloadWriteError) {
Darin Petkov9ce452b2010-11-17 14:33:28 -080053 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,
David Zeuthena99981f2013-04-29 13:42:47 -070063 ErrorCode 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 {
David Zeuthena99981f2013-04-29 13:42:47 -070068 EXPECT_EQ(kErrorCodeSuccess, code);
Darin Petkovc97435c2010-07-20 12:37:43 -070069 }
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_;
David Zeuthena99981f2013-04-29 13:42:47 -070076 ErrorCode 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
Don Garrette410e0f2011-11-10 15:39:01 -080084 virtual bool Write(const void* bytes, size_t count) {
Darin Petkov9ce452b2010-11-17 14:33:28 -080085 if (++current_write_ == fail_write_) {
Don Garrette410e0f2011-11-10 15:39:01 -080086 return false;
Darin Petkov9ce452b2010-11-17 14:33:28 -080087 }
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,
Darin Petkov9ce452b2010-11-17 14:33:28 -0800119 int fail_write,
Darin Petkov9d911fa2010-08-19 09:36:08 -0700120 bool use_download_delegate) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000121 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
122
123 // TODO(adlr): see if we need a different file for build bots
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700124 ScopedTempFile output_temp_file;
Darin Petkov9ce452b2010-11-17 14:33:28 -0800125 TestDirectFileWriter writer;
126 writer.set_fail_write(fail_write);
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700127
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700128 // We pull off the first byte from data and seek past it.
129
Darin Petkov7ed561b2011-10-04 02:59:03 -0700130 string hash =
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700131 OmahaHashCalculator::OmahaHashOfBytes(&data[1], data.size() - 1);
Darin Petkov7ed561b2011-10-04 02:59:03 -0700132 uint64_t size = data.size();
133 InstallPlan install_plan(false,
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700134 "",
Darin Petkov50332f12010-09-24 11:44:47 -0700135 size,
Darin Petkovc97435c2010-07-20 12:37:43 -0700136 hash,
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700137 0,
138 "",
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700139 output_temp_file.GetPath(),
140 "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000141 ObjectFeederAction<InstallPlan> feeder_action;
142 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700143 PrefsMock prefs;
Andrew de los Reyes45168102010-11-22 11:13:50 -0800144 MockHttpFetcher* http_fetcher = new MockHttpFetcher(&data[0],
145 data.size(),
146 NULL);
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700147 // takes ownership of passed in HttpFetcher
Jay Srinivasanf0572052012-10-23 18:12:56 -0700148 DownloadAction download_action(&prefs, NULL, 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 }
David Zeuthena99981f2013-04-29 13:42:47 -0700162 ErrorCode expected_code = kErrorCodeSuccess;
Darin Petkov7ed561b2011-10-04 02:59:03 -0700163 if (fail_write > 0)
David Zeuthena99981f2013-04-29 13:42:47 -0700164 expected_code = kErrorCodeDownloadWriteError;
Darin Petkov50332f12010-09-24 11:44:47 -0700165 DownloadActionTestProcessorDelegate delegate(expected_code);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000166 delegate.loop_ = loop;
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700167 delegate.expected_data_ = vector<char>(data.begin() + 1, data.end());
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700168 delegate.path_ = output_temp_file.GetPath();
rspangler@google.com49fdf182009-10-10 00:57:34 +0000169 ActionProcessor processor;
170 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000171 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000172 processor.EnqueueAction(&download_action);
173
Andrew de los Reyes34e41a12010-10-26 20:07:58 -0700174 StartProcessorInRunLoopArgs args;
175 args.processor = &processor;
176 args.http_fetcher = http_fetcher;
177 g_timeout_add(0, &StartProcessorInRunLoop, &args);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000178 g_main_loop_run(loop);
179 g_main_loop_unref(loop);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000180}
181} // namespace {}
182
183TEST(DownloadActionTest, SimpleTest) {
184 vector<char> small;
185 const char* foo = "foo";
186 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700187 TestWithData(small,
Darin Petkov9ce452b2010-11-17 14:33:28 -0800188 0, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700189 true); // use_download_delegate
rspangler@google.com49fdf182009-10-10 00:57:34 +0000190}
191
192TEST(DownloadActionTest, LargeTest) {
193 vector<char> big(5 * kMockHttpFetcherChunkSize);
194 char c = '0';
195 for (unsigned int i = 0; i < big.size(); i++) {
196 big[i] = c;
Darin Petkov9ce452b2010-11-17 14:33:28 -0800197 c = ('9' == c) ? '0' : c + 1;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000198 }
Darin Petkov50332f12010-09-24 11:44:47 -0700199 TestWithData(big,
Darin Petkov9ce452b2010-11-17 14:33:28 -0800200 0, // fail_write
201 true); // use_download_delegate
202}
203
204TEST(DownloadActionTest, FailWriteTest) {
205 vector<char> big(5 * kMockHttpFetcherChunkSize);
206 char c = '0';
207 for (unsigned int i = 0; i < big.size(); i++) {
208 big[i] = c;
209 c = ('9' == c) ? '0' : c + 1;
210 }
211 TestWithData(big,
Darin Petkov9ce452b2010-11-17 14:33:28 -0800212 2, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700213 true); // use_download_delegate
Darin Petkovc97435c2010-07-20 12:37:43 -0700214}
215
Darin Petkov9d911fa2010-08-19 09:36:08 -0700216TEST(DownloadActionTest, NoDownloadDelegateTest) {
217 vector<char> small;
218 const char* foo = "foofoo";
219 small.insert(small.end(), foo, foo + strlen(foo));
Darin Petkov50332f12010-09-24 11:44:47 -0700220 TestWithData(small,
Darin Petkov9ce452b2010-11-17 14:33:28 -0800221 0, // fail_write
Darin Petkov50332f12010-09-24 11:44:47 -0700222 false); // use_download_delegate
rspangler@google.com49fdf182009-10-10 00:57:34 +0000223}
224
225namespace {
226class TerminateEarlyTestProcessorDelegate : public ActionProcessorDelegate {
227 public:
228 void ProcessingStopped(const ActionProcessor* processor) {
229 ASSERT_TRUE(loop_);
230 g_main_loop_quit(loop_);
231 }
232 GMainLoop *loop_;
233};
234
235gboolean TerminateEarlyTestStarter(gpointer data) {
236 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
237 processor->StartProcessing();
238 CHECK(processor->IsRunning());
239 processor->StopProcessing();
240 return FALSE;
241}
242
Darin Petkov9d911fa2010-08-19 09:36:08 -0700243void TestTerminateEarly(bool use_download_delegate) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000244 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
245
246 vector<char> data(kMockHttpFetcherChunkSize + kMockHttpFetcherChunkSize / 2);
247 memset(&data[0], 0, data.size());
248
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700249 ScopedTempFile temp_file;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000250 {
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700251 DirectFileWriter writer;
252
rspangler@google.com49fdf182009-10-10 00:57:34 +0000253 // takes ownership of passed in HttpFetcher
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000254 ObjectFeederAction<InstallPlan> feeder_action;
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700255 InstallPlan install_plan(false, "", 0, "", 0, "", temp_file.GetPath(), "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000256 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700257 PrefsMock prefs;
Jay Srinivasanf0572052012-10-23 18:12:56 -0700258 DownloadAction download_action(&prefs, NULL,
Andrew de los Reyes45168102010-11-22 11:13:50 -0800259 new MockHttpFetcher(&data[0],
260 data.size(),
261 NULL));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700262 download_action.SetTestFileWriter(&writer);
Darin Petkov9d911fa2010-08-19 09:36:08 -0700263 DownloadActionDelegateMock download_delegate;
264 if (use_download_delegate) {
265 InSequence s;
266 download_action.set_delegate(&download_delegate);
267 EXPECT_CALL(download_delegate, SetDownloadStatus(true)).Times(1);
268 EXPECT_CALL(download_delegate, SetDownloadStatus(false)).Times(1);
269 }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000270 TerminateEarlyTestProcessorDelegate delegate;
271 delegate.loop_ = loop;
272 ActionProcessor processor;
273 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000274 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000275 processor.EnqueueAction(&download_action);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000276 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000277
278 g_timeout_add(0, &TerminateEarlyTestStarter, &processor);
279 g_main_loop_run(loop);
280 g_main_loop_unref(loop);
281 }
282
283 // 1 or 0 chunks should have come through
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700284 const off_t resulting_file_size(utils::FileSize(temp_file.GetPath()));
285 EXPECT_GE(resulting_file_size, 0);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000286 if (resulting_file_size != 0)
287 EXPECT_EQ(kMockHttpFetcherChunkSize, resulting_file_size);
288}
289
Darin Petkov9d911fa2010-08-19 09:36:08 -0700290} // namespace {}
291
292TEST(DownloadActionTest, TerminateEarlyTest) {
293 TestTerminateEarly(true);
294}
295
296TEST(DownloadActionTest, TerminateEarlyNoDownloadDelegateTest) {
297 TestTerminateEarly(false);
298}
299
rspangler@google.com49fdf182009-10-10 00:57:34 +0000300class DownloadActionTestAction;
301
302template<>
303class ActionTraits<DownloadActionTestAction> {
304 public:
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000305 typedef InstallPlan OutputObjectType;
306 typedef InstallPlan InputObjectType;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000307};
308
309// This is a simple Action class for testing.
Yunlian Jiang2dac5762013-04-12 09:53:09 -0700310class DownloadActionTestAction : public Action<DownloadActionTestAction> {
311 public:
rspangler@google.com49fdf182009-10-10 00:57:34 +0000312 DownloadActionTestAction() : did_run_(false) {}
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000313 typedef InstallPlan InputObjectType;
314 typedef InstallPlan OutputObjectType;
315 ActionPipe<InstallPlan>* in_pipe() { return in_pipe_.get(); }
316 ActionPipe<InstallPlan>* out_pipe() { return out_pipe_.get(); }
rspangler@google.com49fdf182009-10-10 00:57:34 +0000317 ActionProcessor* processor() { return processor_; }
318 void PerformAction() {
319 did_run_ = true;
320 ASSERT_TRUE(HasInputObject());
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000321 EXPECT_TRUE(expected_input_object_ == GetInputObject());
rspangler@google.com49fdf182009-10-10 00:57:34 +0000322 ASSERT_TRUE(processor());
David Zeuthena99981f2013-04-29 13:42:47 -0700323 processor()->ActionComplete(this, kErrorCodeSuccess);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000324 }
325 string Type() const { return "DownloadActionTestAction"; }
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000326 InstallPlan expected_input_object_;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000327 bool did_run_;
328};
329
330namespace {
331// This class is an ActionProcessorDelegate that simply terminates the
332// run loop when the ActionProcessor has completed processing. It's used
333// only by the test PassObjectOutTest.
334class PassObjectOutTestProcessorDelegate : public ActionProcessorDelegate {
335 public:
David Zeuthena99981f2013-04-29 13:42:47 -0700336 void ProcessingDone(const ActionProcessor* processor, ErrorCode code) {
rspangler@google.com49fdf182009-10-10 00:57:34 +0000337 ASSERT_TRUE(loop_);
338 g_main_loop_quit(loop_);
339 }
340 GMainLoop *loop_;
341};
342
343gboolean PassObjectOutTestStarter(gpointer data) {
344 ActionProcessor *processor = reinterpret_cast<ActionProcessor*>(data);
345 processor->StartProcessing();
346 return FALSE;
347}
348}
349
350TEST(DownloadActionTest, PassObjectOutTest) {
351 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
352
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700353 DirectFileWriter writer;
354
rspangler@google.com49fdf182009-10-10 00:57:34 +0000355 // takes ownership of passed in HttpFetcher
Darin Petkov7ed561b2011-10-04 02:59:03 -0700356 InstallPlan install_plan(false,
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700357 "",
Darin Petkov50332f12010-09-24 11:44:47 -0700358 1,
Andrew de los Reyes1e338b82010-01-22 14:57:27 -0800359 OmahaHashCalculator::OmahaHashOfString("x"),
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700360 0,
361 "",
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700362 "/dev/null",
Andrew de los Reyes1e338b82010-01-22 14:57:27 -0800363 "/dev/null");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000364 ObjectFeederAction<InstallPlan> feeder_action;
365 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700366 PrefsMock prefs;
Jay Srinivasanf0572052012-10-23 18:12:56 -0700367 DownloadAction download_action(&prefs, NULL,
368 new MockHttpFetcher("x", 1, NULL));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700369 download_action.SetTestFileWriter(&writer);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000370
371 DownloadActionTestAction test_action;
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000372 test_action.expected_input_object_ = install_plan;
373 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000374 BondActions(&download_action, &test_action);
375
376 ActionProcessor processor;
377 PassObjectOutTestProcessorDelegate delegate;
378 delegate.loop_ = loop;
379 processor.set_delegate(&delegate);
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000380 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000381 processor.EnqueueAction(&download_action);
382 processor.EnqueueAction(&test_action);
383
384 g_timeout_add(0, &PassObjectOutTestStarter, &processor);
385 g_main_loop_run(loop);
386 g_main_loop_unref(loop);
387
388 EXPECT_EQ(true, test_action.did_run_);
389}
390
391TEST(DownloadActionTest, BadOutFileTest) {
392 GMainLoop *loop = g_main_loop_new(g_main_context_default(), FALSE);
393
394 const string path("/fake/path/that/cant/be/created/because/of/missing/dirs");
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700395 DirectFileWriter writer;
rspangler@google.com49fdf182009-10-10 00:57:34 +0000396
397 // takes ownership of passed in HttpFetcher
Jay Srinivasan51dcf262012-09-13 17:24:32 -0700398 InstallPlan install_plan(false, "", 0, "", 0, "", path, "");
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000399 ObjectFeederAction<InstallPlan> feeder_action;
400 feeder_action.set_obj(install_plan);
Darin Petkov73058b42010-10-06 16:32:19 -0700401 PrefsMock prefs;
Jay Srinivasanf0572052012-10-23 18:12:56 -0700402 DownloadAction download_action(&prefs, NULL,
403 new MockHttpFetcher("x", 1, NULL));
Andrew de los Reyesf9185172010-05-03 11:07:05 -0700404 download_action.SetTestFileWriter(&writer);
Darin Petkovc1a8b422010-07-19 11:34:49 -0700405
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000406 BondActions(&feeder_action, &download_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000407
408 ActionProcessor processor;
adlr@google.comc98a7ed2009-12-04 18:54:03 +0000409 processor.EnqueueAction(&feeder_action);
rspangler@google.com49fdf182009-10-10 00:57:34 +0000410 processor.EnqueueAction(&download_action);
411 processor.StartProcessing();
412 ASSERT_FALSE(processor.IsRunning());
413
414 g_main_loop_unref(loop);
415}
416
417} // namespace chromeos_update_engine