1 // Copyright 2014 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 COMPONENTS_GOOGLE_CORE_BROWSER_GOOGLE_URL_TRACKER_H_
6 #define COMPONENTS_GOOGLE_CORE_BROWSER_GOOGLE_URL_TRACKER_H_
12 #include "base/callback_forward.h"
13 #include "base/callback_list.h"
14 #include "base/gtest_prod_util.h"
15 #include "base/memory/scoped_ptr.h"
16 #include "base/memory/weak_ptr.h"
17 #include "components/google/core/browser/google_url_tracker_client.h"
18 #include "components/google/core/browser/google_url_tracker_map_entry.h"
19 #include "components/keyed_service/core/keyed_service.h"
20 #include "net/base/network_change_notifier.h"
21 #include "net/url_request/url_fetcher.h"
22 #include "net/url_request/url_fetcher_delegate.h"
25 class GoogleURLTrackerNavigationHelper
;
32 // This object is responsible for checking the Google URL once per network
33 // change, and if necessary prompting the user to see if they want to change to
34 // using it. The current and last prompted values are saved to prefs.
36 // Most consumers should only call google_url(). Consumers who need to be
37 // notified when things change should register a callback that provides the
38 // original and updated values via RegisterCallback().
40 // To protect users' privacy and reduce server load, no updates will be
41 // performed (ever) unless at least one consumer registers interest by calling
42 // RequestServerCheck().
43 class GoogleURLTracker
: public net::URLFetcherDelegate
,
44 public net::NetworkChangeNotifier::IPAddressObserver
,
47 // Callback that is called when the Google URL is updated. The arguments are
48 // the old and new URLs.
49 typedef base::Callback
<void()> OnGoogleURLUpdatedCallback
;
50 typedef base::CallbackList
<void()> CallbackList
;
51 typedef CallbackList::Subscription Subscription
;
53 // The constructor does different things depending on which of these values
54 // you pass it. Hopefully these are self-explanatory.
60 static const char kDefaultGoogleHomepage
[];
62 // Only the GoogleURLTrackerFactory and tests should call this.
63 GoogleURLTracker(scoped_ptr
<GoogleURLTrackerClient
> client
, Mode mode
);
65 virtual ~GoogleURLTracker();
67 // Returns the current Google homepage URL.
68 const GURL
& google_url() const { return google_url_
; }
70 // Requests that the tracker perform a server check to update the Google URL
71 // as necessary. If |force| is false, this will happen at most once per
72 // network change, not sooner than five seconds after startup (checks
73 // requested before that time will occur then; checks requested afterwards
74 // will occur immediately, if no other checks have been made during this run).
75 // If |force| is true, and the tracker has already performed any requested
76 // check, it will check again.
77 void RequestServerCheck(bool force
);
79 // Notifies the tracker that the user has started a Google search.
80 // If prompting is necessary, we then listen for the subsequent pending
81 // navigation to get the appropriate NavigationHelper. When the load
82 // commits, we'll show the infobar.
83 void SearchCommitted();
85 // No one but GoogleURLTrackerInfoBarDelegate or test code should call these.
86 void AcceptGoogleURL(bool redo_searches
);
87 void CancelGoogleURL();
88 const GURL
& fetched_google_url() const { return fetched_google_url_
; }
89 GoogleURLTrackerClient
* client() { return client_
.get(); }
91 // No one but GoogleURLTrackerMapEntry should call this.
92 void DeleteMapEntryForManager(
93 const infobars::InfoBarManager
* infobar_manager
);
95 // Called by the client after SearchCommitted() registers listeners,
96 // to indicate that we've received the "load now pending" notification.
97 // |nav_helper| is the GoogleURLTrackerNavigationHelper associated with this
98 // navigation; |infobar_manager| is the InfoBarManager of the associated tab;
99 // and |pending_id| is the unique ID of the newly pending NavigationEntry.
100 // If there is already a visible GoogleURLTracker infobar for this tab, this
101 // function resets its associated pending entry ID to the new ID. Otherwise
102 // this function creates a map entry for the associated tab.
103 virtual void OnNavigationPending(
104 scoped_ptr
<GoogleURLTrackerNavigationHelper
> nav_helper
,
105 infobars::InfoBarManager
* infobar_manager
,
108 // Called by the navigation observer once a load we're watching commits.
109 // |infobar_manager| is the same as for OnNavigationPending();
110 // |search_url| is guaranteed to be valid.
111 virtual void OnNavigationCommitted(infobars::InfoBarManager
* infobar_manager
,
112 const GURL
& search_url
);
114 // Called by the navigation observer when a tab closes.
115 virtual void OnTabClosed(GoogleURLTrackerNavigationHelper
* nav_helper
);
117 scoped_ptr
<Subscription
> RegisterCallback(
118 const OnGoogleURLUpdatedCallback
& cb
);
121 friend class GoogleURLTrackerTest
;
122 friend class SyncTest
;
124 typedef std::map
<const infobars::InfoBarManager
*, GoogleURLTrackerMapEntry
*>
127 static const char kSearchDomainCheckURL
[];
129 // net::URLFetcherDelegate:
130 virtual void OnURLFetchComplete(const net::URLFetcher
* source
) OVERRIDE
;
132 // NetworkChangeNotifier::IPAddressObserver:
133 virtual void OnIPAddressChanged() OVERRIDE
;
136 virtual void Shutdown() OVERRIDE
;
138 // Registers consumer interest in getting an updated URL from the server.
139 // Observe chrome::NOTIFICATION_GOOGLE_URL_UPDATED to be notified when the URL
141 void SetNeedToFetch();
143 // Called when the five second startup sleep has finished. Runs any pending
147 // Starts the fetch of the up-to-date Google URL if we actually want to fetch
148 // it and can currently do so.
149 void StartFetchIfDesirable();
151 // Closes all map entries. If |redo_searches| is true, this also triggers
152 // each tab with an infobar to re-perform the user's search, but on the new
154 void CloseAllEntries(bool redo_searches
);
156 // Unregisters any listeners for the navigation helper in |map_entry|.
157 // This sanity-DCHECKs that these are registered (or not) in the specific
158 // cases we expect. (|must_be_listening_for_commit| is used purely for this
159 // sanity-checking.) This also unregisters the global navigation pending
160 // listener if there are no remaining listeners for navigation commits, as we
161 // no longer need them until another search is committed.
162 void UnregisterForEntrySpecificNotifications(
163 GoogleURLTrackerMapEntry
* map_entry
,
164 bool must_be_listening_for_commit
);
166 void NotifyGoogleURLUpdated();
168 CallbackList callback_list_
;
170 scoped_ptr
<GoogleURLTrackerClient
> client_
;
173 GURL fetched_google_url_
;
174 scoped_ptr
<net::URLFetcher
> fetcher_
;
176 bool in_startup_sleep_
; // True if we're in the five-second "no fetching"
177 // period that begins at browser start.
178 bool already_fetched_
; // True if we've already fetched a URL once this run;
179 // we won't fetch again until after a restart.
180 bool need_to_fetch_
; // True if a consumer actually wants us to fetch an
181 // updated URL. If this is never set, we won't
182 // bother to fetch anything.
183 // Consumers should register a callback via
184 // RegisterCallback().
185 bool need_to_prompt_
; // True if the last fetched Google URL is not
186 // matched with current user's default Google URL
187 // nor the last prompted Google URL.
188 bool search_committed_
; // True when we're expecting a notification of a new
189 // pending search navigation.
191 base::WeakPtrFactory
<GoogleURLTracker
> weak_ptr_factory_
;
193 DISALLOW_COPY_AND_ASSIGN(GoogleURLTracker
);
196 #endif // COMPONENTS_GOOGLE_CORE_BROWSER_GOOGLE_URL_TRACKER_H_