Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(8509)

Unified Diff: components/password_manager/core/browser/affiliation_fetcher.cc

Issue 767163005: Add AffiliationFetcher to fetch authoritative affiliation information regarding facets. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Addressed nits from gcasto@ and rdsmith@. Created 6 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: components/password_manager/core/browser/affiliation_fetcher.cc
diff --git a/components/password_manager/core/browser/affiliation_fetcher.cc b/components/password_manager/core/browser/affiliation_fetcher.cc
new file mode 100644
index 0000000000000000000000000000000000000000..96c999e9c2591e59aa9f98e2b0abbb9d0ba03f19
--- /dev/null
+++ b/components/password_manager/core/browser/affiliation_fetcher.cc
@@ -0,0 +1,153 @@
+// Copyright 2014 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/password_manager/core/browser/affiliation_fetcher.h"
+
+#include "base/json/json_reader.h"
+#include "base/json/json_writer.h"
+#include "base/values.h"
+#include "components/password_manager/core/browser/affiliation_api.pb.h"
+#include "components/password_manager/core/browser/affiliation_utils.h"
+#include "google_apis/google_api_keys.h"
+#include "net/base/load_flags.h"
+#include "net/base/url_util.h"
+#include "net/http/http_status_code.h"
+#include "net/url_request/url_fetcher.h"
+#include "net/url_request/url_request_context_getter.h"
+#include "url/gurl.h"
+
+namespace password_manager {
+
+AffiliationFetcher::AffiliationFetcher(
+ net::URLRequestContextGetter* request_context_getter,
+ const std::vector<FacetURI>& facet_uris,
+ AffiliationFetcherDelegate* delegate)
+ : request_context_getter_(request_context_getter),
+ requested_facet_uris_(facet_uris),
+ delegate_(delegate) {
+ for (const FacetURI& uri : requested_facet_uris_)
+ DCHECK(uri.is_valid());
Ryan Sleevi 2014/12/12 23:02:19 BUG/DANGER: Whenever using debugging macros and co
engedy 2014/12/15 13:54:51 Done.
+}
+
+AffiliationFetcher::~AffiliationFetcher() {
+}
+
+// static
+AffiliationFetcher* AffiliationFetcher::Create(
+ net::URLRequestContextGetter* context_getter,
+ const std::vector<FacetURI>& facet_uris,
+ AffiliationFetcherDelegate* delegate) {
+ return new AffiliationFetcher(context_getter, facet_uris, delegate);
+}
+
+void AffiliationFetcher::StartRequest() {
+ DCHECK(!fetcher_);
+
+ fetcher_.reset(
+ net::URLFetcher::Create(BuildQueryURL(), net::URLFetcher::POST, this));
+ fetcher_->SetRequestContext(request_context_getter_.get());
+ fetcher_->SetUploadData("application/x-protobuf", PreparePayload());
+ fetcher_->SetLoadFlags(
+ net::LOAD_DO_NOT_SAVE_COOKIES | net::LOAD_DO_NOT_SEND_COOKIES |
+ net::LOAD_DO_NOT_SEND_AUTH_DATA | net::LOAD_BYPASS_CACHE |
+ net::LOAD_DISABLE_CACHE | net::LOAD_DO_NOT_PROMPT_FOR_LOGIN);
+ fetcher_->SetAutomaticallyRetryOn5xx(false);
+ fetcher_->SetAutomaticallyRetryOnNetworkChanges(0);
+ fetcher_->Start();
+}
+
+GURL AffiliationFetcher::BuildQueryURL() const {
+ return net::AppendQueryParameter(
+ GURL("https://www.googleapis.com/affiliation/v1/affiliation:lookup"),
+ "key", google_apis::GetAPIKey());
+}
+
+std::string AffiliationFetcher::PreparePayload() const {
+ affiliation_pb::LookupAffiliationRequest lookup_request;
+ for (const FacetURI& uri : requested_facet_uris_)
+ lookup_request.add_facet(uri.canonical_spec());
+
+ std::string serialized_request;
+ bool success = lookup_request.SerializeToString(&serialized_request);
+ DCHECK(success);
+ return serialized_request;
+}
+
+bool AffiliationFetcher::ParseResponse(
Ryan Sleevi 2014/12/12 23:02:19 DOCUMENTATION: I think it'd be helpful for future
engedy 2014/12/15 13:54:51 Done.
+ AffiliationFetcherDelegate::Result* result) const {
+ std::string serialized_response;
+ if (!fetcher_->GetResponseAsString(&serialized_response))
+ NOTREACHED();
+
+ affiliation_pb::LookupAffiliationResponse response;
+ if (!response.ParseFromString(serialized_response))
+ return false;
+
+ result->reserve(requested_facet_uris_.size());
+
+ std::map<FacetURI, size_t> facet_uri_to_class_index;
+ for (int i = 0; i < response.affiliation_size(); ++i) {
+ const affiliation_pb::Affiliation& equivalence_class(
+ response.affiliation(i));
+
+ AffiliatedFacets affiliated_uris;
+ for (int j = 0; j < equivalence_class.facet_size(); ++j) {
+ const std::string uri_spec(equivalence_class.facet(j));
Ryan Sleevi 2014/12/12 23:02:19 Should this be "const std::string&" ? I didn't di
engedy 2014/12/15 13:54:51 Argh, of course. Done.
+ FacetURI uri = FacetURI::FromPotentiallyInvalidSpec(uri_spec);
+ // Ignore potential future kinds of facet URIs (e.g. for new platforms).
+ if (!uri.is_valid())
+ continue;
+ affiliated_uris.push_back(uri);
+ }
+
+ // Be lenient and ignore empty (after filtering) equivalence classes.
+ if (affiliated_uris.empty())
+ continue;
+
+ // Ignore equivalence classes that are duplicates of earlier ones. However,
+ // fail in the case of a partial overlap, which violates the invariant that
+ // affiliations must form an equivalence relation.
+ for (const FacetURI& uri : affiliated_uris) {
+ if (!facet_uri_to_class_index.count(uri))
+ facet_uri_to_class_index[uri] = result->size();
+ if (facet_uri_to_class_index[uri] !=
+ facet_uri_to_class_index[affiliated_uris[0]]) {
+ return false;
+ }
+ }
+
+ // Filter out duplicate equivalence classes in the response.
+ if (facet_uri_to_class_index[affiliated_uris[0]] == result->size())
+ result->push_back(affiliated_uris);
+ }
+
+ // The API contract of this class is to return an equivalence class for all
+ // requested facets; however, the server will not return anything for facets
+ // that are not affiliated with any other facet. Synthesize an equivalence
+ // class (of size one) for each of the missing facets.
+ for (const FacetURI& uri : requested_facet_uris_) {
+ if (!facet_uri_to_class_index.count(uri))
+ result->push_back(AffiliatedFacets(1, uri));
+ }
+
+ return true;
+}
+
+void AffiliationFetcher::OnURLFetchComplete(const net::URLFetcher* source) {
+ DCHECK_EQ(source, fetcher_.get());
+
+ scoped_ptr<AffiliationFetcherDelegate::Result> result(
+ new AffiliationFetcherDelegate::Result);
+ if (fetcher_->GetStatus().status() == net::URLRequestStatus::SUCCESS &&
+ fetcher_->GetResponseCode() == net::HTTP_OK) {
+ if (ParseResponse(result.get()))
+ delegate_->OnFetchSucceeded(result.Pass());
+ else
+ delegate_->OnMalformedResponse();
+ } else {
+ delegate_->OnFetchFailed();
+ }
+}
+
+} // namespace password_manager

Powered by Google App Engine
This is Rietveld 408576698