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.
5 #include "chrome/service/service_process.h"
9 #include "base/basictypes.h"
10 #include "base/callback.h"
11 #include "base/command_line.h"
12 #include "base/environment.h"
13 #include "base/i18n/rtl.h"
14 #include "base/memory/singleton.h"
15 #include "base/path_service.h"
16 #include "base/prefs/json_pref_store.h"
17 #include "base/strings/string16.h"
18 #include "base/strings/utf_string_conversions.h"
19 #include "base/values.h"
20 #include "chrome/common/chrome_constants.h"
21 #include "chrome/common/chrome_paths.h"
22 #include "chrome/common/chrome_switches.h"
23 #include "chrome/common/env_vars.h"
24 #include "chrome/common/pref_names.h"
25 #include "chrome/common/service_process_util.h"
26 #include "chrome/service/cloud_print/cloud_print_proxy.h"
27 #include "chrome/service/net/service_url_request_context.h"
28 #include "chrome/service/service_ipc_server.h"
29 #include "chrome/service/service_process_prefs.h"
30 #include "grit/chromium_strings.h"
31 #include "grit/generated_resources.h"
32 #include "net/base/network_change_notifier.h"
33 #include "net/url_request/url_fetcher.h"
34 #include "ui/base/l10n/l10n_util.h"
35 #include "ui/base/resource/resource_bundle.h"
36 #include "ui/base/ui_base_switches.h"
38 #if defined(OS_LINUX) || defined(OS_OPENBSD)
39 #include <glib-object.h>
42 #if defined(TOOLKIT_GTK)
44 #include "ui/gfx/gtk_util.h"
47 ServiceProcess
* g_service_process
= NULL
;
51 // Delay in seconds after the last service is disabled before we attempt
53 const int kShutdownDelaySeconds
= 60;
55 // Delay in hours between launching a browser process to check the
57 const int64 kPolicyCheckDelayHours
= 8;
59 const char kDefaultServiceProcessLocale
[] = "en-US";
61 class ServiceIOThread
: public base::Thread
{
63 explicit ServiceIOThread(const char* name
);
64 virtual ~ServiceIOThread();
67 virtual void CleanUp() OVERRIDE
;
70 DISALLOW_COPY_AND_ASSIGN(ServiceIOThread
);
73 ServiceIOThread::ServiceIOThread(const char* name
) : base::Thread(name
) {}
74 ServiceIOThread::~ServiceIOThread() {
78 void ServiceIOThread::CleanUp() {
79 net::URLFetcher::CancelAll();
82 // Prepares the localized strings that are going to be displayed to
83 // the user if the service process dies. These strings are stored in the
84 // environment block so they are accessible in the early stages of the
85 // chrome executable's lifetime.
86 void PrepareRestartOnCrashEnviroment(
87 const CommandLine
&parsed_command_line
) {
88 scoped_ptr
<base::Environment
> env(base::Environment::Create());
89 // Clear this var so child processes don't show the dialog by default.
90 env
->UnSetVar(env_vars::kShowRestart
);
92 // For non-interactive tests we don't restart on crash.
93 if (env
->HasVar(env_vars::kHeadless
))
96 // If the known command-line test options are used we don't create the
97 // environment block which means we don't get the restart dialog.
98 if (parsed_command_line
.HasSwitch(switches::kNoErrorDialogs
))
101 // The encoding we use for the info is "title|context|direction" where
102 // direction is either env_vars::kRtlLocale or env_vars::kLtrLocale depending
103 // on the current locale.
104 string16
dlg_strings(l10n_util::GetStringUTF16(IDS_CRASH_RECOVERY_TITLE
));
105 dlg_strings
.push_back('|');
106 string16
adjusted_string(
107 l10n_util::GetStringFUTF16(IDS_SERVICE_CRASH_RECOVERY_CONTENT
,
108 l10n_util::GetStringUTF16(IDS_GOOGLE_CLOUD_PRINT
)));
109 base::i18n::AdjustStringForLocaleDirection(&adjusted_string
);
110 dlg_strings
.append(adjusted_string
);
111 dlg_strings
.push_back('|');
112 dlg_strings
.append(ASCIIToUTF16(
113 base::i18n::IsRTL() ? env_vars::kRtlLocale
: env_vars::kLtrLocale
));
115 env
->SetVar(env_vars::kRestartInfo
, UTF16ToUTF8(dlg_strings
));
120 ServiceProcess::ServiceProcess()
121 : shutdown_event_(true, false),
122 main_message_loop_(NULL
),
123 enabled_services_(0),
124 update_available_(false) {
125 DCHECK(!g_service_process
);
126 g_service_process
= this;
129 bool ServiceProcess::Initialize(base::MessageLoopForUI
* message_loop
,
130 const CommandLine
& command_line
,
131 ServiceProcessState
* state
) {
132 #if defined(TOOLKIT_GTK)
133 // TODO(jamiewalch): Calling GtkInitFromCommandLine here causes the process
134 // to abort if run headless. The correct fix for this is to refactor the
135 // service process to be more modular, a task that is currently underway.
136 // However, since this problem is blocking cloud print, the following quick
137 // hack will have to do. Note that the situation with this hack in place is
138 // no worse than it was when we weren't initializing GTK at all.
140 scoped_ptr
<char*[]> argv(new char*[2]);
141 argv
[0] = strdup(command_line
.argv()[0].c_str());
143 char **argv_pointer
= argv
.get();
144 gtk_init_check(&argc
, &argv_pointer
);
146 #elif defined(OS_LINUX) || defined(OS_OPENBSD)
147 // g_type_init has been deprecated since version 2.35.
148 #if !GLIB_CHECK_VERSION(2, 35, 0)
149 // GLib type system initialization is needed for gconf.
152 #endif // defined(OS_LINUX) || defined(OS_OPENBSD)
153 main_message_loop_
= message_loop
;
154 service_process_state_
.reset(state
);
155 network_change_notifier_
.reset(net::NetworkChangeNotifier::Create());
156 base::Thread::Options options
;
157 options
.message_loop_type
= base::MessageLoop::TYPE_IO
;
158 io_thread_
.reset(new ServiceIOThread("ServiceProcess_IO"));
159 file_thread_
.reset(new base::Thread("ServiceProcess_File"));
160 if (!io_thread_
->StartWithOptions(options
) ||
161 !file_thread_
->StartWithOptions(options
)) {
166 blocking_pool_
= new base::SequencedWorkerPool(3, "ServiceBlocking");
168 request_context_getter_
= new ServiceURLRequestContextGetter();
170 base::FilePath user_data_dir
;
171 PathService::Get(chrome::DIR_USER_DATA
, &user_data_dir
);
172 base::FilePath pref_path
=
173 user_data_dir
.Append(chrome::kServiceStateFileName
);
174 service_prefs_
.reset(new ServiceProcessPrefs(
176 JsonPrefStore::GetTaskRunnerForFile(pref_path
, blocking_pool_
.get())
178 service_prefs_
->ReadPrefs();
180 // This switch it required to run connector with test gaia.
181 if (command_line
.HasSwitch(switches::kIgnoreUrlFetcherCertRequests
))
182 net::URLFetcher::SetIgnoreCertificateRequests(true);
184 // Check if a locale override has been specified on the command-line.
185 std::string locale
= command_line
.GetSwitchValueASCII(switches::kLang
);
186 if (!locale
.empty()) {
187 service_prefs_
->SetString(prefs::kApplicationLocale
, locale
);
188 service_prefs_
->WritePrefs();
190 // If no command-line value was specified, read the last used locale from
193 service_prefs_
->GetString(prefs::kApplicationLocale
, std::string());
194 // If no locale was specified anywhere, use the default one.
196 locale
= kDefaultServiceProcessLocale
;
198 ResourceBundle::InitSharedInstanceWithLocale(locale
, NULL
);
200 PrepareRestartOnCrashEnviroment(command_line
);
202 // Enable Cloud Print if needed. First check the command-line.
203 // Then check if the cloud print proxy was previously enabled.
204 if (command_line
.HasSwitch(switches::kEnableCloudPrintProxy
) ||
205 service_prefs_
->GetBoolean(prefs::kCloudPrintProxyEnabled
, false)) {
206 GetCloudPrintProxy()->EnableForUser();
209 VLOG(1) << "Starting Service Process IPC Server";
210 ipc_server_
.reset(new ServiceIPCServer(
211 service_process_state_
->GetServiceProcessChannel()));
214 // After the IPC server has started we signal that the service process is
216 if (!service_process_state_
->SignalReady(
217 io_thread_
->message_loop_proxy().get(),
218 base::Bind(&ServiceProcess::Terminate
, base::Unretained(this)))) {
222 // See if we need to stay running.
223 ScheduleShutdownCheck();
225 // Occasionally check to see if we need to launch the browser to get the
226 // policy state information.
227 CloudPrintPolicyCheckIfNeeded();
231 bool ServiceProcess::Teardown() {
232 service_prefs_
.reset();
233 cloud_print_proxy_
.reset();
236 // Signal this event before shutting down the service process. That way all
237 // background threads can cleanup.
238 shutdown_event_
.Signal();
240 file_thread_
.reset();
242 if (blocking_pool_
.get()) {
243 // The goal is to make it impossible for chrome to 'infinite loop' during
244 // shutdown, but to reasonably expect that all BLOCKING_SHUTDOWN tasks
245 // queued during shutdown get run. There's nothing particularly scientific
246 // about the number chosen.
247 const int kMaxNewShutdownBlockingTasks
= 1000;
248 blocking_pool_
->Shutdown(kMaxNewShutdownBlockingTasks
);
249 blocking_pool_
= NULL
;
252 // The NetworkChangeNotifier must be destroyed after all other threads that
253 // might use it have been shut down.
254 network_change_notifier_
.reset();
256 service_process_state_
->SignalStopped();
260 // This method is called when a shutdown command is received from IPC channel
261 // or there was an error in the IPC channel.
262 void ServiceProcess::Shutdown() {
263 #if defined(OS_MACOSX)
264 // On MacOS X the service must be removed from the launchd job list.
265 // http://www.chromium.org/developers/design-documents/service-processes
266 // The best way to do that is to go through the ForceServiceProcessShutdown
267 // path. If it succeeds Terminate() will be called from the handler registered
268 // via service_process_state_->SignalReady().
269 // On failure call Terminate() directly to force the process to actually
271 if (!ForceServiceProcessShutdown("", 0)) {
279 void ServiceProcess::Terminate() {
280 main_message_loop_
->PostTask(FROM_HERE
, base::MessageLoop::QuitClosure());
283 bool ServiceProcess::HandleClientDisconnect() {
284 // If there are no enabled services or if there is an update available
285 // we want to shutdown right away. Else we want to keep listening for
287 if (!enabled_services_
|| update_available()) {
294 cloud_print::CloudPrintProxy
* ServiceProcess::GetCloudPrintProxy() {
295 if (!cloud_print_proxy_
.get()) {
296 cloud_print_proxy_
.reset(new cloud_print::CloudPrintProxy());
297 cloud_print_proxy_
->Initialize(service_prefs_
.get(), this);
299 return cloud_print_proxy_
.get();
302 void ServiceProcess::OnCloudPrintProxyEnabled(bool persist_state
) {
304 // Save the preference that we have enabled the cloud print proxy.
305 service_prefs_
->SetBoolean(prefs::kCloudPrintProxyEnabled
, true);
306 service_prefs_
->WritePrefs();
311 void ServiceProcess::OnCloudPrintProxyDisabled(bool persist_state
) {
313 // Save the preference that we have disabled the cloud print proxy.
314 service_prefs_
->SetBoolean(prefs::kCloudPrintProxyEnabled
, false);
315 service_prefs_
->WritePrefs();
320 ServiceURLRequestContextGetter
*
321 ServiceProcess::GetServiceURLRequestContextGetter() {
322 DCHECK(request_context_getter_
.get());
323 return request_context_getter_
.get();
326 void ServiceProcess::OnServiceEnabled() {
328 if ((1 == enabled_services_
) &&
329 !CommandLine::ForCurrentProcess()->HasSwitch(
330 switches::kNoServiceAutorun
)) {
331 if (!service_process_state_
->AddToAutoRun()) {
332 // TODO(scottbyer/sanjeevr/dmaclach): Handle error condition
333 LOG(ERROR
) << "Unable to AddToAutoRun";
338 void ServiceProcess::OnServiceDisabled() {
339 DCHECK_NE(enabled_services_
, 0);
341 if (0 == enabled_services_
) {
342 if (!service_process_state_
->RemoveFromAutoRun()) {
343 // TODO(scottbyer/sanjeevr/dmaclach): Handle error condition
344 LOG(ERROR
) << "Unable to RemoveFromAutoRun";
346 // We will wait for some time to respond to IPCs before shutting down.
347 ScheduleShutdownCheck();
351 void ServiceProcess::ScheduleShutdownCheck() {
352 base::MessageLoop::current()->PostDelayedTask(
354 base::Bind(&ServiceProcess::ShutdownIfNeeded
, base::Unretained(this)),
355 base::TimeDelta::FromSeconds(kShutdownDelaySeconds
));
358 void ServiceProcess::ShutdownIfNeeded() {
359 if (0 == enabled_services_
) {
360 if (ipc_server_
->is_client_connected()) {
361 // If there is a client connected, we need to try again later.
362 // Note that there is still a timing window here because a client may
363 // decide to connect at this point.
364 // TODO(sanjeevr): Fix this timing window.
365 ScheduleShutdownCheck();
372 void ServiceProcess::ScheduleCloudPrintPolicyCheck() {
373 base::MessageLoop::current()->PostDelayedTask(
375 base::Bind(&ServiceProcess::CloudPrintPolicyCheckIfNeeded
,
376 base::Unretained(this)),
377 base::TimeDelta::FromHours(kPolicyCheckDelayHours
));
380 void ServiceProcess::CloudPrintPolicyCheckIfNeeded() {
381 if (enabled_services_
&& !ipc_server_
->is_client_connected()) {
382 GetCloudPrintProxy()->CheckCloudPrintProxyPolicy();
384 ScheduleCloudPrintPolicyCheck();
387 ServiceProcess::~ServiceProcess() {
389 g_service_process
= NULL
;