// Copyright (c) 2013 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 <stddef.h> #include <vector> #include "base/callback.h" #include "base/files/scoped_temp_dir.h" #include "base/json/json_reader.h" #include "base/macros.h" #include "base/run_loop.h" #include "base/values.h" #include "chrome/browser/extensions/extension_service.h" #include "chrome/browser/extensions/test_extension_system.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "chrome/common/url_constants.h" #include "chrome/test/base/in_process_browser_test.h" #include "chrome/test/base/ui_test_utils.h" #include "components/policy/core/browser/browser_policy_connector.h" #include "components/policy/core/common/external_data_fetcher.h" #include "components/policy/core/common/mock_configuration_policy_provider.h" #include "components/policy/core/common/policy_map.h" #include "components/policy/core/common/policy_types.h" #include "components/policy/core/common/schema.h" #include "content/public/browser/web_contents.h" #include "content/public/test/browser_test_utils.h" #include "extensions/common/extension_builder.h" #include "grit/components_strings.h" #include "policy/policy_constants.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" #include "ui/base/l10n/l10n_util.h" #include "url/gurl.h" using testing::Return; using testing::_; namespace { std::vector<std::string> PopulateExpectedPolicy( const std::string& name, const std::string& value, const std::string& source, const policy::PolicyMap::Entry* metadata, bool unknown) { std::vector<std::string> expected_policy; // Populate expected scope. if (metadata) { expected_policy.push_back(l10n_util::GetStringUTF8( metadata->scope == policy::POLICY_SCOPE_MACHINE ? IDS_POLICY_SCOPE_DEVICE : IDS_POLICY_SCOPE_USER)); } else { expected_policy.push_back(std::string()); } // Populate expected level. if (metadata) { expected_policy.push_back(l10n_util::GetStringUTF8( metadata->level == policy::POLICY_LEVEL_RECOMMENDED ? IDS_POLICY_LEVEL_RECOMMENDED : IDS_POLICY_LEVEL_MANDATORY)); } else { expected_policy.push_back(std::string()); } // Populate expected source name. expected_policy.push_back(source); // Populate expected policy name. expected_policy.push_back(name); // Populate expected policy value. expected_policy.push_back(value); // Populate expected status. if (unknown) expected_policy.push_back(l10n_util::GetStringUTF8(IDS_POLICY_UNKNOWN)); else if (metadata) expected_policy.push_back(l10n_util::GetStringUTF8(IDS_POLICY_OK)); else expected_policy.push_back(l10n_util::GetStringUTF8(IDS_POLICY_UNSET)); // Populate expected expanded policy value. expected_policy.push_back(value); return expected_policy; } } // namespace class PolicyUITest : public InProcessBrowserTest { public: PolicyUITest(); ~PolicyUITest() override; protected: // InProcessBrowserTest implementation. void SetUpInProcessBrowserTestFixture() override; void UpdateProviderPolicy(const policy::PolicyMap& policy); void VerifyPolicies(const std::vector<std::vector<std::string> >& expected); protected: policy::MockConfigurationPolicyProvider provider_; private: DISALLOW_COPY_AND_ASSIGN(PolicyUITest); }; PolicyUITest::PolicyUITest() { } PolicyUITest::~PolicyUITest() { } void PolicyUITest::SetUpInProcessBrowserTestFixture() { EXPECT_CALL(provider_, IsInitializationComplete(_)) .WillRepeatedly(Return(true)); policy::BrowserPolicyConnector::SetPolicyProviderForTesting(&provider_); } void PolicyUITest::UpdateProviderPolicy(const policy::PolicyMap& policy) { provider_.UpdateChromePolicy(policy); base::RunLoop loop; loop.RunUntilIdle(); } void PolicyUITest::VerifyPolicies( const std::vector<std::vector<std::string> >& expected_policies) { ui_test_utils::NavigateToURL(browser(), GURL("chrome://policy")); // Retrieve the text contents of the policy table cells for all policies. const std::string javascript = "var entries = document.querySelectorAll(" " 'section.policy-table-section > * > tbody');" "var policies = [];" "for (var i = 0; i < entries.length; ++i) {" " var items = entries[i].querySelectorAll('tr > td');" " var values = [];" " for (var j = 0; j < items.length; ++j) {" " var item = items[j];" " var children = item.getElementsByTagName('div');" " if (children.length == 1)" " item = children[0];" " children = item.getElementsByTagName('span');" " if (children.length == 1)" " item = children[0];" " values.push(item.textContent);" " }" " policies.push(values);" "}" "domAutomationController.send(JSON.stringify(policies));"; content::WebContents* contents = browser()->tab_strip_model()->GetActiveWebContents(); std::string json; ASSERT_TRUE(content::ExecuteScriptAndExtractString(contents, javascript, &json)); scoped_ptr<base::Value> value_ptr = base::JSONReader::Read(json); const base::ListValue* actual_policies = NULL; ASSERT_TRUE(value_ptr.get()); ASSERT_TRUE(value_ptr->GetAsList(&actual_policies)); // Verify that the cells contain the expected strings for all policies. ASSERT_EQ(expected_policies.size(), actual_policies->GetSize()); for (size_t i = 0; i < expected_policies.size(); ++i) { const std::vector<std::string> expected_policy = expected_policies[i]; const base::ListValue* actual_policy; ASSERT_TRUE(actual_policies->GetList(i, &actual_policy)); ASSERT_EQ(expected_policy.size(), actual_policy->GetSize()); for (size_t j = 0; j < expected_policy.size(); ++j) { std::string value; ASSERT_TRUE(actual_policy->GetString(j, &value)); EXPECT_EQ(expected_policy[j], value); } } } IN_PROC_BROWSER_TEST_F(PolicyUITest, SendPolicyNames) { // Verifies that the names of known policies are sent to the UI and processed // there correctly by checking that the policy table contains all policies in // the correct order. // Expect that the policy table contains all known policies in alphabetical // order and none of the policies have a set value. std::vector<std::vector<std::string> > expected_policies; policy::Schema chrome_schema = policy::Schema::Wrap(policy::GetChromeSchemaData()); ASSERT_TRUE(chrome_schema.valid()); for (policy::Schema::Iterator it = chrome_schema.GetPropertiesIterator(); !it.IsAtEnd(); it.Advance()) { expected_policies.push_back( PopulateExpectedPolicy( it.key(), std::string(), std::string(), nullptr, false)); } // Retrieve the contents of the policy table from the UI and verify that it // matches the expectation. VerifyPolicies(expected_policies); } IN_PROC_BROWSER_TEST_F(PolicyUITest, SendPolicyValues) { // Verifies that policy values are sent to the UI and processed there // correctly by setting the values of four known and one unknown policy and // checking that the policy table contains the policy names, values and // metadata in the correct order. policy::PolicyMap values; std::map<std::string, std::string> expected_values; // Set the values of four existing policies. base::ListValue* restore_on_startup_urls = new base::ListValue; restore_on_startup_urls->Append(new base::StringValue("aaa")); restore_on_startup_urls->Append(new base::StringValue("bbb")); restore_on_startup_urls->Append(new base::StringValue("ccc")); values.Set(policy::key::kRestoreOnStartupURLs, policy::POLICY_LEVEL_MANDATORY, policy::POLICY_SCOPE_USER, policy::POLICY_SOURCE_CLOUD, restore_on_startup_urls, NULL); expected_values[policy::key::kRestoreOnStartupURLs] = "aaa,bbb,ccc"; values.Set(policy::key::kHomepageLocation, policy::POLICY_LEVEL_MANDATORY, policy::POLICY_SCOPE_MACHINE, policy::POLICY_SOURCE_CLOUD, new base::StringValue("http://google.com"), NULL); expected_values[policy::key::kHomepageLocation] = "http://google.com"; values.Set(policy::key::kRestoreOnStartup, policy::POLICY_LEVEL_RECOMMENDED, policy::POLICY_SCOPE_USER, policy::POLICY_SOURCE_CLOUD, new base::FundamentalValue(4), NULL); expected_values[policy::key::kRestoreOnStartup] = "4"; values.Set(policy::key::kShowHomeButton, policy::POLICY_LEVEL_RECOMMENDED, policy::POLICY_SCOPE_MACHINE, policy::POLICY_SOURCE_CLOUD, new base::FundamentalValue(true), NULL); expected_values[policy::key::kShowHomeButton] = "true"; // Set the value of a policy that does not exist. const std::string kUnknownPolicy = "NoSuchThing"; values.Set(kUnknownPolicy, policy::POLICY_LEVEL_MANDATORY, policy::POLICY_SCOPE_USER, policy::POLICY_SOURCE_PLATFORM, new base::FundamentalValue(true), NULL); expected_values[kUnknownPolicy] = "true"; UpdateProviderPolicy(values); // Expect that the policy table contains, in order: // * All known policies whose value has been set, in alphabetical order. // * The unknown policy. // * All known policies whose value has not been set, in alphabetical order. std::vector<std::vector<std::string> > expected_policies; size_t first_unset_position = 0; policy::Schema chrome_schema = policy::Schema::Wrap(policy::GetChromeSchemaData()); ASSERT_TRUE(chrome_schema.valid()); for (policy::Schema::Iterator props = chrome_schema.GetPropertiesIterator(); !props.IsAtEnd(); props.Advance()) { std::map<std::string, std::string>::const_iterator it = expected_values.find(props.key()); const std::string value = it == expected_values.end() ? std::string() : it->second; const std::string source = it == expected_values.end() ? std::string() : "Cloud"; const policy::PolicyMap::Entry* metadata = values.Get(props.key()); expected_policies.insert( metadata ? expected_policies.begin() + first_unset_position++ : expected_policies.end(), PopulateExpectedPolicy(props.key(), value, source, metadata, false)); } expected_policies.insert( expected_policies.begin() + first_unset_position++, PopulateExpectedPolicy(kUnknownPolicy, expected_values[kUnknownPolicy], "Platform", values.Get(kUnknownPolicy), true)); // Retrieve the contents of the policy table from the UI and verify that it // matches the expectation. VerifyPolicies(expected_policies); } IN_PROC_BROWSER_TEST_F(PolicyUITest, ExtensionLoadAndSendPolicy) { ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIPolicyURL)); base::ScopedTempDir temp_dir_; ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); const std::string newly_added_policy_name = "new_policy"; std::string json_data = "{\"type\": \"object\",\"properties\": {\"" + newly_added_policy_name + "\": { \"type\": \"string\"}}}"; const std::string schema_file = "schema.json"; base::FilePath schema_path = temp_dir_.path().AppendASCII(schema_file); base::WriteFile(schema_path, json_data.data(), json_data.size()); // Build extension that contains the policy schema. extensions::DictionaryBuilder storage; storage.Set("managed_schema", schema_file); extensions::DictionaryBuilder manifest; manifest.Set("name", "test") .Set("version", "1") .Set("manifest_version", 2) .Set("storage", storage.Build()); extensions::ExtensionBuilder builder; builder.SetPath(temp_dir_.path()); builder.SetManifest(manifest.Build()); // Install extension. ExtensionService* service = extensions::ExtensionSystem::Get( browser()->profile())->extension_service(); EXPECT_CALL(provider_, RefreshPolicies()); service->OnExtensionInstalled(builder.Build().get(), syncer::StringOrdinal(), 0); std::vector<std::vector<std::string>> expected_policies; policy::Schema chrome_schema = policy::Schema::Wrap(policy::GetChromeSchemaData()); ASSERT_TRUE(chrome_schema.valid()); for (policy::Schema::Iterator it = chrome_schema.GetPropertiesIterator(); !it.IsAtEnd(); it.Advance()) { expected_policies.push_back( PopulateExpectedPolicy( it.key(), std::string(), std::string(), nullptr, false)); } // Add newly added policy to expected policy list. expected_policies.push_back(PopulateExpectedPolicy( newly_added_policy_name, std::string(), std::string(), nullptr, false)); // Verify if policy UI includes policy that extension have. VerifyPolicies(expected_policies); }