| Index: content/browser/web_contents/opened_by_dom_browsertest.cc | 
| diff --git a/content/browser/web_contents/opened_by_dom_browsertest.cc b/content/browser/web_contents/opened_by_dom_browsertest.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..1c2d94099ce1d94e0d8c3fa126a974360f523b6a | 
| --- /dev/null | 
| +++ b/content/browser/web_contents/opened_by_dom_browsertest.cc | 
| @@ -0,0 +1,139 @@ | 
| +// Copyright 2014 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/strings/stringprintf.h" | 
| +#include "content/public/browser/web_contents.h" | 
| +#include "content/public/browser/web_contents_delegate.h" | 
| +#include "content/public/common/content_switches.h" | 
| +#include "content/public/test/browser_test_utils.h" | 
| +#include "content/public/test/content_browser_test.h" | 
| +#include "content/public/test/content_browser_test_utils.h" | 
| +#include "content/public/test/test_navigation_observer.h" | 
| +#include "content/shell/browser/shell.h" | 
| +#include "net/dns/mock_host_resolver.h" | 
| +#include "url/gurl.h" | 
| + | 
| +namespace content { | 
| + | 
| +namespace { | 
| + | 
| +// A dummy WebContentsDelegate which tracks whether CloseContents() has been | 
| +// called. It refuses the actual close but keeps track of whether the renderer | 
| +// requested it. | 
| +class CloseTrackingDelegate : public WebContentsDelegate { | 
| + public: | 
| +  CloseTrackingDelegate() : close_contents_called_(false) {} | 
| + | 
| +  bool close_contents_called() const { return close_contents_called_; } | 
| + | 
| +  virtual void CloseContents(WebContents* source) OVERRIDE { | 
| +    close_contents_called_ = true; | 
| +  } | 
| + | 
| + private: | 
| +  bool close_contents_called_; | 
| + | 
| +  DISALLOW_COPY_AND_ASSIGN(CloseTrackingDelegate); | 
| +}; | 
| + | 
| +}  // namespace | 
| + | 
| +class OpenedByDOMTest : public ContentBrowserTest { | 
| + protected: | 
| +  virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { | 
| +    // Use --site-per-process to force process swaps on cross-site navigations. | 
| +    command_line->AppendSwitch(switches::kSitePerProcess); | 
| +  } | 
| + | 
| +  bool AttemptCloseFromJavaScript(WebContents* web_contents) { | 
| +    CloseTrackingDelegate close_tracking_delegate; | 
| +    WebContentsDelegate* old_delegate = web_contents->GetDelegate(); | 
| +    web_contents->SetDelegate(&close_tracking_delegate); | 
| + | 
| +    const char kCloseWindowScript[] = | 
| +        // Close the window. | 
| +        "window.close();" | 
| +        // Report back after an event loop iteration; the close IPC isn't sent | 
| +        // immediately. | 
| +        "setTimeout(function() {" | 
| +        "window.domAutomationController.send(0);" | 
| +        "});"; | 
| +    int dummy; | 
| +    CHECK(ExecuteScriptAndExtractInt(web_contents, kCloseWindowScript, &dummy)); | 
| + | 
| +    web_contents->SetDelegate(old_delegate); | 
| +    return close_tracking_delegate.close_contents_called(); | 
| +  } | 
| + | 
| +  Shell* OpenWindowFromJavaScript(Shell* shell, const GURL& url) { | 
| +    // Wait for the popup to be created and for it to have navigated. | 
| +    ShellAddedObserver new_shell_observer; | 
| +    TestNavigationObserver nav_observer(NULL); | 
| +    nav_observer.StartWatchingNewWebContents(); | 
| +    CHECK(ExecuteScript( | 
| +        shell->web_contents(), | 
| +        base::StringPrintf("window.open('%s')", url.spec().c_str()))); | 
| +    nav_observer.Wait(); | 
| +    return new_shell_observer.GetShell(); | 
| +  } | 
| +}; | 
| + | 
| +// Tests that window.close() does not work on a normal window that has navigated | 
| +// a few times. | 
| +IN_PROC_BROWSER_TEST_F(OpenedByDOMTest, NormalWindow) { | 
| +  ASSERT_TRUE(test_server()->Start()); | 
| + | 
| +  // window.close is allowed if the window was opened by DOM OR the back/forward | 
| +  // list has only one element. Navigate a bit so the second condition is false. | 
| +  GURL url1 = test_server()->GetURL("files/site_isolation/blank.html?1"); | 
| +  GURL url2 = test_server()->GetURL("files/site_isolation/blank.html?2"); | 
| +  NavigateToURL(shell(), url1); | 
| +  NavigateToURL(shell(), url2); | 
| + | 
| +  // This window was not opened by DOM, so close does not reach the browser | 
| +  // process. | 
| +  EXPECT_FALSE(AttemptCloseFromJavaScript(shell()->web_contents())); | 
| +} | 
| + | 
| +// Tests that window.close() works in a popup window that has navigated a few | 
| +// times. | 
| +IN_PROC_BROWSER_TEST_F(OpenedByDOMTest, Popup) { | 
| +  ASSERT_TRUE(test_server()->Start()); | 
| + | 
| +  GURL url1 = test_server()->GetURL("files/site_isolation/blank.html?1"); | 
| +  GURL url2 = test_server()->GetURL("files/site_isolation/blank.html?2"); | 
| +  GURL url3 = test_server()->GetURL("files/site_isolation/blank.html?3"); | 
| +  NavigateToURL(shell(), url1); | 
| + | 
| +  Shell* popup = OpenWindowFromJavaScript(shell(), url2); | 
| +  NavigateToURL(popup, url3); | 
| +  EXPECT_TRUE(AttemptCloseFromJavaScript(popup->web_contents())); | 
| +} | 
| + | 
| +// Tests that window.close() works in a popup window that has navigated a few | 
| +// times and swapped processes. | 
| +IN_PROC_BROWSER_TEST_F(OpenedByDOMTest, CrossProcessPopup) { | 
| +  host_resolver()->AddRule("*", "127.0.0.1"); | 
| +  ASSERT_TRUE(test_server()->Start()); | 
| + | 
| +  GURL url1 = test_server()->GetURL("files/site_isolation/blank.html?1"); | 
| + | 
| +  GURL url2 = test_server()->GetURL("files/site_isolation/blank.html?2"); | 
| +  GURL::Replacements replace_host; | 
| +  std::string foo_com("foo.com"); | 
| +  replace_host.SetHostStr(foo_com); | 
| +  url2 = url2.ReplaceComponents(replace_host); | 
| + | 
| +  GURL url3 = test_server()->GetURL("files/site_isolation/blank.html?3"); | 
| +  url3 = url3.ReplaceComponents(replace_host); | 
| + | 
| +  NavigateToURL(shell(), url1); | 
| + | 
| +  Shell* popup = OpenWindowFromJavaScript(shell(), url2); | 
| +  NavigateToURL(popup, url3); | 
| +  EXPECT_TRUE(AttemptCloseFromJavaScript(popup->web_contents())); | 
| +} | 
| + | 
| +}  // namespace content | 
|  |