Home | History | Annotate | Download | only in chromeos
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "chrome/browser/chromeos/proxy_config_service_impl.h"
      6 
      7 #include "base/bind.h"
      8 #include "base/bind_helpers.h"
      9 #include "base/callback.h"
     10 #include "base/location.h"
     11 #include "base/logging.h"
     12 #include "base/prefs/pref_service.h"
     13 #include "base/values.h"
     14 #include "chrome/browser/browser_process.h"
     15 #include "chrome/browser/chromeos/login/user_manager.h"
     16 #include "chrome/browser/chromeos/net/proxy_config_handler.h"
     17 #include "chrome/browser/policy/browser_policy_connector.h"
     18 #include "chrome/browser/prefs/proxy_config_dictionary.h"
     19 #include "chrome/browser/prefs/proxy_prefs.h"
     20 #include "chrome/common/pref_names.h"
     21 #include "chromeos/network/favorite_state.h"
     22 #include "chromeos/network/network_profile.h"
     23 #include "chromeos/network/network_profile_handler.h"
     24 #include "chromeos/network/network_state.h"
     25 #include "chromeos/network/network_state_handler.h"
     26 #include "chromeos/network/onc/onc_utils.h"
     27 #include "components/policy/core/common/cloud/cloud_policy_constants.h"
     28 
     29 namespace chromeos {
     30 
     31 namespace {
     32 
     33 // Writes the proxy config of |network| to |proxy_config|.  Set |onc_source| to
     34 // the source of this configuration. Returns false if no
     35 // proxy was configured for this network.
     36 bool GetProxyConfig(const PrefService* profile_prefs,
     37                     const PrefService* local_state_prefs,
     38                     const FavoriteState& network,
     39                     net::ProxyConfig* proxy_config,
     40                     ::onc::ONCSource* onc_source) {
     41   scoped_ptr<ProxyConfigDictionary> proxy_dict =
     42       proxy_config::GetProxyConfigForFavoriteNetwork(
     43           profile_prefs, local_state_prefs, network, onc_source);
     44   if (!proxy_dict)
     45     return false;
     46   return PrefProxyConfigTrackerImpl::PrefConfigToNetConfig(*proxy_dict,
     47                                                            proxy_config);
     48 }
     49 
     50 }  // namespace
     51 
     52 ProxyConfigServiceImpl::ProxyConfigServiceImpl(PrefService* profile_prefs,
     53                                                PrefService* local_state_prefs)
     54     : PrefProxyConfigTrackerImpl(profile_prefs ? profile_prefs
     55                                                : local_state_prefs),
     56       active_config_state_(ProxyPrefs::CONFIG_UNSET),
     57       profile_prefs_(profile_prefs),
     58       local_state_prefs_(local_state_prefs),
     59       pointer_factory_(this) {
     60   const base::Closure proxy_change_callback = base::Bind(
     61       &ProxyConfigServiceImpl::OnProxyPrefChanged, base::Unretained(this));
     62 
     63   if (profile_prefs) {
     64     profile_pref_registrar_.Init(profile_prefs);
     65     profile_pref_registrar_.Add(prefs::kOpenNetworkConfiguration,
     66                                 proxy_change_callback);
     67     profile_pref_registrar_.Add(prefs::kUseSharedProxies,
     68                                 proxy_change_callback);
     69   }
     70   local_state_pref_registrar_.Init(local_state_prefs);
     71   local_state_pref_registrar_.Add(prefs::kDeviceOpenNetworkConfiguration,
     72                                   proxy_change_callback);
     73 
     74   // Register for changes to the default network.
     75   NetworkStateHandler* state_handler =
     76       NetworkHandler::Get()->network_state_handler();
     77   state_handler->AddObserver(this, FROM_HERE);
     78   DefaultNetworkChanged(state_handler->DefaultNetwork());
     79 }
     80 
     81 ProxyConfigServiceImpl::~ProxyConfigServiceImpl() {
     82   if (NetworkHandler::IsInitialized()) {
     83     NetworkHandler::Get()->network_state_handler()->RemoveObserver(
     84         this, FROM_HERE);
     85   }
     86 }
     87 
     88 void ProxyConfigServiceImpl::OnProxyConfigChanged(
     89     ProxyPrefs::ConfigState config_state,
     90     const net::ProxyConfig& config) {
     91   VLOG(1) << "Got prefs change: "
     92           << ProxyPrefs::ConfigStateToDebugString(config_state)
     93           << ", mode=" << config.proxy_rules().type;
     94   DetermineEffectiveConfigFromDefaultNetwork();
     95 }
     96 
     97 void ProxyConfigServiceImpl::OnProxyPrefChanged() {
     98   DetermineEffectiveConfigFromDefaultNetwork();
     99 }
    100 
    101 void ProxyConfigServiceImpl::DefaultNetworkChanged(
    102     const NetworkState* new_network) {
    103   std::string new_network_path;
    104   if (new_network)
    105     new_network_path = new_network->path();
    106 
    107   VLOG(1) << "DefaultNetworkChanged to '" << new_network_path << "'.";
    108   VLOG_IF(1, new_network) << "New network: name=" << new_network->name()
    109                           << ", profile=" << new_network->profile_path();
    110 
    111   // Even if the default network is the same, its proxy config (e.g. if private
    112   // version of network replaces the shared version after login), or
    113   // use-shared-proxies setting (e.g. after login) may have changed, so
    114   // re-determine effective proxy config, and activate if different.
    115   DetermineEffectiveConfigFromDefaultNetwork();
    116 }
    117 
    118 // static
    119 bool ProxyConfigServiceImpl::IgnoreProxy(const PrefService* profile_prefs,
    120                                          const std::string network_profile_path,
    121                                          ::onc::ONCSource onc_source) {
    122   if (!profile_prefs) {
    123     // If the profile preference are not available, this must be the object
    124     // associated to local state used for system requests or login-profile. Make
    125     // sure that proxies are enabled.
    126     VLOG(1) << "Use proxy for system requests and sign-in screen.";
    127     return false;
    128   }
    129 
    130   if (network_profile_path.empty())
    131     return true;
    132 
    133   const NetworkProfile* profile = NetworkHandler::Get()
    134       ->network_profile_handler()->GetProfileForPath(network_profile_path);
    135   if (!profile) {
    136     LOG(WARNING) << "Unknown profile_path '" << network_profile_path
    137                  << "'. Ignoring proxy.";
    138     return true;
    139   }
    140   if (profile->type() == NetworkProfile::TYPE_USER) {
    141     VLOG(1) << "Respect proxy of not-shared networks.";
    142     return false;
    143   }
    144   if (onc_source == ::onc::ONC_SOURCE_DEVICE_POLICY) {
    145     policy::BrowserPolicyConnector* connector =
    146         g_browser_process->browser_policy_connector();
    147     const User* logged_in_user = UserManager::Get()->GetLoggedInUser();
    148     if (connector->GetUserAffiliation(logged_in_user->email()) ==
    149         policy::USER_AFFILIATION_MANAGED) {
    150       VLOG(1) << "Respecting proxy for network, as logged-in user belongs to "
    151               << "the domain the device is enrolled to.";
    152       return false;
    153     }
    154   }
    155 
    156   // This network is shared and not managed by the user's domain.
    157   bool use_shared_proxies = profile_prefs->GetBoolean(prefs::kUseSharedProxies);
    158   VLOG(1) << "Use proxy of shared network: " << use_shared_proxies;
    159   return !use_shared_proxies;
    160 }
    161 
    162 void ProxyConfigServiceImpl::DetermineEffectiveConfigFromDefaultNetwork() {
    163   NetworkStateHandler* handler = NetworkHandler::Get()->network_state_handler();
    164   const FavoriteState* network = handler->DefaultFavoriteNetwork();
    165 
    166   // Get prefs proxy config if available.
    167   net::ProxyConfig pref_config;
    168   ProxyPrefs::ConfigState pref_state = GetProxyConfig(&pref_config);
    169 
    170   // Get network proxy config if available.
    171   net::ProxyConfig network_config;
    172   net::ProxyConfigService::ConfigAvailability network_availability =
    173       net::ProxyConfigService::CONFIG_UNSET;
    174   bool ignore_proxy = true;
    175   if (network) {
    176     ::onc::ONCSource onc_source = ::onc::ONC_SOURCE_NONE;
    177     const bool network_proxy_configured = chromeos::GetProxyConfig(
    178         prefs(), local_state_prefs_, *network, &network_config, &onc_source);
    179     ignore_proxy =
    180         IgnoreProxy(profile_prefs_, network->profile_path(), onc_source);
    181 
    182     // If network is shared but use-shared-proxies is off, use direct mode.
    183     if (ignore_proxy) {
    184       network_config = net::ProxyConfig();
    185       network_availability = net::ProxyConfigService::CONFIG_VALID;
    186     } else if (network_proxy_configured) {
    187       // Network is private or shared with user using shared proxies.
    188       VLOG(1) << this << ": using proxy of network " << network->path();
    189       network_availability = net::ProxyConfigService::CONFIG_VALID;
    190     }
    191   }
    192 
    193   // Determine effective proxy config, either from prefs or network.
    194   ProxyPrefs::ConfigState effective_config_state;
    195   net::ProxyConfig effective_config;
    196   GetEffectiveProxyConfig(pref_state, pref_config,
    197                           network_availability, network_config, ignore_proxy,
    198                           &effective_config_state, &effective_config);
    199 
    200   // Activate effective proxy and store into |active_config_|.
    201   // If last update didn't complete, we definitely update now.
    202   bool update_now = update_pending();
    203   if (!update_now) {  // Otherwise, only update now if there're changes.
    204     update_now = active_config_state_ != effective_config_state ||
    205                  (active_config_state_ != ProxyPrefs::CONFIG_UNSET &&
    206                   !active_config_.Equals(effective_config));
    207   }
    208   if (update_now) {  // Activate and store new effective config.
    209     active_config_state_ = effective_config_state;
    210     if (active_config_state_ != ProxyPrefs::CONFIG_UNSET)
    211       active_config_ = effective_config;
    212     // If effective config is from system (i.e. network), it's considered a
    213     // special kind of prefs that ranks below policy/extension but above
    214     // others, so bump it up to CONFIG_OTHER_PRECEDE to force its precedence
    215     // when PrefProxyConfigTrackerImpl pushes it to ChromeProxyConfigService.
    216     if (effective_config_state == ProxyPrefs::CONFIG_SYSTEM)
    217       effective_config_state = ProxyPrefs::CONFIG_OTHER_PRECEDE;
    218     // If config is manual, add rule to bypass local host.
    219     if (effective_config.proxy_rules().type !=
    220         net::ProxyConfig::ProxyRules::TYPE_NO_RULES) {
    221       effective_config.proxy_rules().bypass_rules.AddRuleToBypassLocal();
    222     }
    223     PrefProxyConfigTrackerImpl::OnProxyConfigChanged(effective_config_state,
    224                                                      effective_config);
    225     if (VLOG_IS_ON(1) && !update_pending()) {  // Update was successful.
    226       scoped_ptr<base::DictionaryValue> config_dict(effective_config.ToValue());
    227       VLOG(1) << this << ": Proxy changed: "
    228               << ProxyPrefs::ConfigStateToDebugString(active_config_state_)
    229               << ", " << *config_dict;
    230     }
    231   }
    232 }
    233 
    234 }  // namespace chromeos
    235