blob: 94de4f39ac3adadb9acd2da9a74ca2ee6f755de9 [file] [log] [blame]
Darin Petkovf2065b42011-05-17 16:36:27 -07001// Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
Darin Petkova4a8a8c2010-07-15 22:21:12 -07002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "update_engine/omaha_request_params.h"
6
7#include <errno.h>
8#include <fcntl.h>
9#include <sys/utsname.h>
10
11#include <map>
12#include <string>
Darin Petkova3df55b2010-11-15 13:33:55 -080013#include <vector>
Darin Petkova4a8a8c2010-07-15 22:21:12 -070014
Ben Chan06c76a42014-09-05 08:21:06 -070015#include <base/files/file_util.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070016#include <base/strings/string_util.h>
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020017#include <policy/device_policy.h>
Darin Petkov49d91322010-10-25 16:34:58 -070018
Chris Sosabe45bef2013-04-09 18:25:12 -070019#include "update_engine/constants.h"
J. Richard Barnette522d36f2013-10-28 17:22:12 -070020#include "update_engine/hardware_interface.h"
Darin Petkova4a8a8c2010-07-15 22:21:12 -070021#include "update_engine/simple_key_value_store.h"
Jay Srinivasanae4697c2013-03-18 17:08:08 -070022#include "update_engine/system_state.h"
Darin Petkova4a8a8c2010-07-15 22:21:12 -070023#include "update_engine/utils.h"
24
Darin Petkov49d91322010-10-25 16:34:58 -070025#define CALL_MEMBER_FN(object, member) ((object).*(member))
26
Darin Petkova4a8a8c2010-07-15 22:21:12 -070027using std::map;
28using std::string;
Darin Petkova3df55b2010-11-15 13:33:55 -080029using std::vector;
Darin Petkova4a8a8c2010-07-15 22:21:12 -070030
Darin Petkova4a8a8c2010-07-15 22:21:12 -070031namespace chromeos_update_engine {
32
Darin Petkov5a7f5652010-07-22 21:40:09 -070033const char* const OmahaRequestParams::kAppId(
34 "{87efface-864d-49a5-9bb3-4b050a7c227a}");
35const char* const OmahaRequestParams::kOsPlatform("Chrome OS");
36const char* const OmahaRequestParams::kOsVersion("Indy");
Jay Srinivasan55f50c22013-01-10 19:24:35 -080037const char* const kProductionOmahaUrl(
Darin Petkov5a7f5652010-07-22 21:40:09 -070038 "https://tools.google.com/service/update2");
39
Jay Srinivasanae4697c2013-03-18 17:08:08 -070040const char* const OmahaRequestParams::kUpdateChannelKey(
41 "CHROMEOS_RELEASE_TRACK");
42const char* const OmahaRequestParams::kIsPowerwashAllowedKey(
43 "CHROMEOS_IS_POWERWASH_ALLOWED");
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020044
Jay Srinivasanae4697c2013-03-18 17:08:08 -070045const char* kChannelsByStability[] = {
46 // This list has to be sorted from least stable to most stable channel.
47 "canary-channel",
48 "dev-channel",
49 "beta-channel",
50 "stable-channel",
51};
Darin Petkov49d91322010-10-25 16:34:58 -070052
Jay Srinivasanae4697c2013-03-18 17:08:08 -070053bool OmahaRequestParams::Init(const std::string& in_app_version,
54 const std::string& in_update_url,
55 bool in_interactive) {
Jay Srinivasan1c0fe792013-03-28 16:45:25 -070056 LOG(INFO) << "Initializing parameters for this update attempt";
Jay Srinivasanae4697c2013-03-18 17:08:08 -070057 InitFromLsbValue();
Darin Petkov10d02dd2011-01-10 14:57:39 -080058 bool stateful_override = !ShouldLockDown();
Jay Srinivasanae4697c2013-03-18 17:08:08 -070059 os_platform_ = OmahaRequestParams::kOsPlatform;
60 os_version_ = OmahaRequestParams::kOsVersion;
61 app_version_ = in_app_version.empty() ?
Alex Vakulenko88b591f2014-08-28 16:48:57 -070062 GetLsbValue("CHROMEOS_RELEASE_VERSION", "", nullptr, stateful_override) :
Darin Petkov10d02dd2011-01-10 14:57:39 -080063 in_app_version;
Jay Srinivasanae4697c2013-03-18 17:08:08 -070064 os_sp_ = app_version_ + "_" + GetMachineType();
65 os_board_ = GetLsbValue("CHROMEOS_RELEASE_BOARD",
66 "",
Alex Vakulenko88b591f2014-08-28 16:48:57 -070067 nullptr,
Jay Srinivasanae4697c2013-03-18 17:08:08 -070068 stateful_override);
Jay Srinivasandb0acdf2013-04-02 14:47:45 -070069 string release_app_id = GetLsbValue("CHROMEOS_RELEASE_APPID",
70 OmahaRequestParams::kAppId,
Alex Vakulenko88b591f2014-08-28 16:48:57 -070071 nullptr,
Jay Srinivasandb0acdf2013-04-02 14:47:45 -070072 stateful_override);
Jay Srinivasanae4697c2013-03-18 17:08:08 -070073 board_app_id_ = GetLsbValue("CHROMEOS_BOARD_APPID",
Jay Srinivasandb0acdf2013-04-02 14:47:45 -070074 release_app_id,
Alex Vakulenko88b591f2014-08-28 16:48:57 -070075 nullptr,
Jay Srinivasanae4697c2013-03-18 17:08:08 -070076 stateful_override);
Jay Srinivasandb0acdf2013-04-02 14:47:45 -070077 canary_app_id_ = GetLsbValue("CHROMEOS_CANARY_APPID",
78 release_app_id,
Alex Vakulenko88b591f2014-08-28 16:48:57 -070079 nullptr,
Jay Srinivasandb0acdf2013-04-02 14:47:45 -070080 stateful_override);
Jay Srinivasanae4697c2013-03-18 17:08:08 -070081 app_lang_ = "en-US";
J. Richard Barnette522d36f2013-10-28 17:22:12 -070082 hwid_ = system_state_->hardware()->GetHardwareClass();
Chris Sosac1972482013-04-30 22:31:10 -070083 if (CollectECFWVersions()) {
J. Richard Barnette522d36f2013-10-28 17:22:12 -070084 fw_version_ = system_state_->hardware()->GetFirmwareVersion();
85 ec_version_ = system_state_->hardware()->GetECVersion();
Chris Sosac1972482013-04-30 22:31:10 -070086 }
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020087
Jay Srinivasanae4697c2013-03-18 17:08:08 -070088 if (current_channel_ == target_channel_) {
89 // deltas are only okay if the /.nodelta file does not exist. if we don't
90 // know (i.e. stat() returns some unexpected error), then err on the side of
91 // caution and say deltas are not okay.
92 struct stat stbuf;
93 delta_okay_ = (stat((root_ + "/.nodelta").c_str(), &stbuf) < 0) &&
94 (errno == ENOENT);
95
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +020096 } else {
Jay Srinivasanae4697c2013-03-18 17:08:08 -070097 LOG(INFO) << "Disabling deltas as a channel change is pending";
98 // For now, disable delta updates if the current channel is different from
99 // the channel that we're sending to the update server because such updates
100 // are destined to fail -- the current rootfs hash will be different than
101 // the expected hash due to the different channel in /etc/lsb-release.
102 delta_okay_ = false;
Patrick Dubroy7fbbe8a2011-08-01 17:28:22 +0200103 }
104
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700105 if (in_update_url.empty())
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700106 update_url_ = GetLsbValue("CHROMEOS_AUSERVER", kProductionOmahaUrl, nullptr,
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700107 stateful_override);
108 else
109 update_url_ = in_update_url;
Gilad Arnoldbbdd4902013-01-10 16:06:30 -0800110
111 // Set the interactive flag accordingly.
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700112 interactive_ = in_interactive;
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700113 return true;
114}
115
Chris Sosac1972482013-04-30 22:31:10 -0700116bool OmahaRequestParams::CollectECFWVersions() const {
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700117 return {
Chris Sosac1972482013-04-30 22:31:10 -0700118 StartsWithASCII(hwid_, string("SAMS ALEX"), true) ||
119 StartsWithASCII(hwid_, string("BUTTERFLY"), true) ||
120 StartsWithASCII(hwid_, string("LUMPY"), true) ||
121 StartsWithASCII(hwid_, string("PARROT"), true) ||
122 StartsWithASCII(hwid_, string("SPRING"), true) ||
123 StartsWithASCII(hwid_, string("SNOW"), true)
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700124 };
Chris Sosac1972482013-04-30 22:31:10 -0700125}
126
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700127bool OmahaRequestParams::SetTargetChannel(const std::string& new_target_channel,
128 bool is_powerwash_allowed) {
129 LOG(INFO) << "SetTargetChannel called with " << new_target_channel
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700130 << ", Is Powerwash Allowed = "
131 << utils::ToString(is_powerwash_allowed)
132 << ". Current channel = " << current_channel_
133 << ", existing target channel = " << target_channel_
134 << ", download channel = " << download_channel_;
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700135 TEST_AND_RETURN_FALSE(IsValidChannel(new_target_channel));
Alex Deymoa7658442014-04-17 15:49:37 -0700136 KeyValueStore lsb_release;
Alex Vakulenko75039d72014-03-25 12:36:28 -0700137 base::FilePath kFile(root_ + kStatefulPartition + "/etc/lsb-release");
Alex Deymoa7658442014-04-17 15:49:37 -0700138
139 lsb_release.Load(kFile.value());
140 lsb_release.SetString(kUpdateChannelKey, new_target_channel);
141 lsb_release.SetBoolean(kIsPowerwashAllowedKey, is_powerwash_allowed);
142
Alex Vakulenko75039d72014-03-25 12:36:28 -0700143 TEST_AND_RETURN_FALSE(base::CreateDirectory(kFile.DirName()));
Alex Deymoa7658442014-04-17 15:49:37 -0700144 TEST_AND_RETURN_FALSE(lsb_release.Save(kFile.value()));
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700145 target_channel_ = new_target_channel;
146 is_powerwash_allowed_ = is_powerwash_allowed;
Darin Petkov49d91322010-10-25 16:34:58 -0700147 return true;
148}
149
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700150void OmahaRequestParams::SetTargetChannelFromLsbValue() {
151 string target_channel_new_value = GetLsbValue(
152 kUpdateChannelKey,
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700153 current_channel_,
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700154 &chromeos_update_engine::OmahaRequestParams::IsValidChannel,
155 true); // stateful_override
156
157 if (target_channel_ != target_channel_new_value) {
158 target_channel_ = target_channel_new_value;
159 LOG(INFO) << "Target Channel set to " << target_channel_
160 << " from LSB file";
161 }
Darin Petkov49d91322010-10-25 16:34:58 -0700162}
163
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700164void OmahaRequestParams::SetCurrentChannelFromLsbValue() {
165 string current_channel_new_value = GetLsbValue(
166 kUpdateChannelKey,
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700167 current_channel_,
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700168 nullptr, // No need to validate the read-only rootfs channel.
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700169 false); // stateful_override is false so we get the current channel.
170
171 if (current_channel_ != current_channel_new_value) {
172 current_channel_ = current_channel_new_value;
173 LOG(INFO) << "Current Channel set to " << current_channel_
174 << " from LSB file in rootfs";
175 }
Satoru Takabayashi583667b2010-10-27 13:09:57 +0900176}
177
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700178void OmahaRequestParams::SetIsPowerwashAllowedFromLsbValue() {
179 string is_powerwash_allowed_str = GetLsbValue(
180 kIsPowerwashAllowedKey,
181 "false",
Alex Vakulenko88b591f2014-08-28 16:48:57 -0700182 nullptr, // no need to validate
Alex Vakulenkod2779df2014-06-16 13:19:00 -0700183 true); // always get it from stateful, as that's the only place it'll be
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700184 bool is_powerwash_allowed_new_value = (is_powerwash_allowed_str == "true");
185 if (is_powerwash_allowed_ != is_powerwash_allowed_new_value) {
186 is_powerwash_allowed_ = is_powerwash_allowed_new_value;
187 LOG(INFO) << "Powerwash Allowed set to "
188 << utils::ToString(is_powerwash_allowed_)
189 << " from LSB file in stateful";
190 }
191}
192
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700193void OmahaRequestParams::UpdateDownloadChannel() {
194 if (download_channel_ != target_channel_) {
195 download_channel_ = target_channel_;
196 LOG(INFO) << "Download channel for this attempt = " << download_channel_;
197 }
198}
199
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700200void OmahaRequestParams::InitFromLsbValue() {
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700201 SetCurrentChannelFromLsbValue();
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700202 SetTargetChannelFromLsbValue();
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700203 SetIsPowerwashAllowedFromLsbValue();
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700204 UpdateDownloadChannel();
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700205}
206
207string OmahaRequestParams::GetLsbValue(const string& key,
208 const string& default_value,
209 ValueValidator validator,
210 bool stateful_override) const {
Darin Petkova3df55b2010-11-15 13:33:55 -0800211 vector<string> files;
212 if (stateful_override) {
Chris Sosabe45bef2013-04-09 18:25:12 -0700213 files.push_back(string(kStatefulPartition) + "/etc/lsb-release");
Darin Petkova3df55b2010-11-15 13:33:55 -0800214 }
215 files.push_back("/etc/lsb-release");
216 for (vector<string>::const_iterator it = files.begin();
217 it != files.end(); ++it) {
218 // TODO(adlr): make sure files checked are owned as root (and all their
219 // parents are recursively, too).
Alex Deymoa7658442014-04-17 15:49:37 -0700220 KeyValueStore data;
221 if (!data.Load(root_ + *it))
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700222 continue;
223
Alex Deymoa7658442014-04-17 15:49:37 -0700224 string value;
225 if (data.GetString(key, &value)) {
Darin Petkov49d91322010-10-25 16:34:58 -0700226 if (validator && !CALL_MEMBER_FN(*this, validator)(value)) {
227 continue;
228 }
229 return value;
230 }
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700231 }
232 // not found
233 return default_value;
234}
235
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700236string OmahaRequestParams::GetMachineType() const {
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700237 struct utsname buf;
238 string ret;
239 if (uname(&buf) == 0)
240 ret = buf.machine;
241 return ret;
242}
243
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700244bool OmahaRequestParams::ShouldLockDown() const {
Darin Petkov10d02dd2011-01-10 14:57:39 -0800245 if (force_lock_down_) {
246 return forced_lock_down_;
247 }
J. Richard Barnette056b0ab2013-10-29 15:24:56 -0700248 return system_state_->hardware()->IsOfficialBuild() &&
249 system_state_->hardware()->IsNormalBootMode();
Darin Petkov49d91322010-10-25 16:34:58 -0700250}
251
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700252bool OmahaRequestParams::IsValidChannel(const std::string& channel) const {
253 return GetChannelIndex(channel) >= 0;
Darin Petkov49d91322010-10-25 16:34:58 -0700254}
255
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700256void OmahaRequestParams::set_root(const std::string& root) {
257 root_ = root;
258 InitFromLsbValue();
259}
260
261void OmahaRequestParams::SetLockDown(bool lock) {
Darin Petkov10d02dd2011-01-10 14:57:39 -0800262 force_lock_down_ = true;
263 forced_lock_down_ = lock;
Darin Petkov49d91322010-10-25 16:34:58 -0700264}
265
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700266int OmahaRequestParams::GetChannelIndex(const std::string& channel) const {
267 for (size_t t = 0; t < arraysize(kChannelsByStability); ++t)
268 if (channel == kChannelsByStability[t])
269 return t;
270
271 return -1;
272}
273
274bool OmahaRequestParams::to_more_stable_channel() const {
275 int current_channel_index = GetChannelIndex(current_channel_);
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700276 int download_channel_index = GetChannelIndex(download_channel_);
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700277
Jay Srinivasan1c0fe792013-03-28 16:45:25 -0700278 return download_channel_index > current_channel_index;
Jay Srinivasanae4697c2013-03-18 17:08:08 -0700279}
280
Jay Srinivasandb0acdf2013-04-02 14:47:45 -0700281string OmahaRequestParams::GetAppId() const {
282 return download_channel_ == "canary-channel" ? canary_app_id_ : board_app_id_;
283}
284
Darin Petkova4a8a8c2010-07-15 22:21:12 -0700285} // namespace chromeos_update_engine