Chromium Code Reviews| Index: services/preferences/public/cpp/user_prefs_impl.cc |
| diff --git a/services/preferences/public/cpp/user_prefs_impl.cc b/services/preferences/public/cpp/user_prefs_impl.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..8286f214691cc1300789eeca73b911e059bc01c9 |
| --- /dev/null |
| +++ b/services/preferences/public/cpp/user_prefs_impl.cc |
| @@ -0,0 +1,387 @@ |
| +// Copyright 2017 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +#include "services/preferences/public/cpp/user_prefs_impl.h" |
| + |
| +#include "base/macros.h" |
| +#include "base/values.h" |
| +#include "components/prefs/json_pref_store.h" |
| +#include "components/prefs/pref_filter.h" |
| +#include "components/user_prefs/tracked/pref_hash_store_impl.h" |
| +#include "components/user_prefs/tracked/segregated_pref_store.h" |
| +#include "components/user_prefs/tracked/tracked_preference_validation_delegate.h" |
| +#include "components/user_prefs/tracked/tracked_preferences_migration.h" |
| +#include "mojo/public/cpp/bindings/strong_binding.h" |
| + |
| +#if defined(OS_WIN) |
| +#include "components/user_prefs/tracked/registry_hash_store_contents_win.h" |
| +#endif |
| + |
| +namespace prefs { |
| +namespace { |
| + |
| +class ForwardingTrackedPreferenceValidationDelegate |
|
tibell
2017/03/08 03:39:54
A short comment why this class exists.
|
| + : public TrackedPreferenceValidationDelegate { |
| + public: |
| + ForwardingTrackedPreferenceValidationDelegate( |
| + scoped_refptr<base::SingleThreadTaskRunner> task_runner, |
| + base::WeakPtr<TrackedPreferenceValidationDelegate> target_delegate) |
| + : task_runner_(std::move(task_runner)), |
| + target_delegate_(target_delegate) {} |
| + |
| + void OnAtomicPreferenceValidation( |
| + const std::string& pref_path, |
| + const base::Value* value, |
| + PrefHashStoreTransaction::ValueState value_state, |
| + PrefHashStoreTransaction::ValueState external_validation_value_state, |
| + bool is_personal) override { |
| + task_runner_->PostTask( |
| + FROM_HERE, |
| + base::Bind(&ForwardingTrackedPreferenceValidationDelegate:: |
| + OnAtomicPreferenceValidationOnTaskRunner, |
| + target_delegate_, pref_path, |
| + base::Passed(value ? value->CreateDeepCopy() : nullptr), |
| + value_state, external_validation_value_state, is_personal)); |
| + } |
| + |
| + // Notifies observes of the result (|value_state|) of checking the split |
| + // |dict_value| (which may be NULL) at |pref_path|. |is_personal| indicates |
| + // whether or not the value may contain personal information. |
| + void OnSplitPreferenceValidation( |
| + const std::string& pref_path, |
| + const base::DictionaryValue* dict_value, |
| + const std::vector<std::string>& invalid_keys, |
| + const std::vector<std::string>& external_validation_invalid_keys, |
| + PrefHashStoreTransaction::ValueState value_state, |
| + PrefHashStoreTransaction::ValueState external_validation_value_state, |
| + bool is_personal) override { |
| + task_runner_->PostTask( |
| + FROM_HERE, |
| + base::Bind( |
| + &ForwardingTrackedPreferenceValidationDelegate:: |
| + OnSplitPreferenceValidationOnTaskRunner, |
| + target_delegate_, pref_path, |
| + base::Passed(dict_value ? dict_value->CreateDeepCopy() : nullptr), |
| + invalid_keys, external_validation_invalid_keys, value_state, |
| + external_validation_value_state, is_personal)); |
| + } |
| + |
| + private: |
| + static void OnAtomicPreferenceValidationOnTaskRunner( |
| + base::WeakPtr<TrackedPreferenceValidationDelegate> target_delegate, |
| + const std::string& pref_path, |
| + std::unique_ptr<base::Value> value, |
| + PrefHashStoreTransaction::ValueState value_state, |
| + PrefHashStoreTransaction::ValueState external_validation_value_state, |
| + bool is_personal) { |
| + if (!target_delegate) |
| + return; |
| + |
| + target_delegate->OnAtomicPreferenceValidation( |
| + pref_path, value.get(), value_state, external_validation_value_state, |
| + is_personal); |
| + } |
| + |
| + static void OnSplitPreferenceValidationOnTaskRunner( |
| + base::WeakPtr<TrackedPreferenceValidationDelegate> target_delegate, |
| + const std::string& pref_path, |
| + std::unique_ptr<base::DictionaryValue> dict_value, |
| + const std::vector<std::string>& invalid_keys, |
| + const std::vector<std::string>& external_validation_invalid_keys, |
| + PrefHashStoreTransaction::ValueState value_state, |
| + PrefHashStoreTransaction::ValueState external_validation_value_state, |
| + bool is_personal) { |
| + if (!target_delegate) |
| + return; |
| + |
| + target_delegate->OnSplitPreferenceValidation( |
| + pref_path, dict_value.get(), invalid_keys, |
| + external_validation_invalid_keys, value_state, |
| + external_validation_value_state, is_personal); |
| + } |
| + |
| + const scoped_refptr<base::SingleThreadTaskRunner> task_runner_; |
| + |
| + // The delegate to forward to. May only be used on |task_runner_|. |
| + const base::WeakPtr<TrackedPreferenceValidationDelegate> target_delegate_; |
| +}; |
| + |
| +class PersistentPrefStoreImpl : public mojom::PersistentPrefStore { |
| + public: |
| + explicit PersistentPrefStoreImpl( |
| + scoped_refptr<::PersistentPrefStore> pref_store); |
| + |
| + ~PersistentPrefStoreImpl() override; |
| + |
| + // mojom::PersistentPrefStore: |
| + void SetValue(const std::string& key, |
| + std::unique_ptr<base::Value> value, |
| + uint32_t flags) override; |
| + void RemoveValue(const std::string& key, uint32_t flags) override; |
| + |
| + void CommitPendingWrite() override; |
| + void SchedulePendingLossyWrites() override; |
| + void ClearMutableValues() override; |
| + |
| + private: |
| + scoped_refptr<::PersistentPrefStore> backing_pref_store_; |
| + |
| + DISALLOW_COPY_AND_ASSIGN(PersistentPrefStoreImpl); |
| +}; |
| + |
| +PersistentPrefStoreImpl::PersistentPrefStoreImpl( |
| + scoped_refptr<::PersistentPrefStore> pref_store) |
| + : backing_pref_store_(std::move(pref_store)) {} |
| + |
| +PersistentPrefStoreImpl::~PersistentPrefStoreImpl() = default; |
| + |
| +// mojomJsonPrefStore: |
| +void PersistentPrefStoreImpl::SetValue(const std::string& key, |
| + std::unique_ptr<base::Value> value, |
| + uint32_t flags) { |
| + backing_pref_store_->SetValue(key, std::move(value), flags); |
| +} |
| + |
| +void PersistentPrefStoreImpl::RemoveValue(const std::string& key, |
| + uint32_t flags) { |
| + backing_pref_store_->RemoveValue(key, flags); |
| +} |
| + |
| +void PersistentPrefStoreImpl::CommitPendingWrite() { |
| + backing_pref_store_->CommitPendingWrite(); |
| +} |
| + |
| +void PersistentPrefStoreImpl::SchedulePendingLossyWrites() { |
| + backing_pref_store_->SchedulePendingLossyWrites(); |
| +} |
| + |
| +void PersistentPrefStoreImpl::ClearMutableValues() { |
| + backing_pref_store_->ClearMutableValues(); |
| +} |
| + |
| +void CallConnectCallback( |
| + PersistentPrefStore* pref_store, |
| + const mojom::PersistentPrefStoreConnector::ConnectCallback& callback) { |
| + mojom::PersistentPrefStorePtr pref_store_ptr; |
| + mojo::MakeStrongBinding( |
| + base::MakeUnique<PersistentPrefStoreImpl>(std::move(pref_store)), |
| + mojo::MakeRequest(&pref_store_ptr)); |
| + callback.Run(pref_store->GetReadError(), pref_store->ReadOnly(), |
| + pref_store->IsInitializationComplete() ? pref_store->GetValues() |
| + : nullptr, |
| + std::move(pref_store_ptr)); |
| +} |
| + |
| +void RemoveValueSilently(const base::WeakPtr<JsonPrefStore> pref_store, |
| + const std::string& key) { |
| + if (pref_store) { |
| + pref_store->RemoveValueSilently( |
| + key, WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |
| + } |
| +} |
| + |
| +std::unique_ptr<PrefHashStore> CreatePrefHashStore( |
| + const std::string& seed, |
| + const std::string& legacy_device_id, |
| + bool use_super_mac) { |
| + return std::unique_ptr<PrefHashStore>( |
| + new PrefHashStoreImpl(seed, legacy_device_id, use_super_mac)); |
| +} |
| + |
| +std::pair<std::unique_ptr<PrefHashStore>, std::unique_ptr<HashStoreContents>> |
| +GetExternalVerificationPrefHashStorePair(const std::string& seed, |
| + const std::string& legacy_device_id, |
| + const base::string16& registry_path, |
| + const base::FilePath& prefs_path) { |
| +#if defined(OS_WIN) |
| + return std::make_pair( |
| + base::MakeUnique<PrefHashStoreImpl>(seed, legacy_device_id, |
| + false /* use_super_mac */), |
| + base::MakeUnique<RegistryHashStoreContentsWin>( |
| + registry_path, prefs_path.DirName().BaseName().LossyDisplayName())); |
| +#else |
| + return std::make_pair(nullptr, nullptr); |
| +#endif |
| +} |
| + |
| +class PersistentPrefStoreConnectorImpl |
| + : public mojom::PersistentPrefStoreConnector, |
| + public PrefStore::Observer { |
| + public: |
| + PersistentPrefStoreConnectorImpl( |
| + scoped_refptr<PersistentPrefStore> backing_pref_store, |
| + std::unique_ptr<TrackedPreferenceValidationDelegate> validation_delegate) |
| + : backing_pref_store_(backing_pref_store), |
| + validation_delegate_(std::move(validation_delegate)) {} |
| + |
| + ~PersistentPrefStoreConnectorImpl() override = default; |
| + |
| + // mojom::PersistentPrefStoreConnector override: |
| + void Connect(const ConnectCallback& callback) override { |
| + if (backing_pref_store_->IsInitializationComplete()) { |
| + CallConnectCallback(backing_pref_store_.get(), callback); |
| + return; |
| + } |
| + connect_callbacks_.push_back(callback); |
| + if (loading_) |
| + return; |
| + |
| + backing_pref_store_->AddObserver(this); |
| + loading_ = true; |
| + backing_pref_store_->ReadPrefsAsync(nullptr); |
| + } |
| + |
| + static void CreateUserPrefs( |
| + const base::FilePath& pref_filename, |
| + const scoped_refptr<base::SequencedTaskRunner>& io_task_runner, |
| + mojom::PersistentPrefStoreConnectorRequest request) { |
| + mojo::MakeStrongBinding( |
| + base::MakeUnique<PersistentPrefStoreConnectorImpl>( |
| + new JsonPrefStore(pref_filename, io_task_runner, nullptr), nullptr), |
| + std::move(request)); |
| + } |
| + |
| + static void CreateSegregatedUserPrefs( |
| + const base::FilePath& unprotected_pref_filename, |
| + const base::FilePath& protected_pref_filename, |
| + const std::vector<PrefHashFilter::TrackedPreferenceMetadata>& |
| + tracking_configuration, |
| + size_t reporting_ids_count, |
| + const std::string& seed, |
| + const std::string& legacy_device_id, |
| + const base::string16& registry_path, |
| + std::unique_ptr<TrackedPreferenceValidationDelegate> validation_delegate, |
| + const base::Closure& on_reset_on_load, |
| + const scoped_refptr<base::SequencedTaskRunner>& io_task_runner, |
| + mojom::PersistentPrefStoreConnectorRequest request) { |
| + std::vector<PrefHashFilter::TrackedPreferenceMetadata> |
| + unprotected_configuration; |
| + std::vector<PrefHashFilter::TrackedPreferenceMetadata> |
| + protected_configuration; |
| + std::set<std::string> protected_pref_names; |
| + std::set<std::string> unprotected_pref_names; |
| + for (const auto& metadata : tracking_configuration) { |
| + if (metadata.enforcement_level > PrefHashFilter::NO_ENFORCEMENT) { |
| + protected_configuration.push_back(metadata); |
| + protected_pref_names.insert(metadata.name); |
| + } else { |
| + unprotected_configuration.push_back(metadata); |
| + unprotected_pref_names.insert(metadata.name); |
| + } |
| + } |
| + |
| + std::unique_ptr<PrefHashFilter> unprotected_pref_hash_filter( |
| + new PrefHashFilter(CreatePrefHashStore(seed, legacy_device_id, false), |
| + GetExternalVerificationPrefHashStorePair( |
| + seed, legacy_device_id, registry_path, |
| + unprotected_pref_filename), |
| + unprotected_configuration, base::Closure(), |
| + validation_delegate.get(), reporting_ids_count, |
| + false)); |
| + std::unique_ptr<PrefHashFilter> protected_pref_hash_filter( |
| + new PrefHashFilter(CreatePrefHashStore(seed, legacy_device_id, true), |
| + GetExternalVerificationPrefHashStorePair( |
| + seed, legacy_device_id, registry_path, |
| + unprotected_pref_filename), |
| + protected_configuration, on_reset_on_load, |
| + validation_delegate.get(), reporting_ids_count, |
| + true)); |
| + |
| + PrefHashFilter* raw_unprotected_pref_hash_filter = |
| + unprotected_pref_hash_filter.get(); |
| + PrefHashFilter* raw_protected_pref_hash_filter = |
| + protected_pref_hash_filter.get(); |
| + |
| + scoped_refptr<JsonPrefStore> unprotected_pref_store( |
| + new JsonPrefStore(unprotected_pref_filename, io_task_runner.get(), |
| + std::move(unprotected_pref_hash_filter))); |
| + scoped_refptr<JsonPrefStore> protected_pref_store( |
| + new JsonPrefStore(protected_pref_filename, io_task_runner.get(), |
| + std::move(protected_pref_hash_filter))); |
| + |
| + SetupTrackedPreferencesMigration( |
| + unprotected_pref_names, protected_pref_names, |
| + base::Bind(&RemoveValueSilently, unprotected_pref_store->AsWeakPtr()), |
| + base::Bind(&RemoveValueSilently, protected_pref_store->AsWeakPtr()), |
| + base::Bind(&JsonPrefStore::RegisterOnNextSuccessfulWriteReply, |
| + unprotected_pref_store->AsWeakPtr()), |
| + base::Bind(&JsonPrefStore::RegisterOnNextSuccessfulWriteReply, |
| + protected_pref_store->AsWeakPtr()), |
| + CreatePrefHashStore(seed, legacy_device_id, false), |
| + CreatePrefHashStore(seed, legacy_device_id, true), |
| + raw_unprotected_pref_hash_filter, raw_protected_pref_hash_filter); |
| + |
| + mojo::MakeStrongBinding( |
| + base::MakeUnique<PersistentPrefStoreConnectorImpl>( |
| + new SegregatedPrefStore(unprotected_pref_store, |
| + protected_pref_store, protected_pref_names), |
| + std::move(validation_delegate)), |
| + std::move(request)); |
| + } |
| + |
| + private: |
| + void OnPrefValueChanged(const std::string& key) override {} |
| + |
| + void OnInitializationCompleted(bool succeeded) override { |
| + loading_ = false; |
| + backing_pref_store_->RemoveObserver(this); |
| + for (const auto& callback : connect_callbacks_) { |
| + CallConnectCallback(backing_pref_store_.get(), callback); |
| + } |
| + connect_callbacks_.clear(); |
| + } |
| + |
| + scoped_refptr<PersistentPrefStore> backing_pref_store_; |
| + std::unique_ptr<TrackedPreferenceValidationDelegate> validation_delegate_; |
| + |
| + bool loading_ = false; |
| + std::vector<ConnectCallback> connect_callbacks_; |
| + |
| + DISALLOW_COPY_AND_ASSIGN(PersistentPrefStoreConnectorImpl); |
| +}; |
| + |
| +} // namespace |
| + |
| +void CreateUserPrefs( |
| + const base::FilePath& pref_filename, |
| + const scoped_refptr<base::SingleThreadTaskRunner>& connection_task_runner, |
| + const scoped_refptr<base::SequencedTaskRunner>& io_task_runner, |
| + mojom::PersistentPrefStoreConnectorRequest request) { |
| + connection_task_runner->PostTask( |
| + FROM_HERE, |
| + base::Bind(&PersistentPrefStoreConnectorImpl::CreateUserPrefs, |
| + pref_filename, io_task_runner, base::Passed(&request))); |
| +} |
| + |
| +void CreateSegregatedUserPrefs( |
| + const base::FilePath& unprotected_pref_filename, |
| + const base::FilePath& protected_pref_filename, |
| + const std::vector<PrefHashFilter::TrackedPreferenceMetadata>& |
| + tracking_configuration, |
| + size_t reporting_ids_count, |
| + const std::string& seed, |
| + const std::string& legacy_device_id, |
| + const base::string16& registry_path, |
| + base::WeakPtr<TrackedPreferenceValidationDelegate> validation_delegate, |
| + const base::Closure& on_reset_on_load, |
| + const scoped_refptr<base::SingleThreadTaskRunner>& connection_task_runner, |
| + const scoped_refptr<base::SequencedTaskRunner>& io_task_runner, |
| + mojom::PersistentPrefStoreConnectorRequest request) { |
| + connection_task_runner->PostTask( |
| + FROM_HERE, |
| + base::Bind( |
| + &PersistentPrefStoreConnectorImpl::CreateSegregatedUserPrefs, |
| + unprotected_pref_filename, protected_pref_filename, |
| + tracking_configuration, reporting_ids_count, seed, legacy_device_id, |
| + registry_path, |
| + base::Passed(validation_delegate |
| + ? base::MakeUnique< |
| + ForwardingTrackedPreferenceValidationDelegate>( |
| + base::ThreadTaskRunnerHandle::Get(), |
| + validation_delegate) |
| + : nullptr), |
| + on_reset_on_load, io_task_runner, base::Passed(&request))); |
| +} |
| + |
| +} // namespace prefs |