| // Copyright 2021 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef CHROME_BROWSER_PASSWORD_MANAGER_CHROME_WEBAUTHN_CREDENTIALS_DELEGATE_H_ |
| #define CHROME_BROWSER_PASSWORD_MANAGER_CHROME_WEBAUTHN_CREDENTIALS_DELEGATE_H_ |
| |
| #include <optional> |
| |
| #include "base/functional/callback.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/weak_ptr.h" |
| #include "base/scoped_observation.h" |
| #include "base/timer/timer.h" |
| #include "base/types/expected.h" |
| #include "base/types/strong_alias.h" |
| #include "build/build_config.h" |
| #include "components/password_manager/core/browser/passkey_credential.h" |
| #include "components/password_manager/core/browser/webauthn_credentials_delegate.h" |
| |
| #if !BUILDFLAG(IS_ANDROID) |
| #include "chrome/browser/webauthn/authenticator_request_dialog_model.h" |
| #endif // !BUILDFLAG(IS_ANDROID) |
| |
| namespace base { |
| class ElapsedTimer; |
| } |
| |
| namespace content { |
| class WebContents; |
| } |
| |
| // Chrome implementation of WebAuthnCredentialsDelegate. |
| class ChromeWebAuthnCredentialsDelegate final : |
| #if !BUILDFLAG(IS_ANDROID) |
| public AuthenticatorRequestDialogModel::Observer, |
| #endif //! BUILDFLAG(IS_ANDROID) |
| public password_manager::WebAuthnCredentialsDelegate { |
| public: |
| using SecurityKeyOrHybridFlowAvailable = |
| base::StrongAlias<struct SecurityKeyOrHybridFlowAvailableTag, bool>; |
| |
| explicit ChromeWebAuthnCredentialsDelegate( |
| content::WebContents* web_contents); |
| ~ChromeWebAuthnCredentialsDelegate() override; |
| ChromeWebAuthnCredentialsDelegate(const ChromeWebAuthnCredentialsDelegate&) = |
| delete; |
| ChromeWebAuthnCredentialsDelegate operator=( |
| const ChromeWebAuthnCredentialsDelegate&) = delete; |
| |
| // password_manager::WebAuthnCredentialsDelegate: |
| void LaunchSecurityKeyOrHybridFlow() override; |
| void SelectPasskey( |
| const std::string& backend_id, |
| password_manager::WebAuthnCredentialsDelegate::OnPasskeySelectedCallback |
| callback) override; |
| base::expected<const std::vector<password_manager::PasskeyCredential>*, |
| PasskeysUnavailableReason> |
| GetPasskeys() const override; |
| void NotifyForPasskeysDisplay() override; |
| bool IsSecurityKeyOrHybridFlowAvailable() const override; |
| void RequestNotificationWhenPasskeysReady( |
| base::OnceClosure callback) override; |
| bool HasPendingPasskeySelection() override; |
| base::WeakPtr<WebAuthnCredentialsDelegate> AsWeakPtr() override; |
| |
| #if !BUILDFLAG(IS_ANDROID) |
| // AuthenticatorRequestDialogModel::Observer: |
| void OnStepTransition() override; |
| #endif // !BUILDFLAG(IS_ANDROID) |
| |
| // Method for providing a list of WebAuthn user entities that can be provided |
| // as autofill suggestions. This is called when a WebAuthn Conditional UI |
| // request is received. The `security_key_or_hybrid_flow_available` |
| // argument determines whether an autofill option to use a passkey from |
| // another device should be offered. |
| void OnCredentialsReceived( |
| std::vector<password_manager::PasskeyCredential> credentials, |
| SecurityKeyOrHybridFlowAvailable security_key_or_hybrid_flow_available); |
| |
| // Lets the delegate know that a WebAuthn request has been aborted, and so |
| // WebAuthn options should no longer show up on the autofill popup. |
| void NotifyWebAuthnRequestAborted(); |
| |
| protected: |
| const raw_ptr<content::WebContents> web_contents_; |
| |
| private: |
| void RecordPasskeyRetrievalDelay(); |
| void NotifyClientsOfPasskeyAvailability(); |
| |
| // List of passkeys populated from authenticators. It is returned to the |
| // client via GetPasskeys. |passkeys_| is nullopt until populated by a |
| // WebAuthn request. |
| std::optional<std::vector<password_manager::PasskeyCredential>> passkeys_; |
| |
| // TODO(crbug.com/368283817): Check if this is required. While |
| // SecurityKeyOrHybridFlowAvailable flows are always available on desktop |
| // platforms, they still require a conditional request from the RP. |
| SecurityKeyOrHybridFlowAvailable security_key_or_hybrid_flow_available_ = |
| #if !BUILDFLAG(IS_ANDROID) |
| SecurityKeyOrHybridFlowAvailable(true); |
| #else |
| SecurityKeyOrHybridFlowAvailable(false); |
| #endif // !BUILDFLAG(IS_ANDROID) |
| |
| std::vector<base::OnceClosure> passkeys_available_callbacks_; |
| std::unique_ptr<base::ElapsedTimer> passkey_retrieval_timer_; |
| |
| #if !BUILDFLAG(IS_ANDROID) |
| // Callback to be run to dismiss the autofill popup. The popup will be shown |
| // while the observed model displays no UI or until the request is completed. |
| OnPasskeySelectedCallback passkey_selected_callback_; |
| base::ScopedObservation<AuthenticatorRequestDialogModel, |
| AuthenticatorRequestDialogModel::Observer> |
| authenticator_observation_{this}; |
| base::OneShotTimer flickering_timer_; |
| #endif // !BUILDFLAG(IS_ANDROID) |
| |
| // Set to true when an autofill surface that could have contained passkeys |
| // has been displayed for the current page. Used for the |
| // PasskeysArrivedAfterAutofillDisplay metric. |
| bool passkey_display_has_happened_ = false; |
| |
| // Set to true when the PasskeysArrivedAfterAutofillDisplay metric has been |
| // recorded. |
| bool passkeys_after_fill_recorded_ = false; |
| |
| // Set to true when the timer for the PasskeyRetrievalWaitDuration metric has |
| // been started, since we only want to use it once. |
| bool passkey_retrieval_timer_started_ = false; |
| |
| // Set when `NotifyWebAuthnRequestAborted` has been called. It is reset if |
| // passkeys are provided again after that, indicating that an additional |
| // request has been made. |
| bool last_request_was_aborted_ = false; |
| |
| base::WeakPtrFactory<ChromeWebAuthnCredentialsDelegate> weak_ptr_factory_{ |
| this}; |
| }; |
| |
| #endif // CHROME_BROWSER_PASSWORD_MANAGER_CHROME_WEBAUTHN_CREDENTIALS_DELEGATE_H_ |