// 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 "chrome/browser/policy/configuration_policy_provider.h" #include #include "chrome/browser/policy/policy_map.h" #include "policy/policy_constants.h" namespace policy { namespace { const char* kProxyPolicies[] = { key::kProxyMode, key::kProxyServerMode, key::kProxyServer, key::kProxyPacUrl, key::kProxyBypassList, }; } // namespace ConfigurationPolicyProvider::Observer::~Observer() {} void ConfigurationPolicyProvider::Observer::OnProviderGoingAway( ConfigurationPolicyProvider* provider) {} // Class ConfigurationPolicyProvider. ConfigurationPolicyProvider::ConfigurationPolicyProvider( const PolicyDefinitionList* policy_list) : policy_definition_list_(policy_list) { } ConfigurationPolicyProvider::~ConfigurationPolicyProvider() { FOR_EACH_OBSERVER(ConfigurationPolicyProvider::Observer, observer_list_, OnProviderGoingAway(this)); } bool ConfigurationPolicyProvider::Provide(PolicyMap* result) { #if !defined(OFFICIAL_BUILD) if (override_policies_.get()) { result->CopyFrom(*override_policies_); return true; } #endif if (ProvideInternal(result)) { FixDeprecatedPolicies(result); return true; } return false; } bool ConfigurationPolicyProvider::IsInitializationComplete() const { return true; } #if !defined(OFFICIAL_BUILD) void ConfigurationPolicyProvider::OverridePolicies(PolicyMap* policies) { if (policies) FixDeprecatedPolicies(policies); override_policies_.reset(policies); NotifyPolicyUpdated(); } #endif // static void ConfigurationPolicyProvider::FixDeprecatedPolicies(PolicyMap* policies) { // Proxy settings have been configured by 5 policies that didn't mix well // together, and maps of policies had to take this into account when merging // policy sources. The proxy settings will eventually be configured by a // single Dictionary policy when all providers have support for that. For // now, the individual policies are mapped here to a single Dictionary policy // that the rest of the policy machinery uses. // The highest (level, scope) pair for an existing proxy policy is determined // first, and then only policies with those exact attributes are merged. PolicyMap::Entry current_priority; // Defaults to the lowest priority. scoped_ptr proxy_settings(new DictionaryValue); for (size_t i = 0; i < arraysize(kProxyPolicies); ++i) { const PolicyMap::Entry* entry = policies->Get(kProxyPolicies[i]); if (entry) { if (entry->has_higher_priority_than(current_priority)) { proxy_settings->Clear(); current_priority = *entry; } if (!entry->has_higher_priority_than(current_priority) && !current_priority.has_higher_priority_than(*entry)) { proxy_settings->Set(kProxyPolicies[i], entry->value->DeepCopy()); } policies->Erase(kProxyPolicies[i]); } } if (!proxy_settings->empty() && !policies->Get(key::kProxySettings)) { policies->Set(key::kProxySettings, current_priority.level, current_priority.scope, proxy_settings.release()); } } void ConfigurationPolicyProvider::NotifyPolicyUpdated() { FOR_EACH_OBSERVER(ConfigurationPolicyProvider::Observer, observer_list_, OnUpdatePolicy(this)); } void ConfigurationPolicyProvider::AddObserver(Observer* observer) { observer_list_.AddObserver(observer); } void ConfigurationPolicyProvider::RemoveObserver(Observer* observer) { observer_list_.RemoveObserver(observer); } ConfigurationPolicyObserverRegistrar::ConfigurationPolicyObserverRegistrar() : provider_(NULL), observer_(NULL) {} ConfigurationPolicyObserverRegistrar::~ConfigurationPolicyObserverRegistrar() { if (provider_) provider_->RemoveObserver(this); } void ConfigurationPolicyObserverRegistrar::Init( ConfigurationPolicyProvider* provider, ConfigurationPolicyProvider::Observer* observer) { provider_ = provider; observer_ = observer; provider_->AddObserver(this); } void ConfigurationPolicyObserverRegistrar::OnUpdatePolicy( ConfigurationPolicyProvider* provider) { DCHECK_EQ(provider_, provider); observer_->OnUpdatePolicy(provider_); } void ConfigurationPolicyObserverRegistrar::OnProviderGoingAway( ConfigurationPolicyProvider* provider) { DCHECK_EQ(provider_, provider); observer_->OnProviderGoingAway(provider_); provider_->RemoveObserver(this); provider_ = NULL; } } // namespace policy