about_signin_internals.cc revision 0529e5d033099cbfc42635f6f6183833b09dff6e
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, 455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& field_val) { 465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) scoped_ptr<base::DictionaryValue> entry(new base::DictionaryValue()); 475d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) entry->SetString("label", field_name); 485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) entry->SetString("value", field_val); 495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) section_list->Append(entry.release()); 505d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)std::string SigninStatusFieldToLabel(UntimedSigninStatusField field) { 535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) switch (field) { 545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case USERNAME: 555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return "User Id"; 565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case UNTIMED_FIELDS_END: 575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NOTREACHED(); 585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return std::string(); 595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NOTREACHED(); 615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return std::string(); 625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 64e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochTimedSigninStatusValue SigninStatusFieldToLabel(TimedSigninStatusField field) { 655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) switch (field) { 665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case SIGNIN_TYPE: 675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Type", "Time"); 685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case CLIENT_LOGIN_STATUS: 695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Last OnClientLogin Status", 705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "Last OnClientLogin Time"); 715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case OAUTH_LOGIN_STATUS: 725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Last OnOAuthLogin Status", 735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "Last OnOAuthLogin Time"); 745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case GET_USER_INFO_STATUS: 765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Last OnGetUserInfo Status", 775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "Last OnGetUserInfo Time"); 785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case UBER_TOKEN_STATUS: 795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Last OnUberToken Status", 805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "Last OnUberToken Time"); 815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case MERGE_SESSION_STATUS: 825d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Last OnMergeSession Status", 835d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "Last OnMergeSession Time"); 845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) case TIMED_FIELDS_END: 855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NOTREACHED(); 865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Error", std::string()); 875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NOTREACHED(); 895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return TimedSigninStatusValue("Error", std::string()); 905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} // anonymous namespace 935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 94e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::AboutSigninInternals( 95e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ProfileOAuth2TokenService* token_service, 96e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch SigninManagerBase* signin_manager) 97e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch : token_service_(token_service), 98e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch signin_manager_(signin_manager), 99e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch client_(NULL) {} 1002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 101e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::~AboutSigninInternals() {} 1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::AddSigninObserver( 1042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AboutSigninInternals::Observer* observer) { 1052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_observers_.AddObserver(observer); 1062a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::RemoveSigninObserver( 1092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) AboutSigninInternals::Observer* observer) { 1102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_observers_.RemoveObserver(observer); 1112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::NotifySigninValueChanged( 1142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const UntimedSigninStatusField& field, 1152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& value) { 1162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) unsigned int field_index = field - UNTIMED_FIELDS_BEGIN; 1172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(field_index >= 0 && 1182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) field_index < signin_status_.untimed_signin_fields.size()); 1192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_status_.untimed_signin_fields[field_index] = value; 1212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Also persist these values in the prefs. 1232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string pref_path = SigninStatusFieldToString(field); 124e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch client_->GetPrefs()->SetString(pref_path.c_str(), value); 1252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) NotifyObservers(); 1272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::NotifySigninValueChanged( 1302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const TimedSigninStatusField& field, 1312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string& value) { 1322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) unsigned int field_index = field - TIMED_FIELDS_BEGIN; 1332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) DCHECK(field_index >= 0 && 1342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) field_index < signin_status_.timed_signin_fields.size()); 1352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 136c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) Time now = Time::NowFromSystemTime(); 1375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) std::string time_as_str = 1385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) base::UTF16ToUTF8(base::TimeFormatFriendlyDate(now)); 1392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) TimedSigninStatusValue timed_value(value, time_as_str); 1402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_status_.timed_signin_fields[field_index] = timed_value; 1422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) // Also persist these values in the prefs. 1442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string value_pref = SigninStatusFieldToString(field) + ".value"; 145c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) const std::string time_pref = SigninStatusFieldToString(field) + ".time"; 146e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch client_->GetPrefs()->SetString(value_pref.c_str(), value); 147e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch client_->GetPrefs()->SetString(time_pref.c_str(), time_as_str); 1482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) NotifyObservers(); 1502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::RefreshSigninPrefs() { 153e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch // Return if no client exists. Can occur in unit tests. 154e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch if (!client_) 1552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) return; 1562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 157e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch PrefService* pref_service = client_->GetPrefs(); 1582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) for (int i = UNTIMED_FIELDS_BEGIN; i < UNTIMED_FIELDS_END; ++i) { 1592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) const std::string pref_path = 1602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) SigninStatusFieldToString(static_cast<UntimedSigninStatusField>(i)); 1612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_status_.untimed_signin_fields[i - UNTIMED_FIELDS_BEGIN] = 1632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) pref_service->GetString(pref_path.c_str()); 1642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 165e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch for (int i = TIMED_FIELDS_BEGIN; i < TIMED_FIELDS_END; ++i) { 166e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch const std::string value_pref = 167e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch SigninStatusFieldToString(static_cast<TimedSigninStatusField>(i)) + 168e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ".value"; 169e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch const std::string time_pref = 170e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch SigninStatusFieldToString(static_cast<TimedSigninStatusField>(i)) + 171e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ".time"; 1722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) TimedSigninStatusValue value(pref_service->GetString(value_pref.c_str()), 1742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) pref_service->GetString(time_pref.c_str())); 1752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_status_.timed_signin_fields[i - TIMED_FIELDS_BEGIN] = value; 1762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) } 1772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1784e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles) // TODO(rogerta): Get status and timestamps for oauth2 tokens. 1792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) NotifyObservers(); 1812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 183e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid AboutSigninInternals::Initialize(SigninClient* client) { 184e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch DCHECK(!client_); 185e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch client_ = client; 1862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1872a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) RefreshSigninPrefs(); 1882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 189e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch signin_manager_->AddSigninDiagnosticsObserver(this); 190e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch token_service_->AddDiagnosticsObserver(this); 1912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::Shutdown() { 194e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch signin_manager_->RemoveSigninDiagnosticsObserver(this); 195e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch token_service_->RemoveDiagnosticsObserver(this); 1962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 1972a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 1982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void AboutSigninInternals::NotifyObservers() { 1992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) FOR_EACH_OBSERVER(AboutSigninInternals::Observer, 2002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) signin_observers_, 201e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch OnSigninStateChanged( 202e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch signin_status_.ToValue(client_->GetProductVersion()))); 2032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 2042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) 2055d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)scoped_ptr<base::DictionaryValue> AboutSigninInternals::GetSigninStatus() { 206e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch return signin_status_.ToValue(client_->GetProductVersion()).Pass(); 2072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} 208c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) 2095d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AboutSigninInternals::OnAccessTokenRequested( 2105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& account_id, 2115d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& consumer_id, 2125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const OAuth2TokenService::ScopeSet& scopes) { 2135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) TokenInfo* token = signin_status_.FindToken(account_id, consumer_id, scopes); 2145d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (token) { 2155d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) *token = TokenInfo(consumer_id, scopes); 2165d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } else { 2175d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token = new TokenInfo(consumer_id, scopes); 2185d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) signin_status_.token_info_map[account_id].push_back(token); 2195d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 2205d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2215d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NotifyObservers(); 2225d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 2235d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2245d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AboutSigninInternals::OnFetchAccessTokenComplete( 2255d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& account_id, 2265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& consumer_id, 2275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const OAuth2TokenService::ScopeSet& scopes, 2285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) GoogleServiceAuthError error, 2295d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) base::Time expiration_time) { 2305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) TokenInfo* token = signin_status_.FindToken(account_id, consumer_id, scopes); 2315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (!token) { 2325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) DVLOG(1) << "Can't find token: " << account_id << ", " << consumer_id; 2335d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return; 2345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 2355d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token->receive_time = base::Time::Now(); 2375d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token->error = error; 2385d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token->expiration_time = expiration_time; 2395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NotifyObservers(); 2415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 2425d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)void AboutSigninInternals::OnTokenRemoved( 2445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& account_id, 2455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const OAuth2TokenService::ScopeSet& scopes) { 2465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) for (size_t i = 0; i < signin_status_.token_info_map[account_id].size(); 247e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ++i) { 2485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) TokenInfo* token = signin_status_.token_info_map[account_id][i]; 2495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (token->scopes == scopes) 2505d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token->Invalidate(); 2515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 2525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) NotifyObservers(); 2535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 2545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::TokenInfo::TokenInfo( 2565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& consumer_id, 2575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const OAuth2TokenService::ScopeSet& scopes) 2585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) : consumer_id(consumer_id), 2595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) scopes(scopes), 2605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) request_time(base::Time::Now()), 2615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) error(GoogleServiceAuthError::AuthErrorNone()), 262e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch removed_(false) {} 2635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::TokenInfo::~TokenInfo() {} 2655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)bool AboutSigninInternals::TokenInfo::LessThan(const TokenInfo* a, 2675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const TokenInfo* b) { 2685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return a->consumer_id < b->consumer_id || a->scopes < b->scopes; 2695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 2705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 271e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid AboutSigninInternals::TokenInfo::Invalidate() { removed_ = true; } 2725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)base::DictionaryValue* AboutSigninInternals::TokenInfo::ToValue() const { 2745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) scoped_ptr<base::DictionaryValue> token_info(new base::DictionaryValue()); 2755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString("service", consumer_id); 2765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) std::string scopes_str; 2785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) for (OAuth2TokenService::ScopeSet::const_iterator it = scopes.begin(); 279e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch it != scopes.end(); 280e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ++it) { 2815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) scopes_str += *it + "<br/>"; 2825d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 2835d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString("scopes", scopes_str); 2845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString("request_time", GetTimeStr(request_time).c_str()); 2855d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 2865d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (removed_) { 2875d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString("status", "Token was revoked."); 2885d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } else if (!receive_time.is_null()) { 2895d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (error == GoogleServiceAuthError::AuthErrorNone()) { 2905d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) bool token_expired = expiration_time < base::Time::Now(); 2915d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) std::string status_str = ""; 2925d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (token_expired) 2935d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) status_str = "<p style=\"color: #ffffff; background-color: #ff0000\">"; 2945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) base::StringAppendF(&status_str, 2955d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "Received token at %s. Expire at %s", 2965d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) GetTimeStr(receive_time).c_str(), 2975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) GetTimeStr(expiration_time).c_str()); 2985d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (token_expired) 2995d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) base::StringAppendF(&status_str, "</p>"); 3005d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString("status", status_str); 3015d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } else { 3025d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString( 3035d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) "status", 304a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles) base::StringPrintf("Failure: %s", error.ToString().c_str())); 3055d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 3065d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } else { 3075d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) token_info->SetString("status", "Waiting for response"); 3085d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 3095d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 3105d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return token_info.release(); 3115d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 3125d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 3135d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::SigninStatus::SigninStatus() 314e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch : untimed_signin_fields(UNTIMED_FIELDS_COUNT), 315e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch timed_signin_fields(TIMED_FIELDS_COUNT) {} 3165d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 3175d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)AboutSigninInternals::SigninStatus::~SigninStatus() { 3185d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) for (TokenInfoMap::iterator it = token_info_map.begin(); 319e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch it != token_info_map.end(); 320e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ++it) { 3215d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) STLDeleteElements(&it->second); 3225d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 3235d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)} 3245d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 325e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen MurdochAboutSigninInternals::TokenInfo* AboutSigninInternals::SigninStatus::FindToken( 3265d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& account_id, 3275d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string& consumer_id, 3285d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const OAuth2TokenService::ScopeSet& scopes) { 329e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch for (size_t i = 0; i < token_info_map[account_id].size(); ++i) { 3305d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) TokenInfo* tmp = token_info_map[account_id][i]; 3315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) if (tmp->consumer_id == consumer_id && tmp->scopes == scopes) 3325d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return tmp; 3335d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) } 3345d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) return NULL; 335c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)} 3365d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 337e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochscoped_ptr<base::DictionaryValue> AboutSigninInternals::SigninStatus::ToValue( 338e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch std::string product_version) { 3395d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) scoped_ptr<base::DictionaryValue> signin_status(new base::DictionaryValue()); 3405d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) base::ListValue* signin_info = new base::ListValue(); 3415d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) signin_status->Set("signin_info", signin_info); 3425d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 3435d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) // A summary of signin related info first. 3445d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) base::ListValue* basic_info = AddSection(signin_info, "Basic Information"); 3455d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string signin_status_string = 346e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch untimed_signin_fields[USERNAME - UNTIMED_FIELDS_BEGIN].empty() 347e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch ? "Not Signed In" 348e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch : "Signed In"; 349e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch AddSectionEntry(basic_info, "Chrome Version", product_version); 3505d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) AddSectionEntry(basic_info, "Signin Status", signin_status_string); 3510529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch AddSectionEntry(basic_info, "Web Based Signin Enabled?", 3520529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch switches::IsEnableWebBasedSignin() == true ? "True" : "False"); 3530529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch AddSectionEntry(basic_info, "New Profile Management Enabled?", 3540529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch switches::IsNewProfileManagement() == true ? "True" : "False"); 3550529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch AddSectionEntry(basic_info, "New Avatar Menu Enabled?", 3560529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch switches::IsNewAvatarMenu() == true ? "True" : "False"); 3570529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch bool new_avatar_menu_flag = 3580529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch CommandLine::ForCurrentProcess()->HasSwitch(switches::kNewAvatarMenu); 3590529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch AddSectionEntry(basic_info, "New Avatar Menu Flag Set?", 3600529e5d033099cbfc42635f6f6183833b09dff6eBen Murdoch new_avatar_menu_flag ? "True" : "False"); 3615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 3625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) // Only add username. SID and LSID have moved to tokens section. 3635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string field = 3645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) SigninStatusFieldToLabel(static_cast<UntimedSigninStatusField>(USERNAME)); 365e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch AddSectionEntry(basic_info, 366e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch field, 367e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch untimed_signin_fields[USERNAME - UNTIMED_FIELDS_BEGIN]); 3685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 3695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) // Time and status information of the possible sign in types. 370e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch base::ListValue* detailed_info = 371e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch AddSection(signin_info, "Last Signin Details"); 3725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) for (int i = TIMED_FIELDS_BEGIN; i < TIMED_FIELDS_END; ++i) { 3735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string value_field = 3745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) SigninStatusFieldToLabel(static_cast<TimedSigninStatusField>(i)).first; 3755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) const std::string time_field = 3765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) SigninStatusFieldToLabel(static_cast<TimedSigninStatusField>(i)).second; 3775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) 378e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch AddSectionEntry(detailed_info, 379e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch value_field, 3805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles) timed_signin_fields[i - TIMED_FIELDS_BEGIN].first); 381e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch AddSectionEntry(detailed_info, 382e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch time_field, 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