// 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. #ifndef CHROME_BROWSER_SIGNIN_SIGNIN_BROWSERTEST_H_ #define CHROME_BROWSER_SIGNIN_SIGNIN_BROWSERTEST_H_ #include "base/command_line.h" #include "chrome/browser/signin/signin_manager.h" #include "chrome/browser/signin/signin_manager_factory.h" #include "chrome/browser/signin/signin_promo.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/singleton_tabs.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "chrome/browser/ui/webui/signin/login_ui_service.h" #include "chrome/browser/ui/webui/signin/login_ui_service_factory.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/render_process_host.h" #include "content/public/browser/render_view_host.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/web_contents_observer.h" #include "content/public/common/content_switches.h" #include "google_apis/gaia/gaia_urls.h" #include "net/http/http_status_code.h" #include "net/url_request/test_url_fetcher_factory.h" #include "net/url_request/url_request_status.h" namespace { const char kNonSigninURL[] = "www.google.com"; } class SigninBrowserTest : public InProcessBrowserTest { public: virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { https_server_.reset(new net::SpawnedTestServer( net::SpawnedTestServer::TYPE_HTTPS, net::SpawnedTestServer::kLocalhost, base::FilePath(FILE_PATH_LITERAL("chrome/test/data")))); ASSERT_TRUE(https_server_->Start()); // Add a host resolver rule to map all outgoing requests to the test server. // This allows us to use "real" hostnames in URLs, which we can use to // create arbitrary SiteInstances. command_line->AppendSwitchASCII( switches::kHostResolverRules, "MAP * " + https_server_->host_port_pair().ToString() + ",EXCLUDE localhost"); command_line->AppendSwitch(switches::kIgnoreCertificateErrors); } virtual void SetUp() OVERRIDE { factory_.reset(new net::URLFetcherImplFactory()); fake_factory_.reset(new net::FakeURLFetcherFactory(factory_.get())); fake_factory_->SetFakeResponse( GaiaUrls::GetInstance()->service_login_url(), std::string(), net::HTTP_OK, net::URLRequestStatus::SUCCESS); fake_factory_->SetFakeResponse( GURL(kNonSigninURL), std::string(), net::HTTP_OK, net::URLRequestStatus::SUCCESS); // Yield control back to the InProcessBrowserTest framework. InProcessBrowserTest::SetUp(); } virtual void TearDown() OVERRIDE { if (fake_factory_.get()) { fake_factory_->ClearFakeResponses(); fake_factory_.reset(); } // Cancel any outstanding URL fetches and destroy the URLFetcherImplFactory // we created. net::URLFetcher::CancelAll(); factory_.reset(); InProcessBrowserTest::TearDown(); } private: // Fake URLFetcher factory used to mock out GAIA signin. scoped_ptr<net::FakeURLFetcherFactory> fake_factory_; // The URLFetcherImplFactory instance used to instantiate |fake_factory_|. scoped_ptr<net::URLFetcherImplFactory> factory_; scoped_ptr<net::SpawnedTestServer> https_server_; }; // If the one-click-signin feature is not enabled (e.g Chrome OS), we // never grant signin privileges to any renderer processes. #if defined(ENABLE_ONE_CLICK_SIGNIN) const bool kOneClickSigninEnabled = true; #else const bool kOneClickSigninEnabled = false; #endif // Disabled on Windows due to flakiness. http://crbug.com/249055 #if defined(OS_WIN) #define MAYBE_ProcessIsolation DISABLED_ProcessIsolation #else #define MAYBE_ProcessIsolation ProcessIsolation #endif IN_PROC_BROWSER_TEST_F(SigninBrowserTest, MAYBE_ProcessIsolation) { SigninManager* signin = SigninManagerFactory::GetForProfile( browser()->profile()); EXPECT_FALSE(signin->HasSigninProcess()); ui_test_utils::NavigateToURL(browser(), signin::GetPromoURL( signin::SOURCE_NTP_LINK, true)); EXPECT_EQ(kOneClickSigninEnabled, signin->HasSigninProcess()); // Navigating away should change the process. ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIOmniboxURL)); EXPECT_FALSE(signin->HasSigninProcess()); ui_test_utils::NavigateToURL(browser(), signin::GetPromoURL( signin::SOURCE_NTP_LINK, true)); EXPECT_EQ(kOneClickSigninEnabled, signin->HasSigninProcess()); content::WebContents* active_tab = browser()->tab_strip_model()->GetActiveWebContents(); int active_tab_process_id = active_tab->GetRenderProcessHost()->GetID(); EXPECT_EQ(kOneClickSigninEnabled, signin->IsSigninProcess(active_tab_process_id)); EXPECT_EQ(0, active_tab->GetRenderViewHost()->GetEnabledBindings()); // Entry points to signin request "SINGLETON_TAB" mode, so a new request // shouldn't change anything. chrome::NavigateParams params(chrome::GetSingletonTabNavigateParams( browser(), GURL(signin::GetPromoURL(signin::SOURCE_NTP_LINK, false)))); params.path_behavior = chrome::NavigateParams::IGNORE_AND_NAVIGATE; ShowSingletonTabOverwritingNTP(browser(), params); EXPECT_EQ(active_tab, browser()->tab_strip_model()->GetActiveWebContents()); EXPECT_EQ(kOneClickSigninEnabled, signin->IsSigninProcess(active_tab_process_id)); // Navigating away should change the process. ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUINewTabURL)); EXPECT_FALSE(signin->IsSigninProcess( active_tab->GetRenderProcessHost()->GetID())); } IN_PROC_BROWSER_TEST_F(SigninBrowserTest, NotTrustedAfterRedirect) { SigninManager* signin = SigninManagerFactory::GetForProfile( browser()->profile()); EXPECT_FALSE(signin->HasSigninProcess()); GURL url = signin::GetPromoURL(signin::SOURCE_NTP_LINK, true); ui_test_utils::NavigateToURL(browser(), url); EXPECT_EQ(kOneClickSigninEnabled, signin->HasSigninProcess()); // Navigating in a different tab should not affect the sign-in process. ui_test_utils::NavigateToURLWithDisposition( browser(), GURL(kNonSigninURL), NEW_BACKGROUND_TAB, ui_test_utils::BROWSER_TEST_WAIT_FOR_NAVIGATION); EXPECT_EQ(kOneClickSigninEnabled, signin->HasSigninProcess()); // Navigating away should clear the sign-in process. GURL redirect_url("https://accounts.google.com/server-redirect?" "https://foo.com?service=chromiumsync"); ui_test_utils::NavigateToURL(browser(), redirect_url); EXPECT_FALSE(signin->HasSigninProcess()); } class BackOnNTPCommitObserver : public content::WebContentsObserver { public: explicit BackOnNTPCommitObserver(content::WebContents* web_contents) : content::WebContentsObserver(web_contents) { } virtual void DidCommitProvisionalLoadForFrame( int64 frame_id, const base::string16& frame_unique_name, bool is_main_frame, const GURL& url, content::PageTransition transition_type, content::RenderViewHost* render_view_host) OVERRIDE { if (url == GURL(chrome::kChromeUINewTabURL) || url == GURL(chrome::kChromeSearchLocalNtpUrl)) { content::WindowedNotificationObserver observer( content::NOTIFICATION_NAV_ENTRY_COMMITTED, content::NotificationService::AllSources()); web_contents()->GetController().GoBack(); observer.Wait(); } } private: DISALLOW_COPY_AND_ASSIGN(BackOnNTPCommitObserver); }; // This is a test for http://crbug.com/257277. It simulates the navigations // that occur if the user clicks on the "Skip for now" link at the signin page // and initiates a back navigation between the point of Commit and // DidStopLoading of the NTP. IN_PROC_BROWSER_TEST_F(SigninBrowserTest, SigninSkipForNowAndGoBack) { GURL ntp_url(chrome::kChromeUINewTabURL); GURL start_url = signin::GetPromoURL(signin::SOURCE_START_PAGE, true); GURL skip_url = signin::GetLandingURL("ntp", 1); SigninManager* signin = SigninManagerFactory::GetForProfile( browser()->profile()); EXPECT_FALSE(signin->HasSigninProcess()); ui_test_utils::NavigateToURL(browser(), start_url); EXPECT_EQ(kOneClickSigninEnabled, signin->HasSigninProcess()); content::WebContents* web_contents = browser()->tab_strip_model()->GetActiveWebContents(); // Simulate clicking on the Skip for now link. It's important to have a // link transition so that OneClickSigninHelper removes the blank page // from the history. chrome::NavigateParams navigate_params(browser(), skip_url, content::PAGE_TRANSITION_LINK); ui_test_utils::NavigateToURL(&navigate_params); // Register an observer that will navigate back immediately on the commit of // the NTP. This will allow us to hit the race condition of navigating back // before the DidStopLoading message of NTP gets delivered. This must be // created after the navigation to the skip_url has finished loading, // otherwise this observer will navigate back, before the history cleaner // has had a chance to remove the navigation entry. BackOnNTPCommitObserver commit_observer(web_contents); // Since the navigation to the blank URL is monitored for, the // OneClickSigninHelper initiates immediately a navigation to the NTP. // Thus, we expect the visible URL to be the NTP. EXPECT_EQ(skip_url, web_contents->GetLastCommittedURL()); EXPECT_EQ(ntp_url, web_contents->GetVisibleURL()); content::WindowedNotificationObserver observer( content::NOTIFICATION_LOAD_STOP, content::NotificationService::AllSources()); observer.Wait(); EXPECT_EQ(start_url, web_contents->GetLastCommittedURL()); } #endif // CHROME_BROWSER_SIGNIN_SIGNIN_BROWSERTEST_H_