Home | History | Annotate | Download | only in cloud_print
      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/service/cloud_print/cloud_print_proxy.h"
      6 
      7 #include "base/bind.h"
      8 #include "base/command_line.h"
      9 #include "base/metrics/histogram.h"
     10 #include "base/path_service.h"
     11 #include "base/process/kill.h"
     12 #include "base/process/launch.h"
     13 #include "base/values.h"
     14 #include "chrome/common/chrome_switches.h"
     15 #include "chrome/common/cloud_print/cloud_print_constants.h"
     16 #include "chrome/common/cloud_print/cloud_print_proxy_info.h"
     17 #include "chrome/common/pref_names.h"
     18 #include "chrome/service/cloud_print/print_system.h"
     19 #include "chrome/service/service_process.h"
     20 #include "chrome/service/service_process_prefs.h"
     21 #include "google_apis/gaia/gaia_oauth_client.h"
     22 #include "google_apis/google_api_keys.h"
     23 #include "url/gurl.h"
     24 
     25 namespace {
     26 
     27 void LaunchBrowserProcessWithSwitch(const std::string& switch_string) {
     28   DCHECK(g_service_process->io_thread()->message_loop_proxy()->
     29       BelongsToCurrentThread());
     30   base::FilePath exe_path;
     31   PathService::Get(base::FILE_EXE, &exe_path);
     32   if (exe_path.empty()) {
     33     NOTREACHED() << "Unable to get browser process binary name.";
     34   }
     35   CommandLine cmd_line(exe_path);
     36 
     37   const CommandLine& process_command_line = *CommandLine::ForCurrentProcess();
     38   base::FilePath user_data_dir =
     39       process_command_line.GetSwitchValuePath(switches::kUserDataDir);
     40   if (!user_data_dir.empty())
     41     cmd_line.AppendSwitchPath(switches::kUserDataDir, user_data_dir);
     42   cmd_line.AppendSwitch(switch_string);
     43 
     44 #if defined(OS_POSIX) && !defined(OS_MACOSX)
     45   base::ProcessHandle pid = 0;
     46   base::LaunchProcess(cmd_line, base::LaunchOptions(), &pid);
     47   base::EnsureProcessGetsReaped(pid);
     48 #else
     49   base::LaunchOptions launch_options;
     50 #if defined(OS_WIN)
     51   launch_options.force_breakaway_from_job_ = true;
     52 #endif  // OS_WIN
     53   base::LaunchProcess(cmd_line, launch_options, NULL);
     54 #endif
     55 }
     56 
     57 void CheckCloudPrintProxyPolicyInBrowser() {
     58   LaunchBrowserProcessWithSwitch(switches::kCheckCloudPrintConnectorPolicy);
     59 }
     60 
     61 }  // namespace
     62 
     63 namespace cloud_print {
     64 
     65 CloudPrintProxy::CloudPrintProxy()
     66     : service_prefs_(NULL),
     67       client_(NULL),
     68       enabled_(false) {
     69 }
     70 
     71 CloudPrintProxy::~CloudPrintProxy() {
     72   DCHECK(CalledOnValidThread());
     73   ShutdownBackend();
     74 }
     75 
     76 void CloudPrintProxy::Initialize(ServiceProcessPrefs* service_prefs,
     77                                  Client* client) {
     78   DCHECK(CalledOnValidThread());
     79   service_prefs_ = service_prefs;
     80   client_ = client;
     81 }
     82 
     83 void CloudPrintProxy::EnableForUser() {
     84   DCHECK(CalledOnValidThread());
     85   if (!CreateBackend())
     86     return;
     87   DCHECK(backend_.get());
     88   // Read persisted robot credentials because we may decide to reuse it if the
     89   // passed in LSID belongs the same user.
     90   std::string robot_refresh_token = service_prefs_->GetString(
     91       prefs::kCloudPrintRobotRefreshToken, std::string());
     92   std::string robot_email =
     93       service_prefs_->GetString(prefs::kCloudPrintRobotEmail, std::string());
     94   user_email_ = service_prefs_->GetString(prefs::kCloudPrintEmail, user_email_);
     95 
     96   // See if we have persisted robot credentials.
     97   if (!robot_refresh_token.empty()) {
     98     DCHECK(!robot_email.empty());
     99     backend_->InitializeWithRobotToken(robot_refresh_token, robot_email);
    100   } else {
    101     // Finally see if we have persisted user credentials (legacy case).
    102     std::string cloud_print_token =
    103         service_prefs_->GetString(prefs::kCloudPrintAuthToken, std::string());
    104     DCHECK(!cloud_print_token.empty());
    105     backend_->InitializeWithToken(cloud_print_token);
    106   }
    107   if (client_) {
    108     client_->OnCloudPrintProxyEnabled(true);
    109   }
    110 }
    111 
    112 void CloudPrintProxy::EnableForUserWithRobot(
    113     const std::string& robot_auth_code,
    114     const std::string& robot_email,
    115     const std::string& user_email,
    116     const base::DictionaryValue& user_settings) {
    117   DCHECK(CalledOnValidThread());
    118 
    119   ShutdownBackend();
    120   std::string proxy_id(
    121       service_prefs_->GetString(prefs::kCloudPrintProxyId, std::string()));
    122   service_prefs_->RemovePref(prefs::kCloudPrintRoot);
    123   if (!proxy_id.empty()) {
    124     // Keep only proxy id;
    125     service_prefs_->SetString(prefs::kCloudPrintProxyId, proxy_id);
    126   }
    127   service_prefs_->SetValue(prefs::kCloudPrintUserSettings,
    128                            user_settings.DeepCopy());
    129   service_prefs_->WritePrefs();
    130 
    131   if (!CreateBackend())
    132     return;
    133   DCHECK(backend_.get());
    134   user_email_ = user_email;
    135   backend_->InitializeWithRobotAuthCode(robot_auth_code, robot_email);
    136   if (client_) {
    137     client_->OnCloudPrintProxyEnabled(true);
    138   }
    139 }
    140 
    141 bool CloudPrintProxy::CreateBackend() {
    142   DCHECK(CalledOnValidThread());
    143   if (backend_.get())
    144     return false;
    145 
    146   ConnectorSettings settings;
    147   settings.InitFrom(service_prefs_);
    148 
    149   // By default we don't poll for jobs when we lose XMPP connection. But this
    150   // behavior can be overridden by a preference.
    151   bool enable_job_poll =
    152     service_prefs_->GetBoolean(prefs::kCloudPrintEnableJobPoll, false);
    153 
    154   gaia::OAuthClientInfo oauth_client_info;
    155   oauth_client_info.client_id =
    156     google_apis::GetOAuth2ClientID(google_apis::CLIENT_CLOUD_PRINT);
    157   oauth_client_info.client_secret =
    158     google_apis::GetOAuth2ClientSecret(google_apis::CLIENT_CLOUD_PRINT);
    159   oauth_client_info.redirect_uri = "oob";
    160   backend_.reset(new CloudPrintProxyBackend(
    161       this, settings, oauth_client_info, enable_job_poll));
    162   return true;
    163 }
    164 
    165 void CloudPrintProxy::UnregisterPrintersAndDisableForUser() {
    166   DCHECK(CalledOnValidThread());
    167   if (backend_.get()) {
    168     // Try getting auth and printers info from the backend.
    169     // We'll get notified in this case.
    170     backend_->UnregisterPrinters();
    171   } else {
    172     // If no backend avaialble, disable connector immidiately.
    173     DisableForUser();
    174   }
    175 }
    176 
    177 void CloudPrintProxy::DisableForUser() {
    178   DCHECK(CalledOnValidThread());
    179   user_email_.clear();
    180   enabled_ = false;
    181   if (client_) {
    182     client_->OnCloudPrintProxyDisabled(true);
    183   }
    184   ShutdownBackend();
    185 }
    186 
    187 void CloudPrintProxy::GetProxyInfo(CloudPrintProxyInfo* info) {
    188   info->enabled = enabled_;
    189   info->email.clear();
    190   if (enabled_)
    191     info->email = user_email();
    192   ConnectorSettings settings;
    193   settings.InitFrom(service_prefs_);
    194   info->proxy_id = settings.proxy_id();
    195 }
    196 
    197 void CloudPrintProxy::GetPrinters(std::vector<std::string>* printers) {
    198   ConnectorSettings settings;
    199   settings.InitFrom(service_prefs_);
    200   scoped_refptr<PrintSystem> print_system =
    201       PrintSystem::CreateInstance(settings.print_system_settings());
    202   if (!print_system.get())
    203     return;
    204   PrintSystem::PrintSystemResult result = print_system->Init();
    205   if (!result.succeeded())
    206     return;
    207   printing::PrinterList printer_list;
    208   print_system->EnumeratePrinters(&printer_list);
    209   for (size_t i = 0; i < printer_list.size(); ++i)
    210     printers->push_back(printer_list[i].printer_name);
    211 }
    212 
    213 void CloudPrintProxy::CheckCloudPrintProxyPolicy() {
    214   g_service_process->io_thread()->message_loop_proxy()->PostTask(
    215       FROM_HERE, base::Bind(&CheckCloudPrintProxyPolicyInBrowser));
    216 }
    217 
    218 void CloudPrintProxy::OnAuthenticated(
    219     const std::string& robot_oauth_refresh_token,
    220     const std::string& robot_email,
    221     const std::string& user_email) {
    222   DCHECK(CalledOnValidThread());
    223   service_prefs_->SetString(prefs::kCloudPrintRobotRefreshToken,
    224                             robot_oauth_refresh_token);
    225   service_prefs_->SetString(prefs::kCloudPrintRobotEmail,
    226                             robot_email);
    227   // If authenticating from a robot, the user email will be empty.
    228   if (!user_email.empty()) {
    229     user_email_ = user_email;
    230   }
    231   service_prefs_->SetString(prefs::kCloudPrintEmail, user_email_);
    232   enabled_ = true;
    233   DCHECK(!user_email_.empty());
    234   service_prefs_->WritePrefs();
    235   // When this switch used we don't want connector continue running, we just
    236   // need authentication.
    237   if (CommandLine::ForCurrentProcess()->HasSwitch(
    238           switches::kCloudPrintSetupProxy)) {
    239     ShutdownBackend();
    240     if (client_) {
    241       client_->OnCloudPrintProxyDisabled(false);
    242     }
    243   }
    244 }
    245 
    246 void CloudPrintProxy::OnAuthenticationFailed() {
    247   DCHECK(CalledOnValidThread());
    248   // Don't disable permanently. Could be just connection issue.
    249   ShutdownBackend();
    250   if (client_) {
    251     client_->OnCloudPrintProxyDisabled(false);
    252   }
    253 }
    254 
    255 void CloudPrintProxy::OnPrintSystemUnavailable() {
    256   // If the print system is unavailable, we want to shutdown the proxy and
    257   // disable it non-persistently.
    258   ShutdownBackend();
    259   if (client_) {
    260     client_->OnCloudPrintProxyDisabled(false);
    261   }
    262 }
    263 
    264 void CloudPrintProxy::OnUnregisterPrinters(
    265     const std::string& auth_token,
    266     const std::list<std::string>& printer_ids) {
    267   UMA_HISTOGRAM_COUNTS_10000("CloudPrint.UnregisterPrinters",
    268                              printer_ids.size());
    269   ShutdownBackend();
    270   ConnectorSettings settings;
    271   settings.InitFrom(service_prefs_);
    272   wipeout_.reset(new CloudPrintWipeout(this, settings.server_url()));
    273   wipeout_->UnregisterPrinters(auth_token, printer_ids);
    274 }
    275 
    276 void CloudPrintProxy::OnXmppPingUpdated(int ping_timeout) {
    277   DCHECK(CalledOnValidThread());
    278   service_prefs_->SetInt(prefs::kCloudPrintXmppPingTimeout, ping_timeout);
    279   service_prefs_->WritePrefs();
    280 }
    281 
    282 void CloudPrintProxy::OnUnregisterPrintersComplete() {
    283   wipeout_.reset();
    284   // Finish disabling cloud print for this user.
    285   DisableForUser();
    286 }
    287 
    288 void CloudPrintProxy::ShutdownBackend() {
    289   DCHECK(CalledOnValidThread());
    290   if (backend_.get())
    291     backend_->Shutdown();
    292   backend_.reset();
    293 }
    294 
    295 }  // namespace cloud_print
    296