| // Copyright (c) 2012 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 "chromeos/network/proxy/proxy_config_service_impl.h" |
| |
| #include <memory> |
| |
| #include "base/bind.h" |
| #include "base/bind_helpers.h" |
| #include "base/callback.h" |
| #include "base/location.h" |
| #include "base/logging.h" |
| #include "base/values.h" |
| #include "chromeos/network/network_profile.h" |
| #include "chromeos/network/network_profile_handler.h" |
| #include "chromeos/network/network_state.h" |
| #include "chromeos/network/network_state_handler.h" |
| #include "chromeos/network/onc/onc_utils.h" |
| #include "chromeos/network/proxy/proxy_config_handler.h" |
| #include "components/onc/onc_pref_names.h" |
| #include "components/prefs/pref_service.h" |
| #include "components/proxy_config/pref_proxy_config_tracker_impl.h" |
| #include "components/proxy_config/proxy_config_dictionary.h" |
| #include "components/proxy_config/proxy_config_pref_names.h" |
| #include "components/proxy_config/proxy_prefs.h" |
| #include "components/user_manager/user_manager.h" |
| |
| namespace chromeos { |
| |
| namespace { |
| |
| // Writes the proxy config of |network| to |proxy_config|. Set |onc_source| to |
| // the source of this configuration. Returns false if no proxy was configured |
| // for this network. |
| bool GetNetworkProxyConfig(const PrefService* profile_prefs, |
| const PrefService* local_state_prefs, |
| const NetworkState& network, |
| net::ProxyConfigWithAnnotation* proxy_config, |
| ::onc::ONCSource* onc_source) { |
| std::unique_ptr<ProxyConfigDictionary> proxy_dict = |
| proxy_config::GetProxyConfigForNetwork( |
| profile_prefs, local_state_prefs, network, |
| NetworkHandler::Get()->network_profile_handler(), onc_source); |
| if (!proxy_dict) |
| return false; |
| return PrefProxyConfigTrackerImpl::PrefConfigToNetConfig(*proxy_dict, |
| proxy_config); |
| } |
| |
| } // namespace |
| |
| ProxyConfigServiceImpl::ProxyConfigServiceImpl( |
| PrefService* profile_prefs, |
| PrefService* local_state_prefs, |
| scoped_refptr<base::SingleThreadTaskRunner> io_task_runner) |
| : PrefProxyConfigTrackerImpl( |
| profile_prefs ? profile_prefs : local_state_prefs, |
| io_task_runner), |
| profile_prefs_(profile_prefs), |
| local_state_prefs_(local_state_prefs) { |
| const base::RepeatingClosure proxy_change_callback = base::BindRepeating( |
| &ProxyConfigServiceImpl::OnProxyPrefChanged, base::Unretained(this)); |
| |
| if (profile_prefs) { |
| profile_pref_registrar_.Init(profile_prefs); |
| profile_pref_registrar_.Add(::onc::prefs::kOpenNetworkConfiguration, |
| proxy_change_callback); |
| profile_pref_registrar_.Add(::proxy_config::prefs::kUseSharedProxies, |
| proxy_change_callback); |
| } |
| local_state_pref_registrar_.Init(local_state_prefs); |
| local_state_pref_registrar_.Add(::onc::prefs::kDeviceOpenNetworkConfiguration, |
| proxy_change_callback); |
| |
| if (NetworkHandler::IsInitialized()) { // null in unit tests. |
| // Register for changes to the default network. |
| NetworkStateHandler* state_handler = |
| NetworkHandler::Get()->network_state_handler(); |
| state_handler->AddObserver(this, FROM_HERE); |
| DefaultNetworkChanged(state_handler->DefaultNetwork()); |
| } |
| } |
| |
| ProxyConfigServiceImpl::~ProxyConfigServiceImpl() { |
| if (NetworkHandler::IsInitialized()) { |
| NetworkHandler::Get()->network_state_handler()->RemoveObserver(this, |
| FROM_HERE); |
| } |
| } |
| |
| void ProxyConfigServiceImpl::OnProxyConfigChanged( |
| ProxyPrefs::ConfigState config_state, |
| const net::ProxyConfigWithAnnotation& config) { |
| VLOG(1) << "Got prefs change: " |
| << ProxyPrefs::ConfigStateToDebugString(config_state) |
| << ", mode=" << static_cast<int>(config.value().proxy_rules().type); |
| DetermineEffectiveConfigFromDefaultNetwork(); |
| } |
| |
| void ProxyConfigServiceImpl::OnProxyPrefChanged() { |
| DetermineEffectiveConfigFromDefaultNetwork(); |
| } |
| |
| void ProxyConfigServiceImpl::DefaultNetworkChanged( |
| const NetworkState* new_network) { |
| std::string new_network_path; |
| if (new_network) |
| new_network_path = new_network->path(); |
| |
| VLOG(1) << "DefaultNetworkChanged to '" << new_network_path << "'."; |
| VLOG_IF(1, new_network) << "New network: name=" << new_network->name() |
| << ", profile=" << new_network->profile_path(); |
| |
| // Even if the default network is the same, its proxy config (e.g. if private |
| // version of network replaces the shared version after login), or |
| // use-shared-proxies setting (e.g. after login) may have changed, so |
| // re-determine effective proxy config, and activate if different. |
| DetermineEffectiveConfigFromDefaultNetwork(); |
| } |
| |
| void ProxyConfigServiceImpl::OnShuttingDown() { |
| // Ownership of this class is complicated. Stop observing NetworkStateHandler |
| // when the class shuts down. |
| NetworkHandler::Get()->network_state_handler()->RemoveObserver(this, |
| FROM_HERE); |
| } |
| |
| // static |
| bool ProxyConfigServiceImpl::IgnoreProxy(const PrefService* profile_prefs, |
| const std::string network_profile_path, |
| ::onc::ONCSource onc_source) { |
| if (!profile_prefs) { |
| // If the profile preference are not available, this must be the object |
| // associated to local state used for system requests or login-profile. Make |
| // sure that proxies are enabled. |
| VLOG(1) << "Use proxy for system requests and sign-in screen."; |
| return false; |
| } |
| |
| if (network_profile_path.empty()) |
| return true; |
| |
| const NetworkProfile* profile = |
| NetworkHandler::Get()->network_profile_handler()->GetProfileForPath( |
| network_profile_path); |
| if (!profile) { |
| VLOG(1) << "Unknown profile_path '" << network_profile_path |
| << "'. Ignoring proxy."; |
| return true; |
| } |
| if (profile->type() == NetworkProfile::TYPE_USER) { |
| VLOG(1) << "Respect proxy of not-shared networks."; |
| return false; |
| } |
| if (onc_source == ::onc::ONC_SOURCE_USER_POLICY) { |
| // Note that this case can occur if the network is shared (e.g. ethernet) |
| // but the proxy is determined by user policy. |
| // See https://crbug.com/454966 . |
| VLOG(1) << "Respect proxy from user policy although network is shared."; |
| return false; |
| } |
| if (onc_source == ::onc::ONC_SOURCE_DEVICE_POLICY) { |
| const user_manager::User* primary_user = |
| user_manager::UserManager::Get()->GetPrimaryUser(); |
| if (!primary_user || primary_user->IsAffiliated()) { |
| VLOG(1) << "Respecting proxy for network, as the primary user belongs to " |
| << "the domain the device is enrolled to."; |
| return false; |
| } |
| } |
| |
| // This network is shared and not managed by the user's domain. |
| bool use_shared_proxies = |
| profile_prefs->GetBoolean(::proxy_config::prefs::kUseSharedProxies); |
| VLOG(1) << "Use proxy of shared network: " << use_shared_proxies; |
| return !use_shared_proxies; |
| } |
| |
| // static |
| std::unique_ptr<ProxyConfigDictionary> |
| ProxyConfigServiceImpl::GetActiveProxyConfigDictionary( |
| const PrefService* profile_prefs, |
| const PrefService* local_state_prefs) { |
| // Apply Pref Proxy configuration if available. |
| net::ProxyConfigWithAnnotation pref_proxy_config; |
| ProxyPrefs::ConfigState pref_state = |
| PrefProxyConfigTrackerImpl::ReadPrefConfig(profile_prefs, |
| &pref_proxy_config); |
| if (PrefProxyConfigTrackerImpl::PrefPrecedes(pref_state)) { |
| const PrefService::Preference* const pref = |
| profile_prefs->FindPreference(::proxy_config::prefs::kProxy); |
| DCHECK(pref->GetValue() && pref->GetValue()->is_dict()); |
| return std::make_unique<ProxyConfigDictionary>(pref->GetValue()->Clone()); |
| } |
| |
| const NetworkState* network = |
| NetworkHandler::Get()->network_state_handler()->DefaultNetwork(); |
| // No connected network. |
| if (!network) |
| return nullptr; |
| |
| // Apply network proxy configuration. |
| ::onc::ONCSource onc_source; |
| std::unique_ptr<ProxyConfigDictionary> proxy_config = |
| proxy_config::GetProxyConfigForNetwork( |
| profile_prefs, local_state_prefs, *network, |
| NetworkHandler::Get()->network_profile_handler(), &onc_source); |
| if (!ProxyConfigServiceImpl::IgnoreProxy(profile_prefs, |
| network->profile_path(), onc_source)) |
| return proxy_config; |
| |
| return std::make_unique<ProxyConfigDictionary>( |
| ProxyConfigDictionary::CreateDirect()); |
| } |
| |
| void ProxyConfigServiceImpl::DetermineEffectiveConfigFromDefaultNetwork() { |
| if (!NetworkHandler::IsInitialized()) |
| return; |
| |
| NetworkStateHandler* handler = NetworkHandler::Get()->network_state_handler(); |
| const NetworkState* network = handler->DefaultNetwork(); |
| |
| // Get prefs proxy config if available. |
| net::ProxyConfigWithAnnotation pref_config; |
| ProxyPrefs::ConfigState pref_state = GetProxyConfig(&pref_config); |
| |
| // Get network proxy config if available. |
| net::ProxyConfigWithAnnotation network_config; |
| net::ProxyConfigService::ConfigAvailability network_availability = |
| net::ProxyConfigService::CONFIG_UNSET; |
| bool ignore_proxy = true; |
| if (network) { |
| ::onc::ONCSource onc_source = ::onc::ONC_SOURCE_NONE; |
| const bool network_proxy_configured = GetNetworkProxyConfig( |
| prefs(), local_state_prefs_, *network, &network_config, &onc_source); |
| ignore_proxy = |
| IgnoreProxy(profile_prefs_, network->profile_path(), onc_source); |
| |
| // If network is shared but use-shared-proxies is off, use direct mode. |
| if (ignore_proxy) { |
| network_config = net::ProxyConfigWithAnnotation::CreateDirect(); |
| network_availability = net::ProxyConfigService::CONFIG_VALID; |
| } else if (network_proxy_configured) { |
| // Network is private or shared with user using shared proxies. |
| VLOG(1) << this << ": using proxy of network " << network->path(); |
| network_availability = net::ProxyConfigService::CONFIG_VALID; |
| } |
| } |
| |
| // Determine effective proxy config, either from prefs or network. |
| ProxyPrefs::ConfigState effective_config_state; |
| net::ProxyConfigWithAnnotation effective_config; |
| GetEffectiveProxyConfig(pref_state, pref_config, network_availability, |
| network_config, ignore_proxy, &effective_config_state, |
| &effective_config); |
| |
| // If effective config is from system (i.e. network), it's considered a |
| // special kind of prefs that ranks below policy/extension but above |
| // others, so bump it up to CONFIG_OTHER_PRECEDE to force its precedence |
| // when PrefProxyConfigTrackerImpl pushes it to ChromeProxyConfigService. |
| if (effective_config_state == ProxyPrefs::CONFIG_SYSTEM) |
| effective_config_state = ProxyPrefs::CONFIG_OTHER_PRECEDE; |
| // If config is manual, add rule to bypass local host. |
| if (effective_config.value().proxy_rules().type != |
| net::ProxyConfig::ProxyRules::Type::EMPTY) { |
| net::ProxyConfig proxy_config = effective_config.value(); |
| // TODO(https://crbug.com/902418): Is this rule still needed? |
| proxy_config.proxy_rules() |
| .bypass_rules.PrependRuleToBypassSimpleHostnames(); |
| effective_config = net::ProxyConfigWithAnnotation( |
| proxy_config, effective_config.traffic_annotation()); |
| } |
| PrefProxyConfigTrackerImpl::OnProxyConfigChanged(effective_config_state, |
| effective_config); |
| if (VLOG_IS_ON(1)) { |
| base::Value config_dict = effective_config.value().ToValue(); |
| VLOG(1) << this << ": Proxy changed: " |
| << ProxyPrefs::ConfigStateToDebugString(effective_config_state) |
| << ", " << config_dict; |
| } |
| } |
| |
| } // namespace chromeos |