| // Copyright 2013 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 "components/autofill/core/browser/autocomplete_history_manager.h" |
| |
| #include <unordered_map> |
| #include <utility> |
| #include <vector> |
| |
| #include "base/memory/weak_ptr.h" |
| #include "base/stl_util.h" |
| #include "base/strings/string16.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "components/autofill/core/browser/autofill_experiments.h" |
| #include "components/autofill/core/browser/autofill_metrics.h" |
| #include "components/autofill/core/browser/suggestion.h" |
| #include "components/autofill/core/browser/validation.h" |
| #include "components/autofill/core/browser/webdata/autofill_entry.h" |
| #include "components/autofill/core/common/autofill_clock.h" |
| #include "components/autofill/core/common/autofill_features.h" |
| #include "components/autofill/core/common/autofill_prefs.h" |
| #include "components/autofill/core/common/form_data.h" |
| #include "components/prefs/pref_service.h" |
| #include "components/version_info/version_info.h" |
| |
| namespace autofill { |
| namespace { |
| |
| // Limit on the number of suggestions to appear in the pop-up menu under an |
| // text input element in a form. |
| const int kMaxAutocompleteMenuItems = 6; |
| |
| bool IsTextField(const FormFieldData& field) { |
| return |
| field.form_control_type == "text" || |
| field.form_control_type == "search" || |
| field.form_control_type == "tel" || |
| field.form_control_type == "url" || |
| field.form_control_type == "email"; |
| } |
| |
| } // namespace |
| |
| void AutocompleteHistoryManager::UMARecorder::OnGetAutocompleteSuggestions( |
| const base::string16& name, |
| WebDataServiceBase::Handle pending_query_handle) { |
| // log only if the current field is different than the latest one that has |
| // been logged. we assume that user works at the same field if |
| // measuring_name_ is same as the name of current field. |
| bool should_log_query = measuring_name_ != name; |
| |
| if (should_log_query) { |
| AutofillMetrics::LogAutocompleteQuery(pending_query_handle /* created */); |
| measuring_name_ = name; |
| } |
| // We should track the query and log the suggestions, only if |
| // - query has been logged. |
| // - or, the query we previously tracked has been cancelled |
| // The previous query must be cancelled if measuring_query_handle_ isn't |
| // reset. |
| if (should_log_query || measuring_query_handle_) |
| measuring_query_handle_ = pending_query_handle; |
| } |
| |
| void AutocompleteHistoryManager::UMARecorder::OnWebDataServiceRequestDone( |
| WebDataServiceBase::Handle pending_query_handle, |
| bool has_suggestion) { |
| // If handle of completed query does not match the query we're currently |
| // measuring then we've already logged a query for this name. |
| bool was_already_logged = (pending_query_handle != measuring_query_handle_); |
| measuring_query_handle_ = 0; |
| if (was_already_logged) |
| return; |
| AutofillMetrics::LogAutocompleteSuggestions(has_suggestion); |
| } |
| |
| AutocompleteHistoryManager::QueryHandler::QueryHandler( |
| int client_query_id, |
| bool autoselect_first_suggestion, |
| base::string16 prefix, |
| base::WeakPtr<SuggestionsHandler> handler) |
| : client_query_id_(client_query_id), |
| autoselect_first_suggestion_(autoselect_first_suggestion), |
| prefix_(prefix), |
| handler_(std::move(handler)) {} |
| |
| AutocompleteHistoryManager::QueryHandler::QueryHandler( |
| const QueryHandler& original) = default; |
| |
| AutocompleteHistoryManager::QueryHandler::~QueryHandler() = default; |
| |
| AutocompleteHistoryManager::AutocompleteHistoryManager() |
| // It is safe to base::Unretained a raw pointer to the current instance, |
| // as it is already being owned elsewhere and will be cleaned-up properly. |
| // Also, the map of callbacks will be deleted when this instance is |
| // destroyed, which means we won't attempt to run one of these callbacks |
| // beyond the life of this instance. |
| : request_callbacks_( |
| {{AUTOFILL_VALUE_RESULT, |
| base::BindRepeating( |
| &AutocompleteHistoryManager::OnAutofillValuesReturned, |
| base::Unretained(this))}, |
| {AUTOFILL_CLEANUP_RESULT, |
| base::BindRepeating( |
| &AutocompleteHistoryManager::OnAutofillCleanupReturned, |
| base::Unretained(this))}}), |
| weak_ptr_factory_(this) {} |
| |
| AutocompleteHistoryManager::~AutocompleteHistoryManager() { |
| CancelAllPendingQueries(); |
| } |
| |
| void AutocompleteHistoryManager::Init( |
| scoped_refptr<AutofillWebDataService> profile_database, |
| PrefService* pref_service, |
| bool is_off_the_record) { |
| profile_database_ = profile_database; |
| pref_service_ = pref_service; |
| is_off_the_record_ = is_off_the_record; |
| |
| // No need to run the retention policy in OTR. |
| if (!is_off_the_record_ && |
| base::FeatureList::IsEnabled( |
| autofill::features::kAutocompleteRetentionPolicyEnabled)) { |
| // Upon successful cleanup, the last cleaned-up major version is being |
| // stored in this pref. |
| int last_cleaned_version = pref_service_->GetInteger( |
| prefs::kAutocompleteLastVersionRetentionPolicy); |
| if (CHROME_VERSION_MAJOR > last_cleaned_version) { |
| // Trigger the cleanup. |
| profile_database_->RemoveExpiredAutocompleteEntries(this); |
| } |
| } |
| } |
| |
| base::WeakPtr<AutocompleteHistoryManager> |
| AutocompleteHistoryManager::GetWeakPtr() { |
| return weak_ptr_factory_.GetWeakPtr(); |
| } |
| |
| void AutocompleteHistoryManager::OnGetAutocompleteSuggestions( |
| int query_id, |
| bool is_autocomplete_enabled, |
| bool autoselect_first_suggestion, |
| const base::string16& name, |
| const base::string16& prefix, |
| const std::string& form_control_type, |
| base::WeakPtr<SuggestionsHandler> handler) { |
| CancelPendingQueries(handler.get()); |
| |
| if (!is_autocomplete_enabled || form_control_type == "textarea" || |
| IsInAutofillSuggestionsDisabledExperiment()) { |
| SendSuggestions({}, QueryHandler(query_id, autoselect_first_suggestion, |
| prefix, handler)); |
| uma_recorder_.OnGetAutocompleteSuggestions(name, |
| 0 /* pending_query_handle */); |
| return; |
| } |
| |
| if (profile_database_) { |
| auto query_handle = profile_database_->GetFormValuesForElementName( |
| name, prefix, kMaxAutocompleteMenuItems, this); |
| uma_recorder_.OnGetAutocompleteSuggestions(name, query_handle); |
| |
| // We can simply insert, since |query_handle| is always unique. |
| pending_queries_.insert( |
| {query_handle, |
| QueryHandler(query_id, autoselect_first_suggestion, prefix, handler)}); |
| } |
| } |
| |
| void AutocompleteHistoryManager::OnWillSubmitForm( |
| const FormData& form, |
| bool is_autocomplete_enabled) { |
| if (!is_autocomplete_enabled || is_off_the_record_) |
| return; |
| |
| // We put the following restriction on stored FormFields: |
| // - non-empty name |
| // - non-empty value |
| // - text field |
| // - autocomplete is not disabled |
| // - value is not a credit card number |
| // - value is not a SSN |
| // - field was not identified as a CVC field (this is handled in |
| // AutofillManager) |
| // - field is focusable |
| // - not a presentation field |
| std::vector<FormFieldData> values; |
| for (const FormFieldData& field : form.fields) { |
| if (!field.value.empty() && !field.name.empty() && IsTextField(field) && |
| field.should_autocomplete && !IsValidCreditCardNumber(field.value) && |
| !IsSSN(field.value) && field.is_focusable && |
| field.role != FormFieldData::ROLE_ATTRIBUTE_PRESENTATION) { |
| values.push_back(field); |
| } |
| } |
| |
| if (!values.empty() && profile_database_.get()) |
| profile_database_->AddFormFields(values); |
| } |
| |
| void AutocompleteHistoryManager::OnRemoveAutocompleteEntry( |
| const base::string16& name, const base::string16& value) { |
| if (profile_database_) |
| profile_database_->RemoveFormValueForElementName(name, value); |
| } |
| |
| void AutocompleteHistoryManager::OnAutocompleteEntrySelected( |
| const base::string16& value) { |
| // Try to find the AutofillEntry associated with the given suggestion. |
| auto last_entries_iter = last_entries_.find(value); |
| if (last_entries_iter == last_entries_.end()) { |
| // Not found, therefore nothing to do. Most likely there was a race |
| // condition, but it's not that big of a deal in the current scenario |
| // (logging metrics). |
| NOTREACHED(); |
| return; |
| } |
| |
| // The AutofillEntry was found, use it to log the DaysSinceLastUsed. |
| const AutofillEntry& entry = last_entries_iter->second; |
| base::TimeDelta time_delta = AutofillClock::Now() - entry.date_last_used(); |
| AutofillMetrics::LogAutocompleteDaysSinceLastUse(time_delta.InDays()); |
| } |
| |
| void AutocompleteHistoryManager::CancelPendingQueries( |
| const SuggestionsHandler* handler) { |
| if (handler && profile_database_) { |
| for (auto iter : pending_queries_) { |
| const QueryHandler& query_handler = iter.second; |
| |
| if (query_handler.handler_ && query_handler.handler_.get() == handler) { |
| profile_database_->CancelRequest(iter.first); |
| } |
| } |
| } |
| |
| // Cleaning up the map with the cancelled handler to remove cancelled |
| // requests. |
| CleanupEntries(handler); |
| } |
| |
| void AutocompleteHistoryManager::OnWebDataServiceRequestDone( |
| WebDataServiceBase::Handle current_handle, |
| std::unique_ptr<WDTypedResult> result) { |
| DCHECK(current_handle); |
| DCHECK(result); |
| |
| WDResultType result_type = result->GetType(); |
| |
| auto request_callbacks_iter = request_callbacks_.find(result_type); |
| if (request_callbacks_iter == request_callbacks_.end()) { |
| // There are no callbacks for this response, hence nothing to do. |
| return; |
| } |
| |
| request_callbacks_iter->second.Run(current_handle, std::move(result)); |
| } |
| |
| void AutocompleteHistoryManager::SendSuggestions( |
| const std::vector<AutofillEntry>& entries, |
| const QueryHandler& query_handler) { |
| if (!query_handler.handler_) { |
| // Either the handler has been destroyed, or it is invalid. |
| return; |
| } |
| |
| // If there is only one suggestion that is the exact same string as |
| // what is in the input box, then don't show the suggestion. |
| bool hide_suggestions = |
| entries.size() == 1 && query_handler.prefix_ == entries[0].key().value(); |
| |
| std::vector<Suggestion> suggestions; |
| last_entries_.clear(); |
| |
| if (!hide_suggestions) { |
| for (const AutofillEntry& entry : entries) { |
| suggestions.push_back(Suggestion(entry.key().value())); |
| last_entries_.insert({entry.key().value(), AutofillEntry(entry)}); |
| } |
| } |
| |
| query_handler.handler_->OnSuggestionsReturned( |
| query_handler.client_query_id_, |
| query_handler.autoselect_first_suggestion_, suggestions); |
| } |
| |
| void AutocompleteHistoryManager::OnAutofillValuesReturned( |
| WebDataServiceBase::Handle current_handle, |
| std::unique_ptr<WDTypedResult> result) { |
| DCHECK_EQ(AUTOFILL_VALUE_RESULT, result->GetType()); |
| |
| auto pending_queries_iter = pending_queries_.find(current_handle); |
| if (pending_queries_iter == pending_queries_.end()) { |
| // There's no handler for this query, hence nothing to do. |
| return; |
| } |
| |
| // Moving the handler since we're erasing the entry. |
| auto query_handler = std::move(pending_queries_iter->second); |
| |
| // Removing the query, as it is no longer pending. |
| pending_queries_.erase(pending_queries_iter); |
| |
| // Returning early here if |result| is NULL. We've seen this happen on |
| // Linux due to NFS dismounting and causing sql failures. |
| // See http://crbug.com/68783. |
| if (!result) { |
| SendSuggestions({}, query_handler); |
| uma_recorder_.OnWebDataServiceRequestDone(current_handle, |
| false /* has_suggestion */); |
| return; |
| } |
| |
| const WDResult<std::vector<AutofillEntry>>* autofill_result = |
| static_cast<const WDResult<std::vector<AutofillEntry>>*>(result.get()); |
| std::vector<AutofillEntry> entries = autofill_result->GetValue(); |
| SendSuggestions(entries, query_handler); |
| uma_recorder_.OnWebDataServiceRequestDone( |
| current_handle, !entries.empty() /* has_suggestion */); |
| } |
| |
| void AutocompleteHistoryManager::OnAutofillCleanupReturned( |
| WebDataServiceBase::Handle current_handle, |
| std::unique_ptr<WDTypedResult> result) { |
| DCHECK_EQ(AUTOFILL_CLEANUP_RESULT, result->GetType()); |
| |
| const WDResult<size_t>* cleanup_wdresult = |
| static_cast<const WDResult<size_t>*>(result.get()); |
| |
| AutofillMetrics::LogNumberOfAutocompleteEntriesCleanedUp( |
| cleanup_wdresult->GetValue()); |
| |
| // Cleanup was successful, update the latest run milestone. |
| pref_service_->SetInteger(prefs::kAutocompleteLastVersionRetentionPolicy, |
| CHROME_VERSION_MAJOR); |
| } |
| |
| void AutocompleteHistoryManager::CancelAllPendingQueries() { |
| if (profile_database_) { |
| for (const auto& pending_query : pending_queries_) { |
| profile_database_->CancelRequest(pending_query.first); |
| } |
| } |
| |
| pending_queries_.clear(); |
| } |
| |
| void AutocompleteHistoryManager::CleanupEntries( |
| const SuggestionsHandler* handler) { |
| base::EraseIf(pending_queries_, [handler](const auto& pending_query) { |
| const QueryHandler& query_handler = pending_query.second; |
| return !query_handler.handler_ || query_handler.handler_.get() == handler; |
| }); |
| } |
| |
| } // namespace autofill |