blob: 97a2ed19001a77ee9a1deb903e3ec3dd9597a3ea [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//
Darin Petkova4a8a8c2010-07-15 22:21:12 -070016
17#include "update_engine/omaha_request_params.h"
18
19#include <errno.h>
20#include <fcntl.h>
21#include <sys/utsname.h>
22
23#include <map>
24#include <string>
Darin Petkova3df55b2010-11-15 13:33:55 -080025#include <vector>
Darin Petkova4a8a8c2010-07-15 22:21:12 -070026
Ben Chan06c76a42014-09-05 08:21:06 -070027#include <base/files/file_util.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070028#include <base/strings/string_util.h>
Alex Deymod942f9d2015-11-06 16:11:50 -080029#include <base/strings/stringprintf.h>
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070030#include <brillo/key_value_store.h>
Alex Deymod942f9d2015-11-06 16:11:50 -080031#include <brillo/strings/string_utils.h>
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020032#include <policy/device_policy.h>
Darin Petkov49d91322010-10-25 16:34:58 -070033
Alex Deymo39910dc2015-11-09 17:04:30 -080034#include "update_engine/common/constants.h"
35#include "update_engine/common/hardware_interface.h"
36#include "update_engine/common/platform_constants.h"
37#include "update_engine/common/utils.h"
Jay Srinivasanae4697c2013-03-18 17:08:08 -070038#include "update_engine/system_state.h"
Darin Petkova4a8a8c2010-07-15 22:21:12 -070039
Darin Petkov49d91322010-10-25 16:34:58 -070040#define CALL_MEMBER_FN(object, member) ((object).*(member))
41
Darin Petkova4a8a8c2010-07-15 22:21:12 -070042using std::map;
43using std::string;
Darin Petkova3df55b2010-11-15 13:33:55 -080044using std::vector;
Darin Petkova4a8a8c2010-07-15 22:21:12 -070045
Darin Petkova4a8a8c2010-07-15 22:21:12 -070046namespace chromeos_update_engine {
47
David Pursell02c18642014-11-06 11:26:11 -080048const char OmahaRequestParams::kOsVersion[] = "Indy";
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020049
Jay Srinivasanae4697c2013-03-18 17:08:08 -070050const char* kChannelsByStability[] = {
51 // This list has to be sorted from least stable to most stable channel.
52 "canary-channel",
53 "dev-channel",
54 "beta-channel",
55 "stable-channel",
56};
Darin Petkov49d91322010-10-25 16:34:58 -070057
Alex Deymo3be05c82015-10-23 11:29:11 -070058OmahaRequestParams::~OmahaRequestParams() {
59 if (!root_.empty())
60 test::SetImagePropertiesRootPrefix(nullptr);
61}
62
Alex Deymof329b932014-10-30 01:37:48 -070063bool OmahaRequestParams::Init(const string& in_app_version,
64 const string& in_update_url,
Jay Srinivasanae4697c2013-03-18 17:08:08 -070065 bool in_interactive) {
Jay Srinivasan1c0fe792013-03-28 16:45:25 -070066 LOG(INFO) << "Initializing parameters for this update attempt";
Alex Deymo85616652015-10-15 18:48:31 -070067 image_props_ = LoadImageProperties(system_state_);
68 mutable_image_props_ = LoadMutableImageProperties(system_state_);
69
70 // Sanity check the channel names.
71 if (!IsValidChannel(image_props_.current_channel))
72 image_props_.current_channel = "stable-channel";
73 if (!IsValidChannel(mutable_image_props_.target_channel))
74 mutable_image_props_.target_channel = image_props_.current_channel;
75 UpdateDownloadChannel();
76
77 LOG(INFO) << "Running from channel " << image_props_.current_channel;
78
Alex Deymoac41a822015-09-15 20:52:53 -070079 os_platform_ = constants::kOmahaPlatformName;
Sen Jiangf87e5cd2018-03-06 18:33:39 -080080 if (!image_props_.system_version.empty()) {
81 if (in_app_version == "ForcedUpdate") {
82 image_props_.system_version = in_app_version;
83 }
Sen Jiang3faf4e72017-06-08 11:34:28 -070084 os_version_ = image_props_.system_version;
Sen Jiangf87e5cd2018-03-06 18:33:39 -080085 } else {
Sen Jiang3faf4e72017-06-08 11:34:28 -070086 os_version_ = OmahaRequestParams::kOsVersion;
Sen Jiangf87e5cd2018-03-06 18:33:39 -080087 }
Alex Deymo85616652015-10-15 18:48:31 -070088 if (!in_app_version.empty())
89 image_props_.version = in_app_version;
90
91 os_sp_ = image_props_.version + "_" + GetMachineType();
Jay Srinivasanae4697c2013-03-18 17:08:08 -070092 app_lang_ = "en-US";
J. Richard Barnette522d36f2013-10-28 17:22:12 -070093 hwid_ = system_state_->hardware()->GetHardwareClass();
Chris Sosac1972482013-04-30 22:31:10 -070094 if (CollectECFWVersions()) {
J. Richard Barnette522d36f2013-10-28 17:22:12 -070095 fw_version_ = system_state_->hardware()->GetFirmwareVersion();
96 ec_version_ = system_state_->hardware()->GetECVersion();
Chris Sosac1972482013-04-30 22:31:10 -070097 }
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020098
Alex Deymo85616652015-10-15 18:48:31 -070099 if (image_props_.current_channel == mutable_image_props_.target_channel) {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700100 // deltas are only okay if the /.nodelta file does not exist. if we don't
101 // know (i.e. stat() returns some unexpected error), then err on the side of
102 // caution and say deltas are not okay.
103 struct stat stbuf;
104 delta_okay_ = (stat((root_ + "/.nodelta").c_str(), &stbuf) < 0) &&
105 (errno == ENOENT);
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +0200106 } else {
Alex Deymo85616652015-10-15 18:48:31 -0700107 LOG(INFO) << "Disabling deltas as a channel change to "
108 << mutable_image_props_.target_channel
109 << " is pending, with is_powerwash_allowed="
110 << utils::ToString(mutable_image_props_.is_powerwash_allowed);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700111 // For now, disable delta updates if the current channel is different from
112 // the channel that we're sending to the update server because such updates
113 // are destined to fail -- the current rootfs hash will be different than
114 // the expected hash due to the different channel in /etc/lsb-release.
115 delta_okay_ = false;
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +0200116 }
117
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700118 if (in_update_url.empty())
Alex Deymo85616652015-10-15 18:48:31 -0700119 update_url_ = image_props_.omaha_url;
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700120 else
121 update_url_ = in_update_url;
Gilad Arnoldbbdd4902013-01-10 16:06:30 -0800122
123 // Set the interactive flag accordingly.
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700124 interactive_ = in_interactive;
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700125 return true;
126}
127
David Pursell02c18642014-11-06 11:26:11 -0800128bool OmahaRequestParams::IsUpdateUrlOfficial() const {
Alex Deymoac41a822015-09-15 20:52:53 -0700129 return (update_url_ == constants::kOmahaDefaultAUTestURL ||
Alex Deymo85616652015-10-15 18:48:31 -0700130 update_url_ == image_props_.omaha_url);
David Pursell02c18642014-11-06 11:26:11 -0800131}
132
Chris Sosac1972482013-04-30 22:31:10 -0700133bool OmahaRequestParams::CollectECFWVersions() const {
Alex Vakulenkoa3cf75a2016-01-20 07:56:15 -0800134 return base::StartsWith(hwid_, string("SAMS ALEX"),
135 base::CompareCase::SENSITIVE) ||
136 base::StartsWith(hwid_, string("BUTTERFLY"),
137 base::CompareCase::SENSITIVE) ||
138 base::StartsWith(hwid_, string("LUMPY"),
139 base::CompareCase::SENSITIVE) ||
140 base::StartsWith(hwid_, string("PARROT"),
141 base::CompareCase::SENSITIVE) ||
142 base::StartsWith(hwid_, string("SPRING"),
143 base::CompareCase::SENSITIVE) ||
144 base::StartsWith(hwid_, string("SNOW"), base::CompareCase::SENSITIVE);
Chris Sosac1972482013-04-30 22:31:10 -0700145}
146
Alex Deymof329b932014-10-30 01:37:48 -0700147bool OmahaRequestParams::SetTargetChannel(const string& new_target_channel,
Alex Deymod942f9d2015-11-06 16:11:50 -0800148 bool is_powerwash_allowed,
149 string* error_message) {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700150 LOG(INFO) << "SetTargetChannel called with " << new_target_channel
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700151 << ", Is Powerwash Allowed = "
152 << utils::ToString(is_powerwash_allowed)
Alex Deymo85616652015-10-15 18:48:31 -0700153 << ". Current channel = " << image_props_.current_channel
154 << ", existing target channel = "
155 << mutable_image_props_.target_channel
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700156 << ", download channel = " << download_channel_;
Sen Jiangce3f7cf2018-01-25 14:07:45 -0800157 if (!IsValidChannel(new_target_channel, error_message)) {
Alex Deymod942f9d2015-11-06 16:11:50 -0800158 return false;
159 }
Alex Deymoa7658442014-04-17 15:49:37 -0700160
Alex Deymo85616652015-10-15 18:48:31 -0700161 MutableImageProperties new_props;
162 new_props.target_channel = new_target_channel;
163 new_props.is_powerwash_allowed = is_powerwash_allowed;
Alex Deymoa7658442014-04-17 15:49:37 -0700164
Alex Deymod942f9d2015-11-06 16:11:50 -0800165 if (!StoreMutableImageProperties(system_state_, new_props)) {
166 if (error_message)
167 *error_message = "Error storing the new channel value.";
168 return false;
169 }
Alex Deymo85616652015-10-15 18:48:31 -0700170 mutable_image_props_ = new_props;
Darin Petkov49d91322010-10-25 16:34:58 -0700171 return true;
172}
173
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700174void OmahaRequestParams::UpdateDownloadChannel() {
Alex Deymo85616652015-10-15 18:48:31 -0700175 if (download_channel_ != mutable_image_props_.target_channel) {
176 download_channel_ = mutable_image_props_.target_channel;
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700177 LOG(INFO) << "Download channel for this attempt = " << download_channel_;
178 }
179}
180
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700181string OmahaRequestParams::GetMachineType() const {
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700182 struct utsname buf;
183 string ret;
184 if (uname(&buf) == 0)
185 ret = buf.machine;
186 return ret;
187}
188
Sen Jiangce3f7cf2018-01-25 14:07:45 -0800189bool OmahaRequestParams::IsValidChannel(const string& channel,
190 string* error_message) const {
191 if (image_props_.allow_arbitrary_channels) {
192 if (!base::EndsWith(channel, "-channel", base::CompareCase::SENSITIVE)) {
193 if (error_message) {
194 *error_message = base::StringPrintf(
195 "Invalid channel name \"%s\", must ends with -channel.",
196 channel.c_str());
197 }
198 return false;
199 }
200 return true;
201 }
202 if (GetChannelIndex(channel) < 0) {
203 string valid_channels = brillo::string_utils::JoinRange(
204 ", ", std::begin(kChannelsByStability), std::end(kChannelsByStability));
205 if (error_message) {
206 *error_message =
207 base::StringPrintf("Invalid channel name \"%s\", valid names are: %s",
208 channel.c_str(),
209 valid_channels.c_str());
210 }
211 return false;
212 }
213 return true;
Darin Petkov49d91322010-10-25 16:34:58 -0700214}
215
Alex Deymof329b932014-10-30 01:37:48 -0700216void OmahaRequestParams::set_root(const string& root) {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700217 root_ = root;
Alex Deymo85616652015-10-15 18:48:31 -0700218 test::SetImagePropertiesRootPrefix(root_.c_str());
Darin Petkov49d91322010-10-25 16:34:58 -0700219}
220
Alex Deymof329b932014-10-30 01:37:48 -0700221int OmahaRequestParams::GetChannelIndex(const string& channel) const {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700222 for (size_t t = 0; t < arraysize(kChannelsByStability); ++t)
223 if (channel == kChannelsByStability[t])
224 return t;
225
226 return -1;
227}
228
Sen Jiang8500d3a2018-02-08 12:04:05 -0800229bool OmahaRequestParams::ToMoreStableChannel() const {
Alex Deymo85616652015-10-15 18:48:31 -0700230 int current_channel_index = GetChannelIndex(image_props_.current_channel);
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700231 int download_channel_index = GetChannelIndex(download_channel_);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700232
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700233 return download_channel_index > current_channel_index;
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700234}
235
Sen Jiang8500d3a2018-02-08 12:04:05 -0800236bool OmahaRequestParams::ShouldPowerwash() const {
237 if (!mutable_image_props_.is_powerwash_allowed)
238 return false;
239 // If arbitrary channels are allowed, always powerwash on channel change.
240 if (image_props_.allow_arbitrary_channels)
241 return image_props_.current_channel != download_channel_;
242 // Otherwise only powerwash if we are moving from less stable (higher version)
243 // to more stable channel (lower version).
244 return ToMoreStableChannel();
245}
246
Jay Srinivasandb0acdf2013-04-02 14:47:45 -0700247string OmahaRequestParams::GetAppId() const {
Alex Deymo85616652015-10-15 18:48:31 -0700248 return download_channel_ == "canary-channel" ? image_props_.canary_product_id
249 : image_props_.product_id;
Jay Srinivasandb0acdf2013-04-02 14:47:45 -0700250}
251
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700252} // namespace chromeos_update_engine