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_INTRANET_REDIRECT_DETECTOR_H_
6 #define CHROME_BROWSER_INTRANET_REDIRECT_DETECTOR_H_
12 #include "base/memory/weak_ptr.h"
13 #include "content/public/browser/notification_observer.h"
14 #include "content/public/browser/notification_registrar.h"
15 #include "net/base/network_change_notifier.h"
16 #include "net/dns/host_resolver_proc.h"
17 #include "net/url_request/url_fetcher_delegate.h"
20 class PrefRegistrySimple
;
22 // This object is responsible for determining whether the user is on a network
23 // that redirects requests for intranet hostnames to another site, and if so,
24 // tracking what that site is (including across restarts via a pref). For
25 // example, the user's ISP might convert a request for "http://query/" into a
26 // 302 redirect to "http://isp.domain.com/search?q=query" in order to display
27 // custom pages on typos, nonexistent sites, etc.
29 // We use this information in the OmniboxNavigationObserver to avoid displaying
30 // infobars for these cases. Our infobars are designed to allow users to get at
31 // intranet sites when they were erroneously taken to a search result page. In
32 // these cases, however, users would be shown a confusing and useless infobar
33 // when they really did mean to do a search.
35 // Consumers should call RedirectOrigin(), which is guaranteed to synchronously
36 // return a value at all times (even during startup or in unittest mode). If no
37 // redirection is in place, the returned GURL will be empty.
38 class IntranetRedirectDetector
39 : public net::URLFetcherDelegate
,
40 public net::NetworkChangeNotifier::IPAddressObserver
{
42 // Only the main browser process loop should call this, when setting up
43 // g_browser_process->intranet_redirect_detector_. No code other than the
44 // IntranetRedirectDetector itself should actually use
45 // g_browser_process->intranet_redirect_detector() (which shouldn't be hard,
46 // since there aren't useful public functions on this object for consumers to
48 IntranetRedirectDetector();
49 virtual ~IntranetRedirectDetector();
51 // Returns the current redirect origin. This will be empty if no redirection
53 static GURL
RedirectOrigin();
55 static void RegisterPrefs(PrefRegistrySimple
* registry
);
57 // The number of characters the fetcher will use for its randomly-generated
59 static const size_t kNumCharsInHostnames
;
62 typedef std::set
<net::URLFetcher
*> Fetchers
;
64 // Called when the seven second startup sleep or the one second network
65 // switch sleep has finished. Runs any pending fetch.
68 // net::URLFetcherDelegate
69 virtual void OnURLFetchComplete(const net::URLFetcher
* source
) OVERRIDE
;
71 // NetworkChangeNotifier::IPAddressObserver
72 virtual void OnIPAddressChanged() OVERRIDE
;
74 GURL redirect_origin_
;
76 std::vector
<GURL
> resulting_origins_
;
77 bool in_sleep_
; // True if we're in the seven-second "no fetching" period
78 // that begins at browser start, or the one-second "no
79 // fetching" period that begins after network switches.
80 base::WeakPtrFactory
<IntranetRedirectDetector
> weak_ptr_factory_
;
82 DISALLOW_COPY_AND_ASSIGN(IntranetRedirectDetector
);
85 // This is for use in testing, where we don't want our fetches to actually go
86 // over the network. It captures the requests and causes them to fail.
87 class IntranetRedirectHostResolverProc
: public net::HostResolverProc
{
89 explicit IntranetRedirectHostResolverProc(net::HostResolverProc
* previous
);
91 virtual int Resolve(const std::string
& host
,
92 net::AddressFamily address_family
,
93 net::HostResolverFlags host_resolver_flags
,
94 net::AddressList
* addrlist
,
95 int* os_error
) OVERRIDE
;
98 virtual ~IntranetRedirectHostResolverProc() {}
101 #endif // CHROME_BROWSER_INTRANET_REDIRECT_DETECTOR_H_