// Copyright (c) 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 "chrome/browser/automation/testing_automation_provider.h" #include "ash/new_window_delegate.h" #include "ash/shell.h" #include "ash/system/tray/system_tray_delegate.h" #include "base/command_line.h" #include "base/i18n/time_formatting.h" #include "base/prefs/pref_service.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/time/time.h" #include "chrome/browser/automation/automation_provider_json.h" #include "chrome/browser/automation/automation_provider_observers.h" #include "chrome/browser/automation/automation_util.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/chromeos/accessibility/accessibility_manager.h" #include "chrome/browser/chromeos/accessibility/accessibility_util.h" #include "chrome/browser/chromeos/login/default_user_images.h" #include "chrome/browser/chromeos/login/enrollment/enrollment_screen.h" #include "chrome/browser/chromeos/login/existing_user_controller.h" #include "chrome/browser/chromeos/login/login_display.h" #include "chrome/browser/chromeos/login/login_display_host_impl.h" #include "chrome/browser/chromeos/login/screen_locker.h" #include "chrome/browser/chromeos/login/screens/eula_screen.h" #include "chrome/browser/chromeos/login/screens/network_screen.h" #include "chrome/browser/chromeos/login/screens/update_screen.h" #include "chrome/browser/chromeos/login/screens/user_image_screen.h" #include "chrome/browser/chromeos/login/startup_utils.h" #include "chrome/browser/chromeos/login/webui_login_display.h" #include "chrome/browser/chromeos/login/wizard_controller.h" #include "chrome/browser/chromeos/net/proxy_config_handler.h" #include "chrome/browser/chromeos/settings/cros_settings.h" #include "chrome/browser/prefs/proxy_config_dictionary.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_window.h" #include "chrome/browser/ui/webui/chromeos/login/oobe_ui.h" #include "chrome/browser/ui/webui/chromeos/login/signin_screen_handler.h" #include "chrome/common/pref_names.h" #include "chromeos/audio/cras_audio_handler.h" #include "chromeos/dbus/dbus_thread_manager.h" #include "chromeos/dbus/session_manager_client.h" #include "chromeos/dbus/update_engine_client.h" #include "chromeos/settings/cros_settings_names.h" #include "chromeos/settings/timezone_settings.h" #include "content/public/browser/web_contents.h" #include "policy/policy_constants.h" #include "ui/views/widget/widget.h" using chromeos::DBusThreadManager; using chromeos::ExistingUserController; using chromeos::UpdateEngineClient; using chromeos::User; using chromeos::UserManager; using chromeos::WizardController; namespace { void UpdateCheckCallback(AutomationJSONReply* reply, UpdateEngineClient::UpdateCheckResult result) { if (result == UpdateEngineClient::UPDATE_RESULT_SUCCESS) reply->SendSuccess(NULL); else reply->SendError("update check failed"); delete reply; } } // namespace #if defined(OS_CHROMEOS) void TestingAutomationProvider::PowerChanged( const power_manager::PowerSupplyProperties& proto) { power_supply_properties_ = proto; } #endif void TestingAutomationProvider::AcceptOOBENetworkScreen( base::DictionaryValue* args, IPC::Message* reply_message) { WizardController* wizard_controller = WizardController::default_controller(); if (!wizard_controller || wizard_controller->current_screen()->GetName() != WizardController::kNetworkScreenName) { AutomationJSONReply(this, reply_message).SendError( "Network screen not active."); return; } // Observer will delete itself. new WizardControllerObserver(wizard_controller, this, reply_message); wizard_controller->GetNetworkScreen()->OnContinuePressed(); } void TestingAutomationProvider::AcceptOOBEEula(base::DictionaryValue* args, IPC::Message* reply_message) { bool accepted; bool usage_stats_reporting; if (!args->GetBoolean("accepted", &accepted) || !args->GetBoolean("usage_stats_reporting", &usage_stats_reporting)) { AutomationJSONReply(this, reply_message).SendError( "Invalid or missing args."); return; } WizardController* wizard_controller = WizardController::default_controller(); if (!wizard_controller || wizard_controller->current_screen()->GetName() != WizardController::kEulaScreenName) { AutomationJSONReply(this, reply_message).SendError( "EULA screen not active."); return; } // Observer will delete itself. new WizardControllerObserver(wizard_controller, this, reply_message); wizard_controller->GetEulaScreen()->OnExit(accepted, usage_stats_reporting); } void TestingAutomationProvider::CancelOOBEUpdate(base::DictionaryValue* args, IPC::Message* reply_message) { if (chromeos::StartupUtils::IsOobeCompleted()) { // Update already finished. scoped_ptr return_value(new base::DictionaryValue); return_value->SetString("next_screen", WizardController::kLoginScreenName); AutomationJSONReply(this, reply_message).SendSuccess(return_value.get()); return; } WizardController* wizard_controller = WizardController::default_controller(); if (!wizard_controller || wizard_controller->current_screen()->GetName() != WizardController::kUpdateScreenName) { AutomationJSONReply(this, reply_message).SendError( "Update screen not active."); return; } // Observer will delete itself. new WizardControllerObserver(wizard_controller, this, reply_message); wizard_controller->GetUpdateScreen()->CancelUpdate(); } void TestingAutomationProvider::GetLoginInfo(base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); scoped_ptr return_value(new base::DictionaryValue); const UserManager* user_manager = UserManager::Get(); if (!user_manager) reply.SendError("No user manager!"); const chromeos::ScreenLocker* screen_locker = chromeos::ScreenLocker::default_screen_locker(); return_value->SetString("login_ui_type", "webui"); return_value->SetBoolean("is_owner", user_manager->IsCurrentUserOwner()); return_value->SetBoolean("is_logged_in", user_manager->IsUserLoggedIn()); return_value->SetBoolean("is_screen_locked", screen_locker); if (user_manager->IsUserLoggedIn()) { const User* user = user_manager->GetLoggedInUser(); return_value->SetBoolean("is_guest", user_manager->IsLoggedInAsGuest()); return_value->SetString("email", user->email()); return_value->SetString("display_email", user->display_email()); switch (user->image_index()) { case User::kExternalImageIndex: return_value->SetString("user_image", "file"); break; case User::kProfileImageIndex: return_value->SetString("user_image", "profile"); break; default: return_value->SetInteger("user_image", user->image_index()); break; } } reply.SendSuccess(return_value.get()); } // See the note under LoginAsGuest(). CreateAccount() causes a login as guest. void TestingAutomationProvider::ShowCreateAccountUI( base::DictionaryValue* args, IPC::Message* reply_message) { ExistingUserController* controller = ExistingUserController::current_controller(); // Return immediately, since we're going to die before the login is finished. AutomationJSONReply(this, reply_message).SendSuccess(NULL); controller->CreateAccount(); } // Logging in as guest will cause session_manager to restart Chrome with new // flags. If you used EnableChromeTesting, you will have to call it again. void TestingAutomationProvider::LoginAsGuest(base::DictionaryValue* args, IPC::Message* reply_message) { ExistingUserController* controller = ExistingUserController::current_controller(); // Return immediately, since we're going to die before the login is finished. AutomationJSONReply(this, reply_message).SendSuccess(NULL); controller->LoginAsGuest(); } void TestingAutomationProvider::SubmitLoginForm(base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); std::string username, password; if (!args->GetString("username", &username) || !args->GetString("password", &password)) { reply.SendError("Invalid or missing args."); return; } chromeos::ExistingUserController* controller = chromeos::ExistingUserController::current_controller(); if (!controller) { reply.SendError("Unable to access ExistingUserController"); return; } // WebUI login. chromeos::WebUILoginDisplay* webui_login_display = static_cast(controller->login_display()); VLOG(2) << "TestingAutomationProvider::SubmitLoginForm " << "ShowSigninScreenForCreds(" << username << ", " << password << ")"; webui_login_display->ShowSigninScreenForCreds(username, password); reply.SendSuccess(NULL); } void TestingAutomationProvider::AddLoginEventObserver( base::DictionaryValue* args, IPC::Message* reply_message) { ExistingUserController* controller = ExistingUserController::current_controller(); AutomationJSONReply reply(this, reply_message); if (!controller) { // This may happen due to SkipToLogin not being called. reply.SendError("Unable to access ExistingUserController"); return; } if (!automation_event_queue_.get()) automation_event_queue_.reset(new AutomationEventQueue); int observer_id = automation_event_queue_->AddObserver( new LoginEventObserver(automation_event_queue_.get(), this)); // Return the observer's id. base::DictionaryValue return_value; return_value.SetInteger("observer_id", observer_id); reply.SendSuccess(&return_value); } void TestingAutomationProvider::SignOut(base::DictionaryValue* args, IPC::Message* reply_message) { ash::Shell::GetInstance()->system_tray_delegate()->SignOut(); // Sign out has the side effect of restarting the session_manager // and chrome, thereby severing the automation channel, so it's // not really necessary to send a reply back. The next line is // for consistency with other methods. AutomationJSONReply(this, reply_message).SendSuccess(NULL); } void TestingAutomationProvider::PickUserImage(base::DictionaryValue* args, IPC::Message* reply_message) { std::string image_type; int image_number = -1; if (!args->GetString("image", &image_type) && !args->GetInteger("image", &image_number)) { AutomationJSONReply(this, reply_message).SendError( "Invalid or missing args."); return; } WizardController* wizard_controller = WizardController::default_controller(); if (!wizard_controller || wizard_controller->current_screen()->GetName() != WizardController::kUserImageScreenName) { AutomationJSONReply(this, reply_message).SendError( "User image screen not active."); return; } chromeos::UserImageScreen* image_screen = wizard_controller->GetUserImageScreen(); // Observer will delete itself unless error is returned. WizardControllerObserver* observer = new WizardControllerObserver(wizard_controller, this, reply_message); if (image_type == "profile") { image_screen->OnImageSelected("", image_type, true); image_screen->OnImageAccepted(); } else if (image_type.empty() && image_number >= 0 && image_number < chromeos::kDefaultImagesCount) { image_screen->OnImageSelected( chromeos::GetDefaultImageUrl(image_number), image_type, true); image_screen->OnImageAccepted(); } else { AutomationJSONReply(this, reply_message).SendError( "Invalid or missing args."); delete observer; return; } } void TestingAutomationProvider::SkipToLogin(base::DictionaryValue* args, IPC::Message* reply_message) { bool skip_post_login_screens; // The argument name is a legacy. If set to |true|, this argument causes any // screens that may otherwise be shown after login (registration, Terms of // Service, user image selection) to be skipped. if (!args->GetBoolean("skip_image_selection", &skip_post_login_screens)) { AutomationJSONReply reply(this, reply_message); reply.SendError("Invalid or missing args."); return; } if (skip_post_login_screens) WizardController::SkipPostLoginScreensForTesting(); WizardController* wizard_controller = WizardController::default_controller(); if (!wizard_controller) { AutomationJSONReply reply(this, reply_message); if (ExistingUserController::current_controller()) { // Already at login screen. scoped_ptr return_value(new base::DictionaryValue); return_value->SetString("next_screen", WizardController::kLoginScreenName); reply.SendSuccess(return_value.get()); } else { reply.SendError("OOBE not active."); } return; } // Observer will delete itself. WizardControllerObserver* observer = new WizardControllerObserver(wizard_controller, this, reply_message); observer->set_screen_to_wait_for(WizardController::kLoginScreenName); wizard_controller->SkipToLoginForTesting(chromeos::LoginScreenContext()); } void TestingAutomationProvider::GetOOBEScreenInfo(base::DictionaryValue* args, IPC::Message* reply_message) { static const char kScreenNameKey[] = "screen_name"; AutomationJSONReply reply(this, reply_message); scoped_ptr return_value(new base::DictionaryValue); WizardController* wizard_controller = WizardController::default_controller(); if (wizard_controller) { if (wizard_controller->login_screen_started()) { return_value->SetString(kScreenNameKey, WizardController::kLoginScreenName); } else { return_value->SetString(kScreenNameKey, wizard_controller->current_screen()->GetName()); } } else if (ExistingUserController::current_controller()) { return_value->SetString(kScreenNameKey, WizardController::kLoginScreenName); } else { // Already logged in. reply.SendSuccess(NULL); return; } reply.SendSuccess(return_value.get()); } void TestingAutomationProvider::LockScreen(base::DictionaryValue* args, IPC::Message* reply_message) { new ScreenLockUnlockObserver(this, reply_message, true); DBusThreadManager::Get()->GetSessionManagerClient()->RequestLockScreen(); } void TestingAutomationProvider::UnlockScreen(base::DictionaryValue* args, IPC::Message* reply_message) { std::string password; if (!args->GetString("password", &password)) { AutomationJSONReply(this, reply_message).SendError( "Invalid or missing args."); return; } chromeos::ScreenLocker* screen_locker = chromeos::ScreenLocker::default_screen_locker(); if (!screen_locker) { AutomationJSONReply(this, reply_message).SendError( "No default screen locker. Are you sure the screen is locked?"); return; } new ScreenUnlockObserver(this, reply_message); screen_locker->AuthenticateByPassword(password); } // Signing out could have undesirable side effects: session_manager is // killed, so its children, including chrome and the window manager, will // also be killed. Anything owned by chronos will probably be killed. void TestingAutomationProvider::SignoutInScreenLocker( base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); chromeos::ScreenLocker* screen_locker = chromeos::ScreenLocker::default_screen_locker(); if (!screen_locker) { reply.SendError( "No default screen locker. Are you sure the screen is locked?"); return; } // Send success before stopping session because if we're a child of // session manager then we'll die when the session is stopped. reply.SendSuccess(NULL); screen_locker->Signout(); } void TestingAutomationProvider::GetBatteryInfo(base::DictionaryValue* args, IPC::Message* reply_message) { scoped_ptr return_value(new base::DictionaryValue); const bool battery_is_present = power_supply_properties_.battery_state() != power_manager::PowerSupplyProperties_BatteryState_NOT_PRESENT; const bool line_power_on = power_supply_properties_.external_power() != power_manager::PowerSupplyProperties_ExternalPower_DISCONNECTED; return_value->SetBoolean("battery_is_present", battery_is_present); return_value->SetBoolean("line_power_on", line_power_on); if (battery_is_present) { const bool battery_is_full = power_supply_properties_.battery_state() == power_manager::PowerSupplyProperties_BatteryState_FULL; return_value->SetBoolean("battery_fully_charged", battery_is_full); return_value->SetDouble("battery_percentage", power_supply_properties_.battery_percent()); if (line_power_on) { int64 time = power_supply_properties_.battery_time_to_full_sec(); if (time > 0 || battery_is_full) return_value->SetInteger("battery_seconds_to_full", time); } else { int64 time = power_supply_properties_.battery_time_to_empty_sec(); if (time > 0) return_value->SetInteger("battery_seconds_to_empty", time); } } AutomationJSONReply(this, reply_message).SendSuccess(return_value.get()); } void TestingAutomationProvider::ExecuteJavascriptInOOBEWebUI( base::DictionaryValue* args, IPC::Message* reply_message) { std::string javascript, frame_xpath; if (!args->GetString("javascript", &javascript)) { AutomationJSONReply(this, reply_message) .SendError("'javascript' missing or invalid"); return; } if (!args->GetString("frame_xpath", &frame_xpath)) { AutomationJSONReply(this, reply_message) .SendError("'frame_xpath' missing or invalid"); return; } const UserManager* user_manager = UserManager::Get(); if (!user_manager) { AutomationJSONReply(this, reply_message).SendError( "No user manager!"); return; } if (user_manager->IsUserLoggedIn()) { AutomationJSONReply(this, reply_message).SendError( "User is already logged in."); return; } ExistingUserController* controller = ExistingUserController::current_controller(); if (!controller) { AutomationJSONReply(this, reply_message).SendError( "Unable to access ExistingUserController"); return; } chromeos::LoginDisplayHostImpl* webui_host = static_cast( controller->login_display_host()); content::WebContents* web_contents = webui_host->GetOobeUI()->web_ui()->GetWebContents(); new DomOperationMessageSender(this, reply_message, true); ExecuteJavascriptInRenderViewFrame(base::ASCIIToUTF16(frame_xpath), base::ASCIIToUTF16(javascript), reply_message, web_contents->GetRenderViewHost()); } void TestingAutomationProvider::EnableSpokenFeedback( base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); scoped_ptr return_value(new base::DictionaryValue); bool enabled; if (!args->GetBoolean("enabled", &enabled)) { reply.SendError("Invalid or missing args."); return; } chromeos::AccessibilityManager::Get()->EnableSpokenFeedback( enabled, ash::A11Y_NOTIFICATION_NONE); reply.SendSuccess(return_value.get()); } void TestingAutomationProvider::IsSpokenFeedbackEnabled( base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); scoped_ptr return_value(new base::DictionaryValue); return_value->SetBoolean( "spoken_feedback", chromeos::AccessibilityManager::Get()->IsSpokenFeedbackEnabled()); reply.SendSuccess(return_value.get()); } void TestingAutomationProvider::GetTimeInfo(Browser* browser, base::DictionaryValue* args, IPC::Message* reply_message) { scoped_ptr return_value(new base::DictionaryValue); base::Time time(base::Time::Now()); bool use_24hour_clock = browser && browser->profile()->GetPrefs()->GetBoolean( prefs::kUse24HourClock); base::HourClockType hour_clock_type = use_24hour_clock ? base::k24HourClock : base::k12HourClock; base::string16 display_time = base::TimeFormatTimeOfDayWithHourClockType( time, hour_clock_type, base::kDropAmPm); base::string16 timezone = chromeos::system::TimezoneSettings::GetInstance()->GetCurrentTimezoneID(); return_value->SetString("display_time", display_time); return_value->SetString("display_date", base::TimeFormatFriendlyDate(time)); return_value->SetString("timezone", timezone); AutomationJSONReply(this, reply_message).SendSuccess(return_value.get()); } void TestingAutomationProvider::GetTimeInfo(base::DictionaryValue* args, IPC::Message* reply_message) { GetTimeInfo(NULL, args, reply_message); } void TestingAutomationProvider::SetTimezone(base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); std::string timezone_id; if (!args->GetString("timezone", &timezone_id)) { reply.SendError("Invalid or missing args."); return; } chromeos::CrosSettings* settings = chromeos::CrosSettings::Get(); settings->SetString(chromeos::kSystemTimezone, timezone_id); reply.SendSuccess(NULL); } void TestingAutomationProvider::UpdateCheck( base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply* reply = new AutomationJSONReply(this, reply_message); DBusThreadManager::Get()->GetUpdateEngineClient() ->RequestUpdateCheck(base::Bind(UpdateCheckCallback, reply)); } void TestingAutomationProvider::GetVolumeInfo(base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); scoped_ptr return_value(new base::DictionaryValue); chromeos::CrasAudioHandler* audio_handler = chromeos::CrasAudioHandler::Get(); if (!audio_handler) { reply.SendError("CrasAudioHandler not initialized."); return; } return_value->SetDouble("volume", audio_handler->GetOutputVolumePercent()); return_value->SetBoolean("is_mute", audio_handler->IsOutputMuted()); reply.SendSuccess(return_value.get()); } void TestingAutomationProvider::SetVolume(base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); double volume_percent; if (!args->GetDouble("volume", &volume_percent)) { reply.SendError("Invalid or missing args."); return; } chromeos::CrasAudioHandler* audio_handler = chromeos::CrasAudioHandler::Get(); if (!audio_handler) { reply.SendError("CrasAudioHandler not initialized."); return; } audio_handler->SetOutputVolumePercent(volume_percent); reply.SendSuccess(NULL); } void TestingAutomationProvider::SetMute(base::DictionaryValue* args, IPC::Message* reply_message) { AutomationJSONReply reply(this, reply_message); bool mute; if (!args->GetBoolean("mute", &mute)) { reply.SendError("Invalid or missing args."); return; } chromeos::CrasAudioHandler* audio_handler = chromeos::CrasAudioHandler::Get(); if (!audio_handler) { reply.SendError("CrasAudioHandler not initialized."); return; } audio_handler->SetOutputMute(mute); reply.SendSuccess(NULL); } void TestingAutomationProvider::OpenCrosh(base::DictionaryValue* args, IPC::Message* reply_message) { new NavigationNotificationObserver( NULL, this, reply_message, 1, false, true); ash::Shell::GetInstance()->new_window_delegate()->OpenCrosh(); } void TestingAutomationProvider::AddChromeosObservers() { chromeos::DBusThreadManager::Get()->GetPowerManagerClient()-> AddObserver(this); } void TestingAutomationProvider::RemoveChromeosObservers() { chromeos::DBusThreadManager::Get()->GetPowerManagerClient()-> RemoveObserver(this); }