[email protected] | 32c3c75 | 2012-01-05 17:33:47 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
brettw | f00b9b4 | 2016-02-01 22:11:38 | [diff] [blame] | 5 | #include "components/prefs/overlay_user_pref_store.h" |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 6 | |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 7 | #include <memory> |
danakj | 0c8d4aa | 2015-11-25 05:29:58 | [diff] [blame] | 8 | #include <utility> |
| 9 | |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 10 | #include "base/memory/ptr_util.h" |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 11 | #include "base/values.h" |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 12 | #include "components/prefs/in_memory_pref_store.h" |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 13 | |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 14 | // Allows us to monitor two pref stores and tell updates from them apart. It |
| 15 | // essentially mimics a Callback for the Observer interface (e.g. it allows |
| 16 | // binding additional arguments). |
| 17 | class OverlayUserPrefStore::ObserverAdapter : public PrefStore::Observer { |
| 18 | public: |
| 19 | ObserverAdapter(bool overlay, OverlayUserPrefStore* parent) |
| 20 | : overlay_(overlay), parent_(parent) {} |
| 21 | |
| 22 | // Methods of PrefStore::Observer. |
| 23 | void OnPrefValueChanged(const std::string& key) override { |
| 24 | parent_->OnPrefValueChanged(overlay_, key); |
| 25 | } |
| 26 | void OnInitializationCompleted(bool succeeded) override { |
| 27 | parent_->OnInitializationCompleted(overlay_, succeeded); |
| 28 | } |
| 29 | |
| 30 | private: |
| 31 | // Is the update for the overlay? |
| 32 | const bool overlay_; |
| 33 | OverlayUserPrefStore* const parent_; |
| 34 | }; |
| 35 | |
| 36 | OverlayUserPrefStore::OverlayUserPrefStore(PersistentPrefStore* underlay) |
| 37 | : OverlayUserPrefStore(new InMemoryPrefStore(), underlay) {} |
| 38 | |
| 39 | OverlayUserPrefStore::OverlayUserPrefStore(PersistentPrefStore* overlay, |
| 40 | PersistentPrefStore* underlay) |
| 41 | : overlay_observer_( |
| 42 | base::MakeUnique<OverlayUserPrefStore::ObserverAdapter>(true, this)), |
| 43 | underlay_observer_( |
| 44 | base::MakeUnique<OverlayUserPrefStore::ObserverAdapter>(false, this)), |
| 45 | overlay_(overlay), |
| 46 | underlay_(underlay) { |
| 47 | DCHECK(overlay->IsInitializationComplete()); |
| 48 | overlay_->AddObserver(overlay_observer_.get()); |
| 49 | underlay_->AddObserver(underlay_observer_.get()); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 50 | } |
| 51 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 52 | bool OverlayUserPrefStore::IsSetInOverlay(const std::string& key) const { |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 53 | return overlay_->GetValue(key, NULL); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 54 | } |
| 55 | |
| 56 | void OverlayUserPrefStore::AddObserver(PrefStore::Observer* observer) { |
| 57 | observers_.AddObserver(observer); |
| 58 | } |
| 59 | |
| 60 | void OverlayUserPrefStore::RemoveObserver(PrefStore::Observer* observer) { |
| 61 | observers_.RemoveObserver(observer); |
| 62 | } |
| 63 | |
[email protected] | 14e0ec6 | 2013-08-26 22:01:39 | [diff] [blame] | 64 | bool OverlayUserPrefStore::HasObservers() const { |
| 65 | return observers_.might_have_observers(); |
[email protected] | d3b05ea | 2012-01-24 22:57:05 | [diff] [blame] | 66 | } |
| 67 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 68 | bool OverlayUserPrefStore::IsInitializationComplete() const { |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 69 | return underlay_->IsInitializationComplete() && |
| 70 | overlay_->IsInitializationComplete(); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 71 | } |
| 72 | |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 73 | bool OverlayUserPrefStore::GetValue(const std::string& key, |
[email protected] | a43a667b | 2013-06-14 17:56:08 | [diff] [blame] | 74 | const base::Value** result) const { |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 75 | // If the |key| shall NOT be stored in the overlay store, there must not |
| 76 | // be an entry. |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 77 | DCHECK(ShallBeStoredInOverlay(key) || !overlay_->GetValue(key, NULL)); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 78 | |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 79 | if (overlay_->GetValue(key, result)) |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 80 | return true; |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 81 | return underlay_->GetValue(key, result); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 82 | } |
| 83 | |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 84 | std::unique_ptr<base::DictionaryValue> OverlayUserPrefStore::GetValues() const { |
| 85 | auto values = underlay_->GetValues(); |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 86 | auto overlay_values = overlay_->GetValues(); |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 87 | for (const auto& key : overlay_names_set_) { |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 88 | std::unique_ptr<base::Value> out_value; |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 89 | overlay_values->Remove(key, &out_value); |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 90 | if (out_value) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 91 | values->Set(key, std::move(out_value)); |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 92 | } |
| 93 | } |
| 94 | return values; |
| 95 | } |
| 96 | |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 97 | bool OverlayUserPrefStore::GetMutableValue(const std::string& key, |
[email protected] | a43a667b | 2013-06-14 17:56:08 | [diff] [blame] | 98 | base::Value** result) { |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 99 | if (!ShallBeStoredInOverlay(key)) |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 100 | return underlay_->GetMutableValue(key, result); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 101 | |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 102 | written_overlay_names_.insert(key); |
| 103 | if (overlay_->GetMutableValue(key, result)) |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 104 | return true; |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 105 | |
| 106 | // Try to create copy of underlay if the overlay does not contain a value. |
[email protected] | a43a667b | 2013-06-14 17:56:08 | [diff] [blame] | 107 | base::Value* underlay_value = NULL; |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 108 | if (!underlay_->GetMutableValue(key, &underlay_value)) |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 109 | return false; |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 110 | |
| 111 | *result = underlay_value->DeepCopy(); |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 112 | overlay_->SetValue(key, base::WrapUnique(*result), |
| 113 | WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 114 | return true; |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 115 | } |
| 116 | |
| 117 | void OverlayUserPrefStore::SetValue(const std::string& key, |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 118 | std::unique_ptr<base::Value> value, |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 119 | uint32_t flags) { |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 120 | if (!ShallBeStoredInOverlay(key)) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 121 | underlay_->SetValue(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 122 | return; |
| 123 | } |
| 124 | |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 125 | written_overlay_names_.insert(key); |
| 126 | overlay_->SetValue(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 127 | } |
| 128 | |
| 129 | void OverlayUserPrefStore::SetValueSilently(const std::string& key, |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 130 | std::unique_ptr<base::Value> value, |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 131 | uint32_t flags) { |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 132 | if (!ShallBeStoredInOverlay(key)) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 133 | underlay_->SetValueSilently(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 134 | return; |
| 135 | } |
| 136 | |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 137 | written_overlay_names_.insert(key); |
| 138 | overlay_->SetValueSilently(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 139 | } |
| 140 | |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 141 | void OverlayUserPrefStore::RemoveValue(const std::string& key, uint32_t flags) { |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 142 | if (!ShallBeStoredInOverlay(key)) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 143 | underlay_->RemoveValue(key, flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 144 | return; |
| 145 | } |
| 146 | |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 147 | written_overlay_names_.insert(key); |
| 148 | overlay_->RemoveValue(key, flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 149 | } |
| 150 | |
| 151 | bool OverlayUserPrefStore::ReadOnly() const { |
| 152 | return false; |
| 153 | } |
| 154 | |
[email protected] | 59c1071 | 2012-03-13 02:10:34 | [diff] [blame] | 155 | PersistentPrefStore::PrefReadError OverlayUserPrefStore::GetReadError() const { |
| 156 | return PersistentPrefStore::PREF_READ_ERROR_NONE; |
| 157 | } |
| 158 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 159 | PersistentPrefStore::PrefReadError OverlayUserPrefStore::ReadPrefs() { |
| 160 | // We do not read intentionally. |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 161 | OnInitializationCompleted(/* overlay */ false, true); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 162 | return PersistentPrefStore::PREF_READ_ERROR_NONE; |
| 163 | } |
| 164 | |
| 165 | void OverlayUserPrefStore::ReadPrefsAsync( |
| 166 | ReadErrorDelegate* error_delegate_raw) { |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 167 | std::unique_ptr<ReadErrorDelegate> error_delegate(error_delegate_raw); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 168 | // We do not read intentionally. |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 169 | OnInitializationCompleted(/* overlay */ false, true); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 170 | } |
| 171 | |
Francois Doray | 405dd2d | 2017-06-09 15:23:33 | [diff] [blame^] | 172 | void OverlayUserPrefStore::CommitPendingWrite(base::OnceClosure done_callback) { |
| 173 | underlay_->CommitPendingWrite(std::move(done_callback)); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 174 | // We do not write our content intentionally. |
| 175 | } |
| 176 | |
benwells | 2673059 | 2015-05-28 13:08:08 | [diff] [blame] | 177 | void OverlayUserPrefStore::SchedulePendingLossyWrites() { |
| 178 | underlay_->SchedulePendingLossyWrites(); |
| 179 | } |
| 180 | |
raymes | 76de1af | 2015-05-06 03:22:21 | [diff] [blame] | 181 | void OverlayUserPrefStore::ReportValueChanged(const std::string& key, |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 182 | uint32_t flags) { |
ericwilligers | 42b92c1 | 2016-10-24 20:21:13 | [diff] [blame] | 183 | for (PrefStore::Observer& observer : observers_) |
| 184 | observer.OnPrefValueChanged(key); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 185 | } |
| 186 | |
[email protected] | 32c3c75 | 2012-01-05 17:33:47 | [diff] [blame] | 187 | void OverlayUserPrefStore::RegisterOverlayPref(const std::string& key) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 188 | DCHECK(!key.empty()) << "Key is empty"; |
| 189 | DCHECK(overlay_names_set_.find(key) == overlay_names_set_.end()) |
| 190 | << "Key already registered"; |
| 191 | overlay_names_set_.insert(key); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 192 | } |
| 193 | |
dvadym | 53fc0d4 | 2016-02-05 13:34:57 | [diff] [blame] | 194 | void OverlayUserPrefStore::ClearMutableValues() { |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 195 | for (const auto& key : written_overlay_names_) { |
| 196 | overlay_->RemoveValue(key, WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |
| 197 | } |
dvadym | 53fc0d4 | 2016-02-05 13:34:57 | [diff] [blame] | 198 | } |
| 199 | |
[email protected] | 2dea5c0 | 2012-04-25 07:01:07 | [diff] [blame] | 200 | OverlayUserPrefStore::~OverlayUserPrefStore() { |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 201 | overlay_->RemoveObserver(overlay_observer_.get()); |
| 202 | underlay_->RemoveObserver(underlay_observer_.get()); |
| 203 | } |
| 204 | |
| 205 | void OverlayUserPrefStore::OnPrefValueChanged(bool overlay, |
| 206 | const std::string& key) { |
| 207 | if (overlay) { |
| 208 | ReportValueChanged(key, DEFAULT_PREF_WRITE_FLAGS); |
| 209 | } else { |
| 210 | if (!overlay_->GetValue(key, NULL)) |
| 211 | ReportValueChanged(key, DEFAULT_PREF_WRITE_FLAGS); |
| 212 | } |
| 213 | } |
| 214 | |
| 215 | void OverlayUserPrefStore::OnInitializationCompleted(bool overlay, |
| 216 | bool succeeded) { |
| 217 | if (!IsInitializationComplete()) |
| 218 | return; |
| 219 | for (PrefStore::Observer& observer : observers_) |
| 220 | observer.OnInitializationCompleted(succeeded); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 221 | } |
| 222 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 223 | bool OverlayUserPrefStore::ShallBeStoredInOverlay( |
| 224 | const std::string& key) const { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 225 | return overlay_names_set_.find(key) != overlay_names_set_.end(); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 226 | } |