OLD | NEW |
---|---|
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/prefs/pref_hash_store_impl.h" | 5 #include "chrome/browser/prefs/pref_hash_store_impl.h" |
6 | 6 |
7 #include "base/logging.h" | 7 #include "base/logging.h" |
8 #include "base/prefs/pref_registry_simple.h" | 8 #include "base/prefs/pref_registry_simple.h" |
9 #include "base/prefs/pref_service.h" | 9 #include "base/prefs/pref_service.h" |
10 #include "base/prefs/scoped_user_pref_update.h" | 10 #include "base/prefs/scoped_user_pref_update.h" |
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
61 // Get the dictionary corresponding to the profile name, which may have a '.' | 61 // Get the dictionary corresponding to the profile name, which may have a '.' |
62 if (!update->GetDictionaryWithoutPathExpansion(hash_store_id_, | 62 if (!update->GetDictionaryWithoutPathExpansion(hash_store_id_, |
63 &child_dictionary)) { | 63 &child_dictionary)) { |
64 child_dictionary = new DictionaryValue; | 64 child_dictionary = new DictionaryValue; |
65 update->SetWithoutPathExpansion(hash_store_id_, child_dictionary); | 65 update->SetWithoutPathExpansion(hash_store_id_, child_dictionary); |
66 } | 66 } |
67 | 67 |
68 child_dictionary->SetString( | 68 child_dictionary->SetString( |
69 path, pref_hash_calculator_.Calculate(path, new_value)); | 69 path, pref_hash_calculator_.Calculate(path, new_value)); |
70 } | 70 } |
71 | |
72 PrefHashStore::ValueState PrefHashStoreImpl::CheckSplitValue( | |
73 const std::string& path, | |
74 const base::DictionaryValue* initial_split_value, | |
75 std::vector<std::string>* invalid_keys) const { | |
76 DCHECK(invalid_keys && invalid_keys->empty()); | |
77 | |
78 const bool has_hashes = HasSplitHashesAtPath(path); | |
79 | |
80 // Treat NULL and empty the same; otherwise we would need to store a hash | |
81 // for the entire dictionary (or some other special beacon) to | |
82 // differentiate these two cases which are really the same for | |
83 // dictionaries. | |
84 if (!initial_split_value || initial_split_value->empty()) | |
85 return has_hashes ? CLEARED : UNCHANGED; | |
86 | |
87 if (!has_hashes) | |
88 return UNKNOWN_VALUE; | |
89 | |
90 for (base::DictionaryValue::Iterator it(*initial_split_value); !it.IsAtEnd(); | |
91 it.Advance()) { | |
92 ValueState value_state = CheckValue(path + "." + it.key(), &it.value()); | |
93 switch (value_state) { | |
94 case CLEARED: | |
95 // CLEARED doesn't make sense as a NULL value would never be sampled | |
96 // by the DictionaryValue::Iterator; in fact it is a known weakness of | |
97 // this current algorithm to not detect the case where a single key is | |
98 // cleared entirely from the dictionary pref. | |
99 NOTREACHED(); | |
100 break; | |
101 case MIGRATED: | |
102 // Split tracked preferences were introduced after the migration started | |
103 // so no migration is expected. | |
104 NOTREACHED(); | |
105 break; | |
106 case UNCHANGED: | |
107 break; | |
108 case CHANGED: // Falls through. | |
109 case UNKNOWN_VALUE: | |
110 // Declare this value invalid, whether it was changed or never seen | |
111 // before. | |
112 invalid_keys->push_back(it.key()); | |
113 break; | |
114 } | |
115 } | |
116 return invalid_keys->empty() ? UNCHANGED : CHANGED; | |
117 } | |
118 | |
119 void PrefHashStoreImpl::StoreSplitHash( | |
120 const std::string& path, | |
121 const base::DictionaryValue* split_value) { | |
122 ClearPath(path); | |
erikwright (departed)
2013/12/19 15:21:28
Is it OK that this method might not be atomic?
I'
gab
2013/12/20 18:37:06
DictionaryPrefUpdate is not transactional: its onl
| |
123 | |
124 if (split_value) { | |
125 for (base::DictionaryValue::Iterator it(*split_value); !it.IsAtEnd(); | |
126 it.Advance()) { | |
127 StoreHash(path + "." + it.key(), &it.value()); | |
128 } | |
129 } | |
130 } | |
131 | |
132 void PrefHashStoreImpl::ClearPath(const std::string& path) { | |
133 DictionaryPrefUpdate update(local_state_, prefs::kProfilePreferenceHashes); | |
134 DictionaryValue* child_dictionary = NULL; | |
135 | |
136 if (update->GetDictionaryWithoutPathExpansion(hash_store_id_, | |
137 &child_dictionary)) { | |
138 child_dictionary->Remove(path, NULL); | |
139 } | |
140 } | |
141 | |
142 bool PrefHashStoreImpl::HasSplitHashesAtPath(const std::string& path) const { | |
143 const base::DictionaryValue* pref_hash_dicts = | |
144 local_state_->GetDictionary(prefs::kProfilePreferenceHashes); | |
145 const base::DictionaryValue* hashed_prefs = NULL; | |
146 pref_hash_dicts->GetDictionaryWithoutPathExpansion(hash_store_id_, | |
147 &hashed_prefs); | |
148 return hashed_prefs && hashed_prefs->GetDictionary(path, NULL); | |
erikwright (departed)
2013/12/19 15:21:28
If you are going to rely on this calling pattern f
gab
2013/12/20 18:37:06
I'll prepare a separate CL for this.
| |
149 } | |
OLD | NEW |