blob: adcfc75978cf4109e3f3f271d856061372b8f744 [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
Amin Hassaniec7bc112020-10-29 16:47:58 -070017#include "update_engine/cros/omaha_request_params.h"
Darin Petkova4a8a8c2010-07-15 22:21:12 -070018
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>
hscham00b6aa22020-02-20 12:32:06 +090028#include <base/stl_util.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070029#include <base/strings/string_util.h>
Alex Deymod942f9d2015-11-06 16:11:50 -080030#include <base/strings/stringprintf.h>
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070031#include <brillo/key_value_store.h>
Alex Deymod942f9d2015-11-06 16:11:50 -080032#include <brillo/strings/string_utils.h>
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020033#include <policy/device_policy.h>
Darin Petkov49d91322010-10-25 16:34:58 -070034
Alex Deymo39910dc2015-11-09 17:04:30 -080035#include "update_engine/common/constants.h"
36#include "update_engine/common/hardware_interface.h"
37#include "update_engine/common/platform_constants.h"
Amin Hassaniec7bc112020-10-29 16:47:58 -070038#include "update_engine/common/system_state.h"
Alex Deymo39910dc2015-11-09 17:04:30 -080039#include "update_engine/common/utils.h"
Amin Hassanie8153632020-10-27 15:11:28 -070040#include "update_engine/update_manager/policy.h"
Darin Petkova4a8a8c2010-07-15 22:21:12 -070041
Darin Petkov49d91322010-10-25 16:34:58 -070042#define CALL_MEMBER_FN(object, member) ((object).*(member))
43
Amin Hassanie8153632020-10-27 15:11:28 -070044using chromeos_update_manager::UpdateCheckParams;
Darin Petkova4a8a8c2010-07-15 22:21:12 -070045using std::string;
46
Darin Petkova4a8a8c2010-07-15 22:21:12 -070047namespace chromeos_update_engine {
48
David Pursell02c18642014-11-06 11:26:11 -080049const char OmahaRequestParams::kOsVersion[] = "Indy";
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020050
Jay Srinivasanae4697c2013-03-18 17:08:08 -070051const char* kChannelsByStability[] = {
52 // This list has to be sorted from least stable to most stable channel.
53 "canary-channel",
54 "dev-channel",
55 "beta-channel",
56 "stable-channel",
57};
Darin Petkov49d91322010-10-25 16:34:58 -070058
Alex Deymo3be05c82015-10-23 11:29:11 -070059OmahaRequestParams::~OmahaRequestParams() {
60 if (!root_.empty())
61 test::SetImagePropertiesRootPrefix(nullptr);
62}
63
Amin Hassanie8153632020-10-27 15:11:28 -070064bool OmahaRequestParams::Init(const string& app_version,
65 const string& update_url,
66 const UpdateCheckParams& params) {
Jay Srinivasan1c0fe792013-03-28 16:45:25 -070067 LOG(INFO) << "Initializing parameters for this update attempt";
Amin Hassani538bd592020-11-04 20:46:08 -080068 image_props_ = LoadImageProperties();
69 mutable_image_props_ = LoadMutableImageProperties();
Alex Deymo85616652015-10-15 18:48:31 -070070
Saint Chouaff51c22020-07-29 14:14:21 +000071 // Validation check the channel names.
Alex Deymo85616652015-10-15 18:48:31 -070072 if (!IsValidChannel(image_props_.current_channel))
73 image_props_.current_channel = "stable-channel";
74 if (!IsValidChannel(mutable_image_props_.target_channel))
75 mutable_image_props_.target_channel = image_props_.current_channel;
76 UpdateDownloadChannel();
77
78 LOG(INFO) << "Running from channel " << image_props_.current_channel;
79
Alex Deymoac41a822015-09-15 20:52:53 -070080 os_platform_ = constants::kOmahaPlatformName;
Amin Hassani24e1d722020-10-28 17:47:58 -070081 os_version_ = OmahaRequestParams::kOsVersion;
Amin Hassanie8153632020-10-27 15:11:28 -070082 if (!app_version.empty())
83 image_props_.version = app_version;
Alex Deymo85616652015-10-15 18:48:31 -070084
85 os_sp_ = image_props_.version + "_" + GetMachineType();
Jay Srinivasanae4697c2013-03-18 17:08:08 -070086 app_lang_ = "en-US";
Amin Hassani538bd592020-11-04 20:46:08 -080087 hwid_ = SystemState::Get()->hardware()->GetHardwareClass();
88 device_requisition_ = SystemState::Get()->hardware()->GetDeviceRequisition();
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020089
Alex Deymo85616652015-10-15 18:48:31 -070090 if (image_props_.current_channel == mutable_image_props_.target_channel) {
Jay Srinivasanae4697c2013-03-18 17:08:08 -070091 // deltas are only okay if the /.nodelta file does not exist. if we don't
92 // know (i.e. stat() returns some unexpected error), then err on the side of
93 // caution and say deltas are not okay.
94 struct stat stbuf;
Amin Hassani7cc8bb02019-01-14 16:29:47 -080095 delta_okay_ =
96 (stat((root_ + "/.nodelta").c_str(), &stbuf) < 0) && (errno == ENOENT);
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020097 } else {
Alex Deymo85616652015-10-15 18:48:31 -070098 LOG(INFO) << "Disabling deltas as a channel change to "
99 << mutable_image_props_.target_channel
100 << " is pending, with is_powerwash_allowed="
101 << utils::ToString(mutable_image_props_.is_powerwash_allowed);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700102 // For now, disable delta updates if the current channel is different from
103 // the channel that we're sending to the update server because such updates
104 // are destined to fail -- the current rootfs hash will be different than
105 // the expected hash due to the different channel in /etc/lsb-release.
106 delta_okay_ = false;
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +0200107 }
108
Amin Hassanie8153632020-10-27 15:11:28 -0700109 if (update_url.empty())
Alex Deymo85616652015-10-15 18:48:31 -0700110 update_url_ = image_props_.omaha_url;
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700111 else
Amin Hassanie8153632020-10-27 15:11:28 -0700112 update_url_ = update_url;
Gilad Arnoldbbdd4902013-01-10 16:06:30 -0800113
114 // Set the interactive flag accordingly.
Amin Hassanie8153632020-10-27 15:11:28 -0700115 interactive_ = params.interactive;
Xiaochu Liu88d90382018-08-29 16:09:11 -0700116
Andrewe045aef2020-01-08 16:29:22 -0800117 dlc_apps_params_.clear();
Xiaochu Liu88d90382018-08-29 16:09:11 -0700118 // Set false so it will do update by default.
119 is_install_ = false;
Amin Hassanie8153632020-10-27 15:11:28 -0700120
121 target_version_prefix_ = params.target_version_prefix;
122
123 lts_tag_ = params.lts_tag;
124
Saurabh Nijhara43d7adc2020-11-06 16:13:02 +0100125 autoupdate_token_ = params.quick_fix_build_token;
126
Amin Hassanie8153632020-10-27 15:11:28 -0700127 rollback_allowed_ = params.rollback_allowed;
128
129 // Set whether saving data over rollback is requested.
130 rollback_data_save_requested_ = params.rollback_data_save_requested;
131
132 // Set how many milestones of rollback are allowed.
133 rollback_allowed_milestones_ = params.rollback_allowed_milestones;
134
135 // Set the target channel, if one was provided.
136 if (params.target_channel.empty()) {
137 LOG(INFO) << "No target channel mandated by policy.";
138 } else {
139 LOG(INFO) << "Setting target channel as mandated: "
140 << params.target_channel;
141 string error_message;
142 if (!SetTargetChannel(params.target_channel,
143 params.rollback_on_channel_downgrade,
144 &error_message)) {
145 LOG(ERROR) << "Setting the channel failed: " << error_message;
146 }
147
148 // Since this is the beginning of a new attempt, update the download
149 // channel. The download channel won't be updated until the next attempt,
150 // even if target channel changes meanwhile, so that how we'll know if we
151 // should cancel the current download attempt if there's such a change in
152 // target channel.
153 UpdateDownloadChannel();
154 }
155
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700156 return true;
157}
158
David Pursell02c18642014-11-06 11:26:11 -0800159bool OmahaRequestParams::IsUpdateUrlOfficial() const {
Alex Deymoac41a822015-09-15 20:52:53 -0700160 return (update_url_ == constants::kOmahaDefaultAUTestURL ||
Alex Deymo85616652015-10-15 18:48:31 -0700161 update_url_ == image_props_.omaha_url);
David Pursell02c18642014-11-06 11:26:11 -0800162}
163
Alex Deymof329b932014-10-30 01:37:48 -0700164bool OmahaRequestParams::SetTargetChannel(const string& new_target_channel,
Alex Deymod942f9d2015-11-06 16:11:50 -0800165 bool is_powerwash_allowed,
166 string* error_message) {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700167 LOG(INFO) << "SetTargetChannel called with " << new_target_channel
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700168 << ", Is Powerwash Allowed = "
169 << utils::ToString(is_powerwash_allowed)
Alex Deymo85616652015-10-15 18:48:31 -0700170 << ". Current channel = " << image_props_.current_channel
171 << ", existing target channel = "
172 << mutable_image_props_.target_channel
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700173 << ", download channel = " << download_channel_;
Sen Jiangce3f7cf2018-01-25 14:07:45 -0800174 if (!IsValidChannel(new_target_channel, error_message)) {
Alex Deymod942f9d2015-11-06 16:11:50 -0800175 return false;
176 }
Alex Deymoa7658442014-04-17 15:49:37 -0700177
Alex Deymo85616652015-10-15 18:48:31 -0700178 MutableImageProperties new_props;
179 new_props.target_channel = new_target_channel;
180 new_props.is_powerwash_allowed = is_powerwash_allowed;
Alex Deymoa7658442014-04-17 15:49:37 -0700181
Amin Hassani538bd592020-11-04 20:46:08 -0800182 if (!StoreMutableImageProperties(new_props)) {
Alex Deymod942f9d2015-11-06 16:11:50 -0800183 if (error_message)
184 *error_message = "Error storing the new channel value.";
185 return false;
186 }
Alex Deymo85616652015-10-15 18:48:31 -0700187 mutable_image_props_ = new_props;
Darin Petkov49d91322010-10-25 16:34:58 -0700188 return true;
189}
190
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700191void OmahaRequestParams::UpdateDownloadChannel() {
Alex Deymo85616652015-10-15 18:48:31 -0700192 if (download_channel_ != mutable_image_props_.target_channel) {
193 download_channel_ = mutable_image_props_.target_channel;
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700194 LOG(INFO) << "Download channel for this attempt = " << download_channel_;
195 }
196}
197
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700198string OmahaRequestParams::GetMachineType() const {
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700199 struct utsname buf;
200 string ret;
201 if (uname(&buf) == 0)
202 ret = buf.machine;
203 return ret;
204}
205
Sen Jiangce3f7cf2018-01-25 14:07:45 -0800206bool OmahaRequestParams::IsValidChannel(const string& channel,
207 string* error_message) const {
208 if (image_props_.allow_arbitrary_channels) {
209 if (!base::EndsWith(channel, "-channel", base::CompareCase::SENSITIVE)) {
210 if (error_message) {
211 *error_message = base::StringPrintf(
212 "Invalid channel name \"%s\", must ends with -channel.",
213 channel.c_str());
214 }
215 return false;
216 }
217 return true;
218 }
219 if (GetChannelIndex(channel) < 0) {
220 string valid_channels = brillo::string_utils::JoinRange(
221 ", ", std::begin(kChannelsByStability), std::end(kChannelsByStability));
222 if (error_message) {
223 *error_message =
224 base::StringPrintf("Invalid channel name \"%s\", valid names are: %s",
225 channel.c_str(),
226 valid_channels.c_str());
227 }
228 return false;
229 }
230 return true;
Darin Petkov49d91322010-10-25 16:34:58 -0700231}
232
Alex Deymof329b932014-10-30 01:37:48 -0700233void OmahaRequestParams::set_root(const string& root) {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700234 root_ = root;
Alex Deymo85616652015-10-15 18:48:31 -0700235 test::SetImagePropertiesRootPrefix(root_.c_str());
Darin Petkov49d91322010-10-25 16:34:58 -0700236}
237
Alex Deymof329b932014-10-30 01:37:48 -0700238int OmahaRequestParams::GetChannelIndex(const string& channel) const {
hscham00b6aa22020-02-20 12:32:06 +0900239 for (size_t t = 0; t < base::size(kChannelsByStability); ++t)
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700240 if (channel == kChannelsByStability[t])
241 return t;
242
243 return -1;
244}
245
Sen Jiang8500d3a2018-02-08 12:04:05 -0800246bool OmahaRequestParams::ToMoreStableChannel() const {
Alex Deymo85616652015-10-15 18:48:31 -0700247 int current_channel_index = GetChannelIndex(image_props_.current_channel);
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700248 int download_channel_index = GetChannelIndex(download_channel_);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700249
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700250 return download_channel_index > current_channel_index;
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700251}
252
Sen Jiang8500d3a2018-02-08 12:04:05 -0800253bool OmahaRequestParams::ShouldPowerwash() const {
254 if (!mutable_image_props_.is_powerwash_allowed)
255 return false;
256 // If arbitrary channels are allowed, always powerwash on channel change.
257 if (image_props_.allow_arbitrary_channels)
258 return image_props_.current_channel != download_channel_;
259 // Otherwise only powerwash if we are moving from less stable (higher version)
260 // to more stable channel (lower version).
261 return ToMoreStableChannel();
262}
263
Jay Srinivasandb0acdf2013-04-02 14:47:45 -0700264string OmahaRequestParams::GetAppId() const {
Alex Deymo85616652015-10-15 18:48:31 -0700265 return download_channel_ == "canary-channel" ? image_props_.canary_product_id
266 : image_props_.product_id;
Jay Srinivasandb0acdf2013-04-02 14:47:45 -0700267}
268
Jae Hoon Kim5e9cd712020-02-12 15:28:28 -0800269string OmahaRequestParams::GetDlcAppId(const std::string& dlc_id) const {
Andrewe045aef2020-01-08 16:29:22 -0800270 // Create APP ID according to |dlc_id| (sticking the current AppID to the
271 // DLC module ID with an underscode).
272 return GetAppId() + "_" + dlc_id;
273}
274
Jae Hoon Kim5e9cd712020-02-12 15:28:28 -0800275bool OmahaRequestParams::IsDlcAppId(const std::string& app_id) const {
276 return dlc_apps_params().find(app_id) != dlc_apps_params().end();
277}
278
Jae Hoon Kime2cac612020-11-02 18:30:29 -0800279bool OmahaRequestParams::GetDlcId(const string& app_id, string* dlc_id) const {
280 auto itr = dlc_apps_params_.find(app_id);
281 if (itr == dlc_apps_params_.end())
282 return false;
283 *dlc_id = itr->second.name;
284 return true;
285}
286
Jae Hoon Kim7fdfbf12020-04-10 18:15:50 -0700287void OmahaRequestParams::SetDlcNoUpdate(const string& app_id) {
288 auto itr = dlc_apps_params_.find(app_id);
289 if (itr == dlc_apps_params_.end())
290 return;
291 itr->second.updated = false;
292}
293
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700294} // namespace chromeos_update_engine