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