blob: af0336c8bdd8f92bbe26e82342720298741336f9 [file] [log] [blame]
// Copyright 2012 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_EXTENSIONS_API_IDENTITY_WEB_AUTH_FLOW_H_
#define CHROME_BROWSER_EXTENSIONS_API_IDENTITY_WEB_AUTH_FLOW_H_
#include <optional>
#include <string>
#include "base/feature_list.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/weak_ptr.h"
#include "base/scoped_observation.h"
#include "base/time/time.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/profiles/profile_observer.h"
#include "content/public/browser/web_contents_observer.h"
#include "extensions/buildflags/buildflags.h"
#include "ui/gfx/geometry/rect.h"
#include "url/gurl.h"
static_assert(BUILDFLAG(ENABLE_EXTENSIONS_CORE));
class Profile;
namespace base {
class OneShotTimer;
class TickClock;
} // namespace base
namespace extensions {
class WebAuthFlowInfoBarDelegate;
// Controller class for web based auth flows. The WebAuthFlow creates
// a browser popup window (or a new tab based on the feature setting)
// with a webview that will navigate to the `provider_url` passed to the
// WebAuthFlow constructor.
//
// The WebAuthFlow monitors the WebContents of the webview, and
// notifies its delegate interface any time the WebContents navigates
// to a new URL or changes title. The delegate is expected to delete
// the flow when navigation reaches a known target location.
//
// The window is not displayed until the first page load
// completes. This allows the flow to complete without flashing a
// window on screen if the provider immediately redirects to the
// target URL.
//
// A WebAuthFlow can be started in Mode::SILENT, which never displays
// a window. If a window would be required, the flow fails.
class WebAuthFlow : public content::WebContentsObserver,
public ProfileObserver {
public:
enum Mode {
INTERACTIVE, // Show UI to the user if necessary.
SILENT // No UI should be shown.
};
enum Failure {
WINDOW_CLOSED, // Window closed by user (app or tab).
INTERACTION_REQUIRED, // Non-redirect page load in silent mode.
LOAD_FAILED,
TIMED_OUT,
CANNOT_CREATE_WINDOW, // Couldn't create a browser window.
};
enum class AbortOnLoad {
kYes,
kNo,
};
// Maximum time on the total `WebAuthFlow` execution in silent node. This is
// the default if timeout is not specified.
static constexpr base::TimeDelta kNonInteractiveMaxTimeout = base::Minutes(1);
class Delegate {
public:
// Called when the auth flow fails. This means that the flow did not result
// in a successful redirect to a valid redirect URL.
virtual void OnAuthFlowFailure(Failure failure) = 0;
// Called on redirects and other navigations to see if the URL should stop
// the flow.
virtual void OnAuthFlowURLChange(const GURL& redirect_url) {}
// Called when the title of the current page changes.
virtual void OnAuthFlowTitleChange(const std::string& title) {}
// Called when the web_contents associated with the flow has finished
// navigation.
virtual void OnNavigationFinished(
content::NavigationHandle* navigation_handle) {}
protected:
virtual ~Delegate() = default;
};
// Creates an instance with the given parameters.
// Caller owns `delegate`.
WebAuthFlow(
Delegate* delegate,
Profile* profile,
const GURL& provider_url,
Mode mode,
bool user_gesture,
AbortOnLoad abort_on_load_for_non_interactive = AbortOnLoad::kYes,
std::optional<base::TimeDelta> timeout_for_non_interactive = std::nullopt,
std::optional<gfx::Rect> popup_bounds = std::nullopt);
WebAuthFlow(const WebAuthFlow&) = delete;
WebAuthFlow& operator=(const WebAuthFlow&) = delete;
~WebAuthFlow() override;
// Testing clock used to test the effect of load timeout.
void SetClockForTesting(const base::TickClock* clock,
scoped_refptr<base::SequencedTaskRunner> task_runner);
// Starts the flow.
virtual void Start();
// Prevents further calls to the delegate and deletes the flow.
void DetachDelegateAndDelete();
// This call will make the interactive mode, that opens up a browser tab for
// auth, display an Infobar that shows the extension name.
void SetShouldShowInfoBar(const std::string& extension_display_name);
// Returns nullptr if the InfoBar is not displayed.
base::WeakPtr<WebAuthFlowInfoBarDelegate> GetInfoBarDelegateForTesting();
private:
// WebContentsObserver implementation.
void DidStopLoading() override;
void WebContentsDestroyed() override;
void TitleWasSet(content::NavigationEntry* entry) override;
void DidStartNavigation(
content::NavigationHandle* navigation_handle) override;
void DidRedirectNavigation(
content::NavigationHandle* navigation_handle) override;
void DidFinishNavigation(
content::NavigationHandle* navigation_handle) override;
// ProfileObserver
void OnProfileWillBeDestroyed(Profile* profile) override;
void BeforeUrlLoaded(const GURL& url);
void AfterUrlLoaded();
void MaybeStartTimeout();
void OnTimeout();
// Displays the auth page in a popup window if that is possible.
//
// Returns true if the auth page is displayed and false otherwise (e.g.
// popup is disabled, API is called from incognito).
// TODO(crbug.com/434156398): Android desktop implementation temporarily
// returns false. Update the implementation to display the auth page in a
// popup or a tab and return true.
bool DisplayAuthPageInPopupWindow();
void DisplayInfoBar();
void CloseInfoBar();
raw_ptr<Delegate> delegate_ = nullptr;
raw_ptr<Profile> profile_;
const GURL provider_url_;
const Mode mode_;
#if BUILDFLAG(ENABLE_EXTENSIONS)
const bool user_gesture_;
#endif
// WebContents used to initialize the authentication. It is not displayed
// and not owned by browser window. This WebContents is observed by
// `this`. When this value becomes nullptr, this means that the browser tab
// has taken ownership and the interactive tab was opened.
std::unique_ptr<content::WebContents> web_contents_;
// Internal struct to manage infobar parameters, external calls can only set
// the extension display name which will force show the info bar through
// `SetShouldShowInfoBar()`.
struct InfoBarParameters {
bool should_show = false;
std::string extension_display_name;
};
InfoBarParameters info_bar_parameters_;
// WeakPtr to the info bar delegate attached to the auth tab when opened. Used
// to close the info bar when closing the flow if still valid.
base::WeakPtr<WebAuthFlowInfoBarDelegate> info_bar_delegate_ = nullptr;
const AbortOnLoad abort_on_load_for_non_interactive_;
const std::optional<base::TimeDelta> timeout_for_non_interactive_;
std::unique_ptr<base::OneShotTimer> non_interactive_timeout_timer_;
const std::optional<gfx::Rect> popup_bounds_;
// Flag indicating that the initial URL was successfully loaded. Influences
// the error code when the flow times out.
bool initial_url_loaded_ = false;
base::ScopedObservation<Profile, ProfileObserver> profile_observation_{this};
};
} // namespace extensions
#endif // CHROME_BROWSER_EXTENSIONS_API_IDENTITY_WEB_AUTH_FLOW_H_