skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 1 | // Copyright 2016 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #ifndef HEADLESS_TEST_HEADLESS_BROWSER_TEST_H_ |
| 6 | #define HEADLESS_TEST_HEADLESS_BROWSER_TEST_H_ |
| 7 | |
skyostil | ab258d6d | 2016-04-06 16:32:57 | [diff] [blame] | 8 | #include <memory> |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 9 | #include <string> |
| 10 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 11 | #include "content/public/test/browser_test_base.h" |
altimin | eaffa8e0 | 2016-11-08 23:56:31 | [diff] [blame] | 12 | #include "headless/public/devtools/domains/network.h" |
| 13 | #include "headless/public/devtools/domains/page.h" |
skyostil | f52fa5b | 2016-03-15 11:38:45 | [diff] [blame] | 14 | #include "headless/public/headless_browser.h" |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 15 | #include "headless/public/headless_web_contents.h" |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 16 | |
skyostil | ab258d6d | 2016-04-06 16:32:57 | [diff] [blame] | 17 | namespace base { |
| 18 | class RunLoop; |
| 19 | } |
| 20 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 21 | namespace headless { |
skyostil | a65d5e61 | 2016-06-01 18:04:21 | [diff] [blame] | 22 | namespace runtime { |
| 23 | class EvaluateResult; |
| 24 | } |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 25 | class HeadlessDevToolsClient; |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 26 | |
skyostil | 25c2c012 | 2016-06-09 12:54:12 | [diff] [blame] | 27 | // A utility class for asynchronously observing load events. |
| 28 | class LoadObserver : public page::Observer, public network::Observer { |
| 29 | public: |
| 30 | LoadObserver(HeadlessDevToolsClient* devtools_client, base::Closure callback); |
| 31 | ~LoadObserver() override; |
| 32 | |
| 33 | // page::Observer implementation: |
| 34 | void OnLoadEventFired(const page::LoadEventFiredParams& params) override; |
| 35 | |
| 36 | // network::Observer implementation: |
| 37 | void OnResponseReceived( |
| 38 | const network::ResponseReceivedParams& params) override; |
| 39 | |
| 40 | bool navigation_succeeded() const { return navigation_succeeded_; } |
| 41 | |
| 42 | private: |
| 43 | base::Closure callback_; |
| 44 | HeadlessDevToolsClient* devtools_client_; // Not owned. |
| 45 | |
| 46 | bool navigation_succeeded_; |
| 47 | |
| 48 | DISALLOW_COPY_AND_ASSIGN(LoadObserver); |
| 49 | }; |
| 50 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 51 | // Base class for tests which require a full instance of the headless browser. |
| 52 | class HeadlessBrowserTest : public content::BrowserTestBase { |
skyostil | ab258d6d | 2016-04-06 16:32:57 | [diff] [blame] | 53 | public: |
| 54 | // Notify that an asynchronous test is now complete and the test runner should |
| 55 | // exit. |
| 56 | void FinishAsynchronousTest(); |
| 57 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 58 | protected: |
| 59 | HeadlessBrowserTest(); |
| 60 | ~HeadlessBrowserTest() override; |
| 61 | |
| 62 | // BrowserTestBase: |
| 63 | void RunTestOnMainThreadLoop() override; |
| 64 | void SetUpOnMainThread() override; |
| 65 | void TearDownOnMainThread() override; |
| 66 | |
skyostil | ab258d6d | 2016-04-06 16:32:57 | [diff] [blame] | 67 | // Run an asynchronous test in a nested run loop. The caller should call |
| 68 | // FinishAsynchronousTest() to notify that the test should finish. |
| 69 | void RunAsynchronousTest(); |
| 70 | |
| 71 | // Synchronously waits for a tab to finish loading. |
| 72 | bool WaitForLoad(HeadlessWebContents* web_contents); |
skyostil | f52fa5b | 2016-03-15 11:38:45 | [diff] [blame] | 73 | |
skyostil | a65d5e61 | 2016-06-01 18:04:21 | [diff] [blame] | 74 | // Synchronously evaluates a script and returns the result. |
| 75 | std::unique_ptr<runtime::EvaluateResult> EvaluateScript( |
| 76 | HeadlessWebContents* web_contents, |
| 77 | const std::string& script); |
| 78 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 79 | protected: |
| 80 | // Returns the browser for the test. |
| 81 | HeadlessBrowser* browser() const; |
| 82 | |
eseckler | 9ff0998 | 2016-11-15 09:18:25 | [diff] [blame] | 83 | // Returns the options used by the browser. Modify with caution, since some |
| 84 | // options only take effect if they were set before browser creation. |
| 85 | HeadlessBrowser::Options* options() const; |
| 86 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 87 | private: |
skyostil | ab258d6d | 2016-04-06 16:32:57 | [diff] [blame] | 88 | std::unique_ptr<base::RunLoop> run_loop_; |
| 89 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 90 | DISALLOW_COPY_AND_ASSIGN(HeadlessBrowserTest); |
| 91 | }; |
| 92 | |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 93 | #define HEADLESS_ASYNC_DEVTOOLED_TEST_F(TEST_FIXTURE_NAME) \ |
| 94 | IN_PROC_BROWSER_TEST_F(TEST_FIXTURE_NAME, RunAsyncTest) { RunTest(); } \ |
| 95 | class AsyncHeadlessBrowserTestNeedsSemicolon##TEST_FIXTURE_NAME {} |
| 96 | |
| 97 | // Base class for tests that require access to a DevToolsClient. Subclasses |
| 98 | // should override the RunDevTooledTest() method, which is called asynchronously |
| 99 | // when the DevToolsClient is ready. |
| 100 | class HeadlessAsyncDevTooledBrowserTest : public HeadlessBrowserTest, |
| 101 | public HeadlessWebContents::Observer { |
| 102 | public: |
| 103 | HeadlessAsyncDevTooledBrowserTest(); |
| 104 | ~HeadlessAsyncDevTooledBrowserTest() override; |
| 105 | |
| 106 | // HeadlessWebContentsObserver implementation: |
| 107 | void DevToolsTargetReady() override; |
alexclarke | 5777c83e | 2016-11-18 17:42:16 | [diff] [blame] | 108 | void RenderProcessExited(base::TerminationStatus status, |
| 109 | int exit_code) override; |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 110 | |
| 111 | // Implemented by tests and used to send request(s) to DevTools. Subclasses |
| 112 | // need to ensure that FinishAsynchronousTest() is called after response(s) |
| 113 | // are processed (e.g. in a callback). |
| 114 | virtual void RunDevTooledTest() = 0; |
| 115 | |
| 116 | protected: |
| 117 | void RunTest(); |
| 118 | |
altimin | bf875c9 | 2016-08-04 17:09:07 | [diff] [blame] | 119 | HeadlessBrowserContext* browser_context_; // Not owned. |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 120 | HeadlessWebContents* web_contents_; |
| 121 | std::unique_ptr<HeadlessDevToolsClient> devtools_client_; |
alexclarke | 5777c83e | 2016-11-18 17:42:16 | [diff] [blame] | 122 | bool render_process_exited_; |
eseckler | 9ef5476 | 2016-06-03 17:28:38 | [diff] [blame] | 123 | }; |
| 124 | |
skyostil | fe11616 | 2016-02-26 20:53:33 | [diff] [blame] | 125 | } // namespace headless |
| 126 | |
| 127 | #endif // HEADLESS_TEST_HEADLESS_BROWSER_TEST_H_ |