// // Copyright (C) 2011 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #include "update_engine/cros/omaha_request_params.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "update_engine/common/constants.h" #include "update_engine/common/hardware_interface.h" #include "update_engine/common/platform_constants.h" #include "update_engine/common/system_state.h" #include "update_engine/common/utils.h" #include "update_engine/update_manager/policy.h" #define CALL_MEMBER_FN(object, member) ((object).*(member)) using chromeos_update_manager::UpdateCheckParams; using std::string; namespace chromeos_update_engine { const char OmahaRequestParams::kOsVersion[] = "Indy"; const char* kChannelsByStability[] = { // This list has to be sorted from least stable to most stable channel. "canary-channel", "dev-channel", "beta-channel", "stable-channel", }; OmahaRequestParams::~OmahaRequestParams() { if (!root_.empty()) test::SetImagePropertiesRootPrefix(nullptr); } bool OmahaRequestParams::Init(const string& app_version, const string& update_url, const UpdateCheckParams& params) { LOG(INFO) << "Initializing parameters for this update attempt"; image_props_ = LoadImageProperties(); mutable_image_props_ = LoadMutableImageProperties(); // Validation check the channel names. if (!IsValidChannel(image_props_.current_channel)) image_props_.current_channel = "stable-channel"; if (!IsValidChannel(mutable_image_props_.target_channel)) mutable_image_props_.target_channel = image_props_.current_channel; UpdateDownloadChannel(); LOG(INFO) << "Running from channel " << image_props_.current_channel; os_platform_ = constants::kOmahaPlatformName; os_version_ = OmahaRequestParams::kOsVersion; if (!app_version.empty()) image_props_.version = app_version; os_sp_ = image_props_.version + "_" + GetMachineType(); app_lang_ = "en-US"; hwid_ = SystemState::Get()->hardware()->GetHardwareClass(); device_requisition_ = SystemState::Get()->hardware()->GetDeviceRequisition(); if (image_props_.current_channel == mutable_image_props_.target_channel) { // deltas are only okay if the /.nodelta file does not exist. if we don't // know (i.e. stat() returns some unexpected error), then err on the side of // caution and say deltas are not okay. struct stat stbuf; delta_okay_ = (stat((root_ + "/.nodelta").c_str(), &stbuf) < 0) && (errno == ENOENT); } else { LOG(INFO) << "Disabling deltas as a channel change to " << mutable_image_props_.target_channel << " is pending, with is_powerwash_allowed=" << utils::ToString(mutable_image_props_.is_powerwash_allowed); // For now, disable delta updates if the current channel is different from // the channel that we're sending to the update server because such updates // are destined to fail -- the current rootfs hash will be different than // the expected hash due to the different channel in /etc/lsb-release. delta_okay_ = false; } if (update_url.empty()) update_url_ = image_props_.omaha_url; else update_url_ = update_url; // Set the interactive flag accordingly. interactive_ = params.interactive; dlc_apps_params_.clear(); // Set false so it will do update by default. is_install_ = false; target_version_prefix_ = params.target_version_prefix; lts_tag_ = params.lts_tag; autoupdate_token_ = params.quick_fix_build_token; rollback_allowed_ = params.rollback_allowed; // Set whether saving data over rollback is requested. rollback_data_save_requested_ = params.rollback_data_save_requested; // Set how many milestones of rollback are allowed. rollback_allowed_milestones_ = params.rollback_allowed_milestones; // Set the target channel, if one was provided. if (params.target_channel.empty()) { LOG(INFO) << "No target channel mandated by policy."; } else { LOG(INFO) << "Setting target channel as mandated: " << params.target_channel; string error_message; if (!SetTargetChannel(params.target_channel, params.rollback_on_channel_downgrade, &error_message)) { LOG(ERROR) << "Setting the channel failed: " << error_message; } // Since this is the beginning of a new attempt, update the download // channel. The download channel won't be updated until the next attempt, // even if target channel changes meanwhile, so that how we'll know if we // should cancel the current download attempt if there's such a change in // target channel. UpdateDownloadChannel(); } return true; } bool OmahaRequestParams::IsUpdateUrlOfficial() const { return (update_url_ == constants::kOmahaDefaultAUTestURL || update_url_ == image_props_.omaha_url); } bool OmahaRequestParams::SetTargetChannel(const string& new_target_channel, bool is_powerwash_allowed, string* error_message) { LOG(INFO) << "SetTargetChannel called with " << new_target_channel << ", Is Powerwash Allowed = " << utils::ToString(is_powerwash_allowed) << ". Current channel = " << image_props_.current_channel << ", existing target channel = " << mutable_image_props_.target_channel << ", download channel = " << download_channel_; if (!IsValidChannel(new_target_channel, error_message)) { return false; } MutableImageProperties new_props; new_props.target_channel = new_target_channel; new_props.is_powerwash_allowed = is_powerwash_allowed; if (!StoreMutableImageProperties(new_props)) { if (error_message) *error_message = "Error storing the new channel value."; return false; } mutable_image_props_ = new_props; return true; } void OmahaRequestParams::UpdateDownloadChannel() { if (download_channel_ != mutable_image_props_.target_channel) { download_channel_ = mutable_image_props_.target_channel; LOG(INFO) << "Download channel for this attempt = " << download_channel_; } } string OmahaRequestParams::GetMachineType() const { struct utsname buf; string ret; if (uname(&buf) == 0) ret = buf.machine; return ret; } bool OmahaRequestParams::IsValidChannel(const string& channel, string* error_message) const { if (image_props_.allow_arbitrary_channels) { if (!base::EndsWith(channel, "-channel", base::CompareCase::SENSITIVE)) { if (error_message) { *error_message = base::StringPrintf( "Invalid channel name \"%s\", must ends with -channel.", channel.c_str()); } return false; } return true; } if (GetChannelIndex(channel) < 0) { string valid_channels = brillo::string_utils::JoinRange( ", ", std::begin(kChannelsByStability), std::end(kChannelsByStability)); if (error_message) { *error_message = base::StringPrintf("Invalid channel name \"%s\", valid names are: %s", channel.c_str(), valid_channels.c_str()); } return false; } return true; } void OmahaRequestParams::set_root(const string& root) { root_ = root; test::SetImagePropertiesRootPrefix(root_.c_str()); } int OmahaRequestParams::GetChannelIndex(const string& channel) const { for (size_t t = 0; t < base::size(kChannelsByStability); ++t) if (channel == kChannelsByStability[t]) return t; return -1; } bool OmahaRequestParams::ToMoreStableChannel() const { int current_channel_index = GetChannelIndex(image_props_.current_channel); int download_channel_index = GetChannelIndex(download_channel_); return download_channel_index > current_channel_index; } bool OmahaRequestParams::ShouldPowerwash() const { if (!mutable_image_props_.is_powerwash_allowed) return false; // If arbitrary channels are allowed, always powerwash on channel change. if (image_props_.allow_arbitrary_channels) return image_props_.current_channel != download_channel_; // Otherwise only powerwash if we are moving from less stable (higher version) // to more stable channel (lower version). return ToMoreStableChannel(); } string OmahaRequestParams::GetAppId() const { return download_channel_ == "canary-channel" ? image_props_.canary_product_id : image_props_.product_id; } string OmahaRequestParams::GetDlcAppId(const std::string& dlc_id) const { // Create APP ID according to |dlc_id| (sticking the current AppID to the // DLC module ID with an underscode). return GetAppId() + "_" + dlc_id; } bool OmahaRequestParams::IsDlcAppId(const std::string& app_id) const { return dlc_apps_params().find(app_id) != dlc_apps_params().end(); } bool OmahaRequestParams::GetDlcId(const string& app_id, string* dlc_id) const { auto itr = dlc_apps_params_.find(app_id); if (itr == dlc_apps_params_.end()) return false; *dlc_id = itr->second.name; return true; } void OmahaRequestParams::SetDlcNoUpdate(const string& app_id) { auto itr = dlc_apps_params_.find(app_id); if (itr == dlc_apps_params_.end()) return; itr->second.updated = false; } } // namespace chromeos_update_engine