1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #ifndef CHROME_BROWSER_CAPTIVE_PORTAL_CAPTIVE_PORTAL_DETECTOR_H_
6 #define CHROME_BROWSER_CAPTIVE_PORTAL_CAPTIVE_PORTAL_DETECTOR_H_
8 #include "base/basictypes.h"
9 #include "base/callback.h"
10 #include "base/compiler_specific.h"
11 #include "base/memory/ref_counted.h"
12 #include "base/memory/scoped_ptr.h"
13 #include "base/threading/non_thread_safe.h"
14 #include "base/time/time.h"
15 #include "net/url_request/url_fetcher.h"
16 #include "net/url_request/url_fetcher_delegate.h"
17 #include "net/url_request/url_request_context_getter.h"
21 namespace captive_portal
{
23 // Possible results of an attempt to detect a captive portal.
25 // There's a confirmed connection to the Internet.
26 RESULT_INTERNET_CONNECTED
,
27 // The URL request received a network or HTTP error, or a non-HTTP response.
29 // The URL request apparently encountered a captive portal. It received a
30 // a valid HTTP response with a 2xx other than 204, 3xx, or 511 status code.
31 RESULT_BEHIND_CAPTIVE_PORTAL
,
35 class CaptivePortalDetector
: public net::URLFetcherDelegate
,
36 public base::NonThreadSafe
{
40 : result(RESULT_NO_RESPONSE
),
41 response_code(net::URLFetcher::RESPONSE_CODE_INVALID
) {
46 base::TimeDelta retry_after_delta
;
49 typedef base::Callback
<void(const Results
& results
)> DetectionCallback
;
51 // The test URL. When connected to the Internet, it should return a
52 // blank page with a 204 status code. When behind a captive portal,
53 // requests for this URL should get an HTTP redirect or a login
54 // page. When neither is true, no server should respond to requests
56 static const char kDefaultURL
[];
58 explicit CaptivePortalDetector(
59 const scoped_refptr
<net::URLRequestContextGetter
>& request_context
);
60 virtual ~CaptivePortalDetector();
62 static std::string
CaptivePortalResultToString(Result result
);
64 // Triggers a check for a captive portal. After completion, runs the
66 void DetectCaptivePortal(const GURL
& url
, const DetectionCallback
& callback
);
68 // Cancels captive portal check.
72 friend class CaptivePortalDetectorTestBase
;
74 // net::URLFetcherDelegate:
75 virtual void OnURLFetchComplete(const net::URLFetcher
* source
) OVERRIDE
;
77 // Takes a net::URLFetcher that has finished trying to retrieve the
78 // test URL, and fills a Results struct based on its result. If the
79 // response is a 503 with a Retry-After header, |retry_after| field
80 // of |results| is populated accordingly. Otherwise, it's set to
82 void GetCaptivePortalResultFromResponse(const net::URLFetcher
* url_fetcher
,
83 Results
* results
) const;
85 // Returns the current time. Used only when determining time until a
87 base::Time
GetCurrentTime() const;
89 // Returns true if a captive portal check is currently running.
90 bool FetchingURL() const;
92 // Sets current test time. Used by unit tests.
93 void set_time_for_testing(const base::Time
& time
) {
94 time_for_testing_
= time
;
97 // Advances current test time. Used by unit tests.
98 void advance_time_for_testing(const base::TimeDelta
& delta
) {
99 time_for_testing_
+= delta
;
102 // URL request context.
103 scoped_refptr
<net::URLRequestContextGetter
> request_context_
;
105 DetectionCallback detection_callback_
;
107 scoped_ptr
<net::URLFetcher
> url_fetcher_
;
109 // Test time used by unit tests.
110 base::Time time_for_testing_
;
112 DISALLOW_COPY_AND_ASSIGN(CaptivePortalDetector
);
115 } // namespace captive_portal
117 #endif // CHROME_BROWSER_CAPTIVE_PORTAL_CAPTIVE_PORTAL_DETECTOR_H_