[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: |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 19 | ObserverAdapter(bool ephemeral, OverlayUserPrefStore* parent) |
| 20 | : ephemeral_user_pref_store_(ephemeral), parent_(parent) {} |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 21 | |
| 22 | // Methods of PrefStore::Observer. |
| 23 | void OnPrefValueChanged(const std::string& key) override { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 24 | parent_->OnPrefValueChanged(ephemeral_user_pref_store_, key); |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 25 | } |
| 26 | void OnInitializationCompleted(bool succeeded) override { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 27 | parent_->OnInitializationCompleted(ephemeral_user_pref_store_, succeeded); |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 28 | } |
| 29 | |
| 30 | private: |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 31 | // Is the update for the ephemeral? |
| 32 | const bool ephemeral_user_pref_store_; |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 33 | OverlayUserPrefStore* const parent_; |
| 34 | }; |
| 35 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 36 | OverlayUserPrefStore::OverlayUserPrefStore(PersistentPrefStore* persistent) |
| 37 | : OverlayUserPrefStore(new InMemoryPrefStore(), persistent) {} |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 38 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 39 | OverlayUserPrefStore::OverlayUserPrefStore(PersistentPrefStore* ephemeral, |
| 40 | PersistentPrefStore* persistent) |
| 41 | : ephemeral_pref_store_observer_( |
Jinho Bang | 84b58bd | 2018-01-01 21:44:48 | [diff] [blame] | 42 | std::make_unique<OverlayUserPrefStore::ObserverAdapter>(true, this)), |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 43 | persistent_pref_store_observer_( |
Jinho Bang | 84b58bd | 2018-01-01 21:44:48 | [diff] [blame] | 44 | std::make_unique<OverlayUserPrefStore::ObserverAdapter>(false, this)), |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 45 | ephemeral_user_pref_store_(ephemeral), |
| 46 | persistent_user_pref_store_(persistent) { |
| 47 | DCHECK(ephemeral->IsInitializationComplete()); |
| 48 | ephemeral_user_pref_store_->AddObserver(ephemeral_pref_store_observer_.get()); |
| 49 | persistent_user_pref_store_->AddObserver( |
| 50 | persistent_pref_store_observer_.get()); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 51 | } |
| 52 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 53 | bool OverlayUserPrefStore::IsSetInOverlay(const std::string& key) const { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 54 | return ephemeral_user_pref_store_->GetValue(key, nullptr); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 55 | } |
| 56 | |
| 57 | void OverlayUserPrefStore::AddObserver(PrefStore::Observer* observer) { |
| 58 | observers_.AddObserver(observer); |
| 59 | } |
| 60 | |
| 61 | void OverlayUserPrefStore::RemoveObserver(PrefStore::Observer* observer) { |
| 62 | observers_.RemoveObserver(observer); |
| 63 | } |
| 64 | |
[email protected] | 14e0ec6 | 2013-08-26 22:01:39 | [diff] [blame] | 65 | bool OverlayUserPrefStore::HasObservers() const { |
| 66 | return observers_.might_have_observers(); |
[email protected] | d3b05ea | 2012-01-24 22:57:05 | [diff] [blame] | 67 | } |
| 68 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 69 | bool OverlayUserPrefStore::IsInitializationComplete() const { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 70 | return persistent_user_pref_store_->IsInitializationComplete() && |
| 71 | ephemeral_user_pref_store_->IsInitializationComplete(); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 72 | } |
| 73 | |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 74 | bool OverlayUserPrefStore::GetValue(const std::string& key, |
[email protected] | a43a667b | 2013-06-14 17:56:08 | [diff] [blame] | 75 | const base::Value** result) const { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 76 | // If the |key| shall NOT be stored in the ephemeral store, there must not |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 77 | // be an entry. |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 78 | DCHECK(!ShallBeStoredInPersistent(key) || |
| 79 | !ephemeral_user_pref_store_->GetValue(key, nullptr)); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 80 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 81 | if (ephemeral_user_pref_store_->GetValue(key, result)) |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 82 | return true; |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 83 | return persistent_user_pref_store_->GetValue(key, result); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 84 | } |
| 85 | |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 86 | std::unique_ptr<base::DictionaryValue> OverlayUserPrefStore::GetValues() const { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 87 | auto values = ephemeral_user_pref_store_->GetValues(); |
| 88 | auto persistent_values = persistent_user_pref_store_->GetValues(); |
| 89 | |
| 90 | // Output |values| are read from |ephemeral_user_pref_store_| (in-memory |
| 91 | // store). Then the values of preferences in |persistent_names_set_| are |
| 92 | // overwritten by the content of |persistent_user_pref_store_| (the persistent |
| 93 | // store). |
| 94 | for (const auto& key : persistent_names_set_) { |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 95 | std::unique_ptr<base::Value> out_value; |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 96 | persistent_values->Remove(key, &out_value); |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 97 | if (out_value) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 98 | values->Set(key, std::move(out_value)); |
tibell | e23659b4 | 2017-02-23 01:44:13 | [diff] [blame] | 99 | } |
| 100 | } |
| 101 | return values; |
| 102 | } |
| 103 | |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 104 | bool OverlayUserPrefStore::GetMutableValue(const std::string& key, |
[email protected] | a43a667b | 2013-06-14 17:56:08 | [diff] [blame] | 105 | base::Value** result) { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 106 | if (ShallBeStoredInPersistent(key)) |
| 107 | return persistent_user_pref_store_->GetMutableValue(key, result); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 108 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 109 | written_ephemeral_names_.insert(key); |
| 110 | if (ephemeral_user_pref_store_->GetMutableValue(key, result)) |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 111 | return true; |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 112 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 113 | // Try to create copy of persistent if the ephemeral does not contain a value. |
| 114 | base::Value* persistent_value = nullptr; |
| 115 | if (!persistent_user_pref_store_->GetMutableValue(key, &persistent_value)) |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 116 | return false; |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 117 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 118 | ephemeral_user_pref_store_->SetValue( |
Vladislav Kuzkokov | 8ee75fbc | 2019-01-14 14:14:04 | [diff] [blame^] | 119 | key, persistent_value->CreateDeepCopy(), |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 120 | WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |
Vladislav Kuzkokov | 8ee75fbc | 2019-01-14 14:14:04 | [diff] [blame^] | 121 | ephemeral_user_pref_store_->GetMutableValue(key, result); |
[email protected] | 892f1d6 | 2012-11-08 18:24:34 | [diff] [blame] | 122 | return true; |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 123 | } |
| 124 | |
| 125 | void OverlayUserPrefStore::SetValue(const std::string& key, |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 126 | std::unique_ptr<base::Value> value, |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 127 | uint32_t flags) { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 128 | if (ShallBeStoredInPersistent(key)) { |
| 129 | persistent_user_pref_store_->SetValue(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 130 | return; |
| 131 | } |
| 132 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 133 | // TODO(https://crbug.com/861722): If we always store in in-memory storage |
| 134 | // and conditionally also stored in persistent one, we wouldn't have to do a |
| 135 | // complex merge in GetValues(). |
| 136 | written_ephemeral_names_.insert(key); |
| 137 | ephemeral_user_pref_store_->SetValue(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 138 | } |
| 139 | |
| 140 | void OverlayUserPrefStore::SetValueSilently(const std::string& key, |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 141 | std::unique_ptr<base::Value> value, |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 142 | uint32_t flags) { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 143 | if (ShallBeStoredInPersistent(key)) { |
| 144 | persistent_user_pref_store_->SetValueSilently(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 145 | return; |
| 146 | } |
| 147 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 148 | written_ephemeral_names_.insert(key); |
| 149 | ephemeral_user_pref_store_->SetValueSilently(key, std::move(value), flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 150 | } |
| 151 | |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 152 | void OverlayUserPrefStore::RemoveValue(const std::string& key, uint32_t flags) { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 153 | if (ShallBeStoredInPersistent(key)) { |
| 154 | persistent_user_pref_store_->RemoveValue(key, flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 155 | return; |
| 156 | } |
| 157 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 158 | written_ephemeral_names_.insert(key); |
| 159 | ephemeral_user_pref_store_->RemoveValue(key, flags); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 160 | } |
| 161 | |
| 162 | bool OverlayUserPrefStore::ReadOnly() const { |
| 163 | return false; |
| 164 | } |
| 165 | |
[email protected] | 59c1071 | 2012-03-13 02:10:34 | [diff] [blame] | 166 | PersistentPrefStore::PrefReadError OverlayUserPrefStore::GetReadError() const { |
| 167 | return PersistentPrefStore::PREF_READ_ERROR_NONE; |
| 168 | } |
| 169 | |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 170 | PersistentPrefStore::PrefReadError OverlayUserPrefStore::ReadPrefs() { |
| 171 | // We do not read intentionally. |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 172 | OnInitializationCompleted(/* ephemeral */ false, true); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 173 | return PersistentPrefStore::PREF_READ_ERROR_NONE; |
| 174 | } |
| 175 | |
| 176 | void OverlayUserPrefStore::ReadPrefsAsync( |
| 177 | ReadErrorDelegate* error_delegate_raw) { |
dcheng | 5f043bc | 2016-04-22 19:09:06 | [diff] [blame] | 178 | std::unique_ptr<ReadErrorDelegate> error_delegate(error_delegate_raw); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 179 | // We do not read intentionally. |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 180 | OnInitializationCompleted(/* ephemeral */ false, true); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 181 | } |
| 182 | |
Gabriel Charette | 788eaf6 | 2018-08-07 20:11:46 | [diff] [blame] | 183 | void OverlayUserPrefStore::CommitPendingWrite( |
| 184 | base::OnceClosure reply_callback, |
| 185 | base::OnceClosure synchronous_done_callback) { |
| 186 | persistent_user_pref_store_->CommitPendingWrite( |
| 187 | std::move(reply_callback), std::move(synchronous_done_callback)); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 188 | // We do not write our content intentionally. |
| 189 | } |
| 190 | |
benwells | 2673059 | 2015-05-28 13:08:08 | [diff] [blame] | 191 | void OverlayUserPrefStore::SchedulePendingLossyWrites() { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 192 | persistent_user_pref_store_->SchedulePendingLossyWrites(); |
benwells | 2673059 | 2015-05-28 13:08:08 | [diff] [blame] | 193 | } |
| 194 | |
raymes | 76de1af | 2015-05-06 03:22:21 | [diff] [blame] | 195 | void OverlayUserPrefStore::ReportValueChanged(const std::string& key, |
avi | 9ef8bb0 | 2015-12-24 05:29:36 | [diff] [blame] | 196 | uint32_t flags) { |
ericwilligers | 42b92c1 | 2016-10-24 20:21:13 | [diff] [blame] | 197 | for (PrefStore::Observer& observer : observers_) |
| 198 | observer.OnPrefValueChanged(key); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 199 | } |
| 200 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 201 | void OverlayUserPrefStore::RegisterPersistentPref(const std::string& key) { |
Johan Tibell | 6566c8ac | 2017-05-11 03:22:17 | [diff] [blame] | 202 | DCHECK(!key.empty()) << "Key is empty"; |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 203 | DCHECK(persistent_names_set_.find(key) == persistent_names_set_.end()) |
| 204 | << "Key already registered: " << key; |
| 205 | persistent_names_set_.insert(key); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 206 | } |
| 207 | |
dvadym | 53fc0d4 | 2016-02-05 13:34:57 | [diff] [blame] | 208 | void OverlayUserPrefStore::ClearMutableValues() { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 209 | for (const auto& key : written_ephemeral_names_) { |
| 210 | ephemeral_user_pref_store_->RemoveValue( |
| 211 | key, WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 212 | } |
dvadym | 53fc0d4 | 2016-02-05 13:34:57 | [diff] [blame] | 213 | } |
| 214 | |
proberge | 45e34728 | 2017-08-16 21:24:05 | [diff] [blame] | 215 | void OverlayUserPrefStore::OnStoreDeletionFromDisk() { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 216 | persistent_user_pref_store_->OnStoreDeletionFromDisk(); |
proberge | 45e34728 | 2017-08-16 21:24:05 | [diff] [blame] | 217 | } |
| 218 | |
[email protected] | 2dea5c0 | 2012-04-25 07:01:07 | [diff] [blame] | 219 | OverlayUserPrefStore::~OverlayUserPrefStore() { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 220 | ephemeral_user_pref_store_->RemoveObserver( |
| 221 | ephemeral_pref_store_observer_.get()); |
| 222 | persistent_user_pref_store_->RemoveObserver( |
| 223 | persistent_pref_store_observer_.get()); |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 224 | } |
| 225 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 226 | void OverlayUserPrefStore::OnPrefValueChanged(bool ephemeral, |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 227 | const std::string& key) { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 228 | if (ephemeral) { |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 229 | ReportValueChanged(key, DEFAULT_PREF_WRITE_FLAGS); |
| 230 | } else { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 231 | if (!ephemeral_user_pref_store_->GetValue(key, nullptr)) |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 232 | ReportValueChanged(key, DEFAULT_PREF_WRITE_FLAGS); |
| 233 | } |
| 234 | } |
| 235 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 236 | void OverlayUserPrefStore::OnInitializationCompleted(bool ephemeral, |
Johan Tibell | 0595f76 | 2017-05-25 23:54:50 | [diff] [blame] | 237 | bool succeeded) { |
| 238 | if (!IsInitializationComplete()) |
| 239 | return; |
| 240 | for (PrefStore::Observer& observer : observers_) |
| 241 | observer.OnInitializationCompleted(succeeded); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 242 | } |
| 243 | |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 244 | bool OverlayUserPrefStore::ShallBeStoredInPersistent( |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 245 | const std::string& key) const { |
Ramin Halavati | f623bafe9 | 2018-07-17 12:12:58 | [diff] [blame] | 246 | return persistent_names_set_.find(key) != persistent_names_set_.end(); |
[email protected] | 39565dd | 2011-11-14 12:09:27 | [diff] [blame] | 247 | } |