blob: bdec86f13813c9f8fb36bb7aa27e4b2fd5c54943 [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2011 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
adlr@google.com3defe6a2009-12-04 20:57:17 +000016
Alex Deymoaab50e32014-11-10 19:55:35 -080017#include "update_engine/omaha_response_handler_action.h"
18
Aaron Woodc73fdc12017-12-06 11:09:15 -080019#include <memory>
adlr@google.com3defe6a2009-12-04 20:57:17 +000020#include <string>
Amin Hassanid3f4bea2018-04-30 14:52:40 -070021#include <utility>
Darin Petkov73058b42010-10-06 16:32:19 -070022
Alex Deymo110e0302015-10-19 20:35:21 -070023#include <base/files/file_util.h>
Sen Jiang297e5832016-03-17 14:45:51 -070024#include <base/files/scoped_temp_dir.h>
Aaron Wood23bd3392017-10-06 14:48:25 -070025#include <brillo/message_loops/fake_message_loop.h>
adlr@google.com3defe6a2009-12-04 20:57:17 +000026#include <gtest/gtest.h>
Darin Petkov73058b42010-10-06 16:32:19 -070027
Alex Deymo39910dc2015-11-09 17:04:30 -080028#include "update_engine/common/constants.h"
29#include "update_engine/common/platform_constants.h"
30#include "update_engine/common/test_utils.h"
31#include "update_engine/common/utils.h"
Gilad Arnold5bb4c902014-04-10 12:32:13 -070032#include "update_engine/fake_system_state.h"
Gilad Arnold74b5f552014-10-07 08:17:16 -070033#include "update_engine/mock_payload_state.h"
Alex Deymo39910dc2015-11-09 17:04:30 -080034#include "update_engine/payload_consumer/payload_constants.h"
Aaron Wood23bd3392017-10-06 14:48:25 -070035#include "update_engine/update_manager/mock_policy.h"
adlr@google.com3defe6a2009-12-04 20:57:17 +000036
Alex Deymo10875d92014-11-10 21:52:57 -080037using chromeos_update_engine::test_utils::System;
38using chromeos_update_engine::test_utils::WriteFileString;
Aaron Wood23bd3392017-10-06 14:48:25 -070039using chromeos_update_manager::EvalStatus;
40using chromeos_update_manager::FakeUpdateManager;
41using chromeos_update_manager::MockPolicy;
adlr@google.com3defe6a2009-12-04 20:57:17 +000042using std::string;
Aaron Woodc73fdc12017-12-06 11:09:15 -080043using testing::_;
Aaron Wood23bd3392017-10-06 14:48:25 -070044using testing::DoAll;
Darin Petkov73058b42010-10-06 16:32:19 -070045using testing::Return;
Aaron Wood23bd3392017-10-06 14:48:25 -070046using testing::SetArgPointee;
adlr@google.com3defe6a2009-12-04 20:57:17 +000047
48namespace chromeos_update_engine {
49
Amin Hassanid3f4bea2018-04-30 14:52:40 -070050class OmahaResponseHandlerActionProcessorDelegate
51 : public ActionProcessorDelegate {
52 public:
53 OmahaResponseHandlerActionProcessorDelegate()
54 : code_(ErrorCode::kError), code_set_(false) {}
55 void ActionCompleted(ActionProcessor* processor,
56 AbstractAction* action,
57 ErrorCode code) {
58 if (action->Type() == OmahaResponseHandlerAction::StaticType()) {
59 auto response_handler_action =
60 static_cast<OmahaResponseHandlerAction*>(action);
61 code_ = code;
62 code_set_ = true;
63 response_handler_action_install_plan_.reset(
64 new InstallPlan(response_handler_action->install_plan_));
65 } else if (action->Type() ==
66 ObjectCollectorAction<InstallPlan>::StaticType()) {
67 auto collector_action =
68 static_cast<ObjectCollectorAction<InstallPlan>*>(action);
69 collector_action_install_plan_.reset(
70 new InstallPlan(collector_action->object()));
71 }
72 }
73 ErrorCode code_;
74 bool code_set_;
75 std::unique_ptr<InstallPlan> collector_action_install_plan_;
76 std::unique_ptr<InstallPlan> response_handler_action_install_plan_;
77};
78
adlr@google.com3defe6a2009-12-04 20:57:17 +000079class OmahaResponseHandlerActionTest : public ::testing::Test {
Alex Deymo763e7db2015-08-27 21:08:08 -070080 protected:
81 void SetUp() override {
82 FakeBootControl* fake_boot_control = fake_system_state_.fake_boot_control();
83 fake_boot_control->SetPartitionDevice(
Tudor Brindusdda79e22018-06-28 18:03:21 -070084 kPartitionNameKernel, 0, "/dev/sdz2");
Alex Deymo763e7db2015-08-27 21:08:08 -070085 fake_boot_control->SetPartitionDevice(
Tudor Brindusdda79e22018-06-28 18:03:21 -070086 kPartitionNameRoot, 0, "/dev/sdz3");
Alex Deymo763e7db2015-08-27 21:08:08 -070087 fake_boot_control->SetPartitionDevice(
Tudor Brindusdda79e22018-06-28 18:03:21 -070088 kPartitionNameKernel, 1, "/dev/sdz4");
Alex Deymo763e7db2015-08-27 21:08:08 -070089 fake_boot_control->SetPartitionDevice(
Tudor Brindusdda79e22018-06-28 18:03:21 -070090 kPartitionNameRoot, 1, "/dev/sdz5");
Alex Deymo763e7db2015-08-27 21:08:08 -070091 }
92
adlr@google.com3defe6a2009-12-04 20:57:17 +000093 // Return true iff the OmahaResponseHandlerAction succeeded.
Alex Vakulenko88b591f2014-08-28 16:48:57 -070094 // If out is non-null, it's set w/ the response from the action.
Darin Petkov6a5b3222010-07-13 14:55:28 -070095 bool DoTest(const OmahaResponse& in,
Gilad Arnold4dbd47e2013-07-22 05:39:26 -070096 const string& deadline_file,
adlr@google.com3defe6a2009-12-04 20:57:17 +000097 InstallPlan* out);
Alex Deymo763e7db2015-08-27 21:08:08 -070098
Amin Hassanid3f4bea2018-04-30 14:52:40 -070099 // Delegate passed to the ActionProcessor.
100 OmahaResponseHandlerActionProcessorDelegate delegate_;
101
Aaron Wood23bd3392017-10-06 14:48:25 -0700102 // Captures the action's result code, for tests that need to directly verify
103 // it in non-success cases.
104 ErrorCode action_result_code_;
105
Alex Deymo763e7db2015-08-27 21:08:08 -0700106 FakeSystemState fake_system_state_;
Sen Jiang2703ef42017-03-16 13:36:21 -0700107 // "Hash+"
108 const brillo::Blob expected_hash_ = {0x48, 0x61, 0x73, 0x68, 0x2b};
adlr@google.com3defe6a2009-12-04 20:57:17 +0000109};
110
adlr@google.com3defe6a2009-12-04 20:57:17 +0000111namespace {
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700112const char* const kLongName =
adlr@google.com3defe6a2009-12-04 20:57:17 +0000113 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
114 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
115 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
116 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
117 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
118 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
119 "very_long_name_and_no_slashes-very_long_name_and_no_slashes"
120 "-the_update_a.b.c.d_DELTA_.tgz";
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700121const char* const kBadVersion = "don't update me";
Sen Jiang2703ef42017-03-16 13:36:21 -0700122const char* const kPayloadHashHex = "486173682b";
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700123} // namespace
adlr@google.com3defe6a2009-12-04 20:57:17 +0000124
Aaron Woodc73fdc12017-12-06 11:09:15 -0800125bool OmahaResponseHandlerActionTest::DoTest(const OmahaResponse& in,
126 const string& test_deadline_file,
127 InstallPlan* out) {
Aaron Wood23bd3392017-10-06 14:48:25 -0700128 brillo::FakeMessageLoop loop(nullptr);
129 loop.SetAsCurrent();
adlr@google.com3defe6a2009-12-04 20:57:17 +0000130 ActionProcessor processor;
Amin Hassanid3f4bea2018-04-30 14:52:40 -0700131 processor.set_delegate(&delegate_);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000132
Amin Hassanid3f4bea2018-04-30 14:52:40 -0700133 auto feeder_action = std::make_unique<ObjectFeederAction<OmahaResponse>>();
134 feeder_action->set_obj(in);
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700135 if (in.update_exists && in.version != kBadVersion) {
Sen Jiang0affc2c2017-02-10 15:55:05 -0800136 string expected_hash;
137 for (const auto& package : in.packages)
138 expected_hash += package.hash + ":";
Alex Deymo763e7db2015-08-27 21:08:08 -0700139 EXPECT_CALL(*(fake_system_state_.mock_prefs()),
Sen Jiang0affc2c2017-02-10 15:55:05 -0800140 SetString(kPrefsUpdateCheckResponseHash, expected_hash))
Darin Petkov73058b42010-10-06 16:32:19 -0700141 .WillOnce(Return(true));
Alex Deymo85616652015-10-15 18:48:31 -0700142
143 int slot = 1 - fake_system_state_.fake_boot_control()->GetCurrentSlot();
144 string key = kPrefsChannelOnSlotPrefix + std::to_string(slot);
145 EXPECT_CALL(*(fake_system_state_.mock_prefs()), SetString(key, testing::_))
146 .WillOnce(Return(true));
Darin Petkov73058b42010-10-06 16:32:19 -0700147 }
Jay Srinivasan53173b92013-05-17 17:13:01 -0700148
Sen Jiang0affc2c2017-02-10 15:55:05 -0800149 string current_url = in.packages.size() ? in.packages[0].payload_urls[0] : "";
Alex Deymo763e7db2015-08-27 21:08:08 -0700150 EXPECT_CALL(*(fake_system_state_.mock_payload_state()), GetCurrentUrl())
Jay Srinivasan53173b92013-05-17 17:13:01 -0700151 .WillRepeatedly(Return(current_url));
152
Amin Hassanid3f4bea2018-04-30 14:52:40 -0700153 auto response_handler_action =
154 std::make_unique<OmahaResponseHandlerAction>(&fake_system_state_);
155 if (!test_deadline_file.empty())
156 response_handler_action->deadline_file_ = test_deadline_file;
157
158 auto collector_action =
159 std::make_unique<ObjectCollectorAction<InstallPlan>>();
160
161 BondActions(feeder_action.get(), response_handler_action.get());
162 BondActions(response_handler_action.get(), collector_action.get());
163 processor.EnqueueAction(std::move(feeder_action));
164 processor.EnqueueAction(std::move(response_handler_action));
165 processor.EnqueueAction(std::move(collector_action));
adlr@google.com3defe6a2009-12-04 20:57:17 +0000166 processor.StartProcessing();
167 EXPECT_TRUE(!processor.IsRunning())
Alex Vakulenko072359c2014-07-18 11:41:07 -0700168 << "Update test to handle non-async actions";
Amin Hassanid3f4bea2018-04-30 14:52:40 -0700169
170 if (out && delegate_.collector_action_install_plan_)
171 *out = *delegate_.collector_action_install_plan_;
172
173 EXPECT_TRUE(delegate_.code_set_);
174 action_result_code_ = delegate_.code_;
175 return delegate_.code_ == ErrorCode::kSuccess;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000176}
177
178TEST_F(OmahaResponseHandlerActionTest, SimpleTest) {
Gilad Arnold4dbd47e2013-07-22 05:39:26 -0700179 string test_deadline_file;
Aaron Woodc73fdc12017-12-06 11:09:15 -0800180 CHECK(utils::MakeTempFile("omaha_response_handler_action_unittest-XXXXXX",
181 &test_deadline_file,
182 nullptr));
Gilad Arnold4dbd47e2013-07-22 05:39:26 -0700183 ScopedPathUnlinker deadline_unlinker(test_deadline_file);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000184 {
Darin Petkov6a5b3222010-07-13 14:55:28 -0700185 OmahaResponse in;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000186 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700187 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800188 in.packages.push_back(
189 {.payload_urls = {"http://foo/the_update_a.b.c.d.tgz"},
190 .size = 12,
191 .hash = kPayloadHashHex});
adlr@google.com3defe6a2009-12-04 20:57:17 +0000192 in.more_info_url = "http://more/info";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000193 in.prompt = false;
Darin Petkov6c118642010-10-21 12:06:30 -0700194 in.deadline = "20101020";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000195 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700196 EXPECT_TRUE(DoTest(in, test_deadline_file, &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800197 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
198 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Alex Deymo80f70ff2016-02-10 16:08:11 -0800199 EXPECT_EQ(1U, install_plan.target_slot);
Darin Petkov6c118642010-10-21 12:06:30 -0700200 string deadline;
Gilad Arnold4dbd47e2013-07-22 05:39:26 -0700201 EXPECT_TRUE(utils::ReadFile(test_deadline_file, &deadline));
Darin Petkov6c118642010-10-21 12:06:30 -0700202 EXPECT_EQ("20101020", deadline);
203 struct stat deadline_stat;
Gilad Arnold4dbd47e2013-07-22 05:39:26 -0700204 EXPECT_EQ(0, stat(test_deadline_file.c_str(), &deadline_stat));
Alex Deymo80f70ff2016-02-10 16:08:11 -0800205 EXPECT_EQ(
206 static_cast<mode_t>(S_IFREG | S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH),
207 deadline_stat.st_mode);
Chris Sosafb1020e2013-07-29 17:27:33 -0700208 EXPECT_EQ(in.version, install_plan.version);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000209 }
210 {
Darin Petkov6a5b3222010-07-13 14:55:28 -0700211 OmahaResponse in;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000212 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700213 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800214 in.packages.push_back(
215 {.payload_urls = {"http://foo/the_update_a.b.c.d.tgz"},
216 .size = 12,
217 .hash = kPayloadHashHex});
adlr@google.com3defe6a2009-12-04 20:57:17 +0000218 in.more_info_url = "http://more/info";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000219 in.prompt = true;
220 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700221 // Set the other slot as current.
222 fake_system_state_.fake_boot_control()->SetCurrentSlot(1);
223 EXPECT_TRUE(DoTest(in, test_deadline_file, &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800224 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
225 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Alex Deymo80f70ff2016-02-10 16:08:11 -0800226 EXPECT_EQ(0U, install_plan.target_slot);
Darin Petkov6c118642010-10-21 12:06:30 -0700227 string deadline;
Gilad Arnold4dbd47e2013-07-22 05:39:26 -0700228 EXPECT_TRUE(utils::ReadFile(test_deadline_file, &deadline) &&
229 deadline.empty());
Chris Sosafb1020e2013-07-29 17:27:33 -0700230 EXPECT_EQ(in.version, install_plan.version);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000231 }
232 {
Darin Petkov6a5b3222010-07-13 14:55:28 -0700233 OmahaResponse in;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000234 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700235 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800236 in.packages.push_back(
237 {.payload_urls = {kLongName}, .size = 12, .hash = kPayloadHashHex});
adlr@google.com3defe6a2009-12-04 20:57:17 +0000238 in.more_info_url = "http://more/info";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000239 in.prompt = true;
Darin Petkov6c118642010-10-21 12:06:30 -0700240 in.deadline = "some-deadline";
adlr@google.com3defe6a2009-12-04 20:57:17 +0000241 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700242 fake_system_state_.fake_boot_control()->SetCurrentSlot(0);
Marton Hunyadye58bddb2018-04-10 20:27:26 +0200243 // Because rollback happened, the deadline shouldn't be written into the
244 // file.
245 EXPECT_CALL(*(fake_system_state_.mock_payload_state()),
246 GetRollbackHappened())
247 .WillOnce(Return(true));
248 EXPECT_TRUE(DoTest(in, test_deadline_file, &install_plan));
249 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
250 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
251 EXPECT_EQ(1U, install_plan.target_slot);
252 string deadline;
253 EXPECT_TRUE(utils::ReadFile(test_deadline_file, &deadline));
254 EXPECT_TRUE(deadline.empty());
255 EXPECT_EQ(in.version, install_plan.version);
256 }
257 {
258 OmahaResponse in;
259 in.update_exists = true;
260 in.version = "a.b.c.d";
261 in.packages.push_back(
262 {.payload_urls = {kLongName}, .size = 12, .hash = kPayloadHashHex});
263 in.more_info_url = "http://more/info";
264 in.prompt = true;
265 in.deadline = "some-deadline";
266 InstallPlan install_plan;
267 fake_system_state_.fake_boot_control()->SetCurrentSlot(0);
268 EXPECT_CALL(*(fake_system_state_.mock_payload_state()),
269 GetRollbackHappened())
270 .WillOnce(Return(false));
Alex Deymo763e7db2015-08-27 21:08:08 -0700271 EXPECT_TRUE(DoTest(in, test_deadline_file, &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800272 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
273 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Alex Deymo80f70ff2016-02-10 16:08:11 -0800274 EXPECT_EQ(1U, install_plan.target_slot);
Darin Petkov6c118642010-10-21 12:06:30 -0700275 string deadline;
Gilad Arnold4dbd47e2013-07-22 05:39:26 -0700276 EXPECT_TRUE(utils::ReadFile(test_deadline_file, &deadline));
Darin Petkov6c118642010-10-21 12:06:30 -0700277 EXPECT_EQ("some-deadline", deadline);
Chris Sosafb1020e2013-07-29 17:27:33 -0700278 EXPECT_EQ(in.version, install_plan.version);
adlr@google.com3defe6a2009-12-04 20:57:17 +0000279 }
280}
281
282TEST_F(OmahaResponseHandlerActionTest, NoUpdatesTest) {
Darin Petkov6a5b3222010-07-13 14:55:28 -0700283 OmahaResponse in;
adlr@google.com3defe6a2009-12-04 20:57:17 +0000284 in.update_exists = false;
285 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700286 EXPECT_FALSE(DoTest(in, "", &install_plan));
Alex Deymoe5e5fe92015-10-05 09:28:19 -0700287 EXPECT_TRUE(install_plan.partitions.empty());
adlr@google.com3defe6a2009-12-04 20:57:17 +0000288}
289
Sen Jiang0affc2c2017-02-10 15:55:05 -0800290TEST_F(OmahaResponseHandlerActionTest, MultiPackageTest) {
291 OmahaResponse in;
292 in.update_exists = true;
293 in.version = "a.b.c.d";
294 in.packages.push_back({.payload_urls = {"http://package/1"},
295 .size = 1,
296 .hash = kPayloadHashHex});
297 in.packages.push_back({.payload_urls = {"http://package/2"},
298 .size = 2,
299 .hash = kPayloadHashHex});
300 in.more_info_url = "http://more/info";
301 InstallPlan install_plan;
302 EXPECT_TRUE(DoTest(in, "", &install_plan));
303 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
304 EXPECT_EQ(2u, install_plan.payloads.size());
305 EXPECT_EQ(in.packages[0].size, install_plan.payloads[0].size);
306 EXPECT_EQ(in.packages[1].size, install_plan.payloads[1].size);
307 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
308 EXPECT_EQ(expected_hash_, install_plan.payloads[1].hash);
309 EXPECT_EQ(in.version, install_plan.version);
310}
311
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800312TEST_F(OmahaResponseHandlerActionTest, HashChecksForHttpTest) {
313 OmahaResponse in;
314 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700315 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800316 in.packages.push_back(
317 {.payload_urls = {"http://test.should/need/hash.checks.signed"},
318 .size = 12,
319 .hash = kPayloadHashHex});
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800320 in.more_info_url = "http://more/info";
David Pursell02c18642014-11-06 11:26:11 -0800321 // Hash checks are always skipped for non-official update URLs.
Alex Deymo763e7db2015-08-27 21:08:08 -0700322 EXPECT_CALL(*(fake_system_state_.mock_request_params()),
David Pursell02c18642014-11-06 11:26:11 -0800323 IsUpdateUrlOfficial())
324 .WillRepeatedly(Return(true));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800325 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700326 EXPECT_TRUE(DoTest(in, "", &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800327 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
328 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800329 EXPECT_TRUE(install_plan.hash_checks_mandatory);
Chris Sosafb1020e2013-07-29 17:27:33 -0700330 EXPECT_EQ(in.version, install_plan.version);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800331}
332
David Pursell02c18642014-11-06 11:26:11 -0800333TEST_F(OmahaResponseHandlerActionTest, HashChecksForUnofficialUpdateUrl) {
334 OmahaResponse in;
335 in.update_exists = true;
336 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800337 in.packages.push_back(
338 {.payload_urls = {"http://url.normally/needs/hash.checks.signed"},
339 .size = 12,
340 .hash = kPayloadHashHex});
David Pursell02c18642014-11-06 11:26:11 -0800341 in.more_info_url = "http://more/info";
Alex Deymo763e7db2015-08-27 21:08:08 -0700342 EXPECT_CALL(*(fake_system_state_.mock_request_params()),
David Pursell02c18642014-11-06 11:26:11 -0800343 IsUpdateUrlOfficial())
344 .WillRepeatedly(Return(false));
345 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700346 EXPECT_TRUE(DoTest(in, "", &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800347 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
348 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
David Pursell02c18642014-11-06 11:26:11 -0800349 EXPECT_FALSE(install_plan.hash_checks_mandatory);
350 EXPECT_EQ(in.version, install_plan.version);
351}
352
David Pursell907b4fa2015-01-27 10:27:38 -0800353TEST_F(OmahaResponseHandlerActionTest,
354 HashChecksForOfficialUrlUnofficialBuildTest) {
355 // Official URLs for unofficial builds (dev/test images) don't require hash.
356 OmahaResponse in;
357 in.update_exists = true;
358 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800359 in.packages.push_back(
360 {.payload_urls = {"http://url.normally/needs/hash.checks.signed"},
361 .size = 12,
362 .hash = kPayloadHashHex});
David Pursell907b4fa2015-01-27 10:27:38 -0800363 in.more_info_url = "http://more/info";
Alex Deymo763e7db2015-08-27 21:08:08 -0700364 EXPECT_CALL(*(fake_system_state_.mock_request_params()),
David Pursell907b4fa2015-01-27 10:27:38 -0800365 IsUpdateUrlOfficial())
366 .WillRepeatedly(Return(true));
Alex Deymo763e7db2015-08-27 21:08:08 -0700367 fake_system_state_.fake_hardware()->SetIsOfficialBuild(false);
David Pursell907b4fa2015-01-27 10:27:38 -0800368 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700369 EXPECT_TRUE(DoTest(in, "", &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800370 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
371 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
David Pursell907b4fa2015-01-27 10:27:38 -0800372 EXPECT_FALSE(install_plan.hash_checks_mandatory);
373 EXPECT_EQ(in.version, install_plan.version);
374}
375
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800376TEST_F(OmahaResponseHandlerActionTest, HashChecksForHttpsTest) {
377 OmahaResponse in;
378 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700379 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800380 in.packages.push_back(
381 {.payload_urls = {"https://test.should/need/hash.checks.signed"},
382 .size = 12,
383 .hash = kPayloadHashHex});
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800384 in.more_info_url = "http://more/info";
Alex Deymo763e7db2015-08-27 21:08:08 -0700385 EXPECT_CALL(*(fake_system_state_.mock_request_params()),
David Pursell02c18642014-11-06 11:26:11 -0800386 IsUpdateUrlOfficial())
387 .WillRepeatedly(Return(true));
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800388 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700389 EXPECT_TRUE(DoTest(in, "", &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800390 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
391 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800392 EXPECT_FALSE(install_plan.hash_checks_mandatory);
Chris Sosafb1020e2013-07-29 17:27:33 -0700393 EXPECT_EQ(in.version, install_plan.version);
Jay Srinivasan738fdf32012-12-07 17:40:54 -0800394}
395
Jay Srinivasan6f6ea002012-12-14 11:26:28 -0800396TEST_F(OmahaResponseHandlerActionTest, HashChecksForBothHttpAndHttpsTest) {
397 OmahaResponse in;
398 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700399 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800400 in.packages.push_back(
401 {.payload_urls = {"http://test.should.still/need/hash.checks",
402 "https://test.should.still/need/hash.checks"},
403 .size = 12,
404 .hash = kPayloadHashHex});
Jay Srinivasan6f6ea002012-12-14 11:26:28 -0800405 in.more_info_url = "http://more/info";
Alex Deymo763e7db2015-08-27 21:08:08 -0700406 EXPECT_CALL(*(fake_system_state_.mock_request_params()),
David Pursell02c18642014-11-06 11:26:11 -0800407 IsUpdateUrlOfficial())
408 .WillRepeatedly(Return(true));
Jay Srinivasan6f6ea002012-12-14 11:26:28 -0800409 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700410 EXPECT_TRUE(DoTest(in, "", &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800411 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
412 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Jay Srinivasan6f6ea002012-12-14 11:26:28 -0800413 EXPECT_TRUE(install_plan.hash_checks_mandatory);
Chris Sosafb1020e2013-07-29 17:27:33 -0700414 EXPECT_EQ(in.version, install_plan.version);
Jay Srinivasan6f6ea002012-12-14 11:26:28 -0800415}
416
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700417TEST_F(OmahaResponseHandlerActionTest, ChangeToMoreStableChannelTest) {
418 OmahaResponse in;
419 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700420 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800421 in.packages.push_back({.payload_urls = {"https://MoreStableChannelTest"},
422 .size = 1,
423 .hash = kPayloadHashHex});
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700424 in.more_info_url = "http://more/info";
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700425
Gilad Arnoldeff87cc2013-07-22 18:32:09 -0700426 // Create a uniquely named test directory.
Sen Jiang297e5832016-03-17 14:45:51 -0700427 base::ScopedTempDir tempdir;
428 ASSERT_TRUE(tempdir.CreateUniqueTempDir());
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700429
Alex Deymo763e7db2015-08-27 21:08:08 -0700430 OmahaRequestParams params(&fake_system_state_);
Alex Deymo85616652015-10-15 18:48:31 -0700431 fake_system_state_.fake_hardware()->SetIsOfficialBuild(false);
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900432 params.set_root(tempdir.GetPath().value());
Sen Jiang297e5832016-03-17 14:45:51 -0700433 params.set_current_channel("canary-channel");
434 // The ImageProperties in Android uses prefs to store MutableImageProperties.
435#ifdef __ANDROID__
Sen Jiang297e5832016-03-17 14:45:51 -0700436 EXPECT_CALL(*fake_system_state_.mock_prefs(), SetBoolean(_, true))
437 .WillOnce(Return(true));
438#endif // __ANDROID__
439 EXPECT_TRUE(params.SetTargetChannel("stable-channel", true, nullptr));
440 params.UpdateDownloadChannel();
Sen Jiang8500d3a2018-02-08 12:04:05 -0800441 EXPECT_TRUE(params.ShouldPowerwash());
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700442
Alex Deymo763e7db2015-08-27 21:08:08 -0700443 fake_system_state_.set_request_params(&params);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700444 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700445 EXPECT_TRUE(DoTest(in, "", &install_plan));
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700446 EXPECT_TRUE(install_plan.powerwash_required);
447}
448
449TEST_F(OmahaResponseHandlerActionTest, ChangeToLessStableChannelTest) {
450 OmahaResponse in;
451 in.update_exists = true;
Chris Sosa3b748432013-06-20 16:42:59 -0700452 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800453 in.packages.push_back({.payload_urls = {"https://LessStableChannelTest"},
454 .size = 15,
455 .hash = kPayloadHashHex});
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700456 in.more_info_url = "http://more/info";
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700457
Gilad Arnoldeff87cc2013-07-22 18:32:09 -0700458 // Create a uniquely named test directory.
Sen Jiang297e5832016-03-17 14:45:51 -0700459 base::ScopedTempDir tempdir;
460 ASSERT_TRUE(tempdir.CreateUniqueTempDir());
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700461
Alex Deymo763e7db2015-08-27 21:08:08 -0700462 OmahaRequestParams params(&fake_system_state_);
Alex Deymo85616652015-10-15 18:48:31 -0700463 fake_system_state_.fake_hardware()->SetIsOfficialBuild(false);
Hidehiko Abe2b9d2412017-12-13 18:56:18 +0900464 params.set_root(tempdir.GetPath().value());
Sen Jiang297e5832016-03-17 14:45:51 -0700465 params.set_current_channel("stable-channel");
466 // The ImageProperties in Android uses prefs to store MutableImageProperties.
467#ifdef __ANDROID__
Sen Jiang297e5832016-03-17 14:45:51 -0700468 EXPECT_CALL(*fake_system_state_.mock_prefs(), SetBoolean(_, false))
469 .WillOnce(Return(true));
470#endif // __ANDROID__
471 EXPECT_TRUE(params.SetTargetChannel("canary-channel", false, nullptr));
472 params.UpdateDownloadChannel();
Sen Jiang8500d3a2018-02-08 12:04:05 -0800473 EXPECT_FALSE(params.ShouldPowerwash());
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700474
Alex Deymo763e7db2015-08-27 21:08:08 -0700475 fake_system_state_.set_request_params(&params);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700476 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700477 EXPECT_TRUE(DoTest(in, "", &install_plan));
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700478 EXPECT_FALSE(install_plan.powerwash_required);
479}
Jay Srinivasan6f6ea002012-12-14 11:26:28 -0800480
David Zeuthen8f191b22013-08-06 12:27:50 -0700481TEST_F(OmahaResponseHandlerActionTest, P2PUrlIsUsedAndHashChecksMandatory) {
482 OmahaResponse in;
483 in.update_exists = true;
484 in.version = "a.b.c.d";
Sen Jiang0affc2c2017-02-10 15:55:05 -0800485 in.packages.push_back(
486 {.payload_urls = {"https://would.not/cause/hash/checks"},
487 .size = 12,
488 .hash = kPayloadHashHex});
David Zeuthen8f191b22013-08-06 12:27:50 -0700489 in.more_info_url = "http://more/info";
David Zeuthen8f191b22013-08-06 12:27:50 -0700490
Alex Deymo763e7db2015-08-27 21:08:08 -0700491 OmahaRequestParams params(&fake_system_state_);
David Pursell02c18642014-11-06 11:26:11 -0800492 // We're using a real OmahaRequestParams object here so we can't mock
493 // IsUpdateUrlOfficial(), but setting the update URL to the AutoUpdate test
494 // server will cause IsUpdateUrlOfficial() to return true.
Alex Deymoac41a822015-09-15 20:52:53 -0700495 params.set_update_url(constants::kOmahaDefaultAUTestURL);
Alex Deymo763e7db2015-08-27 21:08:08 -0700496 fake_system_state_.set_request_params(&params);
David Zeuthen8f191b22013-08-06 12:27:50 -0700497
Alex Deymo763e7db2015-08-27 21:08:08 -0700498 EXPECT_CALL(*fake_system_state_.mock_payload_state(),
David Zeuthenbb8bdc72013-09-03 13:43:48 -0700499 SetUsingP2PForDownloading(true));
500
David Zeuthen8f191b22013-08-06 12:27:50 -0700501 string p2p_url = "http://9.8.7.6/p2p";
Alex Deymo763e7db2015-08-27 21:08:08 -0700502 EXPECT_CALL(*fake_system_state_.mock_payload_state(), GetP2PUrl())
Gilad Arnold74b5f552014-10-07 08:17:16 -0700503 .WillRepeatedly(Return(p2p_url));
Alex Deymo763e7db2015-08-27 21:08:08 -0700504 EXPECT_CALL(*fake_system_state_.mock_payload_state(),
Aaron Woodc73fdc12017-12-06 11:09:15 -0800505 GetUsingP2PForDownloading())
506 .WillRepeatedly(Return(true));
David Zeuthen8f191b22013-08-06 12:27:50 -0700507
508 InstallPlan install_plan;
Alex Deymo763e7db2015-08-27 21:08:08 -0700509 EXPECT_TRUE(DoTest(in, "", &install_plan));
Sen Jiang0affc2c2017-02-10 15:55:05 -0800510 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
Sen Jiang2703ef42017-03-16 13:36:21 -0700511 EXPECT_EQ(p2p_url, install_plan.download_url);
David Zeuthen8f191b22013-08-06 12:27:50 -0700512 EXPECT_TRUE(install_plan.hash_checks_mandatory);
513}
514
Marton Hunyady199152d2018-05-07 19:08:48 +0200515TEST_F(OmahaResponseHandlerActionTest, RollbackTest) {
516 OmahaResponse in;
517 in.update_exists = true;
518 in.packages.push_back({.payload_urls = {"https://RollbackTest"},
519 .size = 1,
520 .hash = kPayloadHashHex});
521 in.is_rollback = true;
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -0700522 in.rollback_key_version.kernel = 1;
523 in.rollback_key_version.kernel = 2;
524 in.rollback_key_version.firmware_key = 3;
525 in.rollback_key_version.firmware = 4;
Marton Hunyady199152d2018-05-07 19:08:48 +0200526
527 fake_system_state_.fake_hardware()->SetMinKernelKeyVersion(0x00010002);
528 fake_system_state_.fake_hardware()->SetMinFirmwareKeyVersion(0x00030004);
529
530 OmahaRequestParams params(&fake_system_state_);
531 params.set_rollback_allowed(true);
532
533 fake_system_state_.set_request_params(&params);
534 InstallPlan install_plan;
535 EXPECT_TRUE(DoTest(in, "", &install_plan));
536 EXPECT_TRUE(install_plan.is_rollback);
537}
538
539TEST_F(OmahaResponseHandlerActionTest, RollbackKernelVersionErrorTest) {
540 OmahaResponse in;
541 in.update_exists = true;
542 in.packages.push_back({.payload_urls = {"https://RollbackTest"},
543 .size = 1,
544 .hash = kPayloadHashHex});
545 in.is_rollback = true;
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -0700546 in.rollback_key_version.kernel_key = 1;
547 in.rollback_key_version.kernel = 1; // This is lower than the minimum.
548 in.rollback_key_version.firmware_key = 3;
549 in.rollback_key_version.firmware = 4;
Marton Hunyady199152d2018-05-07 19:08:48 +0200550
551 fake_system_state_.fake_hardware()->SetMinKernelKeyVersion(0x00010002);
552 fake_system_state_.fake_hardware()->SetMinFirmwareKeyVersion(0x00030004);
553
554 OmahaRequestParams params(&fake_system_state_);
555 params.set_rollback_allowed(true);
556
557 fake_system_state_.set_request_params(&params);
558 InstallPlan install_plan;
559 EXPECT_FALSE(DoTest(in, "", &install_plan));
560}
561
562TEST_F(OmahaResponseHandlerActionTest, RollbackFirmwareVersionErrorTest) {
563 OmahaResponse in;
564 in.update_exists = true;
565 in.packages.push_back({.payload_urls = {"https://RollbackTest"},
566 .size = 1,
567 .hash = kPayloadHashHex});
568 in.is_rollback = true;
Zentaro Kavanagh0ff621c2018-07-13 13:06:56 -0700569 in.rollback_key_version.kernel_key = 1;
570 in.rollback_key_version.kernel = 2;
571 in.rollback_key_version.firmware_key = 3;
572 in.rollback_key_version.firmware = 3; // This is lower than the minimum.
Marton Hunyady199152d2018-05-07 19:08:48 +0200573
574 fake_system_state_.fake_hardware()->SetMinKernelKeyVersion(0x00010002);
575 fake_system_state_.fake_hardware()->SetMinFirmwareKeyVersion(0x00030004);
576
577 OmahaRequestParams params(&fake_system_state_);
578 params.set_rollback_allowed(true);
579
580 fake_system_state_.set_request_params(&params);
581 InstallPlan install_plan;
582 EXPECT_FALSE(DoTest(in, "", &install_plan));
583}
584
585TEST_F(OmahaResponseHandlerActionTest, RollbackNotRollbackTest) {
586 OmahaResponse in;
587 in.update_exists = true;
588 in.packages.push_back({.payload_urls = {"https://RollbackTest"},
589 .size = 1,
590 .hash = kPayloadHashHex});
591 in.is_rollback = false;
592
593 OmahaRequestParams params(&fake_system_state_);
594 params.set_rollback_allowed(true);
595
596 fake_system_state_.set_request_params(&params);
597 InstallPlan install_plan;
598 EXPECT_TRUE(DoTest(in, "", &install_plan));
599 EXPECT_FALSE(install_plan.is_rollback);
600}
601
602TEST_F(OmahaResponseHandlerActionTest, RollbackNotAllowedTest) {
603 OmahaResponse in;
604 in.update_exists = true;
605 in.packages.push_back({.payload_urls = {"https://RollbackTest"},
606 .size = 1,
607 .hash = kPayloadHashHex});
608 in.is_rollback = true;
609
610 OmahaRequestParams params(&fake_system_state_);
611 params.set_rollback_allowed(false);
612
613 fake_system_state_.set_request_params(&params);
614 InstallPlan install_plan;
615 EXPECT_FALSE(DoTest(in, "", &install_plan));
616}
617
Aaron Wood7dcdedf2017-09-06 17:17:41 -0700618TEST_F(OmahaResponseHandlerActionTest, SystemVersionTest) {
619 OmahaResponse in;
620 in.update_exists = true;
621 in.version = "a.b.c.d";
622 in.system_version = "b.c.d.e";
623 in.packages.push_back({.payload_urls = {"http://package/1"},
624 .size = 1,
625 .hash = kPayloadHashHex});
626 in.packages.push_back({.payload_urls = {"http://package/2"},
627 .size = 2,
628 .hash = kPayloadHashHex});
629 in.more_info_url = "http://more/info";
630 InstallPlan install_plan;
631 EXPECT_TRUE(DoTest(in, "", &install_plan));
632 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
633 EXPECT_EQ(2u, install_plan.payloads.size());
634 EXPECT_EQ(in.packages[0].size, install_plan.payloads[0].size);
635 EXPECT_EQ(in.packages[1].size, install_plan.payloads[1].size);
636 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
637 EXPECT_EQ(expected_hash_, install_plan.payloads[1].hash);
638 EXPECT_EQ(in.version, install_plan.version);
639 EXPECT_EQ(in.system_version, install_plan.system_version);
640}
641
Aaron Wood23bd3392017-10-06 14:48:25 -0700642TEST_F(OmahaResponseHandlerActionTest, TestDeferredByPolicy) {
643 OmahaResponse in;
644 in.update_exists = true;
645 in.version = "a.b.c.d";
646 in.packages.push_back({.payload_urls = {"http://foo/the_update_a.b.c.d.tgz"},
647 .size = 12,
648 .hash = kPayloadHashHex});
649 // Setup the UpdateManager to disallow the update.
650 FakeClock fake_clock;
651 MockPolicy* mock_policy = new MockPolicy(&fake_clock);
652 FakeUpdateManager* fake_update_manager =
653 fake_system_state_.fake_update_manager();
654 fake_update_manager->set_policy(mock_policy);
655 EXPECT_CALL(*mock_policy, UpdateCanBeApplied(_, _, _, _, _))
656 .WillOnce(
657 DoAll(SetArgPointee<3>(ErrorCode::kOmahaUpdateDeferredPerPolicy),
658 Return(EvalStatus::kSucceeded)));
659 // Perform the Action. It should "fail" with kOmahaUpdateDeferredPerPolicy.
660 InstallPlan install_plan;
661 EXPECT_FALSE(DoTest(in, "", &install_plan));
662 EXPECT_EQ(ErrorCode::kOmahaUpdateDeferredPerPolicy, action_result_code_);
663 // Verify that DoTest() didn't set the output install plan.
664 EXPECT_EQ("", install_plan.version);
Aaron Wood23bd3392017-10-06 14:48:25 -0700665 // Now verify the InstallPlan that was generated.
Amin Hassanid3f4bea2018-04-30 14:52:40 -0700666 install_plan = *delegate_.response_handler_action_install_plan_;
Aaron Wood23bd3392017-10-06 14:48:25 -0700667 EXPECT_EQ(in.packages[0].payload_urls[0], install_plan.download_url);
668 EXPECT_EQ(expected_hash_, install_plan.payloads[0].hash);
669 EXPECT_EQ(1U, install_plan.target_slot);
670 EXPECT_EQ(in.version, install_plan.version);
671}
672
adlr@google.com3defe6a2009-12-04 20:57:17 +0000673} // namespace chromeos_update_engine