1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
|
// Copyright (c) 2006-2008 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_UI_UI_TEST_H_
#define CHROME_TEST_UI_UI_TEST_H_
// This file provides a common base for running UI unit tests, which operate
// the entire browser application in a separate process for holistic
// functional testing.
//
// Tests should #include this file, subclass UITest, and use the TEST_F macro
// to declare individual test cases. This provides a running browser window
// during the test, accessible through the window_ member variable. The window
// will close when the test ends, regardless of whether the test passed.
//
// Tests which need to launch the browser with a particular set of command-line
// arguments should set the value of launch_arguments_ in their constructors.
#include <windows.h>
#include <string>
#include "base/path_service.h"
#include "base/scoped_ptr.h"
#include "base/time.h"
#include "chrome/test/automation/automation_proxy.h"
#include "testing/gtest/include/gtest/gtest.h"
class DictionaryValue;
class GURL;
class TabProxy;
class UITest : public testing::Test {
protected:
// Delay to let browser complete a requested action.
static const int kWaitForActionMsec = 2000;
static const int kWaitForActionMaxMsec = 10000;
// Delay to let the browser complete the test.
static const int kMaxTestExecutionTime = 30000;
// Tries to delete the specified file/directory returning true on success.
// This differs from file_util::Delete in that it repeatedly invokes Delete
// until successful, or a timeout is reached. Returns true on success.
static bool DieFileDie(const std::wstring& file, bool recurse);
// Constructor
UITest();
// Starts the browser using the arguments in launch_arguments_, and
// sets up member variables.
virtual void SetUp();
// Closes the browser window.
virtual void TearDown();
// ********* Utility functions *********
// Launches the browser and IPC testing server.
void LaunchBrowserAndServer();
// Closes the browser and IPC testing server.
void CloseBrowserAndServer();
// Launches the browser with the given arguments.
void LaunchBrowser(const std::wstring& arguments, bool clear_profile);
// Exits out browser instance.
void QuitBrowser();
// Tells the browser to navigate to the given URL in the active tab
// of the first app window.
// This method doesn't return until the navigation is complete.
void NavigateToURL(const GURL& url);
// Returns the URL of the currently active tab. If there is no active tab,
// or some other error, the returned URL will be empty.
GURL GetActiveTabURL();
// Returns the title of the currently active tab.
std::wstring GetActiveTabTitle();
// Returns true when the browser process is running, independent if any
// renderer process exists or not. It will returns false if an user closed the
// window or if the browser process died by itself.
bool IsBrowserRunning();
// Returns true when time_out_ms milliseconds have elapsed.
// Returns false if the browser process died while waiting.
bool CrashAwareSleep(int time_out_ms);
// Returns the number of tabs in the first window. If no windows exist,
// causes a test failure and returns 0.
int GetTabCount();
// Polls the tab for the cookie_name cookie and returns once one of the
// following conditions hold true:
// - The cookie is of expected_value.
// - The browser process died.
// - The time_out value has been exceeded.
bool WaitUntilCookieValue(TabProxy* tab, const GURL& url,
const char* cookie_name,
int interval_ms,
int time_out_ms,
const char* expected_value);
// Polls the tab for the cookie_name cookie and returns once one of the
// following conditions hold true:
// - The cookie is set to any value.
// - The browser process died.
// - The time_out value has been exceeded.
std::string WaitUntilCookieNonEmpty(TabProxy* tab,
const GURL& url,
const char* cookie_name,
int interval_ms,
int time_out_ms);
// Polls up to kWaitForActionMaxMsec ms to attain a specific tab count. Will
// assert that the tab count is valid at the end of the wait.
void WaitUntilTabCount(int tab_count);
// Checks whether the download shelf is visible in the current tab, giving it
// a chance to appear (we don't know the exact timing) while finishing as soon
// as possible.
bool WaitForDownloadShelfVisible(TabProxy* tab);
// Waits until the Find window has become fully visible (and stopped
// animating) in the specified tab. This function can time out (return false)
// if the window doesn't appear within a specific time.
bool WaitForFindWindowFullyVisible(TabProxy* tab);
// Waits until the Bookmark bar has stopped animating and become fully visible
// (if |wait_for_open| is true) or fully hidden (if |wait_for_open| is false).
// This function can time out (in which case it returns false).
bool WaitForBookmarkBarVisibilityChange(BrowserProxy* browser,
bool wait_for_open);
// Closes the specified browser. Returns true if the browser was closed.
// This call is blocking. |application_closed| is set to true if this was
// the last browser window (and therefore as a result of it closing the
// browser process terminated). Note that in that case this method returns
// after the browser process has terminated.
bool CloseBrowser(BrowserProxy* browser, bool* application_closed) const;
// Prints numerical information to stdout in a controlled format, for
// post-processing. |measurement| is a description of the quantity being
// measured, e.g. "vm_peak"; |modifier| is provided as a convenience and
// will be appended directly to the name of the |measurement|, e.g.
// "_browser"; |trace| is a description of the particular data point, e.g.
// "reference"; |value| is the measured value; and |units| is a description
// of the units of measure, e.g. "bytes". If |important| is true, the output
// line will be specially marked, to notify the post-processor. The strings
// may be empty. They should not contain any colons (:) or equals signs (=).
// A typical post-processing step would be to produce graphs of the data
// produced for various builds, using the combined |measurement| + |modifier|
// string to specify a particular graph and the |trace| to identify a trace
// (i.e., data series) on that graph.
void PrintResult(const std::wstring& measurement,
const std::wstring& modifier,
const std::wstring& trace,
size_t value,
const std::wstring& units,
bool important);
// Gets the directory for the currently active profile in the browser.
std::wstring GetDownloadDirectory();
// Get the handle of browser process connected to the automation. This
// function only retruns a reference to the handle so the caller does not
// own the handle returned.
HANDLE process() { return process_; }
public:
// Get/Set a flag to run the renderer in process when running the
// tests.
static bool in_process_renderer() { return in_process_renderer_; }
static void set_in_process_renderer(bool value) {
in_process_renderer_ = value;
}
// Get/Set a flag to run the plugins in the renderer process when running the
// tests.
static bool in_process_plugins() { return in_process_plugins_; }
static void set_in_process_plugins(bool value) {
in_process_plugins_ = value;
}
// Get/Set a flag to run the renderer outside the sandbox when running the
// tests
static bool no_sandbox() { return no_sandbox_; }
static void set_no_sandbox(bool value) {
no_sandbox_ = value;
}
// Get/Set a flag to run with DCHECKs enabled in release.
static bool enable_dcheck() { return enable_dcheck_; }
static void set_enable_dcheck(bool value) {
enable_dcheck_ = value;
}
// Get/Set a flag to dump the process memory without crashing on DCHECKs.
static bool silent_dump_on_dcheck() { return silent_dump_on_dcheck_; }
static void set_silent_dump_on_dcheck(bool value) {
silent_dump_on_dcheck_ = value;
}
// Get/Set a flag to disable breakpad handling.
static bool disable_breakpad() { return disable_breakpad_; }
static void set_disable_breakpad(bool value) {
disable_breakpad_ = value;
}
// Get/Set a flag to run the plugin processes inside the sandbox when running
// the tests
static bool safe_plugins() { return safe_plugins_; }
static void set_safe_plugins(bool value) {
safe_plugins_ = value;
}
static bool show_error_dialogs() { return show_error_dialogs_; }
static void set_show_error_dialogs(bool value) {
show_error_dialogs_ = value;
}
static bool full_memory_dump() { return full_memory_dump_; }
static void set_full_memory_dump(bool value) {
full_memory_dump_ = value;
}
static bool use_existing_browser() { return default_use_existing_browser_; }
static void set_use_existing_browser(bool value) {
default_use_existing_browser_ = value;
}
static bool dump_histograms_on_exit() { return dump_histograms_on_exit_; }
static void set_dump_histograms_on_exit(bool value) {
dump_histograms_on_exit_ = value;
}
static int test_timeout_ms() { return timeout_ms_; }
static void set_test_timeout_ms(int value) {
timeout_ms_ = value;
}
static std::wstring js_flags() { return js_flags_; }
static void set_js_flags(const std::wstring& value) {
js_flags_ = value;
}
// Called by some tests that wish to have a base profile to start from. This
// "user data directory" (containing one or more profiles) will be recursively
// copied into the user data directory for the test and the files will be
// evicted from the OS cache. To start with a blank profile, supply an empty
// string (the default).
std::wstring template_user_data() const { return template_user_data_; }
void set_template_user_data(const std::wstring& template_user_data) {
template_user_data_ = template_user_data;
}
// Return the user data directory being used by the browser instance in
// UITest::SetUp().
std::wstring user_data_dir() const { return user_data_dir_; }
// Count the number of active browser processes. This function only counts
// browser processes that share the same profile directory as the current
// process. The count includes browser sub-processes.
static int GetBrowserProcessCount();
// Returns a copy of local state preferences. The caller is responsible for
// deleting the returned object. Returns NULL if there is an error.
DictionaryValue* GetLocalState();
// Returns a copy of the default profile preferences. The caller is
// responsible for deleting the returned object. Returns NULL if there is an
// error.
DictionaryValue* GetDefaultProfilePreferences();
// Generate the URL for testing a particular test.
// HTML for the tests is all located in
// test_root_directory\test_directory\<testcase>
static GURL GetTestUrl(const std::wstring& test_directory,
const std::wstring &test_case);
// Waits for the test case to finish.
// ASSERTS if there are test failures.
void WaitForFinish(const std::string &name,
const std::string &id, const GURL &url,
const std::string& test_complete_cookie,
const std::string& expected_cookie_value,
const int wait_time);
private:
// Check that no processes related to Chrome exist, displaying
// the given message if any do.
void AssertAppNotRunning(const std::wstring& error_message);
protected:
AutomationProxy* automation() {
EXPECT_TRUE(server_.get());
return server_.get();
}
// Wait a certain amount of time for all the app processes to exit,
// forcibly killing them if they haven't exited by then.
// It has the side-effect of killing every browser window opened in your
// session, even those unrelated in the test.
void CleanupAppProcesses();
// Returns the proxy for the currently active tab, or NULL if there is no
// tab or there was some kind of error. The returned pointer MUST be
// deleted by the caller if non-NULL.
TabProxy* GetActiveTab();
// ********* Member variables *********
std::wstring browser_directory_; // Path to the browser executable,
// with no trailing slash
std::wstring test_data_directory_; // Path to the unit test data,
// with no trailing slash
std::wstring launch_arguments_; // Arguments to the browser on launch.
int expected_errors_; // The number of errors expected during
// the run (generally 0).
int expected_crashes_; // The number of crashes expected during
// the run (generally 0).
std::wstring homepage_; // Homepage used for testing.
bool wait_for_initial_loads_; // Wait for initial loads to complete
// in SetUp() before running test body.
TimeTicks browser_launch_time_; // Time when the browser was run.
bool dom_automation_enabled_; // This can be set to true to have the
// test run the dom automation case.
std::wstring template_user_data_; // See set_template_user_data().
HANDLE process_; // Handle the the first Chrome process.
std::wstring user_data_dir_; // User data directory used for the test
static bool in_process_renderer_; // true if we're in single process mode
bool show_window_; // Determines if the window is shown or
// hidden. Defaults to hidden.
bool clear_profile_; // If true the profile is cleared before
// launching. Default is true.
bool include_testing_id_; // Should we supply the testing channel
// id on the command line? Default is
// true.
bool use_existing_browser_; // Duplicate of the static version.
// Default value comes from static.
private:
FILETIME test_start_time_; // Time the test was started
// (so we can check for new crash dumps)
static bool in_process_plugins_;
static bool no_sandbox_;
static bool safe_plugins_;
static bool full_memory_dump_; // If true, write full memory dump
// during crash.
static bool show_error_dialogs_; // If true, a user is paying attention
// to the test, so show error dialogs.
static bool default_use_existing_browser_; // The test connects to an already
// running browser instance.
static bool dump_histograms_on_exit_; // Include histograms in log on exit.
static bool enable_dcheck_; // Enable dchecks in release mode.
static bool silent_dump_on_dcheck_; // Dump process memory on dcheck without
// crashing.
static bool disable_breakpad_; // Disable breakpad on the browser.
static int timeout_ms_; // Timeout in milliseconds to wait
// for an test to finish.
static std::wstring js_flags_; // Flags passed to the JS engine.
::scoped_ptr<AutomationProxy> server_;
MessageLoop message_loop_; // Enables PostTask to main thread.
};
// These exist only to support the gTest assertion macros, and
// shouldn't be used in normal program code.
#ifdef UNIT_TEST
std::ostream& operator<<(std::ostream& out, const std::wstring& wstr);
template<typename T>
std::ostream& operator<<(std::ostream& out, const ::scoped_ptr<T>& ptr) {
return out << ptr.get();
}
#endif // UNIT_TEST
#endif // CHROME_TEST_UI_UI_TEST_H_
|