about_signin_internals.cc revision 010d83a9304c5a91596085d917d248abff47903a
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";
70010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    case AUTHENTICATION_RESULT_RECEIVED:
71010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      return "Last Authentication Result Received";
72010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)    case REFRESH_TOKEN_RECEIVED:
73010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      return "Last RefreshToken Received";
745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case GET_USER_INFO_STATUS:
75010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      return "Last OnGetUserInfo Received";
765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case UBER_TOKEN_STATUS:
77010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      return "Last OnUberToken Received";
785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    case MERGE_SESSION_STATUS:
79010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)      return "Last OnMergeSession Received";
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)
251010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)void AboutSigninInternals::OnRefreshTokenReceived(std::string status) {
252010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)  NotifySigninValueChanged(REFRESH_TOKEN_RECEIVED, status);
253010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)}
254010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)
255010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)void AboutSigninInternals::OnAuthenticationResultReceived(std::string status) {
256010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)  NotifySigninValueChanged(AUTHENTICATION_RESULT_RECEIVED, status);
257010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)}
258010d83a9304c5a91596085d917d248abff47903aTorne (Richard Coles)
2595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::TokenInfo::TokenInfo(
2605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& consumer_id,
2615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes)
2625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    : consumer_id(consumer_id),
2635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      scopes(scopes),
2645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      request_time(base::Time::Now()),
2655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      error(GoogleServiceAuthError::AuthErrorNone()),
266e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      removed_(false) {}
2675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::TokenInfo::~TokenInfo() {}
2695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)bool AboutSigninInternals::TokenInfo::LessThan(const TokenInfo* a,
2715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                                               const TokenInfo* b) {
2725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return a->consumer_id < b->consumer_id || a->scopes < b->scopes;
2735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
2745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
275e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid AboutSigninInternals::TokenInfo::Invalidate() { removed_ = true; }
2765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::DictionaryValue* AboutSigninInternals::TokenInfo::ToValue() const {
2785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::DictionaryValue> token_info(new base::DictionaryValue());
2795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token_info->SetString("service", consumer_id);
2805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  std::string scopes_str;
2825d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (OAuth2TokenService::ScopeSet::const_iterator it = scopes.begin();
283e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       it != scopes.end();
284e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++it) {
2855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    scopes_str += *it + "<br/>";
2865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
2875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token_info->SetString("scopes", scopes_str);
2885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  token_info->SetString("request_time", GetTimeStr(request_time).c_str());
2895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (removed_) {
2915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    token_info->SetString("status", "Token was revoked.");
2925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  } else if (!receive_time.is_null()) {
2935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (error == GoogleServiceAuthError::AuthErrorNone()) {
2945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      bool token_expired = expiration_time < base::Time::Now();
2955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      std::string status_str = "";
2965d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if (token_expired)
2975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        status_str = "<p style=\"color: #ffffff; background-color: #ff0000\">";
2985d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::StringAppendF(&status_str,
2995d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                          "Received token at %s. Expire at %s",
3005d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                          GetTimeStr(receive_time).c_str(),
3015d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                          GetTimeStr(expiration_time).c_str());
3025d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      if (token_expired)
3035d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)        base::StringAppendF(&status_str, "</p>");
3045d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token_info->SetString("status", status_str);
3055d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    } else {
3065d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token_info->SetString(
3075d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)          "status",
308a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)          base::StringPrintf("Failure: %s", error.ToString().c_str()));
3095d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
3105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  } else {
3115d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    token_info->SetString("status", "Waiting for response");
3125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3145d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return token_info.release();
3155d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
3165d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3175d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::SigninStatus::SigninStatus()
318e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    : untimed_signin_fields(UNTIMED_FIELDS_COUNT),
319e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      timed_signin_fields(TIMED_FIELDS_COUNT) {}
3205d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3215d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::SigninStatus::~SigninStatus() {
3225d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (TokenInfoMap::iterator it = token_info_map.begin();
323e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       it != token_info_map.end();
324e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++it) {
3255d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    STLDeleteElements(&it->second);
3265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
3285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
329e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::TokenInfo* AboutSigninInternals::SigninStatus::FindToken(
3305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& account_id,
3315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::string& consumer_id,
3325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const OAuth2TokenService::ScopeSet& scopes) {
333e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  for (size_t i = 0; i < token_info_map[account_id].size(); ++i) {
3345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    TokenInfo* tmp = token_info_map[account_id][i];
3355d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (tmp->consumer_id == consumer_id && tmp->scopes == scopes)
3365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      return tmp;
3375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return NULL;
339c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)}
3405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
341e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochscoped_ptr<base::DictionaryValue> AboutSigninInternals::SigninStatus::ToValue(
342e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    std::string product_version) {
3435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  scoped_ptr<base::DictionaryValue> signin_status(new base::DictionaryValue());
3445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* signin_info = new base::ListValue();
3455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  signin_status->Set("signin_info", signin_info);
3465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3475d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // A summary of signin related info first.
3485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* basic_info = AddSection(signin_info, "Basic Information");
3495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  const std::string signin_status_string =
350e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      untimed_signin_fields[USERNAME - UNTIMED_FIELDS_BEGIN].empty()
351e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch          ? "Not Signed In"
352e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch          : "Signed In";
353e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  AddSectionEntry(basic_info, "Chrome Version", product_version);
3545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  AddSectionEntry(basic_info, "Signin Status", signin_status_string);
3550529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "Web Based Signin Enabled?",
3560529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      switches::IsEnableWebBasedSignin() == true ? "True" : "False");
3570529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "New Profile Management Enabled?",
3580529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      switches::IsNewProfileManagement() == true ? "True" : "False");
3590529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "New Avatar Menu Enabled?",
3600529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      switches::IsNewAvatarMenu() == true ? "True" : "False");
3610529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  bool new_avatar_menu_flag =
3620529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      CommandLine::ForCurrentProcess()->HasSwitch(switches::kNewAvatarMenu);
3630529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch  AddSectionEntry(basic_info, "New Avatar Menu Flag Set?",
3640529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch      new_avatar_menu_flag ? "True" : "False");
3655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // Only add username.  SID and LSID have moved to tokens section.
3675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  const std::string field =
3685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      SigninStatusFieldToLabel(static_cast<UntimedSigninStatusField>(USERNAME));
369e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  AddSectionEntry(basic_info,
370e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch                  field,
371e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch                  untimed_signin_fields[USERNAME - UNTIMED_FIELDS_BEGIN]);
3725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // Time and status information of the possible sign in types.
374e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  base::ListValue* detailed_info =
375e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      AddSection(signin_info, "Last Signin Details");
3765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (int i = TIMED_FIELDS_BEGIN; i < TIMED_FIELDS_END; ++i) {
3775c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu    const std::string status_field_label =
3785c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu        SigninStatusFieldToLabel(static_cast<TimedSigninStatusField>(i));
3795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
380e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    AddSectionEntry(detailed_info,
3815c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                    status_field_label,
3825c02ac1a9c1b504631c0a3d2b6e737b5d738bae1Bo Liu                    timed_signin_fields[i - TIMED_FIELDS_BEGIN].first,
3835d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)                    timed_signin_fields[i - TIMED_FIELDS_BEGIN].second);
3845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
3855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // Token information for all services.
3875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* token_info = new base::ListValue();
3885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  signin_status->Set("token_info", token_info);
3895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (TokenInfoMap::iterator it = token_info_map.begin();
390e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       it != token_info_map.end();
391e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch       ++it) {
3925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    base::ListValue* token_details = AddSection(token_info, it->first);
3935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
3945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    std::sort(it->second.begin(), it->second.end(), TokenInfo::LessThan);
3955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    const std::vector<TokenInfo*>& tokens = it->second;
3965d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    for (size_t i = 0; i < tokens.size(); ++i) {
3975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      base::DictionaryValue* token_info = tokens[i]->ToValue();
3985d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      token_details->Append(token_info);
3995d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    }
4005d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
4015d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
4025d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  return signin_status.Pass();
4035d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)}
404