Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(162)

Side by Side Diff: chrome/browser/extensions/app_host_installer_impl_win.cc

Issue 11054006: Make application shortcuts point to app_host.exe, install App Host during app installation. (Closed) Base URL: http://git.chromium.org/chromium/src.git@master
Patch Set: Refactoring AppHostInstallerImpl: checking for is_platform_app() before flow; entry now done by sta… Created 8 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright (c) 2012 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.
4
5 #include "chrome/browser/extensions/app_host_installer_impl_win.h"
6
7 #include <windows.h>
8 #include "base/basictypes.h"
9 #include "base/bind.h"
10 #include "base/callback.h"
11 #include "base/logging.h"
12 #include "base/process_util.h"
13 #include "base/string16.h"
14 #include "base/win/object_watcher.h"
15 #include "base/win/registry.h"
16 #include "chrome/browser/extensions/app_host_installer.h"
17 #include "chrome/installer/launcher_support/chrome_launcher_support.h"
18 #include "content/public/browser/browser_thread.h"
19
20 namespace {
21
22 // TODO(huangs) Refactor the constants: http://crbug.com/148538
23 const wchar_t kGoogleRegClientsKey[] = L"Software\\Google\\Update\\Clients";
24
25 // Copied from chrome_appid.cc.
26 const wchar_t kBinariesAppGuid[] = L"{4DC8B4CA-1BDA-483e-B5FA-D3C12E15B62D}";
27
28 // Copied from google_update_constants.cc
29 const wchar_t kRegCommandLineField[] = L"CommandLine";
30 const wchar_t kRegCommandsKey[] = L"Commands";
31
32 // Copied from util_constants.cc.
33 const wchar_t kCmdQuickEnableApplicationHost[] =
34 L"quick-enable-application-host";
35
36 // QuickEnableDelegate handles the completion event of App Host installation
37 // via the quick-enable-application host command. At construction, the
38 // class is given |callback_| that takes a bool parameter.
39 // Upon completion, |callback_| is invoked, and is passed a boolean to
40 // indicate success or failure of installation.
41 class QuickEnableDelegate : public base::win::ObjectWatcher::Delegate {
42 public:
43 QuickEnableDelegate(const base::Callback<void(bool)>& callback)
44 : callback_(callback) {}
45
46 // base::win::ObjectWatcher::Delegate implementation.
47 virtual void OnObjectSignaled(HANDLE object) OVERRIDE {
48 int exit_code = 0;
49 base::TerminationStatus status(
50 base::GetTerminationStatus(object, &exit_code));
51 if (status == base::TERMINATION_STATUS_NORMAL_TERMINATION) {
52 callback_.Run(true);
53 } else {
54 LOG(ERROR) << "App Host install failed, status = " << status
55 << ", exit code = " << exit_code;
56 callback_.Run(false);
57 }
58 callback_.Reset();
59 }
60
61 private:
62 base::Callback<void(bool)> callback_;
63
64 DISALLOW_COPY_AND_ASSIGN(QuickEnableDelegate);
65 };
66
67 // Reads the path to app_host.exe from the value "UninstallString" within the
68 // App Host's "ClientState" registry key. Returns an empty string if the path
69 // does not exist or cannot be read.
70 string16 GetQuickEnableAppHostCommand(bool system_level) {
71 HKEY root_key = system_level ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
72 string16 subkey(kGoogleRegClientsKey);
73 subkey.append(1, L'\\').append(kBinariesAppGuid)
74 .append(1, L'\\').append(kRegCommandsKey)
75 .append(1, L'\\').append(kCmdQuickEnableApplicationHost);
76 base::win::RegKey reg_key;
77 string16 cmd;
78 if (reg_key.Open(root_key, subkey.c_str(),
79 KEY_QUERY_VALUE) == ERROR_SUCCESS) {
80 // If read is unsuccessful, |cmd| remains empty.
81 reg_key.ReadValue(kRegCommandLineField, &cmd);
82 }
83 return cmd;
84 }
85
86 // Launches the Google Update command to quick-enable App Host.
87 // Returns true if the command is launched.
88 bool LaunchQuickEnableAppHost(base::win::ScopedHandle* process) {
89 DCHECK(!process->IsValid());
90 bool success = false;
91
92 string16 cmd_str(GetQuickEnableAppHostCommand(true));
93 if (cmd_str.empty()) // Try user-level if absent from system-level.
94 cmd_str = GetQuickEnableAppHostCommand(false);
95 if (!cmd_str.empty()) {
96 VLOG(1) << "Quick-enabling application host: " << cmd_str;
97 if (!base::LaunchProcess(cmd_str, base::LaunchOptions(),
98 process->Receive())) {
99 LOG(ERROR) << "Failed to quick-enable application host.";
100 }
101 success = process->IsValid();
102 }
103 return success;
104 }
105
106 } // namespace
107
108 namespace extensions {
109
110 namespace app_host_installer {
111
112 using content::BrowserThread;
113
114 // static function.
erikwright (departed) 2012/10/09 21:51:52 just "// static"
huangs 2012/10/09 22:44:26 Done.
115 void AppHostInstallerImpl::EnsureAppHostInstalled(
116 const base::Callback<void(bool)>& completion_callback) {
117 BrowserThread::ID caller_thread_id;
118 if (!BrowserThread::GetCurrentThreadIdentifier(&caller_thread_id)) {
119 NOTREACHED();
120 return;
121 }
122
123 AppHostInstallerImpl *impl =
erikwright (departed) 2012/10/09 21:51:52 the assignment to a local variable is unnecessary.
huangs 2012/10/09 22:44:26 Done. Needed (new ...)->... to make compiler happ
124 new AppHostInstallerImpl(completion_callback, caller_thread_id);
125 impl->EnsureAppHostInstalledInternal();
126 // impl will delete itself.
erikwright (departed) 2012/10/09 21:51:52 // AppHostInstalerImpl will delete itself new AppH
huangs 2012/10/09 22:44:26 Done.
127 }
128
129 // AppHostInstallerImpl checks the presence of app_host.exe, and launches
erikwright (departed) 2012/10/09 21:51:52 This comment should probably be at the top of this
huangs 2012/10/09 22:44:26 Done.
130 // the installer if missing. The check must be performed on the FILE thread.
131 // The installation is also launched on the FILE thread as an asynchronous
132 // process. Once installation completes, QuickEnableWatcher is notified.
133 // Finish() is called in the end, which returns to the caller thread and
134 // calls |completion_callback|, followed by self-destruction.
135
136 // Private constructor.
137 AppHostInstallerImpl::AppHostInstallerImpl(
erikwright (departed) 2012/10/09 21:51:52 "// Private constructor." is not necessary.
huangs 2012/10/09 22:44:26 Done.
138 const base::Callback<void(bool)>& completion_callback,
139 BrowserThread::ID caller_thread_id)
140 : completion_callback_(completion_callback),
141 caller_thread_id_(caller_thread_id) {}
142
143 // Private destructor.
erikwright (departed) 2012/10/09 21:51:52 not necessary.
huangs 2012/10/09 22:44:26 Done.
144 AppHostInstallerImpl::~AppHostInstallerImpl() {}
145
146 // Checks the system state on the FILE thread. Will trigger InstallAppHost()
erikwright (departed) 2012/10/09 21:51:52 Unless you have implementation details to add beyo
huangs 2012/10/09 22:44:26 Done.
147 // if App Host is not installed.
148 void AppHostInstallerImpl::EnsureAppHostInstalledInternal() {
149 if (!BrowserThread::CurrentlyOn(BrowserThread::FILE)) {
150 // Redo on FILE thread.
151 if (!BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE,
152 base::Bind(&AppHostInstallerImpl::EnsureAppHostInstalledInternal,
153 base::Unretained(this)))) {
154 Finish(false);
155 }
156 return;
157 }
158
159 if (chrome_launcher_support::IsAppHostPresent())
160 Finish(true);
161 else
162 InstallAppHost();
163 }
164
165 // Asynchronously triggers the App Host installation. Will call "Finish" upon
erikwright (departed) 2012/10/09 21:51:52 Ditto.
huangs 2012/10/09 22:44:26 Done. Moved the comment to the .h file.
166 // completion (successful or otherwise).
167 void AppHostInstallerImpl::InstallAppHost() {
168 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
169 DCHECK(!process_.IsValid());
170 if (!LaunchQuickEnableAppHost(&process_)) {
171 Finish(false);
172 } else {
173 DCHECK(process_.IsValid());
174 DCHECK(!delegate_.get());
175 watcher_.StopWatching();
176 delegate_.reset(new QuickEnableDelegate(
177 base::Bind(&AppHostInstallerImpl::Finish, base::Unretained(this))));
178 watcher_.StartWatching(process_, delegate_.get());
179 }
180 }
181
182 // Passes |success| to |completion_callback| on the original caller thread.
erikwright (departed) 2012/10/09 21:51:52 Ditto.
huangs 2012/10/09 22:44:26 Done.
183 // Deletes the AppHostInstallerImpl.
184 void AppHostInstallerImpl::Finish(bool success) {
185 if (!BrowserThread::CurrentlyOn(caller_thread_id_)) {
186 // Redo on caller thread.
187 if (!BrowserThread::PostTask(caller_thread_id_, FROM_HERE, base::Bind(
188 &AppHostInstallerImpl::Finish, base::Unretained(this), success))) {
189 // This could happen in Shutdown....
190 delete this;
191 }
192 return;
193 }
194
195 completion_callback_.Run(success);
196 delete this;
197 }
198
199 } // namespace app_host_installer
200
201 } // namespace extensions
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698