| // Copyright 2017 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/omnibox/browser/remote_suggestions_service.h" |
| |
| #include <memory> |
| #include <utility> |
| |
| #include "base/functional/bind.h" |
| #include "base/metrics/histogram_functions.h" |
| #include "components/omnibox/browser/base_search_provider.h" |
| #include "components/omnibox/browser/document_suggestions_service.h" |
| #include "components/search/search.h" |
| #include "components/search_engines/template_url_service.h" |
| #include "components/variations/net/variations_http_headers.h" |
| #include "net/base/load_flags.h" |
| #include "net/base/url_util.h" |
| #include "net/traffic_annotation/network_traffic_annotation.h" |
| #include "services/network/public/cpp/resource_request.h" |
| #include "services/network/public/cpp/shared_url_loader_factory.h" |
| #include "services/network/public/cpp/simple_url_loader.h" |
| #include "services/network/public/mojom/url_response_head.mojom.h" |
| #include "third_party/lens_server_proto/lens_overlay_service_deps.pb.h" |
| #include "third_party/metrics_proto/omnibox_event.pb.h" |
| |
| namespace { |
| |
| void LogSuggestRequestSent(RemoteRequestType request_type) { |
| base::UmaHistogramEnumeration("Omnibox.SuggestRequestsSent", request_type); |
| } |
| |
| void AddVariationHeaders(network::ResourceRequest* request) { |
| // Note: It's OK to pass InIncognito::kNo since we are expected to be in |
| // non-incognito state here (i.e. remote suggestions are not served in |
| // incognito mode). |
| variations::AppendVariationsHeaderUnknownSignedIn( |
| request->url, variations::InIncognito::kNo, request); |
| } |
| |
| } // namespace |
| |
| RemoteSuggestionsService::RemoteSuggestionsService( |
| DocumentSuggestionsService* document_suggestions_service, |
| scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory) |
| : document_suggestions_service_(document_suggestions_service), |
| url_loader_factory_(url_loader_factory) { |
| DCHECK(url_loader_factory); |
| } |
| |
| RemoteSuggestionsService::~RemoteSuggestionsService() = default; |
| |
| // static |
| GURL RemoteSuggestionsService::EndpointUrl( |
| const TemplateURL* template_url, |
| TemplateURLRef::SearchTermsArgs search_terms_args, |
| const SearchTermsData& search_terms_data) { |
| GURL url = GURL(template_url->suggestions_url_ref().ReplaceSearchTerms( |
| search_terms_args, search_terms_data)); |
| |
| // Return early for non-Google template URLs. |
| if (!search::TemplateURLIsGoogle(template_url, search_terms_data)) { |
| return url; |
| } |
| |
| // Append or replace query params based on `page_classification`. |
| switch (search_terms_args.page_classification) { |
| case metrics::OmniboxEventProto::CHROMEOS_APP_LIST: { |
| // Append `sclient=cros-launcher` for CrOS app_list launcher entry point. |
| url = net::AppendOrReplaceQueryParameter(url, "sclient", "cros-launcher"); |
| break; |
| } |
| case metrics::OmniboxEventProto::LENS_SIDE_PANEL_SEARCHBOX: { |
| // Append `iil=` for the multimodal searchbox entry point, if available. |
| // TODO(b/328763711): Replace this with a TemplateURL substitution. |
| if (search_terms_args.lens_overlay_interaction_response.has_value() && |
| search_terms_args.lens_overlay_interaction_response |
| ->has_encoded_response()) { |
| url = net::AppendOrReplaceQueryParameter( |
| url, "iil", |
| search_terms_args.lens_overlay_interaction_response |
| ->encoded_response()); |
| } |
| break; |
| } |
| default: |
| break; |
| } |
| |
| return url; |
| } |
| |
| std::unique_ptr<network::SimpleURLLoader> |
| RemoteSuggestionsService::StartSuggestionsRequest( |
| RemoteRequestType request_type, |
| const TemplateURL* template_url, |
| TemplateURLRef::SearchTermsArgs search_terms_args, |
| const SearchTermsData& search_terms_data, |
| CompletionCallback completion_callback) { |
| DCHECK(template_url); |
| |
| const GURL suggest_url = |
| EndpointUrl(template_url, search_terms_args, search_terms_data); |
| if (!suggest_url.is_valid()) { |
| return nullptr; |
| } |
| |
| net::NetworkTrafficAnnotationTag traffic_annotation = |
| net::DefineNetworkTrafficAnnotation("omnibox_suggest", R"( |
| semantics { |
| sender: "Omnibox" |
| description: |
| "Chrome can provide search and navigation suggestions from the " |
| "currently-selected search provider in the omnibox dropdown, based " |
| "on user input." |
| trigger: "User typing in the omnibox." |
| data: |
| "The text typed into the address bar. Potentially other metadata, " |
| "such as the current cursor position or URL of the current page." |
| destination: WEBSITE |
| } |
| policy { |
| cookies_allowed: YES |
| cookies_store: "user" |
| setting: |
| "Users can control this feature via the 'Use a prediction service " |
| "to help complete searches and URLs typed in the address bar' " |
| "setting under 'Privacy'. The feature is enabled by default." |
| chrome_policy { |
| SearchSuggestEnabled { |
| policy_options {mode: MANDATORY} |
| SearchSuggestEnabled: false |
| } |
| } |
| })"); |
| auto request = std::make_unique<network::ResourceRequest>(); |
| request->url = suggest_url; |
| request->load_flags = net::LOAD_DO_NOT_SAVE_COOKIES; |
| // Set the SiteForCookies to the request URL's site to avoid cookie blocking. |
| request->site_for_cookies = net::SiteForCookies::FromUrl(suggest_url); |
| // Add Chrome experiment state to the request headers. |
| AddVariationHeaders(request.get()); |
| |
| // Create a unique identifier for the request. |
| const base::UnguessableToken request_id = base::UnguessableToken::Create(); |
| |
| // Notify the observers that request has been created. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestCreated(request_id, request.get()); |
| } |
| |
| // Make loader and start download. |
| std::unique_ptr<network::SimpleURLLoader> loader = |
| network::SimpleURLLoader::Create(std::move(request), traffic_annotation); |
| loader->DownloadToStringOfUnboundedSizeUntilCrashAndDie( |
| url_loader_factory_.get(), |
| base::BindOnce(&RemoteSuggestionsService::OnURLLoadComplete, |
| weak_ptr_factory_.GetWeakPtr(), request_id, |
| std::move(completion_callback), loader.get())); |
| |
| // Notify the observers that the transfer started. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestStarted(request_id, loader.get(), |
| /*request_body*/ ""); |
| } |
| LogSuggestRequestSent(request_type); |
| return loader; |
| } |
| |
| std::unique_ptr<network::SimpleURLLoader> |
| RemoteSuggestionsService::StartZeroPrefixSuggestionsRequest( |
| RemoteRequestType request_type, |
| const TemplateURL* template_url, |
| TemplateURLRef::SearchTermsArgs search_terms_args, |
| const SearchTermsData& search_terms_data, |
| CompletionCallback completion_callback) { |
| DCHECK(template_url); |
| |
| const GURL suggest_url = |
| EndpointUrl(template_url, search_terms_args, search_terms_data); |
| if (!suggest_url.is_valid()) { |
| return nullptr; |
| } |
| |
| net::NetworkTrafficAnnotationTag traffic_annotation = |
| net::DefineNetworkTrafficAnnotation("omnibox_zerosuggest", R"( |
| semantics { |
| sender: "Omnibox" |
| description: |
| "When the user focuses the omnibox, Chrome can provide search or " |
| "navigation suggestions from the default search provider in the " |
| "omnibox dropdown, based on the current page URL.\n" |
| "This is limited to users whose default search engine is Google, " |
| "as no other search engines currently support this kind of " |
| "suggestion." |
| trigger: "The omnibox receives focus." |
| data: "The URL of the current page." |
| destination: GOOGLE_OWNED_SERVICE |
| } |
| policy { |
| cookies_allowed: YES |
| cookies_store: "user" |
| setting: |
| "Users can control this feature via the 'Use a prediction service " |
| "to help complete searches and URLs typed in the address bar' " |
| "settings under 'Privacy'. The feature is enabled by default." |
| chrome_policy { |
| SearchSuggestEnabled { |
| policy_options {mode: MANDATORY} |
| SearchSuggestEnabled: false |
| } |
| } |
| })"); |
| |
| auto request = std::make_unique<network::ResourceRequest>(); |
| request->url = suggest_url; |
| request->load_flags = net::LOAD_DO_NOT_SAVE_COOKIES; |
| if (search_terms_args.bypass_cache) { |
| request->load_flags |= net::LOAD_BYPASS_CACHE; |
| } |
| // Set the SiteForCookies to the request URL's site to avoid cookie blocking. |
| request->site_for_cookies = net::SiteForCookies::FromUrl(suggest_url); |
| // Add Chrome experiment state to the request headers. |
| AddVariationHeaders(request.get()); |
| |
| // Create a unique identifier for the request. |
| const base::UnguessableToken request_id = base::UnguessableToken::Create(); |
| |
| // Notify the observers that request has been created. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestCreated(request_id, request.get()); |
| } |
| |
| // Make loader and start download. |
| std::unique_ptr<network::SimpleURLLoader> loader = |
| network::SimpleURLLoader::Create(std::move(request), traffic_annotation); |
| loader->DownloadToStringOfUnboundedSizeUntilCrashAndDie( |
| url_loader_factory_.get(), |
| base::BindOnce(&RemoteSuggestionsService::OnURLLoadComplete, |
| weak_ptr_factory_.GetWeakPtr(), request_id, |
| std::move(completion_callback), loader.get())); |
| |
| // Notify the observers that the transfer started. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestStarted(request_id, loader.get(), |
| /*request_body*/ ""); |
| } |
| LogSuggestRequestSent(request_type); |
| return loader; |
| } |
| |
| void RemoteSuggestionsService::CreateDocumentSuggestionsRequest( |
| const std::u16string& query, |
| bool is_incognito, |
| DocumentStartCallback start_callback, |
| CompletionCallback completion_callback) { |
| // Create a unique identifier for the request. |
| const base::UnguessableToken request_id = base::UnguessableToken::Create(); |
| |
| document_suggestions_service_->CreateDocumentSuggestionsRequest( |
| query, is_incognito, |
| base::BindOnce( |
| &RemoteSuggestionsService::OnDocumentSuggestionsRequestAvailable, |
| weak_ptr_factory_.GetWeakPtr(), request_id), |
| base::BindOnce( |
| &RemoteSuggestionsService::OnDocumentSuggestionsLoaderAvailable, |
| weak_ptr_factory_.GetWeakPtr(), request_id, |
| std::move(start_callback)), |
| base::BindOnce(&RemoteSuggestionsService::OnURLLoadComplete, |
| weak_ptr_factory_.GetWeakPtr(), request_id, |
| std::move(completion_callback))); |
| } |
| |
| void RemoteSuggestionsService::StopCreatingDocumentSuggestionsRequest() { |
| document_suggestions_service_->StopCreatingDocumentSuggestionsRequest(); |
| } |
| |
| std::unique_ptr<network::SimpleURLLoader> |
| RemoteSuggestionsService::StartDeletionRequest( |
| const std::string& deletion_url, |
| CompletionCallback completion_callback) { |
| const GURL url(deletion_url); |
| DCHECK(url.is_valid()); |
| |
| net::NetworkTrafficAnnotationTag traffic_annotation = |
| net::DefineNetworkTrafficAnnotation("omnibox_suggest_deletion", R"( |
| semantics { |
| sender: "Omnibox" |
| description: |
| "When users attempt to delete server-provided personalized search " |
| "or navigation suggestions from the omnibox dropdown, Chrome sends " |
| "a message to the server requesting deletion of the suggestion." |
| trigger: |
| "A user attempt to delete a server-provided omnibox suggestion, " |
| "for which the server provided a custom deletion URL." |
| data: |
| "No user data is explicitly sent with the request, but because the " |
| "requested URL is provided by the server for each specific " |
| "suggestion, it necessarily uniquely identifies the suggestion the " |
| "user is attempting to delete." |
| destination: WEBSITE |
| } |
| policy { |
| cookies_allowed: YES |
| cookies_store: "user" |
| setting: |
| "Since this can only be triggered on seeing server-provided " |
| "suggestions in the omnibox dropdown, whether it is enabled is the " |
| "same as whether those suggestions are enabled.\n" |
| "Users can control this feature via the 'Use a prediction service " |
| "to help complete searches and URLs typed in the address bar' " |
| "setting under 'Privacy'. The feature is enabled by default." |
| chrome_policy { |
| SearchSuggestEnabled { |
| policy_options {mode: MANDATORY} |
| SearchSuggestEnabled: false |
| } |
| } |
| })"); |
| auto request = std::make_unique<network::ResourceRequest>(); |
| request->url = url; |
| // Set the SiteForCookies to the request URL's site to avoid cookie blocking. |
| request->site_for_cookies = net::SiteForCookies::FromUrl(url); |
| // Add Chrome experiment state to the request headers. |
| AddVariationHeaders(request.get()); |
| |
| // Create a unique identifier for the request. |
| const base::UnguessableToken request_id = base::UnguessableToken::Create(); |
| |
| // Notify the observers that request has been created. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestCreated(request_id, request.get()); |
| } |
| |
| // Make loader and start download. |
| std::unique_ptr<network::SimpleURLLoader> loader = |
| network::SimpleURLLoader::Create(std::move(request), traffic_annotation); |
| loader->DownloadToStringOfUnboundedSizeUntilCrashAndDie( |
| url_loader_factory_.get(), |
| base::BindOnce(&RemoteSuggestionsService::OnURLLoadComplete, |
| weak_ptr_factory_.GetWeakPtr(), request_id, |
| std::move(completion_callback), loader.get())); |
| |
| // Notify the observers that the transfer started. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestStarted(request_id, loader.get(), |
| /*request_body*/ ""); |
| } |
| LogSuggestRequestSent(RemoteRequestType::kDeletion); |
| return loader; |
| } |
| |
| void RemoteSuggestionsService::AddObserver(Observer* observer) { |
| observers_.AddObserver(observer); |
| } |
| |
| void RemoteSuggestionsService::RemoveObserver(Observer* observer) { |
| observers_.RemoveObserver(observer); |
| } |
| |
| void RemoteSuggestionsService::set_url_loader_factory_for_testing( |
| scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory) { |
| url_loader_factory_ = std::move(url_loader_factory); |
| } |
| |
| void RemoteSuggestionsService::OnDocumentSuggestionsRequestAvailable( |
| const base::UnguessableToken& request_id, |
| network::ResourceRequest* request) { |
| // Notify the observers that request has been created. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestCreated(request_id, request); |
| } |
| } |
| |
| void RemoteSuggestionsService::OnDocumentSuggestionsLoaderAvailable( |
| const base::UnguessableToken& request_id, |
| DocumentStartCallback start_callback, |
| std::unique_ptr<network::SimpleURLLoader> loader, |
| const std::string& request_body) { |
| // Notify the observers that the transfer started. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestStarted(request_id, loader.get(), request_body); |
| } |
| LogSuggestRequestSent(RemoteRequestType::kDocumentSuggest); |
| std::move(start_callback).Run(std::move(loader)); |
| } |
| |
| void RemoteSuggestionsService::OnURLLoadComplete( |
| const base::UnguessableToken& request_id, |
| CompletionCallback completion_callback, |
| const network::SimpleURLLoader* source, |
| std::unique_ptr<std::string> response_body) { |
| const int response_code = |
| source->ResponseInfo() && source->ResponseInfo()->headers |
| ? source->ResponseInfo()->headers->response_code() |
| : 0; |
| |
| // Notify the observers that the transfer is done. |
| for (Observer& observer : observers_) { |
| observer.OnSuggestRequestCompleted(request_id, response_code, |
| response_body); |
| } |
| |
| std::move(completion_callback) |
| .Run(source, response_code, std::move(response_body)); |
| } |