| // Copyright 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. |
| |
| #include "ash/constants/ash_switches.h" |
| #include "base/command_line.h" |
| #include "build/chromeos_buildflags.h" |
| #include "chrome/browser/ash/login/chrome_restart_request.h" |
| #include "chrome/browser/ash/profiles/profile_helper.h" |
| #include "chrome/browser/ui/ash/multi_user/multi_user_util.h" |
| #include "chrome/browser/ui/ash/multi_user/multi_user_window_manager_helper.h" |
| #include "chrome/browser/ui/ash/multi_user/test_multi_user_window_manager.h" |
| #include "chrome/browser/ui/ash/window_pin_util.h" |
| #include "chrome/browser/ui/browser.h" |
| #include "chrome/browser/ui/browser_finder.h" |
| #include "chrome/browser/ui/browser_navigator.h" |
| #include "chrome/browser/ui/browser_navigator_browsertest.h" |
| #include "chrome/browser/ui/browser_window.h" |
| #include "chrome/browser/ui/tabs/tab_strip_model.h" |
| #include "chrome/common/chrome_switches.h" |
| #include "chrome/common/webui_url_constants.h" |
| #include "chrome/test/base/ui_test_utils.h" |
| #include "components/account_id/account_id.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/test/browser_test.h" |
| #include "ui/aura/window.h" |
| |
| #if BUILDFLAG(IS_CHROMEOS_LACROS) |
| #include "chromeos/crosapi/mojom/crosapi.mojom.h" |
| #include "chromeos/crosapi/mojom/test_controller.mojom-test-utils.h" |
| #include "chromeos/lacros/lacros_test_helper.h" |
| #include "chromeos/startup/browser_init_params.h" |
| #endif |
| |
| namespace { |
| |
| using BrowserNavigatorTestChromeOS = BrowserNavigatorTest; |
| |
| #if BUILDFLAG(IS_CHROMEOS_ASH) |
| |
| GURL GetGoogleURL() { |
| return GURL("http://www.google.com/"); |
| } |
| |
| // Verifies that new browser is not opened for Signin profile. |
| IN_PROC_BROWSER_TEST_F(BrowserNavigatorTestChromeOS, RestrictSigninProfile) { |
| EXPECT_EQ(chrome::GetTotalBrowserCount(), 1u); |
| |
| EXPECT_EQ(Browser::CreationStatus::kErrorProfileUnsuitable, |
| Browser::GetCreationStatusForProfile( |
| ash::ProfileHelper::GetSigninProfile())); |
| } |
| |
| // Verify that page navigation is blocked in locked fullscreen mode. |
| IN_PROC_BROWSER_TEST_F(BrowserNavigatorTestChromeOS, |
| NavigationBlockedInLockedFullscreen) { |
| // Set locked fullscreen state. |
| aura::Window* window = browser()->window()->GetNativeWindow(); |
| PinWindow(window, /*trusted=*/true); |
| |
| // Navigate to a page. |
| auto url = GURL(chrome::kChromeUIVersionURL); |
| NavigateParams params(MakeNavigateParams(browser())); |
| params.disposition = WindowOpenDisposition::NEW_WINDOW; |
| params.url = url; |
| params.window_action = NavigateParams::SHOW_WINDOW; |
| Navigate(¶ms); |
| |
| // The page should not be opened, and the browser should still sit at the |
| // default about:blank page. |
| EXPECT_EQ(1u, chrome::GetTotalBrowserCount()); |
| EXPECT_EQ(1, browser()->tab_strip_model()->count()); |
| EXPECT_EQ(GURL(url::kAboutBlankURL), |
| browser()->tab_strip_model()->GetActiveWebContents()->GetURL()); |
| |
| // As a sanity check unset the locked fullscreen state and make sure that the |
| // navigation happens (the following EXPECTs fail if the next line isn't |
| // executed). |
| UnpinWindow(window); |
| |
| Navigate(¶ms); |
| |
| // The original browser should still be at the same page, but the newly |
| // opened browser should sit on the chrome:version page. |
| EXPECT_EQ(2u, chrome::GetTotalBrowserCount()); |
| EXPECT_EQ(1, browser()->tab_strip_model()->count()); |
| EXPECT_EQ(GURL(url::kAboutBlankURL), |
| browser()->tab_strip_model()->GetActiveWebContents()->GetURL()); |
| EXPECT_EQ(1, params.browser->tab_strip_model()->count()); |
| EXPECT_EQ( |
| GURL(chrome::kChromeUIVersionURL), |
| params.browser->tab_strip_model()->GetActiveWebContents()->GetURL()); |
| } |
| |
| // Subclass that tests navigation while in the Guest session. |
| class BrowserGuestSessionNavigatorTest : public BrowserNavigatorTest { |
| protected: |
| void SetUpCommandLine(base::CommandLine* command_line) override { |
| base::CommandLine command_line_copy = *command_line; |
| command_line_copy.AppendSwitchASCII(ash::switches::kLoginProfile, "user"); |
| command_line_copy.AppendSwitch(ash::switches::kGuestSession); |
| ash::GetOffTheRecordCommandLine(GetGoogleURL(), command_line_copy, |
| command_line); |
| } |
| }; |
| |
| // This test verifies that the settings page is opened in the incognito window |
| // in Guest Session (as well as all other windows in Guest session). |
| IN_PROC_BROWSER_TEST_F(BrowserGuestSessionNavigatorTest, |
| Disposition_Settings_UseIncognitoWindow) { |
| Browser* incognito_browser = CreateIncognitoBrowser(); |
| |
| EXPECT_EQ(2u, chrome::GetTotalBrowserCount()); |
| EXPECT_EQ(1, browser()->tab_strip_model()->count()); |
| EXPECT_EQ(1, incognito_browser->tab_strip_model()->count()); |
| |
| // Navigate to the settings page. |
| NavigateParams params(MakeNavigateParams(incognito_browser)); |
| params.disposition = WindowOpenDisposition::SINGLETON_TAB; |
| params.url = GURL("chrome://settings"); |
| params.window_action = NavigateParams::SHOW_WINDOW; |
| params.path_behavior = NavigateParams::IGNORE_AND_NAVIGATE; |
| Navigate(¶ms); |
| |
| // Settings page should be opened in incognito window. |
| EXPECT_NE(browser(), params.browser); |
| EXPECT_EQ(incognito_browser, params.browser); |
| EXPECT_EQ(2, incognito_browser->tab_strip_model()->count()); |
| EXPECT_EQ( |
| GURL("chrome://settings"), |
| incognito_browser->tab_strip_model()->GetActiveWebContents()->GetURL()); |
| } |
| |
| // Test that in multi user environments a newly created browser gets created |
| // on the same desktop as the browser is shown on. |
| IN_PROC_BROWSER_TEST_F(BrowserGuestSessionNavigatorTest, |
| Browser_Gets_Created_On_Visiting_Desktop) { |
| // Test 1: Test that a browser created from a visiting browser will be on the |
| // same visiting desktop. |
| { |
| const AccountId desktop_account_id( |
| AccountId::FromUserEmail("desktop_user_id@fake.com")); |
| TestMultiUserWindowManager* window_manager = |
| TestMultiUserWindowManager::Create(browser(), desktop_account_id); |
| |
| EXPECT_EQ(1u, chrome::GetTotalBrowserCount()); |
| |
| // Navigate to the settings page. |
| NavigateParams params(MakeNavigateParams(browser())); |
| params.disposition = WindowOpenDisposition::NEW_POPUP; |
| params.url = GURL("chrome://settings"); |
| params.window_action = NavigateParams::SHOW_WINDOW; |
| params.path_behavior = NavigateParams::IGNORE_AND_NAVIGATE; |
| params.browser = browser(); |
| Navigate(¶ms); |
| |
| EXPECT_EQ(2u, chrome::GetTotalBrowserCount()); |
| |
| aura::Window* created_window = window_manager->created_window(); |
| ASSERT_TRUE(created_window); |
| EXPECT_TRUE( |
| MultiUserWindowManagerHelper::GetInstance()->IsWindowOnDesktopOfUser( |
| created_window, desktop_account_id)); |
| } |
| // Test 2: Test that a window which is not visiting does not cause an owner |
| // assignment of a newly created browser. |
| { |
| const AccountId browser_owner = |
| multi_user_util::GetAccountIdFromProfile(browser()->profile()); |
| TestMultiUserWindowManager* window_manager = |
| TestMultiUserWindowManager::Create(browser(), browser_owner); |
| |
| // Navigate to the settings page. |
| NavigateParams params(MakeNavigateParams(browser())); |
| params.disposition = WindowOpenDisposition::NEW_POPUP; |
| params.url = GURL("chrome://settings"); |
| params.window_action = NavigateParams::SHOW_WINDOW; |
| params.path_behavior = NavigateParams::IGNORE_AND_NAVIGATE; |
| params.browser = browser(); |
| Navigate(¶ms); |
| |
| EXPECT_EQ(3u, chrome::GetTotalBrowserCount()); |
| |
| // The ShowWindowForUser should not have been called since the window is |
| // already on the correct desktop. |
| ASSERT_FALSE(window_manager->created_window()); |
| } |
| } |
| |
| #endif // BUILDFLAG(IS_CHROMEOS_ASH) |
| |
| #if BUILDFLAG(IS_CHROMEOS_LACROS) |
| // Verifies that the navigation is trying to open the os:// scheme page in |
| // Ash, will fail and then open it as chrome:// in Lacros to show a 404 error. |
| IN_PROC_BROWSER_TEST_F(BrowserNavigatorTestChromeOS, OsSchemeRedirectFail) { |
| EXPECT_EQ(1u, chrome::GetTotalBrowserCount()); |
| EXPECT_EQ(1, browser()->tab_strip_model()->count()); |
| |
| // Navigate to an unknown page with an os:// scheme. |
| NavigateParams params(MakeNavigateParams(browser())); |
| params.disposition = WindowOpenDisposition::SINGLETON_TAB; |
| params.url = GURL("os://foobar"); |
| params.window_action = NavigateParams::SHOW_WINDOW; |
| params.path_behavior = NavigateParams::IGNORE_AND_NAVIGATE; |
| Navigate(¶ms); |
| |
| // A new 404 page should be shown in the browser. |
| EXPECT_EQ(browser(), params.browser); |
| EXPECT_EQ(2, browser()->tab_strip_model()->count()); |
| EXPECT_EQ(GURL("chrome://foobar"), |
| browser()->tab_strip_model()->GetActiveWebContents()->GetURL()); |
| } |
| |
| // Verifies that the navigation of an os:// scheme page is opening an app on |
| // the ash side and does not produce a navigation on the Lacros side. |
| IN_PROC_BROWSER_TEST_F(BrowserNavigatorTestChromeOS, OsSchemeRedirectSucceed) { |
| if (chromeos::LacrosService::Get()->GetInterfaceVersion( |
| crosapi::mojom::TestController::Uuid_) < |
| static_cast<int>(crosapi::mojom::TestController::MethodMinVersions:: |
| kGetOpenAshBrowserWindowsMinVersion)) { |
| LOG(WARNING) << "Unsupported ash version."; |
| return; |
| } |
| |
| crosapi::mojom::TestControllerAsyncWaiter waiter( |
| chromeos::LacrosService::Get() |
| ->GetRemote<crosapi::mojom::TestController>() |
| .get()); |
| |
| // Ash shouldn't have a browser window open by now. |
| uint32_t number = 1; |
| waiter.GetOpenAshBrowserWindows(&number); |
| EXPECT_EQ(0u, number); |
| |
| // First we make sure that the GURL we are interested in is in our allow list. |
| auto init_params = crosapi::mojom::BrowserInitParams::New(); |
| init_params->accepted_internal_ash_urls = |
| std::vector<GURL>{GURL(chrome::kOsUIFlagsURL)}; |
| chromeos::BrowserInitParams::SetInitParamsForTests(std::move(init_params)); |
| |
| EXPECT_EQ(1u, chrome::GetTotalBrowserCount()); |
| EXPECT_EQ(1, browser()->tab_strip_model()->count()); |
| GURL url_before_navigation = |
| browser()->tab_strip_model()->GetActiveWebContents()->GetURL(); |
| |
| // Navigate to a known Ash page. |
| NavigateParams params(MakeNavigateParams(browser())); |
| params.disposition = WindowOpenDisposition::SINGLETON_TAB; |
| params.url = GURL(chrome::kOsUIFlagsURL); |
| params.window_action = NavigateParams::SHOW_WINDOW; |
| params.path_behavior = NavigateParams::IGNORE_AND_NAVIGATE; |
| params.transition = ui::PageTransitionFromInt( |
| ui::PAGE_TRANSITION_TYPED | ui::PAGE_TRANSITION_FROM_ADDRESS_BAR); |
| Navigate(¶ms); |
| |
| // No change should have happened on the Lacros side. |
| EXPECT_EQ(browser(), params.browser); |
| EXPECT_EQ(1, browser()->tab_strip_model()->count()); |
| EXPECT_EQ(url_before_navigation, |
| browser()->tab_strip_model()->GetActiveWebContents()->GetURL()); |
| |
| // Clean up the window we have created. |
| |
| // Wait until we have the app running. |
| while (0 == number) { |
| usleep(25000); |
| waiter.GetOpenAshBrowserWindows(&number); |
| } |
| |
| // Close it. |
| bool success = false; |
| waiter.CloseAllBrowserWindows(&success); |
| EXPECT_TRUE(success); |
| |
| // Wait until all are gone. |
| while (0 != number) { |
| usleep(25000); |
| waiter.GetOpenAshBrowserWindows(&number); |
| } |
| } |
| |
| #endif |
| |
| } // namespace |