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/sync/sync_ui_util.h"
6
7 #include "base/i18n/number_formatting.h"
8 #include "base/i18n/time_formatting.h"
9 #include "base/prefs/pref_service.h"
10 #include "base/strings/string_util.h"
11 #include "base/strings/utf_string_conversions.h"
12 #include "chrome/browser/profiles/profile.h"
13 #include "chrome/browser/profiles/profile_manager.h"
14 #include "chrome/browser/signin/profile_oauth2_token_service_factory.h"
15 #include "chrome/browser/signin/signin_ui_util.h"
16 #include "chrome/browser/sync/profile_sync_service.h"
17 #include "chrome/browser/sync/profile_sync_service_factory.h"
18 #include "chrome/browser/ui/browser.h"
19 #include "chrome/browser/ui/browser_window.h"
20 #include "chrome/browser/ui/webui/signin/login_ui_service.h"
21 #include "chrome/browser/ui/webui/signin/login_ui_service_factory.h"
22 #include "chrome/common/chrome_switches.h"
23 #include "chrome/common/pref_names.h"
24 #include "chrome/common/url_constants.h"
25 #include "chrome/grit/chromium_strings.h"
26 #include "chrome/grit/generated_resources.h"
27 #include "chrome/grit/locale_settings.h"
28 #include "components/signin/core/browser/profile_oauth2_token_service.h"
29 #include "components/signin/core/browser/signin_error_controller.h"
30 #include "components/signin/core/browser/signin_manager_base.h"
31 #include "google_apis/gaia/google_service_auth_error.h"
32 #include "sync/internal_api/public/base/model_type.h"
33 #include "sync/internal_api/public/sessions/sync_session_snapshot.h"
34 #include "sync/protocol/proto_enum_conversions.h"
35 #include "sync/protocol/sync_protocol_error.h"
36 #include "ui/base/l10n/l10n_util.h"
37
38 #if defined(OS_CHROMEOS)
39 #include "components/user_manager/user_manager.h"
40 #endif // defined(OS_CHROMEOS)
41
42 typedef GoogleServiceAuthError AuthError;
43
44 namespace sync_ui_util {
45
46 namespace {
47
48 // Returns the message that should be displayed when the user is authenticated
49 // and can connect to the sync server. If the user hasn't yet authenticated, an
50 // empty string is returned.
GetSyncedStateStatusLabel(ProfileSyncService * service,const SigninManagerBase & signin,StatusLabelStyle style)51 base::string16 GetSyncedStateStatusLabel(ProfileSyncService* service,
52 const SigninManagerBase& signin,
53 StatusLabelStyle style) {
54 std::string user_display_name = signin.GetAuthenticatedUsername();
55
56 #if defined(OS_CHROMEOS)
57 if (user_manager::UserManager::IsInitialized()) {
58 // On CrOS user email is sanitized and then passed to the signin manager.
59 // Original email (containing dots) is stored as "display email".
60 user_display_name = user_manager::UserManager::Get()->GetUserDisplayEmail(
61 user_display_name);
62 }
63 #endif // defined(OS_CHROMEOS)
64
65 base::string16 user_name = base::UTF8ToUTF16(user_display_name);
66
67 if (!user_name.empty()) {
68 if (!service || service->IsManaged()) {
69 // User is signed in, but sync is disabled.
70 return l10n_util::GetStringFUTF16(IDS_SIGNED_IN_WITH_SYNC_DISABLED,
71 user_name);
72 } else if (service->IsStartSuppressed()) {
73 // User is signed in, but sync has been stopped.
74 return l10n_util::GetStringFUTF16(IDS_SIGNED_IN_WITH_SYNC_SUPPRESSED,
75 user_name);
76 }
77 }
78
79 if (!service || !service->sync_initialized()) {
80 // User is not signed in, or sync is still initializing.
81 return base::string16();
82 }
83
84 DCHECK(!user_name.empty());
85
86 // Message may also carry additional advice with an HTML link, if acceptable.
87 switch (style) {
88 case PLAIN_TEXT:
89 return l10n_util::GetStringFUTF16(
90 IDS_SYNC_ACCOUNT_SYNCING_TO_USER,
91 user_name);
92 case WITH_HTML:
93 return l10n_util::GetStringFUTF16(
94 IDS_SYNC_ACCOUNT_SYNCING_TO_USER_WITH_MANAGE_LINK,
95 user_name,
96 base::ASCIIToUTF16(chrome::kSyncGoogleDashboardURL));
97 default:
98 NOTREACHED();
99 return NULL;
100 }
101 }
102
GetStatusForActionableError(const syncer::SyncProtocolError & error,base::string16 * status_label)103 void GetStatusForActionableError(
104 const syncer::SyncProtocolError& error,
105 base::string16* status_label) {
106 DCHECK(status_label);
107 switch (error.action) {
108 case syncer::STOP_AND_RESTART_SYNC:
109 status_label->assign(
110 l10n_util::GetStringUTF16(IDS_SYNC_STOP_AND_RESTART_SYNC));
111 break;
112 case syncer::UPGRADE_CLIENT:
113 status_label->assign(
114 l10n_util::GetStringFUTF16(IDS_SYNC_UPGRADE_CLIENT,
115 l10n_util::GetStringUTF16(IDS_PRODUCT_NAME)));
116 break;
117 case syncer::ENABLE_SYNC_ON_ACCOUNT:
118 status_label->assign(
119 l10n_util::GetStringUTF16(IDS_SYNC_ENABLE_SYNC_ON_ACCOUNT));
120 break;
121 case syncer::CLEAR_USER_DATA_AND_RESYNC:
122 status_label->assign(
123 l10n_util::GetStringUTF16(IDS_SYNC_CLEAR_USER_DATA));
124 break;
125 default:
126 NOTREACHED();
127 }
128 }
129
130 // TODO(akalin): Write unit tests for these three functions below.
131
132 // status_label and link_label must either be both NULL or both non-NULL.
GetStatusInfo(ProfileSyncService * service,const SigninManagerBase & signin,StatusLabelStyle style,base::string16 * status_label,base::string16 * link_label)133 MessageType GetStatusInfo(ProfileSyncService* service,
134 const SigninManagerBase& signin,
135 StatusLabelStyle style,
136 base::string16* status_label,
137 base::string16* link_label) {
138 DCHECK_EQ(status_label == NULL, link_label == NULL);
139
140 MessageType result_type(SYNCED);
141
142 if (!signin.IsAuthenticated())
143 return PRE_SYNCED;
144
145 if (!service || service->IsManaged() || service->HasSyncSetupCompleted() ||
146 service->IsStartSuppressed()) {
147 // The order or priority is going to be: 1. Unrecoverable errors.
148 // 2. Auth errors. 3. Protocol errors. 4. Passphrase errors.
149
150 if (service && service->HasUnrecoverableError()) {
151 if (status_label) {
152 status_label->assign(l10n_util::GetStringFUTF16(
153 IDS_SYNC_STATUS_UNRECOVERABLE_ERROR,
154 l10n_util::GetStringUTF16(IDS_SYNC_UNRECOVERABLE_ERROR_HELP_URL)));
155 }
156 return SYNC_ERROR;
157 }
158
159 // For auth errors first check if an auth is in progress.
160 if (signin.AuthInProgress()) {
161 if (status_label) {
162 status_label->assign(
163 l10n_util::GetStringUTF16(IDS_SYNC_AUTHENTICATING_LABEL));
164 }
165 return PRE_SYNCED;
166 }
167
168 // Check for sync errors if the sync service is enabled.
169 if (service) {
170 // Since there is no auth in progress, check for an auth error first.
171 AuthError auth_error =
172 ProfileOAuth2TokenServiceFactory::GetForProfile(service->profile())->
173 signin_error_controller()->auth_error();
174 if (auth_error.state() != AuthError::NONE) {
175 if (status_label && link_label)
176 signin_ui_util::GetStatusLabelsForAuthError(
177 service->profile(), signin, status_label, link_label);
178 return SYNC_ERROR;
179 }
180
181 // We don't have an auth error. Check for an actionable error.
182 ProfileSyncService::Status status;
183 service->QueryDetailedSyncStatus(&status);
184 if (ShouldShowActionOnUI(status.sync_protocol_error)) {
185 if (status_label) {
186 GetStatusForActionableError(status.sync_protocol_error,
187 status_label);
188 }
189 return SYNC_ERROR;
190 }
191
192 // Check for a passphrase error.
193 if (service->IsPassphraseRequired()) {
194 if (service->IsPassphraseRequiredForDecryption()) {
195 // TODO(lipalani) : Ask tim if this is still needed.
196 // NOT first machine.
197 // Show a link ("needs attention"), but still indicate the
198 // current synced status. Return SYNC_PROMO so that
199 // the configure link will still be shown.
200 if (status_label && link_label) {
201 status_label->assign(GetSyncedStateStatusLabel(
202 service, signin, style));
203 link_label->assign(
204 l10n_util::GetStringUTF16(IDS_SYNC_PASSWORD_SYNC_ATTENTION));
205 }
206 return SYNC_PROMO;
207 }
208 }
209
210 // Check to see if sync has been disabled via the dasboard and needs to be
211 // set up once again.
212 if (service->IsStartSuppressed() &&
213 status.sync_protocol_error.error_type == syncer::NOT_MY_BIRTHDAY) {
214 if (status_label) {
215 status_label->assign(GetSyncedStateStatusLabel(service,
216 signin,
217 style));
218 }
219 return PRE_SYNCED;
220 }
221 }
222
223 // There is no error. Display "Last synced..." message.
224 if (status_label)
225 status_label->assign(GetSyncedStateStatusLabel(service, signin, style));
226 return SYNCED;
227 } else {
228 // Either show auth error information with a link to re-login, auth in prog,
229 // or provide a link to continue with setup.
230 if (service->FirstSetupInProgress()) {
231 result_type = PRE_SYNCED;
232 ProfileSyncService::Status status;
233 service->QueryDetailedSyncStatus(&status);
234 AuthError auth_error =
235 ProfileOAuth2TokenServiceFactory::GetForProfile(service->profile())->
236 signin_error_controller()->auth_error();
237 if (status_label) {
238 status_label->assign(
239 l10n_util::GetStringUTF16(IDS_SYNC_NTP_SETUP_IN_PROGRESS));
240 }
241 if (signin.AuthInProgress()) {
242 if (status_label) {
243 status_label->assign(
244 l10n_util::GetStringUTF16(IDS_SYNC_AUTHENTICATING_LABEL));
245 }
246 } else if (auth_error.state() != AuthError::NONE &&
247 auth_error.state() != AuthError::TWO_FACTOR) {
248 if (status_label && link_label) {
249 status_label->clear();
250 signin_ui_util::GetStatusLabelsForAuthError(
251 service->profile(), signin, status_label, link_label);
252 }
253 result_type = SYNC_ERROR;
254 }
255 } else if (service->HasUnrecoverableError()) {
256 result_type = SYNC_ERROR;
257 ProfileSyncService::Status status;
258 service->QueryDetailedSyncStatus(&status);
259 if (ShouldShowActionOnUI(status.sync_protocol_error)) {
260 if (status_label) {
261 GetStatusForActionableError(status.sync_protocol_error,
262 status_label);
263 }
264 } else if (status_label) {
265 status_label->assign(l10n_util::GetStringUTF16(IDS_SYNC_SETUP_ERROR));
266 }
267 } else if (signin.IsAuthenticated()) {
268 // The user is signed in, but sync has been stopped.
269 if (status_label) {
270 base::string16 label = l10n_util::GetStringFUTF16(
271 IDS_SIGNED_IN_WITH_SYNC_SUPPRESSED,
272 base::UTF8ToUTF16(signin.GetAuthenticatedUsername()));
273 status_label->assign(label);
274 result_type = PRE_SYNCED;
275 }
276 }
277 }
278 return result_type;
279 }
280
281 // Returns the status info for use on the new tab page, where we want slightly
282 // different information than in the settings panel.
GetStatusInfoForNewTabPage(ProfileSyncService * service,const SigninManagerBase & signin,base::string16 * status_label,base::string16 * link_label)283 MessageType GetStatusInfoForNewTabPage(ProfileSyncService* service,
284 const SigninManagerBase& signin,
285 base::string16* status_label,
286 base::string16* link_label) {
287 DCHECK(status_label);
288 DCHECK(link_label);
289
290 if (service->HasSyncSetupCompleted() &&
291 service->IsPassphraseRequired()) {
292 if (service->passphrase_required_reason() == syncer::REASON_ENCRYPTION) {
293 // First machine migrating to passwords. Show as a promotion.
294 if (status_label && link_label) {
295 status_label->assign(
296 l10n_util::GetStringFUTF16(
297 IDS_SYNC_NTP_PASSWORD_PROMO,
298 l10n_util::GetStringUTF16(IDS_PRODUCT_NAME)));
299 link_label->assign(
300 l10n_util::GetStringUTF16(IDS_SYNC_NTP_PASSWORD_ENABLE));
301 }
302 return SYNC_PROMO;
303 } else {
304 // NOT first machine.
305 // Show a link and present as an error ("needs attention").
306 if (status_label && link_label) {
307 status_label->assign(base::string16());
308 link_label->assign(
309 l10n_util::GetStringUTF16(IDS_SYNC_CONFIGURE_ENCRYPTION));
310 }
311 return SYNC_ERROR;
312 }
313 }
314
315 // Fallback to default.
316 return GetStatusInfo(service, signin, WITH_HTML, status_label, link_label);
317 }
318
319 } // namespace
320
GetStatusLabels(ProfileSyncService * service,const SigninManagerBase & signin,StatusLabelStyle style,base::string16 * status_label,base::string16 * link_label)321 MessageType GetStatusLabels(ProfileSyncService* service,
322 const SigninManagerBase& signin,
323 StatusLabelStyle style,
324 base::string16* status_label,
325 base::string16* link_label) {
326 DCHECK(status_label);
327 DCHECK(link_label);
328 return sync_ui_util::GetStatusInfo(
329 service, signin, style, status_label, link_label);
330 }
331
GetStatusLabelsForNewTabPage(ProfileSyncService * service,const SigninManagerBase & signin,base::string16 * status_label,base::string16 * link_label)332 MessageType GetStatusLabelsForNewTabPage(ProfileSyncService* service,
333 const SigninManagerBase& signin,
334 base::string16* status_label,
335 base::string16* link_label) {
336 DCHECK(status_label);
337 DCHECK(link_label);
338 return sync_ui_util::GetStatusInfoForNewTabPage(
339 service, signin, status_label, link_label);
340 }
341
342 #if !defined(OS_CHROMEOS)
GetStatusLabelsForSyncGlobalError(const ProfileSyncService * service,base::string16 * menu_label,base::string16 * bubble_message,base::string16 * bubble_accept_label)343 void GetStatusLabelsForSyncGlobalError(const ProfileSyncService* service,
344 base::string16* menu_label,
345 base::string16* bubble_message,
346 base::string16* bubble_accept_label) {
347 DCHECK(menu_label);
348 DCHECK(bubble_message);
349 DCHECK(bubble_accept_label);
350 *menu_label = base::string16();
351 *bubble_message = base::string16();
352 *bubble_accept_label = base::string16();
353
354 // Only display an error if we've completed sync setup.
355 if (!service->HasSyncSetupCompleted())
356 return;
357
358 // Display a passphrase error if we have one.
359 if (service->IsPassphraseRequired() &&
360 service->IsPassphraseRequiredForDecryption()) {
361 // This is not the first machine so ask user to enter passphrase.
362 *menu_label = l10n_util::GetStringUTF16(
363 IDS_SYNC_PASSPHRASE_ERROR_WRENCH_MENU_ITEM);
364 *bubble_message = l10n_util::GetStringUTF16(
365 IDS_SYNC_PASSPHRASE_ERROR_BUBBLE_VIEW_MESSAGE);
366 *bubble_accept_label = l10n_util::GetStringUTF16(
367 IDS_SYNC_PASSPHRASE_ERROR_BUBBLE_VIEW_ACCEPT);
368 return;
369 }
370 }
371 #endif
372
GetStatus(ProfileSyncService * service,const SigninManagerBase & signin)373 MessageType GetStatus(
374 ProfileSyncService* service, const SigninManagerBase& signin) {
375 return sync_ui_util::GetStatusInfo(service, signin, WITH_HTML, NULL, NULL);
376 }
377
ConstructTime(int64 time_in_int)378 base::string16 ConstructTime(int64 time_in_int) {
379 base::Time time = base::Time::FromInternalValue(time_in_int);
380
381 // If time is null the format function returns a time in 1969.
382 if (time.is_null())
383 return base::string16();
384 return base::TimeFormatFriendlyDateAndTime(time);
385 }
386
387 } // namespace sync_ui_util
388