// 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 "base/command_line.h" #include "base/memory/scoped_vector.h" #include "base/stringprintf.h" #include "chrome/browser/extensions/extension_apitest.h" #include "chrome/browser/extensions/extension_host.h" #include "chrome/browser/extensions/extension_process_manager.h" #include "chrome/browser/extensions/extension_system.h" #include "chrome/browser/extensions/extension_test_message_listener.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_finder.h" #include "chrome/browser/ui/browser_list.h" #include "chrome/browser/ui/browser_tabstrip.h" #include "chrome/browser/ui/panels/panel_manager.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/extensions/extension.h" #include "chrome/test/base/ui_test_utils.h" #include "content/public/browser/render_process_host.h" #include "content/public/browser/web_contents.h" #include "content/public/common/result_codes.h" #include "content/public/test/browser_test_utils.h" #include "testing/gtest/include/gtest/gtest.h" #include "net/base/mock_host_resolver.h" using content::OpenURLParams; using content::Referrer; using content::WebContents; // Disabled, http://crbug.com/64899. IN_PROC_BROWSER_TEST_F(ExtensionApiTest, DISABLED_WindowOpen) { CommandLine::ForCurrentProcess()->AppendSwitch( switches::kEnableExperimentalExtensionApis); ResultCatcher catcher; ASSERT_TRUE(LoadExtensionIncognito(test_data_dir_ .AppendASCII("window_open").AppendASCII("spanning"))); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } void WaitForTabsAndPopups(Browser* browser, int num_tabs, int num_popups, int num_panels) { SCOPED_TRACE( StringPrintf("WaitForTabsAndPopups tabs:%d, popups:%d, panels:%d", num_tabs, num_popups, num_panels)); // We start with one tab and one browser already open. ++num_tabs; size_t num_browsers = static_cast(num_popups) + 1; const base::TimeDelta kWaitTime = base::TimeDelta::FromSeconds(15); base::TimeTicks end_time = base::TimeTicks::Now() + kWaitTime; while (base::TimeTicks::Now() < end_time) { if (chrome::GetBrowserCount(browser->profile()) == num_browsers && browser->tab_count() == num_tabs && PanelManager::GetInstance()->num_panels() == num_panels) break; content::RunAllPendingInMessageLoop(); } EXPECT_EQ(num_browsers, chrome::GetBrowserCount(browser->profile())); EXPECT_EQ(num_tabs, browser->tab_count()); EXPECT_EQ(num_panels, PanelManager::GetInstance()->num_panels()); int num_popups_seen = 0; for (BrowserList::const_iterator iter = BrowserList::begin(); iter != BrowserList::end(); ++iter) { if (*iter == browser) continue; // Check for TYPE_POPUP. #if defined(USE_ASH) // On Ash, panel windows open as popup windows. EXPECT_TRUE((*iter)->is_type_popup() || (*iter)->is_type_panel()); #else EXPECT_TRUE((*iter)->is_type_popup()); #endif ++num_popups_seen; } EXPECT_EQ(num_popups, num_popups_seen); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, BrowserIsApp) { host_resolver()->AddRule("a.com", "127.0.0.1"); ASSERT_TRUE(StartTestServer()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII("browser_is_app"))); WaitForTabsAndPopups(browser(), 0, 2, 0); for (BrowserList::const_iterator iter = BrowserList::begin(); iter != BrowserList::end(); ++iter) { if (*iter == browser()) ASSERT_FALSE((*iter)->is_app()); else ASSERT_TRUE((*iter)->is_app()); } } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, WindowOpenPopupDefault) { ASSERT_TRUE(StartTestServer()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII("popup"))); const int num_tabs = 1; const int num_popups = 0; WaitForTabsAndPopups(browser(), num_tabs, num_popups, 0); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, WindowOpenPopupLarge) { ASSERT_TRUE(StartTestServer()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII("popup_large"))); // On other systems this should open a new popup window. const int num_tabs = 0; const int num_popups = 1; WaitForTabsAndPopups(browser(), num_tabs, num_popups, 0); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, WindowOpenPopupSmall) { ASSERT_TRUE(StartTestServer()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII("popup_small"))); // On ChromeOS this should open a new panel (acts like a new popup window). // On other systems this should open a new popup window. const int num_tabs = 0; const int num_popups = 1; WaitForTabsAndPopups(browser(), num_tabs, num_popups, 0); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, PopupBlockingExtension) { host_resolver()->AddRule("*", "127.0.0.1"); ASSERT_TRUE(StartTestServer()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII("popup_blocking") .AppendASCII("extension"))); WaitForTabsAndPopups(browser(), 5, 3, 0); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, PopupBlockingHostedApp) { host_resolver()->AddRule("*", "127.0.0.1"); ASSERT_TRUE(test_server()->Start()); ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII("popup_blocking") .AppendASCII("hosted_app"))); // The app being tested owns the domain a.com . The test URLs we navigate // to below must be within that domain, so that they fall within the app's // web extent. GURL::Replacements replace_host; std::string a_dot_com = "a.com"; replace_host.SetHostStr(a_dot_com); const std::string popup_app_contents_path( "files/extensions/api_test/window_open/popup_blocking/hosted_app/"); GURL open_tab = test_server()->GetURL(popup_app_contents_path + "open_tab.html") .ReplaceComponents(replace_host); GURL open_popup = test_server()->GetURL(popup_app_contents_path + "open_popup.html") .ReplaceComponents(replace_host); browser()->OpenURL(OpenURLParams( open_tab, Referrer(), NEW_FOREGROUND_TAB, content::PAGE_TRANSITION_TYPED, false)); browser()->OpenURL(OpenURLParams( open_popup, Referrer(), NEW_FOREGROUND_TAB, content::PAGE_TRANSITION_TYPED, false)); WaitForTabsAndPopups(browser(), 3, 1, 0); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, WindowArgumentsOverflow) { ASSERT_TRUE(RunExtensionTest("window_open/argument_overflow")) << message_; } class WindowOpenPanelDisabledTest : public ExtensionApiTest { virtual void SetUpCommandLine(CommandLine* command_line) { ExtensionApiTest::SetUpCommandLine(command_line); // TODO(jennb): Re-enable when panels are enabled by default. // command_line->AppendSwitch(switches::kDisablePanels); } }; IN_PROC_BROWSER_TEST_F(WindowOpenPanelDisabledTest, DISABLED_WindowOpenPanelNotEnabled) { ASSERT_TRUE(RunExtensionTest("window_open/panel_not_enabled")) << message_; } class WindowOpenPanelTest : public ExtensionApiTest { virtual void SetUpCommandLine(CommandLine* command_line) { ExtensionApiTest::SetUpCommandLine(command_line); command_line->AppendSwitch(switches::kEnablePanels); } }; #if defined(USE_ASH) // On Ash, this currently fails because we're currently opening new panel // windows as popup windows instead. #define MAYBE_WindowOpenPanel FAILS_WindowOpenPanel #else #define MAYBE_WindowOpenPanel WindowOpenPanel #endif IN_PROC_BROWSER_TEST_F(WindowOpenPanelTest, MAYBE_WindowOpenPanel) { ASSERT_TRUE(RunExtensionTest("window_open/panel")) << message_; } #if defined(USE_ASH) // On Ash, this currently fails because we're currently opening new panel // windows as popup windows instead. #define MAYBE_WindowOpenPanelDetached FAILS_WindowOpenPanelDetached #else #define MAYBE_WindowOpenPanelDetached WindowOpenPanelDetached #endif IN_PROC_BROWSER_TEST_F(WindowOpenPanelTest, MAYBE_WindowOpenPanelDetached) { ASSERT_TRUE(RunExtensionTest("window_open/panel_detached")) << message_; } #if defined(OS_MACOSX) || defined(OS_WIN) // Focus test fails if there is no window manager on Linux. IN_PROC_BROWSER_TEST_F(WindowOpenPanelTest, WindowOpenFocus) { ASSERT_TRUE(RunExtensionTest("window_open/focus")) << message_; } #endif IN_PROC_BROWSER_TEST_F(WindowOpenPanelTest, CloseNonExtensionPanelsOnUninstall) { #if defined(USE_ASH) // On Ash, new panel windows open as popup windows instead. int num_popups = 4; int num_panels = 0; #else int num_popups = 2; int num_panels = 2; #endif ASSERT_TRUE(StartTestServer()); // Setup listeners to wait on strings we expect the extension pages to send. std::vector test_strings; test_strings.push_back("content_tab"); if (num_panels) test_strings.push_back("content_panel"); test_strings.push_back("content_popup"); ScopedVector listeners; for (size_t i = 0; i < test_strings.size(); ++i) { listeners.push_back( new ExtensionTestMessageListener(test_strings[i], false)); } const extensions::Extension* extension = LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII( "close_panels_on_uninstall")); ASSERT_TRUE(extension); // Two tabs. One in extension domain and one in non-extension domain. // Two popups - one in extension domain and one in non-extension domain. // Two panels - one in extension domain and one in non-extension domain. WaitForTabsAndPopups(browser(), 2, num_popups, num_panels); // Wait on test messages to make sure the pages loaded. for (size_t i = 0; i < listeners.size(); ++i) ASSERT_TRUE(listeners[i]->WaitUntilSatisfied()); UninstallExtension(extension->id()); // Wait for the tabs and popups in non-extension domain to stay open. // Expect everything else, including panels, to close. #if defined(USE_ASH) // On Ash, new panel windows open as popup windows instead, so there are 2 // extension domain popups that will close (instead of 1 popup on non-Ash). num_popups -= 2; #else num_popups -= 1; #endif WaitForTabsAndPopups(browser(), 1, num_popups, 0); } IN_PROC_BROWSER_TEST_F(WindowOpenPanelTest, ClosePanelsOnExtensionCrash) { #if defined(USE_ASH) // On Ash, new panel windows open as popup windows instead. int num_popups = 4; int num_panels = 0; #else int num_popups = 2; int num_panels = 2; #endif ASSERT_TRUE(StartTestServer()); // Setup listeners to wait on strings we expect the extension pages to send. std::vector test_strings; test_strings.push_back("content_tab"); if (num_panels) test_strings.push_back("content_panel"); test_strings.push_back("content_popup"); ScopedVector listeners; for (size_t i = 0; i < test_strings.size(); ++i) { listeners.push_back( new ExtensionTestMessageListener(test_strings[i], false)); } const extensions::Extension* extension = LoadExtension( test_data_dir_.AppendASCII("window_open").AppendASCII( "close_panels_on_uninstall")); ASSERT_TRUE(extension); // Two tabs. One in extension domain and one in non-extension domain. // Two popups - one in extension domain and one in non-extension domain. // Two panels - one in extension domain and one in non-extension domain. WaitForTabsAndPopups(browser(), 2, num_popups, num_panels); // Wait on test messages to make sure the pages loaded. for (size_t i = 0; i < listeners.size(); ++i) ASSERT_TRUE(listeners[i]->WaitUntilSatisfied()); // Crash the extension. extensions::ExtensionHost* extension_host = extensions::ExtensionSystem::Get(browser()->profile())-> process_manager()->GetBackgroundHostForExtension(extension->id()); ASSERT_TRUE(extension_host); base::KillProcess(extension_host->render_process_host()->GetHandle(), content::RESULT_CODE_KILLED, false); WaitForExtensionCrash(extension->id()); // Only expect panels to close. The rest stay open to show a sad-tab. WaitForTabsAndPopups(browser(), 2, num_popups, 0); } #if defined(USE_ASH) // This test is not applicable on Ash. Ash opens panel windows as popup // windows. The modified window.open behavior only applies to panel windows. #define MAYBE_WindowOpenFromPanel DISABLED_WindowOpenFromPanel #else #define MAYBE_WindowOpenFromPanel WindowOpenFromPanel #endif IN_PROC_BROWSER_TEST_F(WindowOpenPanelTest, MAYBE_WindowOpenFromPanel) { ASSERT_TRUE(StartTestServer()); // Load the extension that will open a panel which then calls window.open. ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("window_open"). AppendASCII("panel_window_open"))); // Expect one panel (opened by extension) and one tab (from the panel calling // window.open). Panels modify the WindowOpenDisposition in window.open // to always open in a tab. WaitForTabsAndPopups(browser(), 1, 0, 1); } IN_PROC_BROWSER_TEST_F(ExtensionApiTest, DISABLED_WindowOpener) { ASSERT_TRUE(RunExtensionTest("window_open/opener")) << message_; } // Tests that an extension page can call window.open to an extension URL and // the new window has extension privileges. IN_PROC_BROWSER_TEST_F(ExtensionBrowserTest, WindowOpenExtension) { ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("uitest").AppendASCII("window_open"))); GURL start_url(std::string("chrome-extension://") + last_loaded_extension_id_ + "/test.html"); ui_test_utils::NavigateToURL(browser(), start_url); WebContents* newtab = NULL; ASSERT_NO_FATAL_FAILURE(OpenWindow(chrome::GetActiveWebContents(browser()), start_url.Resolve("newtab.html"), true, &newtab)); bool result = false; ASSERT_TRUE(content::ExecuteJavaScriptAndExtractBool( newtab->GetRenderViewHost(), L"", L"testExtensionApi()", &result)); EXPECT_TRUE(result); } // Tests that if an extension page calls window.open to an invalid extension // URL, the browser doesn't crash. IN_PROC_BROWSER_TEST_F(ExtensionBrowserTest, WindowOpenInvalidExtension) { ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("uitest").AppendASCII("window_open"))); GURL start_url(std::string("chrome-extension://") + last_loaded_extension_id_ + "/test.html"); ui_test_utils::NavigateToURL(browser(), start_url); ASSERT_NO_FATAL_FAILURE(OpenWindow(chrome::GetActiveWebContents(browser()), GURL("chrome-extension://thisissurelynotavalidextensionid/newtab.html"), false, NULL)); // If we got to this point, we didn't crash, so we're good. } // Tests that calling window.open from the newtab page to an extension URL // gives the new window extension privileges - even though the opening page // does not have extension privileges, we break the script connection, so // there is no privilege leak. IN_PROC_BROWSER_TEST_F(ExtensionBrowserTest, WindowOpenNoPrivileges) { ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("uitest").AppendASCII("window_open"))); ui_test_utils::NavigateToURL(browser(), GURL("about:blank")); WebContents* newtab = NULL; ASSERT_NO_FATAL_FAILURE(OpenWindow(chrome::GetActiveWebContents(browser()), GURL(std::string("chrome-extension://") + last_loaded_extension_id_ + "/newtab.html"), false, &newtab)); // Extension API should succeed. bool result = false; ASSERT_TRUE(content::ExecuteJavaScriptAndExtractBool( newtab->GetRenderViewHost(), L"", L"testExtensionApi()", &result)); EXPECT_TRUE(result); }