Home | History | Annotate | Download | only in signin
      1 // Copyright (c) 2013 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/signin/signin_global_error.h"
      6 
      7 #include "base/logging.h"
      8 #include "base/prefs/pref_service.h"
      9 #include "chrome/app/chrome_command_ids.h"
     10 #include "chrome/browser/profiles/profile.h"
     11 #include "chrome/browser/signin/profile_oauth2_token_service.h"
     12 #include "chrome/browser/signin/profile_oauth2_token_service_factory.h"
     13 #include "chrome/browser/signin/signin_manager.h"
     14 #include "chrome/browser/signin/signin_manager_factory.h"
     15 #include "chrome/browser/ui/browser_commands.h"
     16 #include "chrome/browser/ui/chrome_pages.h"
     17 #include "chrome/browser/ui/global_error/global_error_service.h"
     18 #include "chrome/browser/ui/global_error/global_error_service_factory.h"
     19 #include "chrome/browser/ui/webui/signin/login_ui_service.h"
     20 #include "chrome/browser/ui/webui/signin/login_ui_service_factory.h"
     21 #include "chrome/common/pref_names.h"
     22 #include "chrome/common/url_constants.h"
     23 #include "grit/chromium_strings.h"
     24 #include "grit/generated_resources.h"
     25 #include "ui/base/l10n/l10n_util.h"
     26 
     27 SigninGlobalError::SigninGlobalError(Profile* profile)
     28     : auth_error_(GoogleServiceAuthError::AuthErrorNone()), profile_(profile) {
     29 }
     30 
     31 SigninGlobalError::~SigninGlobalError() {
     32   DCHECK(provider_set_.empty())
     33       << "All AuthStatusProviders should be unregistered before"
     34       << " SigninManager::Shutdown() is called";
     35 }
     36 
     37 void SigninGlobalError::AddProvider(const AuthStatusProvider* provider) {
     38   DCHECK(provider_set_.find(provider) == provider_set_.end())
     39       << "Adding same AuthStatusProvider multiple times";
     40   provider_set_.insert(provider);
     41   AuthStatusChanged();
     42 }
     43 
     44 void SigninGlobalError::RemoveProvider(const AuthStatusProvider* provider) {
     45   std::set<const AuthStatusProvider*>::iterator iter =
     46       provider_set_.find(provider);
     47   DCHECK(iter != provider_set_.end())
     48       << "Removing provider that was never added";
     49   provider_set_.erase(iter);
     50   AuthStatusChanged();
     51 }
     52 
     53 SigninGlobalError::AuthStatusProvider::AuthStatusProvider() {
     54 }
     55 
     56 SigninGlobalError::AuthStatusProvider::~AuthStatusProvider() {
     57 }
     58 
     59 void SigninGlobalError::AuthStatusChanged() {
     60   // Walk all of the status providers and collect any error.
     61   GoogleServiceAuthError current_error(GoogleServiceAuthError::AuthErrorNone());
     62   std::string current_account_id;
     63   for (std::set<const AuthStatusProvider*>::const_iterator it =
     64            provider_set_.begin(); it != provider_set_.end(); ++it) {
     65     current_account_id = (*it)->GetAccountId();
     66     current_error = (*it)->GetAuthStatus();
     67 
     68     // Break out if any provider reports an error (ignoring ordinary network
     69     // errors, which are not surfaced to the user). This logic may eventually
     70     // need to be extended to prioritize different auth errors, but for now
     71     // all auth errors are treated the same.
     72     if (current_error.state() != GoogleServiceAuthError::NONE &&
     73         current_error.state() != GoogleServiceAuthError::CONNECTION_FAILED) {
     74       break;
     75     }
     76   }
     77   if (current_error.state() != auth_error_.state() ||
     78       account_id_ != current_account_id) {
     79     auth_error_ = current_error;
     80     if (auth_error_.state() == GoogleServiceAuthError::NONE) {
     81       account_id_.clear();
     82     } else {
     83       account_id_ = current_account_id;
     84     }
     85 
     86     GlobalErrorServiceFactory::GetForProfile(profile_)->NotifyErrorsChanged(
     87         this);
     88   }
     89 }
     90 
     91 bool SigninGlobalError::HasMenuItem() {
     92   return !MenuItemLabel().empty();
     93 }
     94 
     95 int SigninGlobalError::MenuItemCommandID() {
     96   return IDC_SHOW_SIGNIN_ERROR;
     97 }
     98 
     99 base::string16 SigninGlobalError::MenuItemLabel() {
    100   if (account_id_.empty() ||
    101       auth_error_.state() == GoogleServiceAuthError::NONE ||
    102       auth_error_.state() == GoogleServiceAuthError::CONNECTION_FAILED) {
    103     // If the user isn't signed in, or there's no auth error worth elevating to
    104     // the user, don't display any menu item.
    105     return base::string16();
    106   } else {
    107     // There's an auth error the user should know about - notify the user.
    108     return l10n_util::GetStringUTF16(IDS_SYNC_SIGN_IN_ERROR_WRENCH_MENU_ITEM);
    109   }
    110 }
    111 
    112 void SigninGlobalError::ExecuteMenuItem(Browser* browser) {
    113 #if defined(OS_CHROMEOS)
    114   if (auth_error_.state() != GoogleServiceAuthError::NONE) {
    115     DLOG(INFO) << "Signing out the user to fix a sync error.";
    116     // TODO(beng): seems like this could just call chrome::AttemptUserExit().
    117     chrome::ExecuteCommand(browser, IDC_EXIT);
    118     return;
    119   }
    120 #endif
    121 
    122   // TODO(rogerta): what we do depends on which account is reporting an error.
    123   // This will be needed once the account reconcilor is implemented.  The
    124   // LoginUIService will support multi-login as well.
    125 
    126   // Global errors don't show up in the wrench menu on android.
    127 #if !defined(OS_ANDROID)
    128   LoginUIService* login_ui = LoginUIServiceFactory::GetForProfile(profile_);
    129   if (login_ui->current_login_ui()) {
    130     login_ui->current_login_ui()->FocusUI();
    131     return;
    132   }
    133   // Need to navigate to the settings page and display the UI.
    134   chrome::ShowSettingsSubPage(browser, chrome::kSyncSetupSubPage);
    135 #endif
    136 }
    137 
    138 bool SigninGlobalError::HasBubbleView() {
    139   return !GetBubbleViewMessages().empty();
    140 }
    141 
    142 base::string16 SigninGlobalError::GetBubbleViewTitle() {
    143   return l10n_util::GetStringUTF16(IDS_SIGNIN_ERROR_BUBBLE_VIEW_TITLE);
    144 }
    145 
    146 std::vector<base::string16> SigninGlobalError::GetBubbleViewMessages() {
    147   std::vector<base::string16> messages;
    148 
    149   // If the user isn't signed in, no need to display an error bubble.
    150   SigninManagerBase* signin_manager =
    151       SigninManagerFactory::GetForProfileIfExists(profile_);
    152   if (signin_manager) {
    153     std::string username = signin_manager->GetAuthenticatedUsername();
    154     if (username.empty())
    155       return messages;
    156   }
    157 
    158   switch (auth_error_.state()) {
    159     // In the case of no error, or a simple network error, don't bother
    160     // displaying a popup bubble.
    161     case GoogleServiceAuthError::CONNECTION_FAILED:
    162     case GoogleServiceAuthError::NONE:
    163       return messages;
    164 
    165     // TODO(rogerta): use account id in error messages.
    166 
    167     // User credentials are invalid (bad acct, etc).
    168     case GoogleServiceAuthError::INVALID_GAIA_CREDENTIALS:
    169     case GoogleServiceAuthError::SERVICE_ERROR:
    170     case GoogleServiceAuthError::ACCOUNT_DELETED:
    171     case GoogleServiceAuthError::ACCOUNT_DISABLED:
    172       messages.push_back(l10n_util::GetStringFUTF16(
    173           IDS_SYNC_SIGN_IN_ERROR_BUBBLE_VIEW_MESSAGE,
    174           l10n_util::GetStringUTF16(IDS_PRODUCT_NAME)));
    175       break;
    176 
    177     // Sync service is not available for this account's domain.
    178     case GoogleServiceAuthError::SERVICE_UNAVAILABLE:
    179       messages.push_back(l10n_util::GetStringFUTF16(
    180           IDS_SYNC_UNAVAILABLE_ERROR_BUBBLE_VIEW_MESSAGE,
    181           l10n_util::GetStringUTF16(IDS_PRODUCT_NAME)));
    182       break;
    183 
    184     // Generic message for "other" errors.
    185     default:
    186       messages.push_back(l10n_util::GetStringFUTF16(
    187           IDS_SYNC_OTHER_SIGN_IN_ERROR_BUBBLE_VIEW_MESSAGE,
    188           l10n_util::GetStringUTF16(IDS_PRODUCT_NAME)));
    189   }
    190   return messages;
    191 }
    192 
    193 base::string16 SigninGlobalError::GetBubbleViewAcceptButtonLabel() {
    194   // If the service is unavailable, don't give the user the option to try
    195   // signing in again.
    196   if (auth_error_.state() == GoogleServiceAuthError::SERVICE_UNAVAILABLE) {
    197     return l10n_util::GetStringUTF16(
    198         IDS_SYNC_UNAVAILABLE_ERROR_BUBBLE_VIEW_ACCEPT);
    199   } else {
    200     return l10n_util::GetStringUTF16(IDS_SYNC_SIGN_IN_ERROR_BUBBLE_VIEW_ACCEPT);
    201   }
    202 }
    203 
    204 base::string16 SigninGlobalError::GetBubbleViewCancelButtonLabel() {
    205   return base::string16();
    206 }
    207 
    208 void SigninGlobalError::OnBubbleViewDidClose(Browser* browser) {
    209 }
    210 
    211 void SigninGlobalError::BubbleViewAcceptButtonPressed(Browser* browser) {
    212   ExecuteMenuItem(browser);
    213 }
    214 
    215 void SigninGlobalError::BubbleViewCancelButtonPressed(Browser* browser) {
    216   NOTREACHED();
    217 }
    218 
    219 // static
    220 SigninGlobalError* SigninGlobalError::GetForProfile(Profile* profile) {
    221   return ProfileOAuth2TokenServiceFactory::GetForProfile(profile)->
    222       signin_global_error();
    223 }
    224