| // Copyright 2014 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "services/preferences/tracked/tracked_atomic_preference.h" |
| |
| #include "base/values.h" |
| #include "services/preferences/public/mojom/tracked_preference_validation_delegate.mojom.h" |
| #include "services/preferences/tracked/pref_hash_store_transaction.h" |
| |
| using ValueState = |
| prefs::mojom::TrackedPreferenceValidationDelegate::ValueState; |
| |
| TrackedAtomicPreference::TrackedAtomicPreference( |
| const std::string& pref_path, |
| size_t reporting_id, |
| size_t reporting_ids_count, |
| prefs::mojom::TrackedPreferenceMetadata::EnforcementLevel enforcement_level, |
| prefs::mojom::TrackedPreferenceMetadata::ValueType value_type, |
| prefs::mojom::TrackedPreferenceValidationDelegate* delegate) |
| : pref_path_(pref_path), |
| helper_(pref_path, |
| reporting_id, |
| reporting_ids_count, |
| enforcement_level, |
| value_type), |
| delegate_(delegate) {} |
| |
| TrackedPreferenceType TrackedAtomicPreference::GetType() const { |
| return TrackedPreferenceType::ATOMIC; |
| } |
| |
| void TrackedAtomicPreference::OnNewValue( |
| const base::Value* value, |
| PrefHashStoreTransaction* transaction, |
| const os_crypt_async::Encryptor* encryptor) const { |
| transaction->StoreHash(pref_path_, value); |
| |
| if (encryptor) { |
| transaction->StoreEncryptedHash(pref_path_, value); |
| } |
| } |
| |
| bool TrackedAtomicPreference::EnforceAndReport( |
| base::Value::Dict& pref_store_contents, |
| PrefHashStoreTransaction* transaction, |
| PrefHashStoreTransaction* external_validation_transaction, |
| const os_crypt_async::Encryptor* encryptor) const { |
| const base::Value* value = pref_store_contents.FindByDottedPath(pref_path_); |
| // TODO(zackhan@): Currently this function support dual-hash validation. |
| // Revisit and double check this function later on when the feature is fully |
| // rolled out and the hmac based validation is removed. |
| // transaction->CheckValue() (from CL1) is dual-hash aware and uses the |
| // encryptor with which `transaction` was initialized by PrefHashFilter. |
| ValueState value_state = transaction->CheckValue(pref_path_, value); |
| helper_.ReportValidationResult(value_state, transaction->GetStoreUMASuffix()); |
| |
| ValueState external_validation_value_state = ValueState::UNSUPPORTED; |
| if (external_validation_transaction) { |
| external_validation_value_state = |
| external_validation_transaction->CheckValue(pref_path_, value); |
| helper_.ReportValidationResult( |
| external_validation_value_state, |
| external_validation_transaction->GetStoreUMASuffix()); |
| } |
| |
| if (delegate_) { |
| delegate_->OnAtomicPreferenceValidation( |
| pref_path_, value ? std::make_optional(value->Clone()) : std::nullopt, |
| value_state, external_validation_value_state, helper_.IsPersonal()); |
| } |
| TrackedPreferenceHelper::ResetAction reset_action = |
| helper_.GetAction(value_state); |
| helper_.ReportAction(reset_action); |
| |
| bool was_reset = false; |
| if (reset_action == TrackedPreferenceHelper::DO_RESET || |
| reset_action == TrackedPreferenceHelper::DO_RESET_LEGACY || |
| reset_action == TrackedPreferenceHelper::DO_RESET_ENCRYPTED) { |
| pref_store_contents.RemoveByDottedPath(pref_path_); |
| was_reset = true; |
| } |
| |
| // A hash needs to be stored if the state is anything other than the two |
| // ideal "unchanged" states. This includes writing an encrypted hash to a |
| // preference that was UNCHANGED_VIA_HMAC_FALLBACK. |
| if (value_state != ValueState::UNCHANGED && |
| value_state != ValueState::UNCHANGED_ENCRYPTED) { |
| // Get the new value (which could be the default if a reset occurred). |
| const base::Value* new_value = |
| pref_store_contents.FindByDottedPath(pref_path_); |
| |
| // Store the legacy MAC for backward compatibility. |
| transaction->StoreHash(pref_path_, new_value); |
| |
| if (encryptor) { |
| transaction->StoreEncryptedHash(pref_path_, new_value); |
| } |
| } |
| |
| // Update MACs in the external store if there is one and there either was a |
| // reset or external validation failed. |
| if (external_validation_transaction && |
| (was_reset || external_validation_value_state != ValueState::UNCHANGED)) { |
| const base::Value* new_value = |
| pref_store_contents.FindByDottedPath(pref_path_); |
| external_validation_transaction->StoreHash(pref_path_, new_value); |
| if (encryptor) { |
| external_validation_transaction->StoreEncryptedHash( |
| pref_path_, pref_store_contents.FindByDottedPath(pref_path_)); |
| } |
| } |
| |
| return was_reset; |
| } |