| // Copyright 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 "chrome/browser/ui/webui/extensions/extension_settings_browsertest.h" |
| |
| #include <string> |
| |
| #include "base/command_line.h" |
| #include "base/files/file_util.h" |
| #include "base/path_service.h" |
| #include "base/strings/string_util.h" |
| #include "base/threading/thread_restrictions.h" |
| #include "build/build_config.h" |
| #include "chrome/browser/extensions/activity_log/activity_log.h" |
| #include "chrome/browser/extensions/api/developer_private/developer_private_api.h" |
| #include "chrome/browser/extensions/chrome_test_extension_loader.h" |
| #include "chrome/browser/extensions/unpacked_installer.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/chrome_paths.h" |
| #include "chrome/common/chrome_switches.h" |
| #include "chrome/common/pref_names.h" |
| #include "chrome/test/base/ui_test_utils.h" |
| #include "components/prefs/pref_service.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/test/browser_test.h" |
| #include "content/public/test/browser_test_utils.h" |
| #include "content/public/test/test_utils.h" |
| #include "extensions/browser/extension_dialog_auto_confirm.h" |
| #include "extensions/browser/extension_system.h" |
| #include "extensions/test/extension_test_message_listener.h" |
| |
| using extensions::Extension; |
| using extensions::TestManagementPolicyProvider; |
| |
| ExtensionSettingsUIBrowserTest::ExtensionSettingsUIBrowserTest() |
| : policy_provider_(TestManagementPolicyProvider::PROHIBIT_MODIFY_STATUS | |
| TestManagementPolicyProvider::MUST_REMAIN_ENABLED | |
| TestManagementPolicyProvider::MUST_REMAIN_INSTALLED), |
| test_data_dir_(base::PathService::CheckedGet(chrome::DIR_TEST_DATA) |
| .AppendASCII("extensions")) {} |
| |
| ExtensionSettingsUIBrowserTest::~ExtensionSettingsUIBrowserTest() {} |
| |
| void ExtensionSettingsUIBrowserTest::InstallGoodExtension() { |
| EXPECT_TRUE(InstallExtension(test_data_dir_.AppendASCII("good.crx"))); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::InstallErrorsExtension() { |
| EXPECT_TRUE( |
| InstallExtension(test_data_dir_.AppendASCII("error_console") |
| .AppendASCII("runtime_and_manifest_errors"))); |
| EXPECT_TRUE(InstallExtension(test_data_dir_.AppendASCII("error_console") |
| .AppendASCII("deep_stack_trace"))); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::InstallSharedModule() { |
| base::FilePath shared_module_path = |
| test_data_dir_.AppendASCII("api_test").AppendASCII("shared_module"); |
| EXPECT_TRUE(InstallExtension(shared_module_path.AppendASCII("shared"))); |
| EXPECT_TRUE(InstallExtension(shared_module_path.AppendASCII("import_pass"))); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::InstallPackagedApp() { |
| EXPECT_TRUE(InstallExtension(test_data_dir_.AppendASCII("packaged_app"))); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::InstallHostedApp() { |
| EXPECT_TRUE(InstallExtension(test_data_dir_.AppendASCII("hosted_app"))); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::InstallPlatformApp() { |
| EXPECT_TRUE(InstallExtension( |
| test_data_dir_.AppendASCII("platform_apps").AppendASCII("minimal"))); |
| } |
| |
| const extensions::Extension* |
| ExtensionSettingsUIBrowserTest::InstallExtensionWithInPageOptions() { |
| const extensions::Extension* extension = |
| InstallExtension(test_data_dir_.AppendASCII("options_page_in_view")); |
| EXPECT_TRUE(extension); |
| return extension; |
| } |
| |
| void ExtensionSettingsUIBrowserTest::AddManagedPolicyProvider() { |
| extensions::ExtensionSystem* extension_system = |
| extensions::ExtensionSystem::Get(browser()->profile()); |
| extension_system->management_policy()->RegisterProvider(&policy_provider_); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::SetAutoConfirmUninstall() { |
| uninstall_auto_confirm_ = |
| std::make_unique<extensions::ScopedTestDialogAutoConfirm>( |
| extensions::ScopedTestDialogAutoConfirm::ACCEPT); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::SetDevModeEnabled(bool enabled) { |
| browser()->profile()->GetPrefs()->SetBoolean( |
| prefs::kExtensionsUIDeveloperMode, enabled); |
| } |
| |
| void ExtensionSettingsUIBrowserTest::ShrinkWebContentsView() { |
| content::WebContents* web_contents = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| CHECK(web_contents); |
| web_contents->Resize(gfx::Rect(0, 0, 400, 400)); |
| } |
| |
| const Extension* ExtensionSettingsUIBrowserTest::InstallExtension( |
| const base::FilePath& path) { |
| extensions::ChromeTestExtensionLoader loader(browser()->profile()); |
| loader.set_ignore_manifest_warnings(true); |
| return loader.LoadExtension(path).get(); |
| } |
| |
| // Tests that viewing a source of the options page works fine. |
| // This is a regression test for https://crbug.com/796080. |
| IN_PROC_BROWSER_TEST_F(ExtensionSettingsUIBrowserTest, ViewSource) { |
| // Navigate to an in-page (guest-view-based) extension options page |
| // and grab the WebContents hosting the options page. |
| const extensions::Extension* extension = InstallExtensionWithInPageOptions(); |
| GURL options_url("chrome://extensions/?options=" + extension->id()); |
| content::WebContents* options_contents = nullptr; |
| { |
| content::WebContentsAddedObserver options_contents_added_observer; |
| ui_test_utils::NavigateToURL(browser(), options_url); |
| options_contents = options_contents_added_observer.GetWebContents(); |
| } |
| ASSERT_TRUE(options_contents); |
| EXPECT_TRUE(content::WaitForLoadStop(options_contents)); |
| EXPECT_EQ(extension->GetResourceURL("options.html"), |
| options_contents->GetLastCommittedURL()); |
| |
| // Open the view-source of the options page. |
| int old_tabs_count = browser()->tab_strip_model()->count(); |
| content::WebContentsAddedObserver view_source_contents_added_observer; |
| options_contents->GetMainFrame()->ViewSource(); |
| content::WebContents* view_source_contents = |
| view_source_contents_added_observer.GetWebContents(); |
| ASSERT_TRUE(view_source_contents); |
| EXPECT_TRUE(content::WaitForLoadStop(view_source_contents)); |
| |
| // Verify that the view-source is present in the tab-strip. |
| int new_tabs_count = browser()->tab_strip_model()->count(); |
| EXPECT_EQ(new_tabs_count, old_tabs_count + 1); |
| EXPECT_EQ(view_source_contents, |
| browser()->tab_strip_model()->GetActiveWebContents()); |
| |
| // Verify the contents of the view-source tab. |
| std::string actual_source_text; |
| std::string view_source_extraction_script = R"( |
| output = ""; |
| document.querySelectorAll(".line-content").forEach(function(elem) { |
| output += elem.innerText; |
| }); |
| domAutomationController.send(output); )"; |
| EXPECT_TRUE(content::ExecuteScriptAndExtractString( |
| view_source_contents, view_source_extraction_script, |
| &actual_source_text)); |
| base::FilePath source_path = |
| test_data_dir().AppendASCII("options_page_in_view/options.html"); |
| std::string expected_source_text; |
| { |
| base::ScopedAllowBlockingForTesting scoped_allow_blocking; |
| EXPECT_TRUE(base::ReadFileToString(source_path, &expected_source_text)); |
| } |
| EXPECT_TRUE( |
| base::RemoveChars(expected_source_text, "\n", &expected_source_text)); |
| EXPECT_EQ(expected_source_text, actual_source_text); |
| } |
| |
| // Verify that listeners for the developer private API are only registered |
| // when there is a chrome://extensions page open. This is important, since some |
| // of the event construction can be expensive. |
| IN_PROC_BROWSER_TEST_F(ExtensionSettingsUIBrowserTest, ListenerRegistration) { |
| Profile* profile = browser()->profile(); |
| extensions::EventRouter* event_router = extensions::EventRouter::Get(profile); |
| extensions::DeveloperPrivateAPI* dev_private_api = |
| extensions::DeveloperPrivateAPI::Get(profile); |
| auto expect_has_listeners = [event_router, |
| dev_private_api](bool has_listeners) { |
| EXPECT_EQ(has_listeners, event_router->HasEventListener( |
| "developerPrivate.onItemStateChanged")); |
| EXPECT_EQ(has_listeners, event_router->HasEventListener( |
| "developerPrivate.onProfileStateChanged")); |
| EXPECT_EQ(has_listeners, |
| dev_private_api->developer_private_event_router() != nullptr); |
| }; |
| |
| { |
| SCOPED_TRACE("Before page load"); |
| expect_has_listeners(false); |
| } |
| |
| ui_test_utils::NavigateToURLWithDisposition( |
| browser(), GURL("chrome://extensions"), |
| WindowOpenDisposition::NEW_FOREGROUND_TAB, |
| ui_test_utils::BROWSER_TEST_WAIT_FOR_LOAD_STOP); |
| |
| { |
| SCOPED_TRACE("With page loaded"); |
| expect_has_listeners(true); |
| } |
| |
| TabStripModel* tab_strip = browser()->tab_strip_model(); |
| tab_strip->CloseWebContentsAt(tab_strip->active_index(), |
| TabStripModel::CLOSE_NONE); |
| base::RunLoop().RunUntilIdle(); |
| content::RunAllTasksUntilIdle(); |
| |
| { |
| SCOPED_TRACE("After page unload"); |
| expect_has_listeners(false); |
| } |
| } |
| |
| IN_PROC_BROWSER_TEST_F(ExtensionSettingsUIBrowserTest, |
| ActivityLogInactiveWithoutSwitch) { |
| // Navigate to chrome://extensions which is a allowlisted URL for the |
| // chrome.activityLogPrivate API. |
| GURL extensions_url("chrome://extensions"); |
| ui_test_utils::NavigateToURL(browser(), extensions_url); |
| content::WebContents* page_contents = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| ASSERT_TRUE(page_contents); |
| |
| // Attempt to add an event listener for the |
| // activityLogPrivate.onExtensionActivity event. |
| ASSERT_TRUE(content::ExecuteScript(page_contents, R"( |
| let activityLogListener = () => {}; |
| chrome.activityLogPrivate.onExtensionActivity.addListener( |
| activityLogListener); |
| )")); |
| |
| // Activity log will be inactive as the command line switch is not present and |
| // no allowlisted extensions for activityLogPrivate are enabled. |
| extensions::ActivityLog* activity_log = |
| extensions::ActivityLog::GetInstance(browser()->profile()); |
| ASSERT_FALSE(activity_log->is_active()); |
| } |
| |
| class ExtensionsActivityLogTest : public ExtensionSettingsUIBrowserTest { |
| protected: |
| // Enable command line flags for test. |
| void SetUpCommandLine(base::CommandLine* command_line) override { |
| command_line->AppendSwitch(switches::kEnableExtensionActivityLogging); |
| } |
| }; |
| |
| IN_PROC_BROWSER_TEST_F(ExtensionsActivityLogTest, TestActivityLogVisible) { |
| base::FilePath test_data_dir; |
| ASSERT_TRUE(base::PathService::Get(chrome::DIR_TEST_DATA, &test_data_dir)); |
| test_data_dir = test_data_dir.AppendASCII("extensions"); |
| extensions::ChromeTestExtensionLoader loader(browser()->profile()); |
| |
| ExtensionTestMessageListener listener("ready", false); |
| scoped_refptr<const extensions::Extension> extension = loader.LoadExtension( |
| test_data_dir.AppendASCII("activity_log/simple_call")); |
| ASSERT_TRUE(listener.WaitUntilSatisfied()); |
| |
| GURL activity_log_url("chrome://extensions/?activity=" + extension->id()); |
| ui_test_utils::NavigateToURL(browser(), activity_log_url); |
| content::WebContents* activity_log_contents = |
| browser()->tab_strip_model()->GetActiveWebContents(); |
| ASSERT_TRUE(activity_log_contents); |
| EXPECT_EQ(activity_log_url, activity_log_contents->GetLastCommittedURL()); |
| |
| // We are looking for the 'test.sendMessage' entry in the activity log as |
| // that is the only API call the simple_call.crx extension does. |
| // The querySelectors and shadowRoots are used here in order to penetrate |
| // multiple nested shadow DOMs created by Polymer components |
| // in the chrome://extensions page. |
| // See chrome/browser/resources/extensions for the Polymer code. |
| // This test only serves as an end to end test, and most of the functionality |
| // is covered in the JS unit tests. |
| bool has_api_call = false; |
| ASSERT_TRUE(content::ExecuteScriptAndExtractBool( |
| activity_log_contents, |
| R"(let manager = document.querySelector('extensions-manager'); |
| let activityLog = |
| manager.shadowRoot.querySelector('extensions-activity-log'); |
| let activityLogHistory = |
| activityLog.shadowRoot.querySelector('activity-log-history'); |
| const polymerPath = |
| 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; |
| Promise.all([ |
| activityLogHistory.whenDataFetched(), |
| import(polymerPath), |
| ]).then((results) => { |
| const polymerModule = results[1]; |
| polymerModule.flush(); |
| let item = activityLogHistory.shadowRoot.querySelector( |
| 'activity-log-history-item'); |
| let activityKey = item.shadowRoot.getElementById('activity-key'); |
| window.domAutomationController.send( |
| activityKey.innerText === 'test.sendMessage'); |
| }); |
| )", |
| &has_api_call)); |
| EXPECT_TRUE(has_api_call); |
| } |