| // Copyright 2018 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/update_client/protocol_serializer_json.h" |
| |
| #include <memory> |
| #include <utility> |
| #include <vector> |
| |
| #include "base/json/json_writer.h" |
| #include "base/values.h" |
| #include "build/build_config.h" |
| #include "components/update_client/updater_state.h" |
| |
| namespace update_client { |
| |
| using Value = base::Value; |
| |
| std::string ProtocolSerializerJSON::Serialize( |
| const protocol_request::Request& request) const { |
| base::Value::Dict root_node; |
| base::Value::Dict request_node; |
| request_node.Set("protocol", request.protocol_version); |
| request_node.Set("dedup", "cr"); |
| request_node.Set("acceptformat", Value("crx2,crx3")); |
| if (!request.additional_attributes.empty()) { |
| for (const auto& attr : request.additional_attributes) |
| request_node.Set(attr.first, attr.second); |
| } |
| request_node.Set("sessionid", request.session_id); |
| request_node.Set("requestid", request.request_id); |
| request_node.Set("@updater", request.updatername); |
| request_node.Set("prodversion", request.prodversion); |
| request_node.Set("updaterversion", request.updaterversion); |
| request_node.Set("@os", request.operating_system); |
| #if defined(STARBOARD) |
| // Upstream sets the lang in app_nodes, but we want it in request_node until |
| // we fully update //components/update_client. |
| request_node.Set("lang", request.lang); |
| #endif |
| request_node.Set("arch", request.lang); |
| request_node.Set("nacl_arch", request.nacl_arch); |
| #if BUILDFLAG(IS_WIN) |
| if (request.is_wow64) |
| request_node.Set("wow64", request.is_wow64); |
| #endif // BUILDFLAG(IS_WIN) |
| if (!request.updaterchannel.empty()) |
| request_node.Set("updaterchannel", request.updaterchannel); |
| if (!request.prodchannel.empty()) |
| request_node.Set("prodchannel", request.prodchannel); |
| if (!request.dlpref.empty()) |
| request_node.Set("dlpref", request.dlpref); |
| if (request.domain_joined) |
| request_node.Set("domainjoined", *request.domain_joined); |
| |
| // HW platform information. |
| base::Value::Dict hw_node; |
| hw_node.Set("physmemory", static_cast<int>(request.hw.physmemory)); |
| request_node.Set("hw", std::move(hw_node)); |
| |
| // OS version and platform information. |
| base::Value::Dict os_node; |
| os_node.Set("platform", request.os.platform); |
| os_node.Set("arch", request.os.arch); |
| if (!request.os.version.empty()) |
| os_node.Set("version", request.os.version); |
| if (!request.os.service_pack.empty()) |
| os_node.Set("sp", request.os.service_pack); |
| request_node.Set("os", std::move(os_node)); |
| |
| #if defined(GOOGLE_CHROME_BUILD) |
| if (request.updater) { |
| const auto& updater = *request.updater; |
| base::Value::Dict updater_node; |
| updater_node.Set("name", updater.name); |
| updater_node.Set("ismachine", updater.is_machine); |
| updater_node.Set("autoupdatecheckenabled", |
| updater.autoupdate_check_enabled); |
| updater_node.Set("updatepolicy", updater.update_policy); |
| if (!updater.version.empty()) |
| updater_node.Set("version", updater.version); |
| if (updater.last_checked) |
| updater_node.Set("lastchecked", *updater.last_checked); |
| if (updater.last_started) |
| updater_node.Set("laststarted", *updater.last_started); |
| request_node.Set("updater", std::move(updater_node)); |
| } |
| #endif |
| |
| base::Value::List app_nodes; |
| for (const auto& app : request.apps) { |
| base::Value::Dict app_node; |
| app_node.Set("appid", app.app_id); |
| app_node.Set("version", app.version); |
| if (!app.brand_code.empty()) |
| app_node.Set("brand", app.brand_code); |
| #if !defined(STARBOARD) |
| if (!app.lang.empty()) |
| app_node.Set("lang", app.lang); |
| #endif |
| if (!app.install_source.empty()) |
| app_node.Set("installsource", app.install_source); |
| if (!app.install_location.empty()) |
| app_node.Set("installedby", app.install_location); |
| if (!app.cohort.empty()) |
| app_node.Set("cohort", app.cohort); |
| if (!app.cohort_name.empty()) |
| app_node.Set("cohortname", app.cohort_name); |
| if (!app.cohort_hint.empty()) |
| app_node.Set("cohorthint", app.cohort_hint); |
| if (app.enabled) |
| app_node.Set("enabled", *app.enabled); |
| |
| if (app.disabled_reasons && !app.disabled_reasons->empty()) { |
| base::Value::List disabled_nodes; |
| for (const int disabled_reason : *app.disabled_reasons) { |
| base::Value::Dict disabled_node; |
| disabled_node.Set("reason", disabled_reason); |
| disabled_nodes.Append(std::move(disabled_node)); |
| } |
| app_node.Set("disabled", std::move(disabled_nodes)); |
| } |
| |
| for (const auto& attr : app.installer_attributes) |
| app_node.Set(attr.first, attr.second); |
| |
| if (app.update_check) { |
| base::Value::Dict update_check_node; |
| if (app.update_check->is_update_disabled) |
| update_check_node.Set("updatedisabled", true); |
| app_node.Set("updatecheck", std::move(update_check_node)); |
| } |
| |
| if (app.ping) { |
| const auto& ping = *app.ping; |
| base::Value::Dict ping_node; |
| if (!ping.ping_freshness.empty()) |
| ping_node.Set("ping_freshness", ping.ping_freshness); |
| |
| // Output "ad" or "a" only if the this app has been seen 'active'. |
| if (ping.date_last_active) { |
| ping_node.Set("ad", *ping.date_last_active); |
| } else if (ping.days_since_last_active_ping) { |
| ping_node.Set("a", *ping.days_since_last_active_ping); |
| } |
| |
| // Output "rd" if valid or "r" as a last resort roll call metric. |
| if (ping.date_last_roll_call) |
| ping_node.Set("rd", *ping.date_last_roll_call); |
| else |
| ping_node.Set("r", ping.days_since_last_roll_call); |
| app_node.Set("ping", std::move(ping_node)); |
| } |
| |
| if (!app.fingerprint.empty()) { |
| base::Value::List package_nodes; |
| base::Value::Dict package; |
| package.Set("fp", app.fingerprint); |
| package_nodes.Append(std::move(package)); |
| base::Value::Dict packages_node; |
| packages_node.Set("package", std::move(package_nodes)); |
| app_node.Set("packages", std::move(packages_node)); |
| } |
| |
| if (app.events) { |
| base::Value::List event_nodes; |
| for (const auto& event : *app.events) { |
| DCHECK(!event.empty()); |
| event_nodes.Append(event.Clone()); |
| } |
| app_node.Set("event", std::move(event_nodes)); |
| } |
| |
| app_nodes.Append(std::move(app_node)); |
| } |
| |
| if (!app_nodes.empty()) |
| request_node.Set("app", std::move(app_nodes)); |
| |
| root_node.Set("request", std::move(request_node)); |
| |
| std::string msg; |
| return base::JSONWriter::WriteWithOptions( |
| root_node, base::JSONWriter::OPTIONS_OMIT_DOUBLE_TYPE_PRESERVATION, |
| &msg) |
| ? msg |
| : std::string(); |
| } |
| |
| } // namespace update_client |