| // 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/content/browser/content_credential_manager_dispatcher.h" |
| |
| #include "base/strings/string16.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "components/autofill/core/common/password_form.h" |
| #include "components/password_manager/content/browser/credential_manager_password_form_manager.h" |
| #include "components/password_manager/content/common/credential_manager_messages.h" |
| #include "components/password_manager/content/common/credential_manager_types.h" |
| #include "components/password_manager/core/browser/password_manager_client.h" |
| #include "components/password_manager/core/browser/password_store.h" |
| #include "content/public/browser/render_view_host.h" |
| #include "content/public/browser/web_contents.h" |
| #include "ipc/ipc_message_macros.h" |
| |
| namespace password_manager { |
| |
| ContentCredentialManagerDispatcher::ContentCredentialManagerDispatcher( |
| content::WebContents* web_contents, |
| PasswordManagerClient* client) |
| : WebContentsObserver(web_contents), |
| client_(client), |
| pending_request_id_(0) { |
| DCHECK(web_contents); |
| } |
| |
| ContentCredentialManagerDispatcher::~ContentCredentialManagerDispatcher() {} |
| |
| bool ContentCredentialManagerDispatcher::OnMessageReceived( |
| const IPC::Message& message) { |
| bool handled = true; |
| IPC_BEGIN_MESSAGE_MAP(ContentCredentialManagerDispatcher, message) |
| IPC_MESSAGE_HANDLER(CredentialManagerHostMsg_NotifyFailedSignIn, |
| OnNotifyFailedSignIn); |
| IPC_MESSAGE_HANDLER(CredentialManagerHostMsg_NotifySignedIn, |
| OnNotifySignedIn); |
| IPC_MESSAGE_HANDLER(CredentialManagerHostMsg_NotifySignedOut, |
| OnNotifySignedOut); |
| IPC_MESSAGE_HANDLER(CredentialManagerHostMsg_RequestCredential, |
| OnRequestCredential); |
| IPC_MESSAGE_UNHANDLED(handled = false) |
| IPC_END_MESSAGE_MAP() |
| return handled; |
| } |
| |
| void ContentCredentialManagerDispatcher::OnNotifyFailedSignIn( |
| int request_id, const CredentialInfo&) { |
| DCHECK(request_id); |
| // TODO(mkwst): This is a stub. |
| web_contents()->GetRenderViewHost()->Send( |
| new CredentialManagerMsg_AcknowledgeFailedSignIn( |
| web_contents()->GetRenderViewHost()->GetRoutingID(), request_id)); |
| } |
| |
| void ContentCredentialManagerDispatcher::OnNotifySignedIn( |
| int request_id, |
| const password_manager::CredentialInfo& credential) { |
| DCHECK(request_id); |
| scoped_ptr<autofill::PasswordForm> form( |
| CreatePasswordFormFromCredentialInfo(credential, |
| web_contents()->GetLastCommittedURL().GetOrigin())); |
| |
| // TODO(mkwst): This is a stub; we should be checking the PasswordStore to |
| // determine whether or not the credential exists, and calling UpdateLogin |
| // accordingly. |
| form_manager_.reset( |
| new CredentialManagerPasswordFormManager(client_, *form, this)); |
| |
| web_contents()->GetRenderViewHost()->Send( |
| new CredentialManagerMsg_AcknowledgeSignedIn( |
| web_contents()->GetRenderViewHost()->GetRoutingID(), request_id)); |
| } |
| |
| void ContentCredentialManagerDispatcher::OnProvisionalSaveComplete() { |
| DCHECK(form_manager_); |
| client_->PromptUserToSavePassword(form_manager_.Pass()); |
| } |
| |
| void ContentCredentialManagerDispatcher::OnNotifySignedOut(int request_id) { |
| DCHECK(request_id); |
| // TODO(mkwst): This is a stub. |
| web_contents()->GetRenderViewHost()->Send( |
| new CredentialManagerMsg_AcknowledgeSignedOut( |
| web_contents()->GetRenderViewHost()->GetRoutingID(), request_id)); |
| } |
| |
| void ContentCredentialManagerDispatcher::OnRequestCredential( |
| int request_id, |
| bool /* zero_click_only */, |
| const std::vector<GURL>& federations) { |
| DCHECK(request_id); |
| PasswordStore* store = GetPasswordStore(); |
| if (pending_request_id_ || !store) { |
| web_contents()->GetRenderViewHost()->Send( |
| new CredentialManagerMsg_RejectCredentialRequest( |
| web_contents()->GetRenderViewHost()->GetRoutingID(), |
| request_id, |
| pending_request_id_ |
| ? blink::WebCredentialManagerError::ErrorTypePendingRequest |
| : blink::WebCredentialManagerError:: |
| ErrorTypePasswordStoreUnavailable)); |
| return; |
| } |
| |
| pending_request_id_ = request_id; |
| |
| autofill::PasswordForm form; |
| form.scheme = autofill::PasswordForm::SCHEME_HTML; |
| form.origin = web_contents()->GetLastCommittedURL().GetOrigin(); |
| form.signon_realm = form.origin.spec(); |
| |
| store->GetLogins(form, PasswordStore::DISALLOW_PROMPT, this); |
| } |
| |
| void ContentCredentialManagerDispatcher::OnGetPasswordStoreResults( |
| const std::vector<autofill::PasswordForm*>& results) { |
| DCHECK(pending_request_id_); |
| |
| // Take ownership of all the password form objects in the |results| vector. |
| ScopedVector<autofill::PasswordForm> entries; |
| entries.assign(results.begin(), results.end()); |
| |
| // TODO(mkwst): This is a stub. We're just grabbing the first result and |
| // piping it down into Blink. Really, we should be kicking off some sort |
| // of UI full of magic moments and delight. Also, we should deal with |
| // federated login types. |
| CredentialInfo info = results.empty() ? CredentialInfo() |
| : CredentialInfo(*entries[0]); |
| web_contents()->GetRenderViewHost()->Send( |
| new CredentialManagerMsg_SendCredential( |
| web_contents()->GetRenderViewHost()->GetRoutingID(), |
| pending_request_id_, |
| info)); |
| pending_request_id_ = 0; |
| } |
| |
| PasswordStore* ContentCredentialManagerDispatcher::GetPasswordStore() { |
| return client_ ? client_->GetPasswordStore() : nullptr; |
| } |
| |
| } // namespace password_manager |