OLD | NEW |
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2013 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "base/command_line.h" | 5 #include "base/command_line.h" |
6 #include "base/path_service.h" | 6 #include "base/path_service.h" |
7 #include "base/stringprintf.h" | |
8 #include "chrome/browser/browser_process.h" | |
9 #include "chrome/browser/chrome_browser_main.h" | 7 #include "chrome/browser/chrome_browser_main.h" |
10 #include "chrome/browser/chrome_browser_main_extra_parts.h" | 8 #include "chrome/browser/chrome_browser_main_extra_parts.h" |
11 #include "chrome/browser/chrome_content_browser_client.h" | 9 #include "chrome/browser/chrome_content_browser_client.h" |
12 #include "chrome/browser/chromeos/app_mode/kiosk_app_launch_error.h" | |
13 #include "chrome/browser/chromeos/app_mode/kiosk_app_manager.h" | |
14 #include "chrome/browser/chromeos/cros/cros_in_process_browser_test.h" | 10 #include "chrome/browser/chromeos/cros/cros_in_process_browser_test.h" |
15 #include "chrome/browser/chromeos/login/existing_user_controller.h" | 11 #include "chrome/browser/chromeos/login/existing_user_controller.h" |
16 #include "chrome/browser/chromeos/login/webui_login_display.h" | 12 #include "chrome/browser/chromeos/login/webui_login_display.h" |
17 #include "chrome/browser/chromeos/login/webui_login_display_host.h" | |
18 #include "chrome/browser/chromeos/login/wizard_controller.h" | 13 #include "chrome/browser/chromeos/login/wizard_controller.h" |
19 #include "chrome/browser/extensions/extension_service.h" | |
20 #include "chrome/browser/extensions/extension_system.h" | |
21 #include "chrome/browser/google_apis/test_server/http_request.h" | 14 #include "chrome/browser/google_apis/test_server/http_request.h" |
22 #include "chrome/browser/google_apis/test_server/http_response.h" | 15 #include "chrome/browser/google_apis/test_server/http_response.h" |
23 #include "chrome/browser/google_apis/test_server/http_server.h" | 16 #include "chrome/browser/google_apis/test_server/http_server.h" |
24 #include "chrome/browser/lifetime/application_lifetime.h" | |
25 #include "chrome/browser/prefs/scoped_user_pref_update.h" | |
26 #include "chrome/browser/profiles/profile_manager.h" | |
27 #include "chrome/browser/ui/browser.h" | 17 #include "chrome/browser/ui/browser.h" |
28 #include "chrome/browser/ui/webui/chromeos/login/oobe_ui.h" | |
29 #include "chrome/common/chrome_notification_types.h" | 18 #include "chrome/common/chrome_notification_types.h" |
30 #include "chrome/common/chrome_paths.h" | 19 #include "chrome/common/chrome_paths.h" |
31 #include "chrome/common/chrome_switches.h" | 20 #include "chrome/common/chrome_switches.h" |
32 #include "chrome/common/extensions/extension.h" | |
33 #include "chrome/test/base/in_process_browser_test.h" | 21 #include "chrome/test/base/in_process_browser_test.h" |
34 #include "chrome/test/base/interactive_test_utils.h" | 22 #include "chrome/test/base/interactive_test_utils.h" |
35 #include "chrome/test/base/ui_test_utils.h" | 23 #include "chrome/test/base/ui_test_utils.h" |
36 #include "chromeos/chromeos_switches.h" | 24 #include "chromeos/chromeos_switches.h" |
37 #include "content/public/browser/notification_observer.h" | 25 #include "content/public/browser/notification_observer.h" |
38 #include "content/public/browser/notification_registrar.h" | 26 #include "content/public/browser/notification_registrar.h" |
39 #include "content/public/browser/notification_service.h" | 27 #include "content/public/browser/notification_service.h" |
40 #include "content/public/test/test_utils.h" | 28 #include "content/public/test/test_utils.h" |
41 #include "google_apis/gaia/gaia_switches.h" | 29 #include "google_apis/gaia/gaia_switches.h" |
42 #include "net/base/host_port_pair.h" | |
43 #include "net/dns/mock_host_resolver.h" | |
44 #include "testing/gmock/include/gmock/gmock.h" | 30 #include "testing/gmock/include/gmock/gmock.h" |
45 #include "testing/gtest/include/gtest/gtest.h" | 31 #include "testing/gtest/include/gtest/gtest.h" |
46 | 32 |
47 using namespace google_apis; | 33 using namespace google_apis; |
48 using namespace google_apis::test_server; | 34 using namespace google_apis::test_server; |
49 | 35 |
50 namespace chromeos { | |
51 | |
52 namespace { | 36 namespace { |
53 | 37 |
54 const char kWebstoreDomain[] = "cws.com"; | |
55 | |
56 // Webstore data json is in | |
57 // chrome/test/data/chromeos/app_mode/webstore/inlineinstall/ | |
58 // detail/ggbflgnkafappblpkiflbgpmkfdpnhhe | |
59 const char kTestKioskApp[] = "ggbflgnkafappblpkiflbgpmkfdpnhhe"; | |
60 | |
61 // Used to add an observer to NotificationService after it's created. | 38 // Used to add an observer to NotificationService after it's created. |
62 class TestBrowserMainExtraParts | 39 class TestBrowserMainExtraParts |
63 : public ChromeBrowserMainExtraParts, | 40 : public ChromeBrowserMainExtraParts, |
64 public content::NotificationObserver { | 41 public content::NotificationObserver { |
65 public: | 42 public: |
66 TestBrowserMainExtraParts() {} | 43 TestBrowserMainExtraParts() {} |
67 | |
68 virtual ~TestBrowserMainExtraParts() {} | 44 virtual ~TestBrowserMainExtraParts() {} |
69 | 45 |
70 // ChromeBrowserMainExtraParts implementation. | 46 // ChromeBrowserMainExtraParts implementation. |
71 virtual void PreEarlyInitialization() OVERRIDE { | 47 virtual void PreEarlyInitialization() OVERRIDE { |
72 registrar_.Add(this, chrome::NOTIFICATION_LOGIN_WEBUI_VISIBLE, | 48 registrar_.Add(this, chrome::NOTIFICATION_LOGIN_WEBUI_VISIBLE, |
73 content::NotificationService::AllSources()); | 49 content::NotificationService::AllSources()); |
74 registrar_.Add(this, chrome::NOTIFICATION_KIOSK_APPS_LOADED, | 50 registrar_.Add(this, chrome::NOTIFICATION_SESSION_STARTED, |
75 content::NotificationService::AllSources()); | |
76 registrar_.Add(this, chrome::NOTIFICATION_KIOSK_APP_LAUNCHED, | |
77 content::NotificationService::AllSources()); | 51 content::NotificationService::AllSources()); |
78 } | 52 } |
79 | 53 |
80 void set_quit_task(const base::Closure& quit_task) { quit_task_ = quit_task; } | 54 void set_quit_task(const base::Closure& quit_task) { quit_task_ = quit_task; } |
| 55 void set_gaia_url(const std::string& url) { gaia_url_ = url; } |
81 | 56 |
82 private: | 57 private: |
83 // Overridden from content::NotificationObserver: | 58 // Overridden from content::NotificationObserver: |
84 virtual void Observe(int type, | 59 virtual void Observe(int type, |
85 const content::NotificationSource& source, | 60 const content::NotificationSource& source, |
86 const content::NotificationDetails& details) OVERRIDE { | 61 const content::NotificationDetails& details) OVERRIDE { |
87 if (type == chrome::NOTIFICATION_LOGIN_WEBUI_VISIBLE) { | 62 if (type == chrome::NOTIFICATION_LOGIN_WEBUI_VISIBLE) { |
88 LOG(INFO) << "NOTIFICATION_LOGIN_WEBUI_VISIBLE"; | 63 LOG(INFO) << "NOTIFICATION_LOGIN_WEBUI_VISIBLE"; |
89 } else if (type == chrome::NOTIFICATION_KIOSK_APPS_LOADED) { | 64 chromeos::ExistingUserController* controller = |
90 LOG(INFO) << "chrome::NOTIFICATION_KIOSK_APPS_LOADED"; | 65 chromeos::ExistingUserController::current_controller(); |
91 content::WebUI* web_ui = static_cast<chromeos::WebUILoginDisplayHost*>( | 66 CHECK(controller); |
92 chromeos::WebUILoginDisplayHost::default_host())-> | 67 chromeos::WebUILoginDisplay* webui_login_display = |
93 GetOobeUI()->web_ui(); | 68 static_cast<chromeos::WebUILoginDisplay*>( |
94 web_ui->CallJavascriptFunction("login.AppsMenuButton.runAppForTesting", | 69 controller->login_display()); |
95 base::StringValue(kTestKioskApp)); | 70 CHECK(webui_login_display); |
96 } else if (type == chrome::NOTIFICATION_KIOSK_APP_LAUNCHED) { | 71 webui_login_display->SetGaiaOriginForTesting(gaia_url_); |
97 LOG(INFO) << "chrome::NOTIFICATION_KIOSK_APP_LAUNCHED"; | 72 webui_login_display->ShowSigninScreenForCreds("username", "password"); |
98 registrar_.Add(this, content::NOTIFICATION_RENDERER_PROCESS_CLOSED, | 73 // TODO(glotov): mock GAIA server (test_server_) should support |
99 content::NotificationService::AllSources()); | 74 // username/password configuration. |
100 quit_task_.Run(); | 75 } else if (type == chrome::NOTIFICATION_SESSION_STARTED) { |
101 } else if (type == content::NOTIFICATION_RENDERER_PROCESS_CLOSED) { | 76 LOG(INFO) << "chrome::NOTIFICATION_SESSION_STARTED"; |
102 LOG(INFO) << "content::NOTIFICATION_RENDERER_PROCESS_CLOSED"; | |
103 quit_task_.Run(); | 77 quit_task_.Run(); |
104 } else { | 78 } else { |
105 NOTREACHED(); | 79 NOTREACHED(); |
106 } | 80 } |
107 } | 81 } |
108 | 82 |
109 content::NotificationRegistrar registrar_; | 83 content::NotificationRegistrar registrar_; |
110 base::Closure quit_task_; | 84 base::Closure quit_task_; |
| 85 std::string gaia_url_; |
111 | 86 |
112 DISALLOW_COPY_AND_ASSIGN(TestBrowserMainExtraParts); | 87 DISALLOW_COPY_AND_ASSIGN(TestBrowserMainExtraParts); |
113 }; | 88 }; |
114 | 89 |
115 class TestContentBrowserClient : public chrome::ChromeContentBrowserClient { | 90 class TestContentBrowserClient : public chrome::ChromeContentBrowserClient { |
116 public: | 91 public: |
117 TestContentBrowserClient() {} | 92 TestContentBrowserClient() {} |
118 virtual ~TestContentBrowserClient() {} | 93 virtual ~TestContentBrowserClient() {} |
119 | 94 |
120 virtual content::BrowserMainParts* CreateBrowserMainParts( | 95 virtual content::BrowserMainParts* CreateBrowserMainParts( |
121 const content::MainFunctionParams& parameters) OVERRIDE { | 96 const content::MainFunctionParams& parameters) OVERRIDE { |
122 ChromeBrowserMainParts* main_parts = static_cast<ChromeBrowserMainParts*>( | 97 ChromeBrowserMainParts* main_parts = static_cast<ChromeBrowserMainParts*>( |
123 ChromeContentBrowserClient::CreateBrowserMainParts(parameters)); | 98 ChromeContentBrowserClient::CreateBrowserMainParts(parameters)); |
124 | 99 |
125 browser_main_extra_parts_ = new TestBrowserMainExtraParts(); | 100 browser_main_extra_parts_ = new TestBrowserMainExtraParts(); |
126 main_parts->AddParts(browser_main_extra_parts_); | 101 main_parts->AddParts(browser_main_extra_parts_); |
127 return main_parts; | 102 return main_parts; |
128 } | 103 } |
129 | 104 |
130 TestBrowserMainExtraParts* browser_main_extra_parts_; | 105 TestBrowserMainExtraParts* browser_main_extra_parts_; |
131 | 106 |
132 private: | 107 private: |
133 DISALLOW_COPY_AND_ASSIGN(TestContentBrowserClient); | 108 DISALLOW_COPY_AND_ASSIGN(TestContentBrowserClient); |
134 }; | 109 }; |
135 | 110 |
136 } // namespace | 111 const base::FilePath kServiceLogin("chromeos/service_login.html"); |
137 | 112 |
138 class KioskTest : public chromeos::CrosInProcessBrowserTest { | 113 class OobeTest : public chromeos::CrosInProcessBrowserTest { |
139 public: | |
140 KioskTest() : chromeos::CrosInProcessBrowserTest() { | |
141 SetExitWhenLastBrowserCloses(false); | |
142 } | |
143 | |
144 protected: | 114 protected: |
145 | |
146 virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { | 115 virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { |
147 command_line->AppendSwitch(chromeos::switches::kLoginManager); | 116 command_line->AppendSwitch(chromeos::switches::kLoginManager); |
148 command_line->AppendSwitch(chromeos::switches::kForceLoginManagerInTests); | 117 command_line->AppendSwitch(chromeos::switches::kForceLoginManagerInTests); |
149 command_line->AppendSwitch(::switches::kDisableChromeCaptivePortalDetector); | 118 command_line->AppendSwitch(switches::kDisableChromeCaptivePortalDetector); |
150 command_line->AppendSwitch(::switches::kDisableBackgroundNetworking); | |
151 command_line->AppendSwitchASCII(chromeos::switches::kLoginProfile, "user"); | 119 command_line->AppendSwitchASCII(chromeos::switches::kLoginProfile, "user"); |
152 | 120 command_line->AppendSwitchASCII(switches::kAuthExtensionPath, "gaia_auth"); |
153 ASSERT_TRUE(test_server()->Start()); | |
154 net::HostPortPair host_port = test_server()->host_port_pair(); | |
155 std::string test_gallery_url = base::StringPrintf( | |
156 "http://%s:%d/files/chromeos/app_mode/webstore", | |
157 kWebstoreDomain, host_port.port()); | |
158 command_line->AppendSwitchASCII( | |
159 ::switches::kAppsGalleryURL, test_gallery_url); | |
160 | |
161 std::string test_gallery_download_url = test_gallery_url; | |
162 test_gallery_download_url.append("/downloads/%s.crx"); | |
163 command_line->AppendSwitchASCII( | |
164 ::switches::kAppsGalleryDownloadURL, test_gallery_download_url); | |
165 } | 121 } |
166 | 122 |
167 virtual void SetUpInProcessBrowserTestFixture() OVERRIDE { | 123 virtual void SetUpInProcessBrowserTestFixture() OVERRIDE { |
168 content_browser_client_.reset(new TestContentBrowserClient()); | 124 content_browser_client_.reset(new TestContentBrowserClient()); |
169 original_content_browser_client_ = content::SetBrowserClientForTesting( | 125 original_content_browser_client_ = content::SetBrowserClientForTesting( |
170 content_browser_client_.get()); | 126 content_browser_client_.get()); |
171 host_resolver()->AddRule(kWebstoreDomain, "127.0.0.1"); | 127 base::FilePath test_data_dir; |
| 128 PathService::Get(chrome::DIR_TEST_DATA, &test_data_dir); |
| 129 CHECK(file_util::ReadFileToString(test_data_dir.Append(kServiceLogin), |
| 130 &service_login_response_)); |
| 131 } |
| 132 |
| 133 virtual void SetUpOnMainThread() OVERRIDE { |
| 134 test_server_ = new HttpServer(); // Constructor wants UI thread. |
| 135 CHECK(test_server_->InitializeAndWaitUntilReady()); |
| 136 test_server_->RegisterRequestHandler( |
| 137 base::Bind(&OobeTest::HandleRequest, base::Unretained(this))); |
| 138 LOG(INFO) << "Set up http server at " << test_server_->base_url(); |
| 139 CHECK(test_server_->port() >= 8040 && test_server_->port() < 8045) |
| 140 << "Current manifest_test.json for gaia_login restrictions " |
| 141 << "does not allow this port"; |
| 142 |
| 143 const std::string gaia_url = |
| 144 "http://localhost:" + test_server_->base_url().port(); |
| 145 content_browser_client_->browser_main_extra_parts_->set_gaia_url(gaia_url); |
172 } | 146 } |
173 | 147 |
174 virtual void CleanUpOnMainThread() OVERRIDE { | 148 virtual void CleanUpOnMainThread() OVERRIDE { |
175 // Clean up while main thread still runs. | 149 LOG(INFO) << "Stopping the http server."; |
176 // See http://crbug.com/176659. | 150 test_server_->ShutdownAndWaitUntilComplete(); |
177 chromeos::KioskAppManager::Get()->CleanUp(); | 151 delete test_server_; // Destructor wants UI thread. |
178 } | 152 } |
179 | 153 |
180 void ReloadKioskApps() { | 154 scoped_ptr<HttpResponse> HandleRequest(const HttpRequest& request) { |
181 chromeos::KioskAppManager::Get()->AddApp(kTestKioskApp); | 155 GURL url = test_server_->GetURL(request.relative_url); |
| 156 LOG(INFO) << "Http request: " << url.spec(); |
| 157 |
| 158 scoped_ptr<HttpResponse> http_response(new HttpResponse()); |
| 159 if (url.path() == "/ServiceLogin") { |
| 160 http_response->set_code(test_server::SUCCESS); |
| 161 http_response->set_content(service_login_response_); |
| 162 http_response->set_content_type("text/html"); |
| 163 } else if (url.path() == "/ServiceLoginAuth") { |
| 164 LOG(INFO) << "Params: " << request.content; |
| 165 static const char kContinueParam[] = "continue="; |
| 166 int continue_arg_begin = request.content.find(kContinueParam) + |
| 167 arraysize(kContinueParam) - 1; |
| 168 int continue_arg_end = request.content.find("&", continue_arg_begin); |
| 169 const std::string continue_url = request.content.substr( |
| 170 continue_arg_begin, continue_arg_end - continue_arg_begin); |
| 171 http_response->set_code(test_server::SUCCESS); |
| 172 const std::string redirect_js = |
| 173 "document.location.href = unescape('" + continue_url + "');"; |
| 174 http_response->set_content( |
| 175 "<HTML><HEAD><SCRIPT>\n" + redirect_js + "\n</SCRIPT></HEAD></HTML>"); |
| 176 http_response->set_content_type("text/html"); |
| 177 } else { |
| 178 NOTREACHED() << url.path(); |
| 179 } |
| 180 return http_response.Pass(); |
182 } | 181 } |
183 | 182 |
184 scoped_ptr<TestContentBrowserClient> content_browser_client_; | 183 scoped_ptr<TestContentBrowserClient> content_browser_client_; |
185 content::ContentBrowserClient* original_content_browser_client_; | 184 content::ContentBrowserClient* original_content_browser_client_; |
186 std::string service_login_response_; | 185 std::string service_login_response_; |
| 186 HttpServer* test_server_; // cant use scoped_ptr because destructor |
| 187 // needs UI thread. |
187 }; | 188 }; |
188 | 189 |
189 IN_PROC_BROWSER_TEST_F(KioskTest, InstallAndLaunchApp) { | 190 IN_PROC_BROWSER_TEST_F(OobeTest, NewUser) { |
190 // Start UI, find menu entry for this app and launch it. | |
191 chromeos::WizardController::SkipPostLoginScreensForTesting(); | 191 chromeos::WizardController::SkipPostLoginScreensForTesting(); |
192 chromeos::WizardController* wizard_controller = | 192 chromeos::WizardController* wizard_controller = |
193 chromeos::WizardController::default_controller(); | 193 chromeos::WizardController::default_controller(); |
194 CHECK(wizard_controller); | 194 CHECK(wizard_controller); |
195 wizard_controller->SkipToLoginForTesting(); | 195 wizard_controller->SkipToLoginForTesting(); |
196 | 196 |
197 ReloadKioskApps(); | |
198 | |
199 // The first loop exits after we receive NOTIFICATION_KIOSK_APP_LAUNCHED | |
200 // notification - right at app launch. | |
201 scoped_refptr<content::MessageLoopRunner> runner = | 197 scoped_refptr<content::MessageLoopRunner> runner = |
202 new content::MessageLoopRunner; | 198 new content::MessageLoopRunner; |
203 content_browser_client_->browser_main_extra_parts_->set_quit_task( | 199 content_browser_client_->browser_main_extra_parts_->set_quit_task( |
204 runner->QuitClosure()); | 200 runner->QuitClosure()); |
205 runner->Run(); | 201 runner->Run(); |
206 | |
207 // Check installer status. | |
208 EXPECT_EQ(chromeos::KioskAppLaunchError::NONE, | |
209 chromeos::KioskAppLaunchError::Get()); | |
210 | |
211 // Check if the kiosk webapp is really installed for the default profile. | |
212 ASSERT_TRUE(ProfileManager::GetDefaultProfile()); | |
213 const extensions::Extension* app = | |
214 extensions::ExtensionSystem::Get(ProfileManager::GetDefaultProfile())-> | |
215 extension_service()->GetInstalledExtension(kTestKioskApp); | |
216 EXPECT_TRUE(app); | |
217 | |
218 // The second loop exits when kiosk app terminates and we receive | |
219 // NOTIFICATION_RENDERER_PROCESS_CLOSED. | |
220 scoped_refptr<content::MessageLoopRunner> runner2 = | |
221 new content::MessageLoopRunner; | |
222 content_browser_client_->browser_main_extra_parts_->set_quit_task( | |
223 runner2->QuitClosure()); | |
224 runner2->Run(); | |
225 } | 202 } |
226 | 203 |
227 } // namespace chromeos | 204 } |
OLD | NEW |