|  | // 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_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<size_t>(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 (browser::GetBrowserCount(browser->profile()) == num_browsers && | 
|  | browser->tab_count() == num_tabs && | 
|  | PanelManager::GetInstance()->num_panels() == num_panels) | 
|  | break; | 
|  |  | 
|  | content::RunAllPendingInMessageLoop(); | 
|  | } | 
|  |  | 
|  | EXPECT_EQ(num_browsers, browser::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<std::string> test_strings; | 
|  | test_strings.push_back("content_tab"); | 
|  | if (num_panels) | 
|  | test_strings.push_back("content_panel"); | 
|  | test_strings.push_back("content_popup"); | 
|  |  | 
|  | ScopedVector<ExtensionTestMessageListener> 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<std::string> test_strings; | 
|  | test_strings.push_back("content_tab"); | 
|  | if (num_panels) | 
|  | test_strings.push_back("content_panel"); | 
|  | test_strings.push_back("content_popup"); | 
|  |  | 
|  | ScopedVector<ExtensionTestMessageListener> 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 = | 
|  | browser()->profile()->GetExtensionProcessManager()-> | 
|  | 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; | 
|  | 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; | 
|  | 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); | 
|  | } |