// 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 (ProvideInternal(result)) { FixDeprecatedPolicies(result); return true; } return false; } bool ConfigurationPolicyProvider::IsInitializationComplete() const { return true; } // 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]); } } // Sets the new |proxy_settings| if kProxySettings isn't set yet, or if the // new priority is higher. const PolicyMap::Entry* existing = policies->Get(key::kProxySettings); if (!proxy_settings->empty() && (!existing || current_priority.has_higher_priority_than(*existing))) { 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); // The |observer_| might delete |this| during this callback. Don't touch any // of |this| field's after it returns. provider_->RemoveObserver(this); provider_ = NULL; observer_->OnProviderGoingAway(provider); } } // namespace policy