1 // Copyright 2013 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "chrome/browser/signin/local_auth.h" 6 7 #include "base/base64.h" 8 #include "base/logging.h" 9 #include "base/memory/scoped_ptr.h" 10 #include "base/metrics/histogram.h" 11 #include "base/prefs/pref_service.h" 12 #include "base/strings/string_util.h" 13 #include "chrome/browser/browser_process.h" 14 #include "chrome/browser/profiles/profile.h" 15 #include "chrome/browser/profiles/profile_manager.h" 16 #include "chrome/common/pref_names.h" 17 #include "components/user_prefs/pref_registry_syncable.h" 18 #include "components/webdata/encryptor/encryptor.h" 19 #include "crypto/random.h" 20 #include "crypto/secure_util.h" 21 #include "crypto/symmetric_key.h" 22 23 namespace { 24 25 // WARNING: Changing these values will make it impossible to do off-line 26 // authentication until the next successful on-line authentication. To change 27 // these safely, change the "encoding" version below and make verification 28 // handle multiple values. 29 const char kHash1Encoding = '1'; 30 const unsigned kHash1Bits = 256; 31 const unsigned kHash1Bytes = kHash1Bits / 8; 32 const unsigned kHash1IterationCount = 100000; 33 34 std::string CreateSecurePasswordHash(const std::string& salt, 35 const std::string& password, 36 char encoding) { 37 DCHECK_EQ(kHash1Bytes, salt.length()); 38 DCHECK_EQ(kHash1Encoding, encoding); // Currently support only one method. 39 40 base::Time start_time = base::Time::Now(); 41 42 // Library call to create secure password hash as SymmetricKey (uses PBKDF2). 43 scoped_ptr<crypto::SymmetricKey> password_key( 44 crypto::SymmetricKey::DeriveKeyFromPassword( 45 crypto::SymmetricKey::AES, 46 password, salt, 47 kHash1IterationCount, kHash1Bits)); 48 std::string password_hash; 49 const bool success = password_key->GetRawKey(&password_hash); 50 DCHECK(success); 51 DCHECK_EQ(kHash1Bytes, password_hash.length()); 52 53 UMA_HISTOGRAM_TIMES("PasswordHash.CreateTime", 54 base::Time::Now() - start_time); 55 56 return password_hash; 57 } 58 59 std::string EncodePasswordHashRecord(const std::string& record, 60 char encoding) { 61 DCHECK_EQ(kHash1Encoding, encoding); // Currently support only one method. 62 63 // Encrypt the hash using the OS account-password protection (if available). 64 std::string encoded; 65 const bool success = Encryptor::EncryptString(record, &encoded); 66 DCHECK(success); 67 68 // Convert binary record to text for preference database. 69 std::string encoded64; 70 base::Base64Encode(encoded, &encoded64); 71 72 // Stuff the "encoding" value into the first byte. 73 encoded64.insert(0, &encoding, sizeof(encoding)); 74 75 return encoded64; 76 } 77 78 bool DecodePasswordHashRecord(const std::string& encoded, 79 std::string* decoded, 80 char* encoding) { 81 // Extract the "encoding" value from the first byte and validate. 82 if (encoded.length() < 1) 83 return false; 84 *encoding = encoded[0]; 85 if (*encoding != kHash1Encoding) 86 return false; 87 88 // Stored record is base64; convert to binary. 89 std::string unbase64; 90 if (!base::Base64Decode(encoded.substr(1), &unbase64)) 91 return false; 92 93 // Decrypt the record using the OS account-password protection (if available). 94 return Encryptor::DecryptString(unbase64, decoded); 95 } 96 97 } // namespace 98 99 namespace chrome { 100 101 void RegisterLocalAuthPrefs(user_prefs::PrefRegistrySyncable* registry) { 102 registry->RegisterStringPref( 103 prefs::kGoogleServicesPasswordHash, 104 std::string(), 105 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); 106 } 107 108 void SetLocalAuthCredentials(size_t info_index, 109 const std::string& password) { 110 DCHECK(password.length()); 111 112 // Salt should be random data, as long as the hash length, and different with 113 // every save. 114 std::string salt_str; 115 crypto::RandBytes(WriteInto(&salt_str, kHash1Bytes + 1), kHash1Bytes); 116 DCHECK_EQ(kHash1Bytes, salt_str.length()); 117 118 // Perform secure hash of password for storage. 119 std::string password_hash = CreateSecurePasswordHash( 120 salt_str, password, kHash1Encoding); 121 DCHECK_EQ(kHash1Bytes, password_hash.length()); 122 123 // Group all fields into a single record for storage; 124 std::string record; 125 record.append(salt_str); 126 record.append(password_hash); 127 128 // Encode it and store it. 129 std::string encoded = EncodePasswordHashRecord(record, kHash1Encoding); 130 ProfileInfoCache& info = 131 g_browser_process->profile_manager()->GetProfileInfoCache(); 132 info.SetLocalAuthCredentialsOfProfileAtIndex(info_index, encoded); 133 } 134 135 void SetLocalAuthCredentials(const Profile* profile, 136 const std::string& password) { 137 DCHECK(profile); 138 139 ProfileInfoCache& info = 140 g_browser_process->profile_manager()->GetProfileInfoCache(); 141 size_t info_index = info.GetIndexOfProfileWithPath(profile->GetPath()); 142 if (info_index == std::string::npos) { 143 NOTREACHED(); 144 return; 145 } 146 SetLocalAuthCredentials(info_index, password); 147 } 148 149 bool ValidateLocalAuthCredentials(size_t info_index, 150 const std::string& password) { 151 std::string record; 152 char encoding; 153 154 ProfileInfoCache& info = 155 g_browser_process->profile_manager()->GetProfileInfoCache(); 156 157 std::string encodedhash = 158 info.GetLocalAuthCredentialsOfProfileAtIndex(info_index); 159 if (encodedhash.length() == 0 && password.length() == 0) 160 return true; 161 if (!DecodePasswordHashRecord(encodedhash, &record, &encoding)) 162 return false; 163 164 std::string password_hash; 165 const char* password_saved; 166 const char* password_check; 167 size_t password_length; 168 169 if (encoding == '1') { 170 // Validate correct length; extract salt and password hash. 171 if (record.length() != 2 * kHash1Bytes) 172 return false; 173 std::string salt_str(record.data(), kHash1Bytes); 174 password_saved = record.data() + kHash1Bytes; 175 password_hash = CreateSecurePasswordHash(salt_str, password, encoding); 176 password_check = password_hash.data(); 177 password_length = kHash1Bytes; 178 } else { 179 // unknown encoding 180 return false; 181 } 182 183 return crypto::SecureMemEqual(password_saved, password_check, 184 password_length); 185 } 186 187 bool ValidateLocalAuthCredentials(const Profile* profile, 188 const std::string& password) { 189 DCHECK(profile); 190 191 ProfileInfoCache& info = 192 g_browser_process->profile_manager()->GetProfileInfoCache(); 193 size_t info_index = info.GetIndexOfProfileWithPath(profile->GetPath()); 194 if (info_index == std::string::npos) { 195 NOTREACHED(); // This should never happen but fail safely if it does. 196 return false; 197 } 198 return ValidateLocalAuthCredentials(info_index, password); 199 } 200 201 } // namespace chrome 202