diff options
Diffstat (limited to 'chrome/browser/first_run_win.cc')
-rw-r--r-- | chrome/browser/first_run_win.cc | 667 |
1 files changed, 0 insertions, 667 deletions
diff --git a/chrome/browser/first_run_win.cc b/chrome/browser/first_run_win.cc deleted file mode 100644 index 3a24488..0000000 --- a/chrome/browser/first_run_win.cc +++ /dev/null @@ -1,667 +0,0 @@ -// 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. - -#include "chrome/browser/first_run.h" - -#include <atlbase.h> -#include <atlcom.h> -#include <windows.h> -#include <shlobj.h> - -#include <sstream> - -#include "app/app_switches.h" -#include "app/resource_bundle.h" -#include "base/command_line.h" -#include "base/file_util.h" -#include "base/logging.h" -#include "base/object_watcher.h" -#include "base/path_service.h" -#include "base/process.h" -#include "base/process_util.h" -#include "base/registry.h" -#include "base/string_util.h" -#include "chrome/common/chrome_constants.h" -#include "chrome/common/chrome_switches.h" -#include "chrome/common/pref_names.h" -#include "chrome/browser/browser_process.h" -#include "chrome/browser/hang_monitor/hung_window_detector.h" -#include "chrome/browser/importer/importer.h" -#include "chrome/browser/process_singleton.h" -#include "chrome/browser/profile.h" -#include "chrome/browser/profile_manager.h" -#include "chrome/browser/shell_integration.h" -#include "chrome/browser/views/first_run_view.h" -#include "chrome/common/chrome_constants.h" -#include "chrome/common/chrome_paths.h" -#include "chrome/common/pref_service.h" -#include "chrome/common/result_codes.h" -#include "chrome/installer/util/browser_distribution.h" -#include "chrome/installer/util/google_update_constants.h" -#include "chrome/installer/util/google_update_settings.h" -#include "chrome/installer/util/install_util.h" -#include "chrome/installer/util/master_preferences.h" -#include "chrome/installer/util/shell_util.h" -#include "chrome/installer/util/util_constants.h" -#include "google_update_idl.h" -#include "grit/locale_settings.h" -#include "views/widget/accelerator_handler.h" -#include "views/window/window.h" - -namespace { - -// The kSentinelFile file absence will tell us it is a first run. -const wchar_t kSentinelFile[] = L"First Run"; - -// Gives the full path to the sentinel file. The file might not exist. -bool GetFirstRunSentinelFilePath(std::wstring* path) { - std::wstring exe_path; - if (!PathService::Get(base::DIR_EXE, &exe_path)) - return false; - - std::wstring first_run_sentinel; - if (InstallUtil::IsPerUserInstall(exe_path.c_str())) { - first_run_sentinel = exe_path; - } else { - if (!PathService::Get(chrome::DIR_USER_DATA, &first_run_sentinel)) - return false; - } - - file_util::AppendToPath(&first_run_sentinel, kSentinelFile); - *path = first_run_sentinel; - return true; -} - -bool GetNewerChromeFile(std::wstring* path) { - if (!PathService::Get(base::DIR_EXE, path)) - return false; - file_util::AppendToPath(path, installer_util::kChromeNewExe); - return true; -} - -bool GetBackupChromeFile(std::wstring* path) { - if (!PathService::Get(base::DIR_EXE, path)) - return false; - file_util::AppendToPath(path, installer_util::kChromeOldExe); - return true; -} - -std::wstring GetDefaultPrefFilePath(bool create_profile_dir, - const std::wstring& user_data_dir) { - FilePath default_pref_dir = ProfileManager::GetDefaultProfileDir( - FilePath::FromWStringHack(user_data_dir)); - if (create_profile_dir) { - if (!file_util::PathExists(default_pref_dir)) { - if (!file_util::CreateDirectory(default_pref_dir)) - return std::wstring(); - } - } - return ProfileManager::GetDefaultProfilePath(default_pref_dir) - .ToWStringHack(); -} - -bool InvokeGoogleUpdateForRename() { - CComPtr<IProcessLauncher> ipl; - if (!FAILED(ipl.CoCreateInstance(__uuidof(ProcessLauncherClass)))) { - ULONG_PTR phandle = NULL; - DWORD id = GetCurrentProcessId(); - if (!FAILED(ipl->LaunchCmdElevated(google_update::kChromeGuid, - google_update::kRegRenameCmdField, - id, &phandle))) { - HANDLE handle = HANDLE(phandle); - DWORD exit_code; - ::GetExitCodeProcess(handle, &exit_code); - ::CloseHandle(handle); - if (exit_code == installer_util::RENAME_SUCCESSFUL) - return true; - } - } - return false; -} - -bool LaunchSetupWithParam(const std::wstring& param, const std::wstring& value, - int* ret_code) { - FilePath exe_path; - if (!PathService::Get(base::DIR_MODULE, &exe_path)) - return false; - exe_path = exe_path.Append(installer_util::kInstallerDir); - exe_path = exe_path.Append(installer_util::kSetupExe); - base::ProcessHandle ph; - CommandLine cl(exe_path.ToWStringHack()); - cl.AppendSwitchWithValue(param, value); - if (!base::LaunchApp(cl, false, false, &ph)) - return false; - DWORD wr = ::WaitForSingleObject(ph, INFINITE); - if (wr != WAIT_OBJECT_0) - return false; - return (TRUE == ::GetExitCodeProcess(ph, reinterpret_cast<DWORD*>(ret_code))); -} - -bool WriteEULAtoTempFile(FilePath* eula_path) { - std::string terms = - ResourceBundle::GetSharedInstance().GetDataResource(IDR_TERMS_HTML); - if (terms.empty()) - return false; - FilePath temp_dir; - if (!file_util::GetTempDir(&temp_dir)) - return false; - *eula_path = temp_dir.Append(L"chrome_eula_iframe.html"); - return (file_util::WriteFile(*eula_path, terms.c_str(), terms.size()) > 0); -} - -} // namespace - -bool FirstRun::IsChromeFirstRun() { - // A troolean, 0 means not yet set, 1 means set to true, 2 set to false. - static int first_run = 0; - if (first_run != 0) - return first_run == 1; - - std::wstring first_run_sentinel; - if (!GetFirstRunSentinelFilePath(&first_run_sentinel) || - file_util::PathExists(first_run_sentinel)) { - first_run = 2; - return false; - } - first_run = 1; - return true; -} - -bool FirstRun::CreateChromeDesktopShortcut() { - std::wstring chrome_exe; - if (!PathService::Get(base::FILE_EXE, &chrome_exe)) - return false; - BrowserDistribution *dist = BrowserDistribution::GetDistribution(); - if (!dist) - return false; - return ShellUtil::CreateChromeDesktopShortcut(chrome_exe, - dist->GetAppDescription(), ShellUtil::CURRENT_USER, - false, true); // create if doesn't exist. -} - -bool FirstRun::CreateChromeQuickLaunchShortcut() { - std::wstring chrome_exe; - if (!PathService::Get(base::FILE_EXE, &chrome_exe)) - return false; - return ShellUtil::CreateChromeQuickLaunchShortcut(chrome_exe, - ShellUtil::CURRENT_USER, // create only for current user. - true); // create if doesn't exist. -} - -bool FirstRun::RemoveSentinel() { - std::wstring first_run_sentinel; - if (!GetFirstRunSentinelFilePath(&first_run_sentinel)) - return false; - return file_util::Delete(first_run_sentinel, false); -} - -bool FirstRun::CreateSentinel() { - std::wstring first_run_sentinel; - if (!GetFirstRunSentinelFilePath(&first_run_sentinel)) - return false; - HANDLE file = ::CreateFileW(first_run_sentinel.c_str(), - FILE_READ_DATA | FILE_WRITE_DATA, - FILE_SHARE_READ | FILE_SHARE_WRITE, - NULL, CREATE_ALWAYS, 0, NULL); - if (INVALID_HANDLE_VALUE == file) - return false; - ::CloseHandle(file); - return true; -} - -bool FirstRun::ProcessMasterPreferences(const FilePath& user_data_dir, - const FilePath& master_prefs_path, - int* preference_details, - std::vector<std::wstring>* new_tabs) { - DCHECK(!user_data_dir.empty()); - if (preference_details) - *preference_details = 0; - - std::wstring master_prefs; - if (master_prefs_path.empty()) { - // The default location of the master prefs is next to the chrome exe. - std::wstring master_path; - if (!PathService::Get(base::DIR_EXE, &master_path)) - return true; - file_util::AppendToPath(&master_path, installer_util::kDefaultMasterPrefs); - master_prefs = master_path; - } else { - master_prefs = master_prefs_path.ToWStringHack(); - } - - int parse_result = installer_util::ParseDistributionPreferences(master_prefs); - if (preference_details) - *preference_details = parse_result; - - if (parse_result & installer_util::MASTER_PROFILE_ERROR) - return true; - - if (new_tabs) - *new_tabs = installer_util::ParseFirstRunTabs(master_prefs); - - if (parse_result & installer_util::MASTER_PROFILE_REQUIRE_EULA) { - // Show the post-installation EULA. This is done by setup.exe and the - // result determines if we continue or not. We wait here until the user - // dismisses the dialog. - - // The actual eula text is in a resource in chrome. We extract it to - // a text file so setup.exe can use it as an inner frame. - FilePath inner_html; - if (WriteEULAtoTempFile(&inner_html)) { - int retcode = 0; - const std::wstring& eula = installer_util::switches::kShowEula; - if (!LaunchSetupWithParam(eula, inner_html.ToWStringHack(), &retcode) || - (retcode == installer_util::EULA_REJECTED)) { - LOG(WARNING) << "EULA rejected. Fast exit."; - ::ExitProcess(1); - } - if (retcode == installer_util::EULA_ACCEPTED) { - LOG(INFO) << "EULA : no collection"; - GoogleUpdateSettings::SetCollectStatsConsent(false); - } else if (retcode == installer_util::EULA_ACCEPTED_OPT_IN) { - LOG(INFO) << "EULA : collection consent"; - GoogleUpdateSettings::SetCollectStatsConsent(true); - } - } - } - - if (parse_result & installer_util::MASTER_PROFILE_OEM_FIRST_RUN_BUBBLE) - FirstRun::SetOEMFirstRunBubblePref(); - - FilePath user_prefs = FilePath::FromWStringHack( - GetDefaultPrefFilePath(true, user_data_dir.ToWStringHack())); - if (user_prefs.empty()) - return true; - - // The master prefs are regular prefs so we can just copy the file - // to the default place and they just work. - if (!file_util::CopyFile(master_prefs, user_prefs.ToWStringHack())) - return true; - - if (!(parse_result & installer_util::MASTER_PROFILE_NO_FIRST_RUN_UI)) - return true; - - // From here on we won't show first run so we need to do the work to set the - // required state given that FirstRunView is not going to be called. - FirstRun::SetShowFirstRunBubblePref(); - - // We need to be able to create the first run sentinel or else we cannot - // proceed because ImportSettings will launch the importer process which - // would end up here if the sentinel is not present. - if (!FirstRun::CreateSentinel()) - return false; - - if (parse_result & installer_util::MASTER_PROFILE_SHOW_WELCOME) - FirstRun::SetShowWelcomePagePref(); - - int import_items = 0; - if (parse_result & installer_util::MASTER_PROFILE_IMPORT_SEARCH_ENGINE) - import_items += SEARCH_ENGINES; - if (parse_result & installer_util::MASTER_PROFILE_IMPORT_HISTORY) - import_items += HISTORY; - if (parse_result & installer_util::MASTER_PROFILE_IMPORT_BOOKMARKS) - import_items += FAVORITES; - - if (import_items) { - // There is something to import from the default browser. This launches - // the importer process and blocks until done or until it fails. - scoped_refptr<ImporterHost> importer_host = new ImporterHost(); - if (!FirstRun::ImportSettings(NULL, - importer_host->GetSourceProfileInfoAt(0).browser_type, - import_items, NULL)) { - LOG(WARNING) << "silent import failed"; - } - } - - if (parse_result & - installer_util::MASTER_PROFILE_MAKE_CHROME_DEFAULT_FOR_USER) - ShellIntegration::SetAsDefaultBrowser(); - - return false; -} - -bool Upgrade::IsBrowserAlreadyRunning() { - static HANDLE handle = NULL; - std::wstring exe; - PathService::Get(base::FILE_EXE, &exe); - std::replace(exe.begin(), exe.end(), '\\', '!'); - std::transform(exe.begin(), exe.end(), exe.begin(), tolower); - exe = L"Global\\" + exe; - if (handle != NULL) - CloseHandle(handle); - handle = CreateEvent(NULL, TRUE, TRUE, exe.c_str()); - int error = GetLastError(); - return (error == ERROR_ALREADY_EXISTS || error == ERROR_ACCESS_DENIED); -} - -bool Upgrade::RelaunchChromeBrowser(const CommandLine& command_line) { - ::SetEnvironmentVariable(google_update::kEnvProductVersionKey, NULL); - return base::LaunchApp(command_line.command_line_string(), - false, false, NULL); -} - -bool Upgrade::SwapNewChromeExeIfPresent() { - std::wstring new_chrome_exe; - if (!GetNewerChromeFile(&new_chrome_exe)) - return false; - if (!file_util::PathExists(new_chrome_exe)) - return false; - std::wstring curr_chrome_exe; - if (!PathService::Get(base::FILE_EXE, &curr_chrome_exe)) - return false; - - // First try to rename exe by launching rename command ourselves. - bool user_install = InstallUtil::IsPerUserInstall(curr_chrome_exe.c_str()); - HKEY reg_root = user_install ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE; - BrowserDistribution *dist = BrowserDistribution::GetDistribution(); - RegKey key; - std::wstring rename_cmd; - if (key.Open(reg_root, dist->GetVersionKey().c_str(), KEY_READ) && - key.ReadValue(google_update::kRegRenameCmdField, &rename_cmd)) { - base::ProcessHandle handle; - if (base::LaunchApp(rename_cmd, true, true, &handle)) { - DWORD exit_code; - ::GetExitCodeProcess(handle, &exit_code); - ::CloseHandle(handle); - if (exit_code == installer_util::RENAME_SUCCESSFUL) - return true; - } - } - - // Rename didn't work so try to rename by calling Google Update - if (InvokeGoogleUpdateForRename()) - return true; - - // Rename still didn't work so just try to rename exe ourselves (for - // backward compatibility, can be deleted once the new process works). - std::wstring backup_exe; - if (!GetBackupChromeFile(&backup_exe)) - return false; - if (::ReplaceFileW(curr_chrome_exe.c_str(), new_chrome_exe.c_str(), - backup_exe.c_str(), REPLACEFILE_IGNORE_MERGE_ERRORS, - NULL, NULL)) { - return true; - } - return false; -} - -void OpenFirstRunDialog(Profile* profile, - ProcessSingleton* process_singleton) { - DCHECK(profile); - DCHECK(process_singleton); - - views::Window* first_run_ui = views::Window::CreateChromeWindow( - NULL, gfx::Rect(), new FirstRunView(profile)); - DCHECK(first_run_ui); - - // We need to avoid dispatching new tabs when we are doing the import - // because that will lead to data corruption or a crash. Lock() does that. - // If a CopyData message does come in while the First Run UI is visible, - // then we will attempt to set first_run_ui as the foreground window. - process_singleton->Lock(first_run_ui->GetNativeWindow()); - - first_run_ui->Show(); - - // We must now run a message loop (will be terminated when the First Run UI - // is closed) so that the window can receive messages and we block the - // browser window from showing up. We pass the accelerator handler here so - // that keyboard accelerators (Enter, Esc, etc) work in the dialog box. - MessageLoopForUI::current()->Run(g_browser_process->accelerator_handler()); - process_singleton->Unlock(); -} - -namespace { - -// This class is used by FirstRun::ImportSettings to determine when the import -// process has ended and what was the result of the operation as reported by -// the process exit code. This class executes in the context of the main chrome -// process. -class ImportProcessRunner : public base::ObjectWatcher::Delegate { - public: - // The constructor takes the importer process to watch and then it does a - // message loop blocking wait until the process ends. This object now owns - // the import_process handle. - explicit ImportProcessRunner(base::ProcessHandle import_process) - : import_process_(import_process), - exit_code_(ResultCodes::NORMAL_EXIT) { - watcher_.StartWatching(import_process, this); - MessageLoop::current()->Run(); - } - virtual ~ImportProcessRunner() { - ::CloseHandle(import_process_); - } - // Returns the child process exit code. There are 3 expected values: - // NORMAL_EXIT, IMPORTER_CANCEL or IMPORTER_HUNG. - int exit_code() const { - return exit_code_; - } - // The child process has terminated. Find the exit code and quit the loop. - virtual void OnObjectSignaled(HANDLE object) { - DCHECK(object == import_process_); - if (!::GetExitCodeProcess(import_process_, &exit_code_)) { - NOTREACHED(); - } - MessageLoop::current()->Quit(); - } - - private: - base::ObjectWatcher watcher_; - base::ProcessHandle import_process_; - DWORD exit_code_; -}; - -// Check every 3 seconds if the importer UI has hung. -const int kPollHangFrequency = 3000; - -// This class specializes on finding hung 'owned' windows. Unfortunately, the -// HungWindowDetector class cannot be used here because it assumes child -// windows and not owned top-level windows. -// This code is executed in the context of the main browser process and will -// terminate the importer process if it is hung. -class HungImporterMonitor : public WorkerThreadTicker::Callback { - public: - // The ctor takes the owner popup window and the process handle of the - // process to kill in case the popup or its owned active popup become - // unresponsive. - HungImporterMonitor(HWND owner_window, base::ProcessHandle import_process) - : owner_window_(owner_window), - import_process_(import_process), - ticker_(kPollHangFrequency) { - ticker_.RegisterTickHandler(this); - ticker_.Start(); - } - virtual ~HungImporterMonitor() { - ticker_.Stop(); - ticker_.UnregisterTickHandler(this); - } - - private: - virtual void OnTick() { - if (!import_process_) - return; - // We find the top active popup that we own, this will be either the - // owner_window_ itself or the dialog window of the other process. In - // both cases it is worth hung testing because both windows share the - // same message queue and at some point the other window could be gone - // while the other process still not pumping messages. - HWND active_window = ::GetLastActivePopup(owner_window_); - if (::IsHungAppWindow(active_window) || ::IsHungAppWindow(owner_window_)) { - ::TerminateProcess(import_process_, ResultCodes::IMPORTER_HUNG); - import_process_ = NULL; - } - } - - HWND owner_window_; - base::ProcessHandle import_process_; - WorkerThreadTicker ticker_; - DISALLOW_EVIL_CONSTRUCTORS(HungImporterMonitor); -}; - -// This class is used by FirstRun::ImportNow to get notified of the outcome of -// the import operation. It differs from ImportProcessRunner in that this -// class executes in the context of importing child process. -// The values that it handles are meant to be used as the process exit code. -class FirstRunImportObserver : public ImportObserver { - public: - FirstRunImportObserver() - : loop_running_(false), import_result_(ResultCodes::NORMAL_EXIT) { - } - int import_result() const { - return import_result_; - } - virtual void ImportCanceled() { - import_result_ = ResultCodes::IMPORTER_CANCEL; - Finish(); - } - virtual void ImportComplete() { - import_result_ = ResultCodes::NORMAL_EXIT; - Finish(); - } - - void RunLoop() { - loop_running_ = true; - MessageLoop::current()->Run(); - } - - private: - void Finish() { - if (loop_running_) - MessageLoop::current()->Quit(); - } - - bool loop_running_; - int import_result_; - DISALLOW_EVIL_CONSTRUCTORS(FirstRunImportObserver); -}; - -std::wstring EncodeImportParams(int browser_type, int options, HWND window) { - return StringPrintf(L"%d@%d@%d", browser_type, options, window); -} - -bool DecodeImportParams(const std::wstring& encoded, - int* browser_type, int* options, HWND* window) { - std::vector<std::wstring> v; - SplitString(encoded, L'@', &v); - if (v.size() != 3) - return false; - *browser_type = static_cast<int>(StringToInt64(v[0])); - *options = static_cast<int>(StringToInt64(v[1])); - *window = reinterpret_cast<HWND>(StringToInt64(v[2])); - return true; -} - -} // namespace - -bool FirstRun::ImportSettings(Profile* profile, int browser_type, - int items_to_import, HWND parent_window) { - const CommandLine& cmdline = *CommandLine::ForCurrentProcess(); - CommandLine import_cmd(cmdline.program()); - // Propagate user data directory switch. - if (cmdline.HasSwitch(switches::kUserDataDir)) { - import_cmd.AppendSwitchWithValue( - switches::kUserDataDir, - cmdline.GetSwitchValue(switches::kUserDataDir)); - } - - // Since ImportSettings is called before the local state is stored on disk - // we pass the language as an argument. GetApplicationLocale checks the - // current command line as fallback. - import_cmd.AppendSwitchWithValue( - switches::kLang, - g_browser_process->GetApplicationLocale()); - - import_cmd.CommandLine::AppendSwitchWithValue(switches::kImport, - EncodeImportParams(browser_type, items_to_import, parent_window)); - - // Time to launch the process that is going to do the import. - base::ProcessHandle import_process; - if (!base::LaunchApp(import_cmd, false, false, &import_process)) - return false; - - // Activate the importer monitor. It awakes periodically in another thread - // and checks that the importer UI is still pumping messages. - if (parent_window) - HungImporterMonitor hang_monitor(parent_window, import_process); - - // We block inside the import_runner ctor, pumping messages until the - // importer process ends. This can happen either by completing the import - // or by hang_monitor killing it. - ImportProcessRunner import_runner(import_process); - - // Import process finished. Reload the prefs, because importer may set - // the pref value. - if (profile) - profile->GetPrefs()->ReloadPersistentPrefs(); - - return (import_runner.exit_code() == ResultCodes::NORMAL_EXIT); -} - -int FirstRun::ImportNow(Profile* profile, const CommandLine& cmdline) { - std::wstring import_info = cmdline.GetSwitchValue(switches::kImport); - if (import_info.empty()) { - NOTREACHED(); - return false; - } - int browser_type = 0; - int items_to_import = 0; - HWND parent_window = NULL; - if (!DecodeImportParams(import_info, &browser_type, &items_to_import, - &parent_window)) { - NOTREACHED(); - return false; - } - scoped_refptr<ImporterHost> importer_host = new ImporterHost(); - FirstRunImportObserver observer; - - // If there is no parent window, we run in headless mode which amounts - // to having the windows hidden and if there is user action required the - // import is automatically canceled. - if (!parent_window) - importer_host->set_headless(); - - StartImportingWithUI( - parent_window, - items_to_import, - importer_host, - importer_host->GetSourceProfileInfoForBrowserType(browser_type), - profile, - &observer, - true); - observer.RunLoop(); - return observer.import_result(); -} - -bool FirstRun::SetShowFirstRunBubblePref() { - PrefService* local_state = g_browser_process->local_state(); - if (!local_state) - return false; - if (!local_state->IsPrefRegistered(prefs::kShouldShowFirstRunBubble)) { - local_state->RegisterBooleanPref(prefs::kShouldShowFirstRunBubble, false); - local_state->SetBoolean(prefs::kShouldShowFirstRunBubble, true); - } - return true; -} - -bool FirstRun::SetOEMFirstRunBubblePref() { - PrefService* local_state = g_browser_process->local_state(); - if (!local_state) - return false; - if (!local_state->IsPrefRegistered(prefs::kShouldUseOEMFirstRunBubble)) { - local_state->RegisterBooleanPref(prefs::kShouldUseOEMFirstRunBubble, - false); - local_state->SetBoolean(prefs::kShouldUseOEMFirstRunBubble, true); - } - return true; -} - -bool FirstRun::SetShowWelcomePagePref() { - PrefService* local_state = g_browser_process->local_state(); - if (!local_state) - return false; - if (!local_state->IsPrefRegistered(prefs::kShouldShowWelcomePage)) { - local_state->RegisterBooleanPref(prefs::kShouldShowWelcomePage, false); - local_state->SetBoolean(prefs::kShouldShowWelcomePage, true); - } - return true; -} |