// 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 #include "base/base_switches.h" #include "base/command_line.h" #include "base/files/file_path.h" #include "base/macros.h" #include "base/strings/stringprintf.h" #include "build/build_config.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_commands.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 "content/public/browser/notification_service.h" #include "content/public/browser/notification_types.h" #include "content/public/browser/web_contents.h" #include "content/public/test/browser_test_utils.h" #include "net/test/embedded_test_server/embedded_test_server.h" #include "net/test/embedded_test_server/http_request.h" #include "net/test/embedded_test_server/http_response.h" #include "testing/gtest/include/gtest/gtest.h" #include "ui/base/page_transition_types.h" using content::OpenURLParams; using content::Referrer; using content::WebContents; // TODO(jam): http://crbug.com/350550 #if !(defined(OS_CHROMEOS) && defined(ADDRESS_SANITIZER)) namespace { void SimulateRendererCrash(Browser* browser) { content::WindowedNotificationObserver observer( content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, content::NotificationService::AllSources()); browser->OpenURL(OpenURLParams( GURL(content::kChromeUICrashURL), Referrer(), CURRENT_TAB, ui::PAGE_TRANSITION_TYPED, false)); observer.Wait(); } // A request handler which returns a different result each time but stays fresh // into the far future. class CacheMaxAgeHandler { public: explicit CacheMaxAgeHandler(const std::string& path) : path_(path), request_count_(0) { } scoped_ptr HandleRequest( const net::test_server::HttpRequest& request) { if (request.relative_url != path_) return scoped_ptr(); request_count_++; scoped_ptr response( new net::test_server::BasicHttpResponse); response->set_content(base::StringPrintf("%d", request_count_)); response->set_content_type("text/html"); response->AddCustomHeader("Cache-Control", "max-age=99999"); return std::move(response); } private: std::string path_; int request_count_; DISALLOW_COPY_AND_ASSIGN(CacheMaxAgeHandler); }; class CrashRecoveryBrowserTest : public InProcessBrowserTest { protected: WebContents* GetActiveWebContents() { return browser()->tab_strip_model()->GetActiveWebContents(); } private: void SetUpCommandLine(base::CommandLine* command_line) override { command_line->AppendSwitch(switches::kDisableBreakpad); } }; // Test that reload works after a crash. // Disabled, http://crbug.com/29331 , http://crbug.com/69637 . IN_PROC_BROWSER_TEST_F(CrashRecoveryBrowserTest, Reload) { // The title of the active tab should change each time this URL is loaded. GURL url( "data:text/html,"); ui_test_utils::NavigateToURL(browser(), url); base::string16 title_before_crash; base::string16 title_after_crash; ASSERT_TRUE(ui_test_utils::GetCurrentTabTitle(browser(), &title_before_crash)); SimulateRendererCrash(browser()); chrome::Reload(browser(), CURRENT_TAB); content::WaitForLoadStop(GetActiveWebContents()); ASSERT_TRUE(ui_test_utils::GetCurrentTabTitle(browser(), &title_after_crash)); EXPECT_NE(title_before_crash, title_after_crash); } // Test that reload after a crash forces a cache revalidation. IN_PROC_BROWSER_TEST_F(CrashRecoveryBrowserTest, ReloadCacheRevalidate) { const char kTestPath[] = "/test"; // Use the test server so as not to bypass cache behavior. The title of the // active tab should change only when this URL is reloaded. ASSERT_TRUE(embedded_test_server()->Start()); embedded_test_server()->RegisterRequestHandler( base::Bind(&CacheMaxAgeHandler::HandleRequest, base::Owned(new CacheMaxAgeHandler(kTestPath)))); ui_test_utils::NavigateToURL(browser(), embedded_test_server()->GetURL(kTestPath)); base::string16 title_before_crash; base::string16 title_after_crash; ASSERT_TRUE(ui_test_utils::GetCurrentTabTitle(browser(), &title_before_crash)); SimulateRendererCrash(browser()); chrome::Reload(browser(), CURRENT_TAB); content::WaitForLoadStop(GetActiveWebContents()); ASSERT_TRUE(ui_test_utils::GetCurrentTabTitle(browser(), &title_after_crash)); EXPECT_NE(title_before_crash, title_after_crash); } // Tests that loading a crashed page in a new tab correctly updates the title. // There was an earlier bug (1270510) in process-per-site in which the max page // ID of the RenderProcessHost was stale, so the NavigationEntry in the new tab // was not committed. This prevents regression of that bug. IN_PROC_BROWSER_TEST_F(CrashRecoveryBrowserTest, LoadInNewTab) { const base::FilePath::CharType kTitle2File[] = FILE_PATH_LITERAL("title2.html"); ui_test_utils::NavigateToURL( browser(), ui_test_utils::GetTestUrl( base::FilePath(base::FilePath::kCurrentDirectory), base::FilePath(kTitle2File))); base::string16 title_before_crash; base::string16 title_after_crash; ASSERT_TRUE(ui_test_utils::GetCurrentTabTitle(browser(), &title_before_crash)); SimulateRendererCrash(browser()); chrome::Reload(browser(), CURRENT_TAB); content::WaitForLoadStop(GetActiveWebContents()); ASSERT_TRUE(ui_test_utils::GetCurrentTabTitle(browser(), &title_after_crash)); EXPECT_EQ(title_before_crash, title_after_crash); } // Tests that reloads of navigation errors behave correctly after a crash. // Regression test for http://crbug.com/348918 IN_PROC_BROWSER_TEST_F(CrashRecoveryBrowserTest, DoubleReloadWithError) { GURL url("chrome://bogus"); ui_test_utils::NavigateToURL(browser(), url); ASSERT_EQ(url, GetActiveWebContents()->GetVisibleURL()); SimulateRendererCrash(browser()); chrome::Reload(browser(), CURRENT_TAB); content::WaitForLoadStop(GetActiveWebContents()); ASSERT_EQ(url, GetActiveWebContents()->GetVisibleURL()); chrome::Reload(browser(), CURRENT_TAB); content::WaitForLoadStop(GetActiveWebContents()); ASSERT_EQ(url, GetActiveWebContents()->GetVisibleURL()); } } // namespace #endif