// 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. #ifndef CHROME_TEST_BASE_IN_PROCESS_BROWSER_TEST_H_ #define CHROME_TEST_BASE_IN_PROCESS_BROWSER_TEST_H_ #include "base/compiler_specific.h" #include "base/files/scoped_temp_dir.h" #include "base/memory/ref_counted.h" #include "base/memory/scoped_ptr.h" #include "content/public/common/page_transition_types.h" #include "content/public/test/browser_test.h" #include "content/public/test/browser_test_base.h" #include "testing/gtest/include/gtest/gtest.h" namespace base { #if defined(OS_MACOSX) namespace mac { class ScopedNSAutoreleasePool; } // namespace mac #endif // defined(OS_MACOSX) #if defined(OS_WIN) && defined(USE_AURA) namespace win { class ScopedCOMInitializer; } #endif // defined(OS_WIN) && defined(USE_AURA) } // namespace base class Browser; class CommandLine; class Profile; namespace content { class ContentRendererClient; } // Base class for tests wanting to bring up a browser in the unit test process. // Writing tests with InProcessBrowserTest is slightly different than that of // other tests. This is necessitated by InProcessBrowserTest running a message // loop. To use InProcessBrowserTest do the following: // . Use the macro IN_PROC_BROWSER_TEST_F to define your test. // . Your test method is invoked on the ui thread. If you need to block until // state changes you'll need to run the message loop from your test method. // For example, if you need to wait till a find bar has completely been shown // you'll need to invoke content::RunMessageLoop. When the message bar is // shown, invoke MessageLoop::current()->Quit() to return control back to your // test method. // . If you subclass and override SetUp, be sure and invoke // InProcessBrowserTest::SetUp. (But see also SetUpOnMainThread, // SetUpInProcessBrowserTestFixture and other related hook methods for a // cleaner alternative). // // Following three hook methods are called in sequence before calling // BrowserMain(), thus no browser has been created yet. They are mainly for // setting up the environment for running the browser. // . SetUpUserDataDirectory() // . SetUpCommandLine() // . SetUpInProcessBrowserTestFixture() // // SetUpOnMainThread() is called just after creating the default browser object // and before executing the real test code. It's mainly for setting up things // related to the browser object and associated window, like opening a new Tab // with a testing page loaded. // // CleanUpOnMainThread() is called just after executing the real test code to // do necessary cleanup before the browser is torn down. // // TearDownInProcessBrowserTestFixture() is called after BrowserMain() exits to // cleanup things setup for running the browser. // // By default InProcessBrowserTest creates a single Browser (as returned from // the CreateBrowser method). You can obviously create more as needed. // InProcessBrowserTest disables the sandbox when running. // // See ui_test_utils for a handful of methods designed for use with this class. // // It's possible to write browser tests that span a restart by splitting each // run of the browser process into a separate test. Example: // // IN_PROC_BROWSER_TEST_F(Foo, PRE_Bar) { // do something // } // // IN_PROC_BROWSER_TEST_F(Foo, Bar) { // verify something persisted from before // } // // This is recursive, so PRE_PRE_Bar would run before PRE_BAR. class InProcessBrowserTest : public content::BrowserTestBase { public: InProcessBrowserTest(); virtual ~InProcessBrowserTest(); // Configures everything for an in process browser test, then invokes // BrowserMain. BrowserMain ends up invoking RunTestOnMainThreadLoop. virtual void SetUp() OVERRIDE; // Restores state configured in SetUp. virtual void TearDown() OVERRIDE; protected: // Returns the browser created by CreateBrowser. Browser* browser() const { return browser_; } // Convenience methods for adding tabs to a Browser. void AddTabAtIndexToBrowser(Browser* browser, int index, const GURL& url, content::PageTransition transition); void AddTabAtIndex(int index, const GURL& url, content::PageTransition transition); // Initializes the contents of the user data directory. Called by SetUp() // after creating the user data directory, but before any browser is launched. // If a test wishes to set up some initial non-empty state in the user data // directory before the browser starts up, it can do so here. Returns true if // successful. virtual bool SetUpUserDataDirectory() WARN_UNUSED_RESULT; // Override this to add any custom cleanup code that needs to be done on the // main thread before the browser is torn down. virtual void CleanUpOnMainThread() {} // BrowserTestBase: virtual void RunTestOnMainThreadLoop() OVERRIDE; // Creates a browser with a single tab (about:blank), waits for the tab to // finish loading and shows the browser. // // This is invoked from Setup. Browser* CreateBrowser(Profile* profile); // Similar to |CreateBrowser|, but creates an incognito browser. Browser* CreateIncognitoBrowser(); // Creates a browser for a popup window with a single tab (about:blank), waits // for the tab to finish loading, and shows the browser. Browser* CreateBrowserForPopup(Profile* profile); // Creates a browser for an application and waits for it to load and shows // the browser. Browser* CreateBrowserForApp(const std::string& app_name, Profile* profile); // Called from the various CreateBrowser methods to add a blank tab, wait for // the navigation to complete, and show the browser's window. void AddBlankTabAndShow(Browser* browser); #if !defined OS_MACOSX // Return a CommandLine object that is used to relaunch the browser_test // binary as a browser process. This function is deliberately not defined on // the Mac because re-using an existing browser process when launching from // the command line isn't a concept that we support on the Mac; AppleEvents // are the Mac solution for the same need. Any test based on these functions // doesn't apply to the Mac. CommandLine GetCommandLineForRelaunch(); #endif #if defined(OS_MACOSX) // Returns the autorelease pool in use inside RunTestOnMainThreadLoop(). base::mac::ScopedNSAutoreleasePool* AutoreleasePool() const { return autorelease_pool_; } #endif // OS_MACOSX void set_exit_when_last_browser_closes(bool value) { exit_when_last_browser_closes_ = value; } // This must be called before RunTestOnMainThreadLoop() to have any effect. void set_multi_desktop_test(bool multi_desktop_test) { multi_desktop_test_ = multi_desktop_test; } private: // Creates a user data directory for the test if one is needed. Returns true // if successful. virtual bool CreateUserDataDirectory() WARN_UNUSED_RESULT; // Quits all open browsers and waits until there are no more browsers. void QuitBrowsers(); // Prepare command line that will be used to launch the child browser process // with an in-process test. void PrepareTestCommandLine(CommandLine* command_line); // Browser created from CreateBrowser. Browser* browser_; // Temporary user data directory. Used only when a user data directory is not // specified in the command line. base::ScopedTempDir temp_user_data_dir_; // True if we should exit the tests after the last browser instance closes. bool exit_when_last_browser_closes_; // True if this is a multi-desktop test (in which case this browser test will // not ensure that Browsers are only created on the tested desktop). bool multi_desktop_test_; #if defined(OS_MACOSX) base::mac::ScopedNSAutoreleasePool* autorelease_pool_; #endif // OS_MACOSX #if defined(OS_WIN) && defined(USE_AURA) scoped_ptr com_initializer_; #endif }; #endif // CHROME_TEST_BASE_IN_PROCESS_BROWSER_TEST_H_