about_signin_internals.cc revision 5c02ac1a9c1b504631c0a3d2b6e737b5d738bae1
12a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
22a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
32a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// found in the LICENSE file.
42a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
5e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch#include "components/signin/core/browser/about_signin_internals.h"
62a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
70529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch#include "base/command_line.h"
82a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/debug/trace_event.h"
92a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/hash.h"
102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/i18n/time_formatting.h"
112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/logging.h"
122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/prefs/pref_service.h"
135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)#include "base/strings/stringprintf.h"
14868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/utf_string_conversions.h"
15effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch#include "components/signin/core/browser/profile_oauth2_token_service.h"
16e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch#include "components/signin/core/browser/signin_client.h"
17effb81e5f8246d0db0270817048dc992db66e9fbBen Murdoch#include "components/signin/core/browser/signin_internals_util.h"
18e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch#include "components/signin/core/browser/signin_manager.h"
190529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch#include "components/signin/core/common/profile_management_switches.h"
200529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch#include "components/signin/core/common/signin_switches.h"
212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "google_apis/gaia/gaia_constants.h"
222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
23c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)using base::Time;
242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)using namespace signin_internals_util;
252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)namespace {
275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)std::string GetTimeStr(base::Time time) {
295d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return base::UTF16ToUTF8(base::TimeFormatShortDateAndTime(time));
305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::ListValue* AddSection(base::ListValue* parent_list,
335d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                            const std::string& title) {
345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::DictionaryValue> section(new base::DictionaryValue());
355d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* section_contents = new base::ListValue();
365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  section->SetString("title", title);
385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  section->Set("data", section_contents);
395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  parent_list->Append(section.release());
405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return section_contents;
415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
425d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AddSectionEntry(base::ListValue* section_list,
445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                     const std::string& field_name,
455c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                     const std::string& field_status,
465c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                     const std::string& field_time = "") {
475d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::DictionaryValue> entry(new base::DictionaryValue());
485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  entry->SetString("label", field_name);
495c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  entry->SetString("status", field_status);
505c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  entry->SetString("time", field_time);
515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  section_list->Append(entry.release());
525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)std::string SigninStatusFieldToLabel(UntimedSigninStatusField field) {
555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  switch (field) {
565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case USERNAME:
575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      return "User Id";
585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case UNTIMED_FIELDS_END:
595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      NOTREACHED();
605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      return std::string();
615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  NOTREACHED();
635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return std::string();
645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
665c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liustd::string SigninStatusFieldToLabel(TimedSigninStatusField field) {
675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  switch (field) {
685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case SIGNIN_TYPE:
695c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Type";
705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case CLIENT_LOGIN_STATUS:
715c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Last OnClientLogin Status";
725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case OAUTH_LOGIN_STATUS:
735c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Last OnOAuthLogin Status";
745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case GET_USER_INFO_STATUS:
755c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Last OnGetUserInfo Status";
765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case UBER_TOKEN_STATUS:
775c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Last OnUberToken Status";
785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case MERGE_SESSION_STATUS:
795c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Last OnMergeSession Status";
805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case TIMED_FIELDS_END:
815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      NOTREACHED();
825c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu      return "Error";
835d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  NOTREACHED();
855c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu  return "Error";
865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}  // anonymous namespace
895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
90e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::AboutSigninInternals(
91e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    ProfileOAuth2TokenService* token_service,
92e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    SigninManagerBase* signin_manager)
93e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    : token_service_(token_service),
94e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      signin_manager_(signin_manager),
95e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      client_(NULL) {}
962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
97e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::~AboutSigninInternals() {}
982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::AddSigninObserver(
1002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    AboutSigninInternals::Observer* observer) {
1012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  signin_observers_.AddObserver(observer);
1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::RemoveSigninObserver(
1052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    AboutSigninInternals::Observer* observer) {
1062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  signin_observers_.RemoveObserver(observer);
1072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::NotifySigninValueChanged(
1102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const UntimedSigninStatusField& field,
1112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& value) {
1122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  unsigned int field_index = field - UNTIMED_FIELDS_BEGIN;
1132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(field_index >= 0 &&
1142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)         field_index < signin_status_.untimed_signin_fields.size());
1152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  signin_status_.untimed_signin_fields[field_index] = value;
1172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Also persist these values in the prefs.
1192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const std::string pref_path = SigninStatusFieldToString(field);
120e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  client_->GetPrefs()->SetString(pref_path.c_str(), value);
1212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  NotifyObservers();
1232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1242a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::NotifySigninValueChanged(
1262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const TimedSigninStatusField& field,
1272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& value) {
1282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  unsigned int field_index = field - TIMED_FIELDS_BEGIN;
1292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(field_index >= 0 &&
1302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)         field_index < signin_status_.timed_signin_fields.size());
1312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
132c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  Time now = Time::NowFromSystemTime();
1335d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::string time_as_str =
1345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::UTF16ToUTF8(base::TimeFormatFriendlyDate(now));
1352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  TimedSigninStatusValue timed_value(value, time_as_str);
1362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  signin_status_.timed_signin_fields[field_index] = timed_value;
1382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Also persist these values in the prefs.
1402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const std::string value_pref = SigninStatusFieldToString(field) + ".value";
141c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  const std::string time_pref = SigninStatusFieldToString(field) + ".time";
142e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  client_->GetPrefs()->SetString(value_pref.c_str(), value);
143e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  client_->GetPrefs()->SetString(time_pref.c_str(), time_as_str);
1442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  NotifyObservers();
1462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::RefreshSigninPrefs() {
149e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  // Return if no client exists. Can occur in unit tests.
150e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  if (!client_)
1512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return;
1522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
153e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  PrefService* pref_service = client_->GetPrefs();
1542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  for (int i = UNTIMED_FIELDS_BEGIN; i < UNTIMED_FIELDS_END; ++i) {
1552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string pref_path =
1562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        SigninStatusFieldToString(static_cast<UntimedSigninStatusField>(i));
1572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    signin_status_.untimed_signin_fields[i - UNTIMED_FIELDS_BEGIN] =
1592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        pref_service->GetString(pref_path.c_str());
1602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
161e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  for (int i = TIMED_FIELDS_BEGIN; i < TIMED_FIELDS_END; ++i) {
162e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    const std::string value_pref =
163e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        SigninStatusFieldToString(static_cast<TimedSigninStatusField>(i)) +
164e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        ".value";
165e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    const std::string time_pref =
166e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        SigninStatusFieldToString(static_cast<TimedSigninStatusField>(i)) +
167e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        ".time";
1682a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    TimedSigninStatusValue value(pref_service->GetString(value_pref.c_str()),
1702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                 pref_service->GetString(time_pref.c_str()));
1712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    signin_status_.timed_signin_fields[i - TIMED_FIELDS_BEGIN] = value;
1722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1744e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  // TODO(rogerta): Get status and timestamps for oauth2 tokens.
1752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  NotifyObservers();
1772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
179e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid AboutSigninInternals::Initialize(SigninClient* client) {
180e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  DCHECK(!client_);
181e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  client_ = client;
1822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  RefreshSigninPrefs();
1842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
185e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  signin_manager_->AddSigninDiagnosticsObserver(this);
186e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  token_service_->AddDiagnosticsObserver(this);
1872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::Shutdown() {
190e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  signin_manager_->RemoveSigninDiagnosticsObserver(this);
191e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  token_service_->RemoveDiagnosticsObserver(this);
1922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::NotifyObservers() {
1952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  FOR_EACH_OBSERVER(AboutSigninInternals::Observer,
1962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                    signin_observers_,
197e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch                    OnSigninStateChanged(
198e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch                        signin_status_.ToValue(client_->GetProductVersion())));
1992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
2002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2015d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)scoped_ptr<base::DictionaryValue> AboutSigninInternals::GetSigninStatus() {
202e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  return signin_status_.ToValue(client_->GetProductVersion()).Pass();
2032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
204c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
2055d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AboutSigninInternals::OnAccessTokenRequested(
2065d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& account_id,
2075d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& consumer_id,
2085d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes) {
2095d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  TokenInfo* token = signin_status_.FindToken(account_id, consumer_id, scopes);
2105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (token) {
2115d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    *token = TokenInfo(consumer_id, scopes);
2125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  } else {
2135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    token = new TokenInfo(consumer_id, scopes);
2145d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    signin_status_.token_info_map[account_id].push_back(token);
2155d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
2165d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2175d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  NotifyObservers();
2185d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
2195d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2205d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AboutSigninInternals::OnFetchAccessTokenComplete(
2215d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& account_id,
2225d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& consumer_id,
2235d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes,
2245d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    GoogleServiceAuthError error,
2255d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    base::Time expiration_time) {
2265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  TokenInfo* token = signin_status_.FindToken(account_id, consumer_id, scopes);
2275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (!token) {
2285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    DVLOG(1) << "Can't find token: " << account_id << ", " << consumer_id;
2295d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    return;
2305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
2315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token->receive_time = base::Time::Now();
2335d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token->error = error;
2345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token->expiration_time = expiration_time;
2355d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  NotifyObservers();
2375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
2385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AboutSigninInternals::OnTokenRemoved(
2405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& account_id,
2415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes) {
2425d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (size_t i = 0; i < signin_status_.token_info_map[account_id].size();
243e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++i) {
2445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    TokenInfo* token = signin_status_.token_info_map[account_id][i];
2455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (token->scopes == scopes)
2465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token->Invalidate();
2475d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
2485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  NotifyObservers();
2495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
2505d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::TokenInfo::TokenInfo(
2525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& consumer_id,
2535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes)
2545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    : consumer_id(consumer_id),
2555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      scopes(scopes),
2565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      request_time(base::Time::Now()),
2575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      error(GoogleServiceAuthError::AuthErrorNone()),
258e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      removed_(false) {}
2595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::TokenInfo::~TokenInfo() {}
2615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)bool AboutSigninInternals::TokenInfo::LessThan(const TokenInfo* a,
2635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                                               const TokenInfo* b) {
2645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return a->consumer_id < b->consumer_id || a->scopes < b->scopes;
2655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
2665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
267e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid AboutSigninInternals::TokenInfo::Invalidate() { removed_ = true; }
2685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::DictionaryValue* AboutSigninInternals::TokenInfo::ToValue() const {
2705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::DictionaryValue> token_info(new base::DictionaryValue());
2715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token_info->SetString("service", consumer_id);
2725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::string scopes_str;
2745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (OAuth2TokenService::ScopeSet::const_iterator it = scopes.begin();
275e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       it != scopes.end();
276e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++it) {
2775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    scopes_str += *it + "<br/>";
2785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
2795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token_info->SetString("scopes", scopes_str);
2805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token_info->SetString("request_time", GetTimeStr(request_time).c_str());
2815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2825d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (removed_) {
2835d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    token_info->SetString("status", "Token was revoked.");
2845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  } else if (!receive_time.is_null()) {
2855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (error == GoogleServiceAuthError::AuthErrorNone()) {
2865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      bool token_expired = expiration_time < base::Time::Now();
2875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      std::string status_str = "";
2885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if (token_expired)
2895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        status_str = "<p style=\"color: #ffffff; background-color: #ff0000\">";
2905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::StringAppendF(&status_str,
2915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                          "Received token at %s. Expire at %s",
2925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                          GetTimeStr(receive_time).c_str(),
2935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                          GetTimeStr(expiration_time).c_str());
2945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if (token_expired)
2955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        base::StringAppendF(&status_str, "</p>");
2965d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token_info->SetString("status", status_str);
2975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    } else {
2985d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token_info->SetString(
2995d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)          "status",
300a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)          base::StringPrintf("Failure: %s", error.ToString().c_str()));
3015d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
3025d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  } else {
3035d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    token_info->SetString("status", "Waiting for response");
3045d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3055d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3065d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return token_info.release();
3075d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
3085d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3095d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::SigninStatus::SigninStatus()
310e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    : untimed_signin_fields(UNTIMED_FIELDS_COUNT),
311e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      timed_signin_fields(TIMED_FIELDS_COUNT) {}
3125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::SigninStatus::~SigninStatus() {
3145d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (TokenInfoMap::iterator it = token_info_map.begin();
315e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       it != token_info_map.end();
316e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++it) {
3175d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    STLDeleteElements(&it->second);
3185d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3195d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
3205d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
321e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::TokenInfo* AboutSigninInternals::SigninStatus::FindToken(
3225d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& account_id,
3235d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& consumer_id,
3245d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes) {
325e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  for (size_t i = 0; i < token_info_map[account_id].size(); ++i) {
3265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    TokenInfo* tmp = token_info_map[account_id][i];
3275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (tmp->consumer_id == consumer_id && tmp->scopes == scopes)
3285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      return tmp;
3295d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return NULL;
331c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)}
3325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
333e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochscoped_ptr<base::DictionaryValue> AboutSigninInternals::SigninStatus::ToValue(
334e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    std::string product_version) {
3355d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::DictionaryValue> signin_status(new base::DictionaryValue());
3365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* signin_info = new base::ListValue();
3375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  signin_status->Set("signin_info", signin_info);
3385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // A summary of signin related info first.
3405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* basic_info = AddSection(signin_info, "Basic Information");
3415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  const std::string signin_status_string =
342e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      untimed_signin_fields[USERNAME - UNTIMED_FIELDS_BEGIN].empty()
343e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch          ? "Not Signed In"
344e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch          : "Signed In";
345e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  AddSectionEntry(basic_info, "Chrome Version", product_version);
3465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  AddSectionEntry(basic_info, "Signin Status", signin_status_string);
3470529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "Web Based Signin Enabled?",
3480529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      switches::IsEnableWebBasedSignin() == true ? "True" : "False");
3490529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "New Profile Management Enabled?",
3500529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      switches::IsNewProfileManagement() == true ? "True" : "False");
3510529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "New Avatar Menu Enabled?",
3520529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      switches::IsNewAvatarMenu() == true ? "True" : "False");
3530529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  bool new_avatar_menu_flag =
3540529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      CommandLine::ForCurrentProcess()->HasSwitch(switches::kNewAvatarMenu);
3550529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "New Avatar Menu Flag Set?",
3560529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      new_avatar_menu_flag ? "True" : "False");
3575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // Only add username.  SID and LSID have moved to tokens section.
3595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  const std::string field =
3605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      SigninStatusFieldToLabel(static_cast<UntimedSigninStatusField>(USERNAME));
361e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  AddSectionEntry(basic_info,
362e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch                  field,
363e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch                  untimed_signin_fields[USERNAME - UNTIMED_FIELDS_BEGIN]);
3645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // Time and status information of the possible sign in types.
366e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  base::ListValue* detailed_info =
367e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      AddSection(signin_info, "Last Signin Details");
3685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (int i = TIMED_FIELDS_BEGIN; i < TIMED_FIELDS_END; ++i) {
3695c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    const std::string status_field_label =
3705c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        SigninStatusFieldToLabel(static_cast<TimedSigninStatusField>(i));
3715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
372e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    AddSectionEntry(detailed_info,
3735c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                    status_field_label,
3745c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                    timed_signin_fields[i - TIMED_FIELDS_BEGIN].first,
3755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                    timed_signin_fields[i - TIMED_FIELDS_BEGIN].second);
3765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // Token information for all services.
3795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* token_info = new base::ListValue();
3805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  signin_status->Set("token_info", token_info);
3815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (TokenInfoMap::iterator it = token_info_map.begin();
382e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       it != token_info_map.end();
383e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++it) {
3845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    base::ListValue* token_details = AddSection(token_info, it->first);
3855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    std::sort(it->second.begin(), it->second.end(), TokenInfo::LessThan);
3875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::vector<TokenInfo*>& tokens = it->second;
3885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    for (size_t i = 0; i < tokens.size(); ++i) {
3895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::DictionaryValue* token_info = tokens[i]->ToValue();
3905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token_details->Append(token_info);
3915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
3925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return signin_status.Pass();
3955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
396