// Copyright (c) 2010 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 "chrome/browser/autofill/autofill_download.h" #include #include "base/logging.h" #include "base/rand_util.h" #include "base/stl_util-inl.h" #include "chrome/browser/autofill/autofill_xml_parser.h" #include "chrome/browser/pref_service.h" #include "chrome/browser/profile.h" #include "chrome/common/pref_names.h" #include "net/http/http_response_headers.h" #define DISABLED_REQUEST_URL "http://disabled" #if defined(GOOGLE_CHROME_BUILD) #include "chrome/browser/autofill/internal/autofill_download_internal.h" #else #define AUTO_FILL_QUERY_SERVER_REQUEST_URL DISABLED_REQUEST_URL #define AUTO_FILL_UPLOAD_SERVER_REQUEST_URL DISABLED_REQUEST_URL #define AUTO_FILL_QUERY_SERVER_NAME_START_IN_HEADER "SOMESERVER/" #endif AutoFillDownloadManager::AutoFillDownloadManager(Profile* profile) : profile_(profile), observer_(NULL), next_query_request_(base::Time::Now()), next_upload_request_(base::Time::Now()), positive_upload_rate_(0), negative_upload_rate_(0), fetcher_id_for_unittest_(0), is_testing_(false) { // |profile_| could be NULL in some unit-tests. if (profile_) { PrefService* preferences = profile_->GetPrefs(); positive_upload_rate_ = preferences->GetReal(prefs::kAutoFillPositiveUploadRate); negative_upload_rate_ = preferences->GetReal(prefs::kAutoFillNegativeUploadRate); } } AutoFillDownloadManager::~AutoFillDownloadManager() { STLDeleteContainerPairFirstPointers(url_fetchers_.begin(), url_fetchers_.end()); } void AutoFillDownloadManager::SetObserver( AutoFillDownloadManager::Observer *observer) { if (observer) { DCHECK(!observer_); observer_ = observer; } else { observer_ = NULL; } } bool AutoFillDownloadManager::StartQueryRequest( const ScopedVector& forms) { if (next_query_request_ > base::Time::Now()) { // We are in back-off mode: do not do the request. return false; } std::string form_xml; FormStructure::EncodeQueryRequest(forms, &form_xml); FormRequestData request_data; request_data.form_signatures.reserve(forms.size()); for (ScopedVector::const_iterator it = forms.begin(); it != forms.end(); ++it) { request_data.form_signatures.push_back((*it)->FormSignature()); } request_data.request_type = AutoFillDownloadManager::REQUEST_QUERY; return StartRequest(form_xml, request_data); } bool AutoFillDownloadManager::StartUploadRequest( const FormStructure& form, bool form_was_matched) { if (next_upload_request_ > base::Time::Now()) { // We are in back-off mode: do not do the request. return false; } // Check if we need to upload form. double upload_rate = form_was_matched ? GetPositiveUploadRate() : GetNegativeUploadRate(); if (base::RandDouble() > upload_rate) { LOG(INFO) << "AutoFillDownloadManager: Upload request is ignored"; // If we ever need notification that upload was skipped, add it here. return false; } std::string form_xml; form.EncodeUploadRequest(form_was_matched, &form_xml); FormRequestData request_data; request_data.form_signatures.push_back(form.FormSignature()); request_data.request_type = AutoFillDownloadManager::REQUEST_UPLOAD; return StartRequest(form_xml, request_data); } bool AutoFillDownloadManager::CancelRequest( const std::string& form_signature, AutoFillDownloadManager::AutoFillRequestType request_type) { for (std::map::iterator it = url_fetchers_.begin(); it != url_fetchers_.end(); ++it) { if (std::find(it->second.form_signatures.begin(), it->second.form_signatures.end(), form_signature) != it->second.form_signatures.end() && it->second.request_type == request_type) { delete it->first; url_fetchers_.erase(it); return true; } } return false; } double AutoFillDownloadManager::GetPositiveUploadRate() const { return positive_upload_rate_; } double AutoFillDownloadManager::GetNegativeUploadRate() const { return negative_upload_rate_; } void AutoFillDownloadManager::SetPositiveUploadRate(double rate) { if (rate == positive_upload_rate_) return; positive_upload_rate_ = rate; DCHECK_GE(rate, 0.0); DCHECK_LE(rate, 1.0); DCHECK(profile_); PrefService* preferences = profile_->GetPrefs(); preferences->SetReal(prefs::kAutoFillPositiveUploadRate, rate); } void AutoFillDownloadManager::SetNegativeUploadRate(double rate) { if (rate == negative_upload_rate_) return; negative_upload_rate_ = rate; DCHECK_GE(rate, 0.0); DCHECK_LE(rate, 1.0); DCHECK(profile_); PrefService* preferences = profile_->GetPrefs(); preferences->SetReal(prefs::kAutoFillNegativeUploadRate, rate); } bool AutoFillDownloadManager::StartRequest( const std::string& form_xml, const FormRequestData& request_data) { std::string request_url; if (request_data.request_type == AutoFillDownloadManager::REQUEST_QUERY) request_url = AUTO_FILL_QUERY_SERVER_REQUEST_URL; else request_url = AUTO_FILL_UPLOAD_SERVER_REQUEST_URL; if (!request_url.compare(DISABLED_REQUEST_URL) && !is_testing_) { // We have it disabled - return true as if it succeeded, but do nothing. return true; } // Id is ignored for regular chrome, in unit test id's for fake fetcher // factory will be 0, 1, 2, ... URLFetcher *fetcher = URLFetcher::Create(fetcher_id_for_unittest_++, GURL(request_url), URLFetcher::POST, this); url_fetchers_[fetcher] = request_data; fetcher->set_automatcally_retry_on_5xx(false); fetcher->set_request_context(Profile::GetDefaultRequestContext()); fetcher->set_upload_data("text/plain", form_xml); fetcher->Start(); return true; } void AutoFillDownloadManager::OnURLFetchComplete(const URLFetcher* source, const GURL& url, const URLRequestStatus& status, int response_code, const ResponseCookies& cookies, const std::string& data) { std::map::iterator it = url_fetchers_.find(const_cast(source)); if (it == url_fetchers_.end()) { // Looks like crash on Mac is possibly caused with callback entering here // with unknown fetcher when network is refreshed. return; } std::string type_of_request( it->second.request_type == AutoFillDownloadManager::REQUEST_QUERY ? "query" : "upload"); const int kHttpResponseOk = 200; const int kHttpInternalServerError = 500; const int kHttpBadGateway = 502; const int kHttpServiceUnavailable = 503; CHECK(it->second.form_signatures.size()); if (response_code != kHttpResponseOk) { bool back_off = false; std::string server_header; switch (response_code) { case kHttpBadGateway: if (!source->response_headers()->EnumerateHeader(NULL, "server", &server_header) || StartsWithASCII(server_header.c_str(), AUTO_FILL_QUERY_SERVER_NAME_START_IN_HEADER, false) != 0) break; // Bad getaway was received from AutoFill servers. Fall through to back // off. case kHttpInternalServerError: case kHttpServiceUnavailable: back_off = true; break; } if (back_off) { base::Time back_off_time(base::Time::Now() + source->backoff_delay()); if (it->second.request_type == AutoFillDownloadManager::REQUEST_QUERY) { next_query_request_ = back_off_time; } else { next_upload_request_ = back_off_time; } } LOG(INFO) << "AutoFillDownloadManager: " << type_of_request << " request has failed with response" << response_code; if (observer_) { observer_->OnHeuristicsRequestError(it->second.form_signatures[0], it->second.request_type, response_code); } } else { LOG(INFO) << "AutoFillDownloadManager: " << type_of_request << " request has succeeded"; if (it->second.request_type == AutoFillDownloadManager::REQUEST_QUERY) { if (observer_) observer_->OnLoadedAutoFillHeuristics(it->second.form_signatures, data); } else { double new_positive_upload_rate = 0; double new_negative_upload_rate = 0; AutoFillUploadXmlParser parse_handler(&new_positive_upload_rate, &new_negative_upload_rate); buzz::XmlParser parser(&parse_handler); parser.Parse(data.data(), data.length(), true); if (parse_handler.succeeded()) { SetPositiveUploadRate(new_positive_upload_rate); SetNegativeUploadRate(new_negative_upload_rate); } if (observer_) observer_->OnUploadedAutoFillHeuristics(it->second.form_signatures[0]); } } delete it->first; url_fetchers_.erase(it); }