blob: 92901b0d61005d3f5d7cb659db7703c59f3ac48f [file] [log] [blame]
// Copyright (c) 2012 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 "base/callback_forward.h"
#include "base/compiler_specific.h"
#include "base/macros.h"
#include "base/memory/weak_ptr.h"
#include "base/time/time.h"
#include "base/timer/timer.h"
#include "chrome/browser/captive_portal/captive_portal_service.h"
class Profile;
namespace content {
class WebContents;
namespace net {
class SSLInfo;
// Keeps track of whether a tab has encountered a navigation error caused by a
// captive portal. Also triggers captive portal checks when a page load may
// have been broken or be taking longer due to a captive portal. All methods
// may only be called on the UI thread.
// Only supports SSL main frames which end at error pages as a result of
// captive portals, since these make for a particularly bad user experience.
// Non-SSL requests are intercepted by captive portals, which take users to the
// login page. SSL requests, however, may be silently blackholed, or result
// in a variety of error pages, and will continue to do so if a user tries to
// reload them.
class CaptivePortalTabReloader {
enum State {
// The slow load timer is running. Only started on SSL provisional loads.
// If the timer triggers before the page has been committed, a captive
// portal test will be requested.
// The tab may have been broken by a captive portal. A tab switches to
// this state either on a main frame SSL error that may be caused by a
// captive portal, or when an SSL request takes too long to commit. The
// tab will remain in this state until the current load succeeds, a new
// provisional load starts, it gets a captive portal result, or the load
// fails with error that indicates the page was not broken by a captive
// portal.
// The TabHelper switches to this state from STATE_MAYBE_BROKEN_BY_PORTAL in
// response to a RESULT_BEHIND_CAPTIVE_PORTAL. The tab will remain in this
// state until a new provisional load starts, the original load successfully
// commits, the current load is aborted, or the tab reloads the page in
// response to receiving a captive portal result other than
// The page may need to be reloaded. The tab will be reloaded if the page
// fails the next load with a timeout, or immediately upon switching to this
// state, if the page already timed out. If anything else happens
// when in this state (Another error, successful navigation, or the original
// navigation was aborted), the TabHelper transitions to STATE_NONE without
// reloading.
// Function to open a login tab, if there isn't one already.
typedef base::Callback<void()> OpenLoginTabCallback;
// |profile| and |web_contents| will only be dereferenced in ReloadTab,
// MaybeOpenCaptivePortalLoginTab, and CheckForCaptivePortal, so they can
// both be NULL in the unit tests as long as those functions are not called.
CaptivePortalTabReloader(Profile* profile,
content::WebContents* web_contents,
const OpenLoginTabCallback& open_login_tab_callback);
virtual ~CaptivePortalTabReloader();
// The following functions are all invoked by the CaptivePortalTabHelper:
// Called when a non-error main frame load starts. Resets current state,
// unless this is a login tab. Each load will eventually result in a call to
// OnLoadCommitted or OnAbort. The former will be called both on successful
// loads and for error pages.
virtual void OnLoadStart(bool is_ssl);
// Called when the main frame is committed. |net_error| will be net::OK in
// the case of a successful load. For an errror page, the entire 3-step
// process of getting the error, starting a new provisional load for the error
// page, and committing the error page is treated as a single commit.
// The Link Doctor page will typically be one OnLoadCommitted with an error
// code, followed by another OnLoadCommitted with net::OK for the Link Doctor
// page.
virtual void OnLoadCommitted(int net_error);
// This is called when the current provisional main frame load is canceled.
// Sets state to STATE_NONE, unless this is a login tab.
virtual void OnAbort();
// Called whenever a provisional load to the main frame is redirected.
virtual void OnRedirect(bool is_ssl);
// Called whenever a captive portal test completes.
virtual void OnCaptivePortalResults(
captive_portal::CaptivePortalResult previous_result,
captive_portal::CaptivePortalResult result);
// Called on certificate errors, which often indicate a captive portal.
void OnSSLCertError(const net::SSLInfo& ssl_info);
// The following functions are used only when testing:
State state() const { return state_; }
content::WebContents* web_contents() { return web_contents_; }
void set_slow_ssl_load_time(base::TimeDelta slow_ssl_load_time) {
slow_ssl_load_time_ = slow_ssl_load_time;
// Started whenever an SSL tab starts loading, when the state is switched to
// STATE_TIMER_RUNNING. Stopped on any state change, including when a page
// commits or there's an error. If the timer triggers, the state switches to
// STATE_MAYBE_BROKEN_BY_PORTAL and |this| kicks off a captive portal check.
base::OneShotTimer slow_ssl_load_timer_;
friend class CaptivePortalBrowserTest;
// Sets |state_| and takes any action associated with the new state. Also
// stops the timer, if needed.
void SetState(State new_state);
// Called by a timer when an SSL main frame provisional load is taking a
// while to commit.
void OnSlowSSLConnect();
// Reloads the tab if there's no provisional load going on and the current
// state is STATE_NEEDS_RELOAD. Not safe to call synchronously when called
// by from a WebContentsObserver function, since the WebContents is currently
// performing some action.
void ReloadTabIfNeeded();
// Reloads the tab.
virtual void ReloadTab();
// Opens a login tab in the topmost browser window for the |profile_|, if the
// profile has a tabbed browser window and the window doesn't already have a
// login tab. Otherwise, does nothing.
virtual void MaybeOpenCaptivePortalLoginTab();
// Tries to get |profile_|'s CaptivePortalService and have it start a captive
// portal check.
virtual void CheckForCaptivePortal();
Profile* profile_;
content::WebContents* web_contents_;
State state_;
// Tracks if there's a load going on that can't safely be interrupted. This
// is true between the time when a provisional load fails and when an error
// page's provisional load starts, so does not perfectly align with the
// notion of a provisional load used by the WebContents.
bool provisional_main_frame_load_;
// True if there was an SSL URL the in the redirect chain for the current
// provisional main frame load.
bool ssl_url_in_redirect_chain_;
// Time to wait after a provisional HTTPS load before triggering a captive
// portal check.
base::TimeDelta slow_ssl_load_time_;
const OpenLoginTabCallback open_login_tab_callback_;
base::WeakPtrFactory<CaptivePortalTabReloader> weak_factory_;