// 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/policy_bundle.h" #include "base/logging.h" #include "base/stl_util.h" #include "chrome/browser/policy/policy_map.h" namespace policy { PolicyBundle::PolicyBundle() {} PolicyBundle::~PolicyBundle() { Clear(); } PolicyMap& PolicyBundle::Get(PolicyDomain domain, const std::string& component_id) { DCHECK(domain != POLICY_DOMAIN_CHROME || component_id.empty()); PolicyMap*& policy = policy_bundle_[PolicyNamespace(domain, component_id)]; if (!policy) policy = new PolicyMap(); return *policy; } const PolicyMap& PolicyBundle::Get(PolicyDomain domain, const std::string& component_id) const { DCHECK(domain != POLICY_DOMAIN_CHROME || component_id.empty()); PolicyNamespace ns(domain, component_id); const_iterator it = policy_bundle_.find(ns); return it == end() ? kEmpty_ : *it->second; } void PolicyBundle::Swap(PolicyBundle* other) { policy_bundle_.swap(other->policy_bundle_); } void PolicyBundle::CopyFrom(const PolicyBundle& other) { Clear(); for (PolicyBundle::const_iterator it = other.begin(); it != other.end(); ++it) { PolicyMap*& policy = policy_bundle_[it->first]; policy = new PolicyMap(); policy->CopyFrom(*it->second); } } void PolicyBundle::MergeFrom(const PolicyBundle& other) { // Iterate over both |this| and |other| in order; skip what's extra in |this|, // add what's missing, and merge the namespaces in common. MapType::iterator it_this = policy_bundle_.begin(); MapType::iterator end_this = policy_bundle_.end(); const_iterator it_other = other.begin(); const_iterator end_other = other.end(); while (it_this != end_this && it_other != end_other) { if (it_this->first == it_other->first) { // Same namespace: merge existing PolicyMaps. it_this->second->MergeFrom(*it_other->second); ++it_this; ++it_other; } else if (it_this->first < it_other->first) { // |this| has a PolicyMap that |other| doesn't; skip it. ++it_this; } else if (it_this->first > it_other->first) { // |other| has a PolicyMap that |this| doesn't; copy it. PolicyMap*& policy = policy_bundle_[it_other->first]; DCHECK(!policy); policy = new PolicyMap(); policy->CopyFrom(*it_other->second); ++it_other; } else { NOTREACHED(); } } // Add extra PolicyMaps at the end. while (it_other != end_other) { PolicyMap*& policy = policy_bundle_[it_other->first]; DCHECK(!policy); policy = new PolicyMap(); policy->CopyFrom(*it_other->second); ++it_other; } } bool PolicyBundle::Equals(const PolicyBundle& other) const { // Equals() has the peculiarity that an entry with an empty PolicyMap equals // an non-existant entry. This handles usage of non-const Get() that doesn't // insert any policies. const_iterator it_this = begin(); const_iterator it_other = other.begin(); while (true) { // Skip empty PolicyMaps. while (it_this != end() && it_this->second->empty()) ++it_this; while (it_other != other.end() && it_other->second->empty()) ++it_other; if (it_this == end() || it_other == other.end()) break; if (it_this->first != it_other->first || !it_this->second->Equals(*it_other->second)) { return false; } ++it_this; ++it_other; } return it_this == end() && it_other == other.end(); } PolicyBundle::const_iterator PolicyBundle::begin() const { return policy_bundle_.begin(); } PolicyBundle::const_iterator PolicyBundle::end() const { return policy_bundle_.end(); } void PolicyBundle::Clear() { STLDeleteValues(&policy_bundle_); } } // namespace policy