1 // Copyright 2013 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.
5 #include "chrome/test/remoting/remote_desktop_browsertest.h"
7 #include "base/command_line.h"
8 #include "base/files/file_util.h"
9 #include "base/json/json_reader.h"
10 #include "base/path_service.h"
11 #include "chrome/browser/extensions/extension_service.h"
12 #include "chrome/browser/extensions/unpacked_installer.h"
13 #include "chrome/browser/ui/extensions/app_launch_params.h"
14 #include "chrome/browser/ui/extensions/application_launch.h"
15 #include "chrome/common/chrome_switches.h"
16 #include "chrome/test/remoting/key_code_conv.h"
17 #include "chrome/test/remoting/page_load_notification_observer.h"
18 #include "chrome/test/remoting/waiter.h"
19 #include "content/public/browser/native_web_keyboard_event.h"
20 #include "content/public/browser/render_view_host.h"
21 #include "content/public/test/test_utils.h"
22 #include "extensions/browser/extension_registry.h"
23 #include "extensions/common/constants.h"
24 #include "extensions/common/extension.h"
25 #include "extensions/common/extension_set.h"
26 #include "extensions/common/switches.h"
27 #include "ui/base/window_open_disposition.h"
31 RemoteDesktopBrowserTest::RemoteDesktopBrowserTest()
32 : remote_test_helper_(nullptr), extension_(nullptr) {
35 RemoteDesktopBrowserTest::~RemoteDesktopBrowserTest() {}
37 void RemoteDesktopBrowserTest::SetUp() {
39 PlatformAppBrowserTest::SetUp();
42 void RemoteDesktopBrowserTest::SetUpOnMainThread() {
43 PlatformAppBrowserTest::SetUpOnMainThread();
45 // Pushing the initial WebContents instance onto the stack before
46 // RunTestOnMainThread() and after |InProcessBrowserTest::browser_|
47 // is initialized in InProcessBrowserTest::RunTestOnMainThreadLoop()
48 web_contents_stack_
.push_back(
49 browser()->tab_strip_model()->GetActiveWebContents());
52 // Change behavior of the default host resolver to avoid DNS lookup errors,
53 // so we can make network calls.
54 void RemoteDesktopBrowserTest::SetUpInProcessBrowserTestFixture() {
55 // The resolver object lifetime is managed by sync_test_setup, not here.
56 EnableDNSLookupForThisTest(
57 new net::RuleBasedHostResolverProc(host_resolver()));
60 void RemoteDesktopBrowserTest::TearDownInProcessBrowserTestFixture() {
61 DisableDNSLookupForThisTest();
64 void RemoteDesktopBrowserTest::VerifyInternetAccess() {
65 ui_test_utils::NavigateToURLBlockUntilNavigationsComplete(
66 browser(), GURL("http://www.google.com"), 1);
68 EXPECT_EQ(GetCurrentURL().host(), "www.google.com");
71 void RemoteDesktopBrowserTest::OpenClientBrowserPage() {
72 // Open the client browser page in a new tab
73 ui_test_utils::NavigateToURLWithDisposition(
75 GURL(http_server() + "/client.html"),
76 NEW_FOREGROUND_TAB
, ui_test_utils::BROWSER_TEST_WAIT_FOR_TAB
);
78 // Save this web content for later reference
79 client_web_content_
= browser()->tab_strip_model()->GetActiveWebContents();
81 // Go back to the previous tab that has chromoting opened
82 browser()->tab_strip_model()->SelectPreviousTab();
84 // Create the RemoteTestHelper object to use.
85 remote_test_helper_
.reset(new RemoteTestHelper(client_web_content_
));
88 bool RemoteDesktopBrowserTest::HtmlElementVisible(const std::string
& name
) {
89 _ASSERT_TRUE(HtmlElementExists(name
));
92 "function isElementVisible(name) {"
93 " var element = document.getElementById(name);"
94 " /* The existence of the element has already been ASSERTed. */"
96 " if (element.hidden) {"
99 " element = element.parentNode;"
100 " } while (element != null);"
104 return ExecuteScriptAndExtractBool(
105 "isElementVisible(\"" + name
+ "\")");
108 void RemoteDesktopBrowserTest::InstallChromotingAppCrx() {
109 ASSERT_TRUE(!is_unpacked());
111 base::FilePath
install_dir(WebAppCrxPath());
112 scoped_refptr
<const Extension
> extension(InstallExtensionWithUIAutoConfirm(
113 install_dir
, 1, browser()));
115 EXPECT_FALSE(extension
.get() == NULL
);
117 extension_
= extension
.get();
120 void RemoteDesktopBrowserTest::InstallChromotingAppUnpacked() {
121 ASSERT_TRUE(is_unpacked());
123 scoped_refptr
<extensions::UnpackedInstaller
> installer
=
124 extensions::UnpackedInstaller::Create(extension_service());
125 installer
->set_prompt_for_plugins(false);
127 content::WindowedNotificationObserver
observer(
128 extensions::NOTIFICATION_EXTENSION_LOADED_DEPRECATED
,
129 content::NotificationService::AllSources());
131 installer
->Load(webapp_unpacked_
);
136 void RemoteDesktopBrowserTest::UninstallChromotingApp() {
137 UninstallExtension(ChromotingID());
141 void RemoteDesktopBrowserTest::VerifyChromotingLoaded(bool expected
) {
142 bool installed
= false;
144 for (const scoped_refptr
<const extensions::Extension
>& extension
:
145 extensions::ExtensionRegistry::Get(profile())->enabled_extensions()) {
146 // Is there a better way to recognize the chromoting extension
147 // than name comparison?
148 if (extension
->name() == extension_name_
) {
150 EXPECT_EQ(extension
.get(), extension_
);
152 extension_
= extension
.get();
161 // Either a V1 (TYPE_LEGACY_PACKAGED_APP) or a V2 (TYPE_PLATFORM_APP ) app.
162 extensions::Manifest::Type type
= extension_
->GetType();
163 EXPECT_TRUE(type
== extensions::Manifest::TYPE_PLATFORM_APP
||
164 type
== extensions::Manifest::TYPE_LEGACY_PACKAGED_APP
);
166 EXPECT_TRUE(extension_
->ShouldDisplayInAppLauncher());
169 ASSERT_EQ(installed
, expected
);
172 content::WebContents
* RemoteDesktopBrowserTest::LaunchChromotingApp(
174 WindowOpenDisposition window_open_disposition
) {
175 _ASSERT_TRUE(extension_
);
177 GURL chromoting_main
= Chromoting_Main_URL();
178 // We cannot simply wait for any page load because the first page
179 // loaded could be the generated background page. We need to wait
180 // till the chromoting main page is loaded.
181 PageLoadNotificationObserver
observer(chromoting_main
);
182 observer
.set_ignore_url_parameters(true);
184 // If the app should be started in deferred mode, ensure that a "source" URL
185 // parameter is present; if not, ensure that no such parameter is present.
186 // The value of the parameter is determined by the AppLaunchParams ("test",
188 extensions::FeatureSwitch::ScopedOverride
override_trace_app_source(
189 extensions::FeatureSwitch::trace_app_source(),
192 if (is_platform_app()) {
193 window_open_disposition
= NEW_WINDOW
;
196 OpenApplication(AppLaunchParams(browser()->profile(), extension_
,
198 ? extensions::LAUNCH_CONTAINER_NONE
199 : extensions::LAUNCH_CONTAINER_TAB
,
200 window_open_disposition
,
201 extensions::SOURCE_TEST
));
206 // The active WebContents instance should be the source of the LOAD_STOP
208 content::NavigationController
* controller
=
209 content::Source
<content::NavigationController
>(observer
.source()).ptr();
211 content::WebContents
* web_contents
= controller
->GetWebContents();
212 _ASSERT_TRUE(web_contents
);
214 if (web_contents
!= active_web_contents())
215 web_contents_stack_
.push_back(web_contents
);
217 if (is_platform_app()) {
218 EXPECT_EQ(GetFirstAppWindowWebContents(), active_web_contents());
220 // For apps v1 only, the DOMOperationObserver is not ready at the LOAD_STOP
221 // event. A half second wait is necessary for the subsequent javascript
222 // injection to work.
223 // TODO(weitaosu): Find out whether there is a more appropriate notification
224 // to wait for so we can get rid of this wait.
225 _ASSERT_TRUE(TimeoutWaiter(base::TimeDelta::FromSeconds(5)).Wait());
228 EXPECT_EQ(Chromoting_Main_URL(), GetCurrentURL());
232 content::WebContents
* RemoteDesktopBrowserTest::LaunchChromotingApp(
234 return LaunchChromotingApp(defer_start
, CURRENT_TAB
);
237 void RemoteDesktopBrowserTest::StartChromotingApp() {
238 ClickOnControl("browser-test-continue-init");
241 void RemoteDesktopBrowserTest::Authorize() {
242 // The chromoting extension should be installed.
243 ASSERT_TRUE(extension_
);
245 // The chromoting main page should be loaded in the current tab
246 // and isAuthenticated() should be false (auth dialog visible).
247 ASSERT_EQ(Chromoting_Main_URL(), GetCurrentURL());
248 ASSERT_FALSE(IsAuthenticated());
250 // We cannot simply wait for any page load because the first page
251 // loaded will be chrome://chrome-signin in a packaged app. We need to wait
252 // for the Google login page to be loaded (inside an embedded iframe).
253 GURL
google_login("https://accounts.google.com/ServiceLogin");
254 PageLoadNotificationObserver
observer(google_login
);
255 observer
.set_ignore_url_parameters(true);
257 ClickOnControl("auth-button");
261 content::NavigationController
* controller
=
262 content::Source
<content::NavigationController
>(observer
.source()).ptr();
264 content::WebContents
* web_contents
= controller
->GetWebContents();
265 _ASSERT_TRUE(web_contents
);
267 if (web_contents
!= active_web_contents()) {
268 // Pushing the WebContents hosting the Google login page onto the stack.
269 // If this is a packaged app the Google login page will be loaded in an
270 // iframe embedded in the chrome://chrome-signin page. But we can ignore
271 // that WebContents because we never need to interact with it directly.
272 LOG(INFO
) << "Pushing onto the stack: " << web_contents
->GetURL();
273 web_contents_stack_
.push_back(web_contents
);
276 // Verify the active tab is at the "Google Accounts" login page.
277 EXPECT_EQ("accounts.google.com", GetCurrentURL().host());
279 EXPECT_TRUE(HtmlElementExists("Email"));
280 EXPECT_TRUE(HtmlElementExists("Passwd"));
283 void RemoteDesktopBrowserTest::Authenticate() {
284 // The chromoting extension should be installed.
285 ASSERT_TRUE(extension_
);
287 // The active WebContents should have the "Google Accounts" login page loaded.
288 ASSERT_EQ("accounts.google.com", GetCurrentURL().host());
290 ASSERT_TRUE(HtmlElementExists("Email"));
291 ASSERT_TRUE(HtmlElementExists("Passwd"));
293 // Now log in using the username and password passed in from the command line.
294 ExecuteScriptAndWaitForAnyPageLoad(
295 "document.getElementById(\"Email\").value = \"" + username_
+ "\";" +
296 "document.getElementById(\"Passwd\").value = \"" + password_
+"\";" +
297 "document.forms[\"gaia_loginform\"].submit();");
299 // TODO(weitaosu): Is there a better way to verify we are on the
300 // "Request for Permission" page?
301 // V2 app won't ask for approval here because the chromoting test account
302 // has already been granted permissions.
303 if (!is_platform_app()) {
304 EXPECT_EQ(GetCurrentURL().host(), "accounts.google.com");
305 EXPECT_TRUE(HtmlElementExists("submit_approve_access"));
309 void RemoteDesktopBrowserTest::Approve() {
310 // The chromoting extension should be installed.
311 ASSERT_TRUE(extension_
);
313 if (is_platform_app()) {
314 // Popping the login window off the stack to return to the chromoting
316 web_contents_stack_
.pop_back();
318 // There is nothing for the V2 app to approve because the chromoting test
319 // account has already been granted permissions.
320 // TODO(weitaosu): Revoke the permission at the beginning of the test so
321 // that we can test first-time experience here.
322 ConditionalTimeoutWaiter
waiter(
323 base::TimeDelta::FromSeconds(7),
324 base::TimeDelta::FromSeconds(1),
326 &RemoteDesktopBrowserTest::IsAuthenticatedInWindow
,
327 active_web_contents()));
329 ASSERT_TRUE(waiter
.Wait());
331 ASSERT_EQ("accounts.google.com", GetCurrentURL().host());
333 // Is there a better way to verify we are on the "Request for Permission"
335 ASSERT_TRUE(HtmlElementExists("submit_approve_access"));
337 const GURL chromoting_main
= Chromoting_Main_URL();
339 // active_web_contents() is still the login window but the observer
340 // should be set up to observe the chromoting window because that is
341 // where we'll receive the message from the login window and reload the
343 content::WindowedNotificationObserver
observer(
344 content::NOTIFICATION_LOAD_STOP
,
346 &RemoteDesktopBrowserTest::IsAuthenticatedInWindow
,
347 browser()->tab_strip_model()->GetActiveWebContents()));
349 // Click to Approve the web-app.
350 ClickOnControl("submit_approve_access");
354 // Popping the login window off the stack to return to the chromoting
356 web_contents_stack_
.pop_back();
359 ASSERT_TRUE(GetCurrentURL() == Chromoting_Main_URL());
361 EXPECT_TRUE(IsAuthenticated());
364 void RemoteDesktopBrowserTest::ExpandMe2Me() {
365 // The chromoting extension should be installed.
366 ASSERT_TRUE(extension_
);
368 // The active tab should have the chromoting app loaded.
369 ASSERT_EQ(Chromoting_Main_URL(), GetCurrentURL());
370 EXPECT_TRUE(IsAuthenticated());
372 // The Me2Me host list should be hidden.
373 ASSERT_FALSE(HtmlElementVisible("me2me-content"));
374 // The Me2Me "Get Start" button should be visible.
375 ASSERT_TRUE(HtmlElementVisible("get-started-me2me"));
378 ExecuteScript("remoting.showMe2MeUiAndSave();");
380 EXPECT_TRUE(HtmlElementVisible("me2me-content"));
381 EXPECT_FALSE(HtmlElementVisible("me2me-first-run"));
384 void RemoteDesktopBrowserTest::DisconnectMe2Me() {
385 // The chromoting extension should be installed.
386 ASSERT_TRUE(extension_
);
388 ASSERT_TRUE(RemoteDesktopBrowserTest::IsSessionConnected());
390 ExecuteScript("remoting.app.getActivity().stop();");
392 EXPECT_TRUE(HtmlElementVisible("client-dialog"));
393 EXPECT_TRUE(HtmlElementVisible("client-reconnect-button"));
394 EXPECT_TRUE(HtmlElementVisible("client-finished-me2me-button"));
396 ClickOnControl("client-finished-me2me-button");
398 EXPECT_FALSE(HtmlElementVisible("client-dialog"));
401 void RemoteDesktopBrowserTest::SimulateKeyPressWithCode(
402 ui::KeyboardCode keyCode
,
404 SimulateKeyPressWithCode(keyCode
, code
, false, false, false, false);
407 void RemoteDesktopBrowserTest::SimulateKeyPressWithCode(
408 ui::KeyboardCode keyCode
,
414 content::SimulateKeyPressWithCode(
415 active_web_contents(),
424 void RemoteDesktopBrowserTest::SimulateCharInput(char c
) {
426 ui::KeyboardCode keyboard_code
;
428 GetKeyValuesFromChar(c
, &code
, &keyboard_code
, &shift
);
429 ASSERT_TRUE(code
!= NULL
);
430 SimulateKeyPressWithCode(keyboard_code
, code
, false, shift
, false, false);
433 void RemoteDesktopBrowserTest::SimulateStringInput(const std::string
& input
) {
434 for (size_t i
= 0; i
< input
.length(); ++i
)
435 SimulateCharInput(input
[i
]);
438 void RemoteDesktopBrowserTest::SimulateMouseLeftClickAt(int x
, int y
) {
439 SimulateMouseClickAt(0, blink::WebMouseEvent::ButtonLeft
, x
, y
);
442 void RemoteDesktopBrowserTest::SimulateMouseClickAt(
443 int modifiers
, blink::WebMouseEvent::Button button
, int x
, int y
) {
444 // TODO(weitaosu): The coordinate translation doesn't work correctly for
447 "var clientPluginElement = "
448 "document.getElementById('session-client-plugin');"
449 "var clientPluginRect = clientPluginElement.getBoundingClientRect();");
451 int top
= ExecuteScriptAndExtractInt("clientPluginRect.top");
452 int left
= ExecuteScriptAndExtractInt("clientPluginRect.left");
453 int width
= ExecuteScriptAndExtractInt("clientPluginRect.width");
454 int height
= ExecuteScriptAndExtractInt("clientPluginRect.height");
459 ASSERT_LT(y
, height
);
461 content::SimulateMouseClickAt(
462 browser()->tab_strip_model()->GetActiveWebContents(),
465 gfx::Point(left
+ x
, top
+ y
));
468 void RemoteDesktopBrowserTest::Install() {
470 VerifyChromotingLoaded(false);
472 InstallChromotingAppUnpacked();
474 InstallChromotingAppCrx();
477 VerifyChromotingLoaded(true);
480 void RemoteDesktopBrowserTest::LoadBrowserTestJavaScript(
481 content::WebContents
* content
) {
482 LoadScript(content
, FILE_PATH_LITERAL("browser_test.js"));
483 LoadScript(content
, FILE_PATH_LITERAL("mock_client_plugin.js"));
484 LoadScript(content
, FILE_PATH_LITERAL("mock_host_list_api.js"));
485 LoadScript(content
, FILE_PATH_LITERAL("mock_identity.js"));
486 LoadScript(content
, FILE_PATH_LITERAL("mock_oauth2_api.js"));
487 LoadScript(content
, FILE_PATH_LITERAL("mock_session_connector.js"));
488 LoadScript(content
, FILE_PATH_LITERAL("mock_signal_strategy.js"));
489 LoadScript(content
, FILE_PATH_LITERAL("timeout_waiter.js"));
490 LoadScript(content
, FILE_PATH_LITERAL("sinon.js"));
493 void RemoteDesktopBrowserTest::Cleanup() {
494 // TODO(weitaosu): Remove this hack by blocking on the appropriate
496 // The browser may still be loading images embedded in the webapp. If we
497 // uinstall it now those load will fail.
498 ASSERT_TRUE(TimeoutWaiter(base::TimeDelta::FromSeconds(2)).Wait());
501 UninstallChromotingApp();
502 VerifyChromotingLoaded(false);
505 // TODO(chaitali): Remove this additional timeout after we figure out
506 // why this is needed for the v1 app to work.
507 // Without this timeout the test fail with a "CloseWebContents called for
508 // tab not in our strip" error for the v1 app.
509 ASSERT_TRUE(TimeoutWaiter(base::TimeDelta::FromSeconds(2)).Wait());
512 content::WebContents
* RemoteDesktopBrowserTest::SetUpTest() {
513 VerifyInternetAccess();
515 content::WebContents
* app_web_content
= LaunchChromotingApp(false);
517 LoadBrowserTestJavaScript(app_web_content
);
519 // The call to EnsureRemoteConnectionEnabled() does a PIN reset.
520 // This causes the test to fail because of a recent bug:
522 // TODO(anandc): Reactivate this call after above bug is fixed.
523 //EnsureRemoteConnectionEnabled(app_web_content);
524 return app_web_content
;
527 void RemoteDesktopBrowserTest::Auth() {
528 // For this test, we must be given the user-name and password.
529 ASSERT_TRUE(!username_
.empty() && !password_
.empty());
536 void RemoteDesktopBrowserTest::EnsureRemoteConnectionEnabled(
537 content::WebContents
* app_web_contents
) {
538 // browser_test.ensureRemoteConnectionEnabled is defined in
539 // browser_test.js, which must be loaded before calling this function.
540 // TODO(kelvinp): This function currently only works on linux when the user is
541 // already part of the chrome-remote-desktop group. Extend this functionality
542 // to Mac (https://crbug.com/397576) and Windows (https://crbug.com/397575).
544 EXPECT_TRUE(content::ExecuteScriptAndExtractBool(
546 "browserTest.ensureRemoteConnectionEnabled(" + me2me_pin() + ")",
548 EXPECT_TRUE(result
) << "Cannot start the host with Pin:" << me2me_pin();
551 void RemoteDesktopBrowserTest::ConnectToLocalHost(bool remember_pin
) {
552 // Wait for local-host to be ready.
553 ConditionalTimeoutWaiter
waiter(
554 base::TimeDelta::FromSeconds(5),
555 base::TimeDelta::FromMilliseconds(500),
556 base::Bind(&RemoteDesktopBrowserTest::IsLocalHostReady
, this));
557 EXPECT_TRUE(waiter
.Wait());
559 // Verify that the local host is online.
560 ASSERT_TRUE(ExecuteScriptAndExtractBool(
561 "remoting.hostList.localHostSection_.host_.hostName && "
562 "remoting.hostList.localHostSection_.host_.hostId && "
563 "remoting.hostList.localHostSection_.host_.status && "
564 "remoting.hostList.localHostSection_.host_.status == 'ONLINE'"));
567 ClickOnControl("local-host-connect-button");
569 // Enter the pin # passed in from the command line.
570 EnterPin(me2me_pin(), remember_pin
);
575 void RemoteDesktopBrowserTest::ConnectToRemoteHost(
576 const std::string
& host_name
, bool remember_pin
) {
578 // Wait for hosts list to be fetched.
579 // This test typically runs with a clean user-profile, with no host-list
580 // cached. Waiting for the host-list to be null is sufficient to proceed.
581 ConditionalTimeoutWaiter
waiter(
582 base::TimeDelta::FromSeconds(5),
583 base::TimeDelta::FromMilliseconds(500),
584 base::Bind(&RemoteDesktopBrowserTest::IsHostListReady
, this));
585 EXPECT_TRUE(waiter
.Wait());
587 std::string host_id
= ExecuteScriptAndExtractString(
588 "remoting.hostList.getHostIdForName('" + host_name
+ "')");
590 EXPECT_FALSE(host_id
.empty());
591 std::string element_id
= "host_" + host_id
;
593 // Verify the host is online.
594 std::string host_div_class
= ExecuteScriptAndExtractString(
595 "document.getElementById('" + element_id
+ "').parentNode.className");
596 EXPECT_NE(std::string::npos
, host_div_class
.find("host-online"));
598 ClickOnControl(element_id
);
600 // Enter the pin # passed in from the command line.
601 EnterPin(me2me_pin(), remember_pin
);
606 void RemoteDesktopBrowserTest::EnableDNSLookupForThisTest(
607 net::RuleBasedHostResolverProc
* host_resolver
) {
608 // mock_host_resolver_override_ takes ownership of the resolver.
609 scoped_refptr
<net::RuleBasedHostResolverProc
> resolver
=
610 new net::RuleBasedHostResolverProc(host_resolver
);
611 resolver
->AllowDirectLookup("*.google.com");
612 // On Linux, we use Chromium's NSS implementation which uses the following
613 // hosts for certificate verification. Without these overrides, running the
614 // integration tests on Linux causes errors as we make external DNS lookups.
615 resolver
->AllowDirectLookup("*.thawte.com");
616 resolver
->AllowDirectLookup("*.geotrust.com");
617 resolver
->AllowDirectLookup("*.gstatic.com");
618 resolver
->AllowDirectLookup("*.googleapis.com");
619 mock_host_resolver_override_
.reset(
620 new net::ScopedDefaultHostResolverProc(resolver
.get()));
623 void RemoteDesktopBrowserTest::DisableDNSLookupForThisTest() {
624 mock_host_resolver_override_
.reset();
627 void RemoteDesktopBrowserTest::ParseCommandLine() {
628 base::CommandLine
* command_line
= base::CommandLine::ForCurrentProcess();
630 // The test framework overrides any command line user-data-dir
631 // argument with a /tmp/.org.chromium.Chromium.XXXXXX directory.
632 // That happens in the ChromeTestLauncherDelegate, and affects
633 // all unit tests (no opt out available). It intentionally erases
634 // any --user-data-dir switch if present and appends a new one.
635 // Re-override the default data dir if override-user-data-dir
637 if (command_line
->HasSwitch(kOverrideUserDataDir
)) {
638 const base::FilePath
& override_user_data_dir
=
639 command_line
->GetSwitchValuePath(kOverrideUserDataDir
);
641 ASSERT_FALSE(override_user_data_dir
.empty());
643 command_line
->AppendSwitchPath(switches::kUserDataDir
,
644 override_user_data_dir
);
647 base::CommandLine::StringType accounts_file
=
648 command_line
->GetSwitchValueNative(kAccountsFile
);
649 std::string account_type
= command_line
->GetSwitchValueASCII(kAccountType
);
650 if (!accounts_file
.empty()) {
651 // We've been passed in a file containing accounts information.
652 // In this case, we'll obtain the user-name and password information from
653 // the specified file, even if user-name and password have been specified
654 // on the command-line.
655 base::FilePath
accounts_file_path((base::FilePath(accounts_file
)));
656 ASSERT_FALSE(account_type
.empty());
657 ASSERT_TRUE(base::PathExists((base::FilePath(accounts_file
))));
658 SetUserNameAndPassword((base::FilePath(accounts_file
)), account_type
);
660 // No file for accounts specified. Read user-name and password from command
662 username_
= command_line
->GetSwitchValueASCII(kUserName
);
663 password_
= command_line
->GetSwitchValueASCII(kUserPassword
);
666 me2me_pin_
= command_line
->GetSwitchValueASCII(kMe2MePin
);
667 remote_host_name_
= command_line
->GetSwitchValueASCII(kRemoteHostName
);
668 extension_name_
= command_line
->GetSwitchValueASCII(kExtensionName
);
669 http_server_
= command_line
->GetSwitchValueASCII(kHttpServer
);
671 no_cleanup_
= command_line
->HasSwitch(kNoCleanup
);
672 no_install_
= command_line
->HasSwitch(kNoInstall
);
675 webapp_crx_
= command_line
->GetSwitchValuePath(kWebAppCrx
);
676 webapp_unpacked_
= command_line
->GetSwitchValuePath(kWebAppUnpacked
);
677 // One and only one of these two arguments should be provided.
678 ASSERT_NE(webapp_crx_
.empty(), webapp_unpacked_
.empty());
681 // Enable experimental extensions; this is to allow adding the LG extensions
682 command_line
->AppendSwitch(
683 extensions::switches::kEnableExperimentalExtensionApis
);
686 void RemoteDesktopBrowserTest::ExecuteScript(const std::string
& script
) {
687 ASSERT_TRUE(content::ExecuteScript(active_web_contents(), script
));
690 void RemoteDesktopBrowserTest::ExecuteScriptAndWaitForAnyPageLoad(
691 const std::string
& script
) {
692 content::WindowedNotificationObserver
observer(
693 content::NOTIFICATION_LOAD_STOP
,
694 content::Source
<content::NavigationController
>(
695 &active_web_contents()->
698 ExecuteScript(script
);
704 bool RemoteDesktopBrowserTest::LoadScript(
705 content::WebContents
* web_contents
,
706 const base::FilePath::StringType
& path
) {
708 base::FilePath src_dir
;
709 _ASSERT_TRUE(PathService::Get(base::DIR_EXE
, &src_dir
));
710 base::FilePath script_path
=
711 src_dir
.Append(FILE_PATH_LITERAL("remoting/browser_test_resources/"));
712 script_path
= script_path
.Append(path
);
714 if (!base::ReadFileToString(script_path
, &script
)) {
715 LOG(ERROR
) << "Failed to load script " << script_path
.value();
719 return content::ExecuteScript(web_contents
, script
);
723 void RemoteDesktopBrowserTest::RunJavaScriptTest(
724 content::WebContents
* web_contents
,
725 const std::string
& testName
,
726 const std::string
& testData
) {
728 std::string script
= "browserTest.runTest(browserTest." + testName
+ ", " +
731 DVLOG(1) << "Executing " << script
;
734 content::ExecuteScriptAndExtractString(web_contents
, script
, &result
));
737 base::JSONReader reader
;
738 scoped_ptr
<base::Value
> value
;
739 value
.reset(reader
.Read(result
, base::JSON_ALLOW_TRAILING_COMMAS
));
741 // Convert to dictionary
742 base::DictionaryValue
* dict_value
= NULL
;
743 ASSERT_TRUE(value
->GetAsDictionary(&dict_value
));
746 std::string error_message
;
747 std::string stack_trace
;
749 // Extract the fields
750 ASSERT_TRUE(dict_value
->GetBoolean("succeeded", &succeeded
));
751 ASSERT_TRUE(dict_value
->GetString("error_message", &error_message
));
752 ASSERT_TRUE(dict_value
->GetString("stack_trace", &stack_trace
));
754 EXPECT_TRUE(succeeded
) << error_message
<< "\n" << stack_trace
;
757 void RemoteDesktopBrowserTest::ClickOnControl(const std::string
& name
) {
758 ASSERT_TRUE(HtmlElementVisible(name
));
760 std::string has_disabled_attribute
=
761 "document.getElementById('" + name
+ "').hasAttribute('disabled')";
763 if (RemoteTestHelper::ExecuteScriptAndExtractBool(active_web_contents(),
764 has_disabled_attribute
)) {
765 // This element has a disabled attribute. Wait for it become enabled.
766 ConditionalTimeoutWaiter
waiter(
767 base::TimeDelta::FromSeconds(5),
768 base::TimeDelta::FromMilliseconds(500),
769 base::Bind(&RemoteDesktopBrowserTest::IsEnabled
,
770 active_web_contents(), name
));
771 ASSERT_TRUE(waiter
.Wait());
774 ExecuteScript("document.getElementById(\"" + name
+ "\").click();");
777 void RemoteDesktopBrowserTest::EnterPin(const std::string
& pin
,
779 // Wait for the pin-form to be displayed. This can take a while.
780 // We also need to dismiss the host-needs-update dialog if it comes up.
781 // TODO(weitaosu) 1: Instead of polling, can we register a callback to be
782 // called when the pin-form is ready?
783 // TODO(weitaosu) 2: Instead of blindly dismiss the host-needs-update dialog,
784 // we should verify that it only pops up at the right circumstance. That
785 // probably belongs in a separate test case though.
786 ConditionalTimeoutWaiter
waiter(
787 base::TimeDelta::FromSeconds(30),
788 base::TimeDelta::FromSeconds(1),
789 base::Bind(&RemoteDesktopBrowserTest::IsPinFormVisible
, this));
790 EXPECT_TRUE(waiter
.Wait());
793 "document.getElementById(\"pin-entry\").value = \"" + pin
+ "\";");
796 EXPECT_TRUE(HtmlElementVisible("remember-pin"));
797 EXPECT_FALSE(ExecuteScriptAndExtractBool(
798 "document.getElementById('remember-pin-checkbox').checked"));
799 ClickOnControl("remember-pin");
800 EXPECT_TRUE(ExecuteScriptAndExtractBool(
801 "document.getElementById('remember-pin-checkbox').checked"));
804 ClickOnControl("pin-connect-button");
807 void RemoteDesktopBrowserTest::WaitForConnection() {
808 // Wait until the client has connected to the server.
809 // This can take a while.
810 // TODO(weitaosu): Instead of polling, can we register a callback to
811 // remoting.clientSession.onStageChange_?
812 ConditionalTimeoutWaiter
waiter(
813 base::TimeDelta::FromSeconds(30),
814 base::TimeDelta::FromSeconds(1),
815 base::Bind(&RemoteDesktopBrowserTest::IsSessionConnected
, this));
816 EXPECT_TRUE(waiter
.Wait());
818 // The client is not yet ready to take input when the session state becomes
819 // CONNECTED. Wait for 2 seconds for the client to become ready.
820 // TODO(weitaosu): Find a way to detect when the client is truly ready.
821 TimeoutWaiter(base::TimeDelta::FromSeconds(2)).Wait();
824 bool RemoteDesktopBrowserTest::IsLocalHostReady() {
825 // TODO(weitaosu): Instead of polling, can we register a callback to
826 // remoting.hostList.setLocalHost_?
827 return ExecuteScriptAndExtractBool(
828 "remoting.hostList.localHostSection_.host_ != null");
831 bool RemoteDesktopBrowserTest::IsHostListReady() {
832 // Wait until hostList is not null.
833 // The connect-to-host tests are run on the waterfall using a new profile-dir.
834 // No hosts will be cached.
835 return ExecuteScriptAndExtractBool(
836 "remoting.hostList != null && remoting.hostList.hosts_ != null");
839 bool RemoteDesktopBrowserTest::IsSessionConnected() {
840 // If some form of PINless authentication is enabled, the host version
841 // warning may appear while waiting for the session to connect.
842 DismissHostVersionWarningIfVisible();
844 return ExecuteScriptAndExtractBool(
845 "remoting.currentMode === remoting.AppMode.IN_SESSION");
848 bool RemoteDesktopBrowserTest::IsPinFormVisible() {
849 DismissHostVersionWarningIfVisible();
850 return HtmlElementVisible("pin-form");
853 void RemoteDesktopBrowserTest::DismissHostVersionWarningIfVisible() {
854 if (HtmlElementVisible("host-needs-update-connect-button"))
855 ClickOnControl("host-needs-update-connect-button");
858 void RemoteDesktopBrowserTest::SetUserNameAndPassword(
859 const base::FilePath
&accounts_file_path
, const std::string
& account_type
) {
861 // Read contents of accounts file, using its absolute path.
862 base::FilePath absolute_path
= base::MakeAbsoluteFilePath(accounts_file_path
);
863 std::string accounts_info
;
864 ASSERT_TRUE(base::ReadFileToString(absolute_path
, &accounts_info
));
866 // Get the root dictionary from the input json file contents.
867 scoped_ptr
<base::Value
> root(
868 base::JSONReader::Read(accounts_info
, base::JSON_ALLOW_TRAILING_COMMAS
));
870 const base::DictionaryValue
* root_dict
= NULL
;
871 ASSERT_TRUE(root
.get() && root
->GetAsDictionary(&root_dict
));
873 // Now get the dictionary for the specified account type.
874 const base::DictionaryValue
* account_dict
= NULL
;
875 ASSERT_TRUE(root_dict
->GetDictionary(account_type
, &account_dict
));
876 ASSERT_TRUE(account_dict
->GetString(kUserName
, &username_
));
877 ASSERT_TRUE(account_dict
->GetString(kUserPassword
, &password_
));
881 bool RemoteDesktopBrowserTest::IsAuthenticatedInWindow(
882 content::WebContents
* web_contents
) {
883 return RemoteTestHelper::ExecuteScriptAndExtractBool(
884 web_contents
, "remoting.identity.isAuthenticated()");
888 bool RemoteDesktopBrowserTest::IsHostActionComplete(
889 content::WebContents
* client_web_content
,
890 std::string host_action_var
) {
891 return RemoteTestHelper::ExecuteScriptAndExtractBool(
897 bool RemoteDesktopBrowserTest::IsEnabled(
898 content::WebContents
* client_web_content
,
899 const std::string
& element_name
) {
900 return !RemoteTestHelper::ExecuteScriptAndExtractBool(
902 "document.getElementById(\"" + element_name
+ "\").disabled");
905 bool RemoteDesktopBrowserTest::IsAppModeEqualTo(const std::string
& mode
) {
906 return ExecuteScriptAndExtractBool(
907 "remoting.currentMode == " + mode
);
910 void RemoteDesktopBrowserTest::DisableRemoteConnection() {
911 ConditionalTimeoutWaiter
hostReadyWaiter(
912 base::TimeDelta::FromSeconds(5),
913 base::TimeDelta::FromMilliseconds(500),
914 base::Bind(&RemoteDesktopBrowserTest::IsLocalHostReady
, this));
915 EXPECT_TRUE(hostReadyWaiter
.Wait());
917 ClickOnControl("stop-daemon");
919 ConditionalTimeoutWaiter
setupDoneWaiter(
920 base::TimeDelta::FromSeconds(30),
921 base::TimeDelta::FromMilliseconds(500),
922 base::Bind(&RemoteDesktopBrowserTest::IsAppModeEqualTo
,
923 this, "remoting.AppMode.HOST_SETUP_DONE"));
924 EXPECT_TRUE(setupDoneWaiter
.Wait());
926 ClickOnControl("host-config-done-dismiss");
928 ConditionalTimeoutWaiter
homeWaiter(
929 base::TimeDelta::FromSeconds(5),
930 base::TimeDelta::FromMilliseconds(500),
931 base::Bind(&RemoteDesktopBrowserTest::IsAppModeEqualTo
,
932 this, "remoting.AppMode.HOME"));
933 EXPECT_TRUE(homeWaiter
.Wait());
936 } // namespace remoting