// 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.
#ifndef CHROME_BROWSER_GOOGLE_GOOGLE_URL_TRACKER_H_
#define CHROME_BROWSER_GOOGLE_GOOGLE_URL_TRACKER_H_
#include <map>
#include <string>
#include <utility>
#include "base/callback_forward.h"
#include "base/gtest_prod_util.h"
#include "base/memory/scoped_ptr.h"
#include "base/memory/weak_ptr.h"
#include "chrome/browser/google/google_url_tracker_map_entry.h"
#include "components/browser_context_keyed_service/browser_context_keyed_service.h"
#include "net/base/network_change_notifier.h"
#include "net/url_request/url_fetcher.h"
#include "net/url_request/url_fetcher_delegate.h"
#include "url/gurl.h"
class GoogleURLTrackerNavigationHelper;
class InfoBar;
class PrefService;
class Profile;
namespace content {
class NavigationController;
}
// This object is responsible for checking the Google URL once per network
// change, and if necessary prompting the user to see if they want to change to
// using it. The current and last prompted values are saved to prefs.
//
// Most consumers should only call GoogleURL(), which is guaranteed to
// synchronously return a value at all times (even during startup or in unittest
// mode). Consumers who need to be notified when things change should listen to
// the notification service for NOTIFICATION_GOOGLE_URL_UPDATED, which provides
// the original and updated values.
//
// To protect users' privacy and reduce server load, no updates will be
// performed (ever) unless at least one consumer registers interest by calling
// RequestServerCheck().
class GoogleURLTracker : public net::URLFetcherDelegate,
public net::NetworkChangeNotifier::IPAddressObserver,
public BrowserContextKeyedService {
public:
// The contents of the Details for a NOTIFICATION_GOOGLE_URL_UPDATED.
typedef std::pair<GURL, GURL> UpdatedDetails;
// The constructor does different things depending on which of these values
// you pass it. Hopefully these are self-explanatory.
enum Mode {
NORMAL_MODE,
UNIT_TEST_MODE,
};
// Only the GoogleURLTrackerFactory and tests should call this. No code other
// than the GoogleURLTracker itself should actually use
// GoogleURLTrackerFactory::GetForProfile().
GoogleURLTracker(Profile* profile,
scoped_ptr<GoogleURLTrackerNavigationHelper> nav_helper,
Mode mode);
virtual ~GoogleURLTracker();
// Returns the current Google URL. This will return a valid URL even if
// |profile| is NULL or a testing profile.
//
// This is the only function most code should ever call.
static GURL GoogleURL(Profile* profile);
// Requests that the tracker perform a server check to update the Google URL
// as necessary. If |force| is false, this will happen at most once per
// network change, not sooner than five seconds after startup (checks
// requested before that time will occur then; checks requested afterwards
// will occur immediately, if no other checks have been made during this run).
// If |force| is true, and the tracker has already performed any requested
// check, it will check again.
//
// When |profile| is NULL or a testing profile, this function does nothing.
static void RequestServerCheck(Profile* profile, bool force);
// Notifies the tracker that the user has started a Google search.
// If prompting is necessary, we then listen for the subsequent pending
// navigation to get the appropriate NavigationController. When the load
// commits, we'll show the infobar.
//
// When |profile| is NULL or a testing profile, this function does nothing.
static void GoogleURLSearchCommitted(Profile* profile);
// No one but GoogleURLTrackerInfoBarDelegate or test code should call these.
void AcceptGoogleURL(bool redo_searches);
void CancelGoogleURL();
const GURL& google_url() const { return google_url_; }
const GURL& fetched_google_url() const { return fetched_google_url_; }
// No one but GoogleURLTrackerMapEntry should call this.
void DeleteMapEntryForService(const InfoBarService* infobar_service);
// Called by the navigation observer after SearchCommitted() registers
// listeners, to indicate that we've received the "load now pending"
// notification. |navigation_controller| is the NavigationController for this
// load; |infobar_service| is the InfoBarService of the associated tab; and
// |pending_id| is the unique ID of the newly pending NavigationEntry.
// If there is already a visible GoogleURLTracker infobar for this tab, this
// function resets its associated pending entry ID to the new ID. Otherwise
// this function creates a map entry for the associated tab.
virtual void OnNavigationPending(
content::NavigationController* navigation_controller,
InfoBarService* infobar_service,
int pending_id);
// Called by the navigation observer once a load we're watching commits.
// |infobar_service| is the same as for OnNavigationPending();
// |search_url| is guaranteed to be valid.
virtual void OnNavigationCommitted(InfoBarService* infobar_service,
const GURL& search_url);
// Called by the navigation observer when a tab closes.
virtual void OnTabClosed(
content::NavigationController* navigation_controller);
static const char kDefaultGoogleHomepage[];
static const char kSearchDomainCheckURL[];
private:
friend class GoogleURLTrackerTest;
typedef std::map<const InfoBarService*, GoogleURLTrackerMapEntry*> EntryMap;
// net::URLFetcherDelegate:
virtual void OnURLFetchComplete(const net::URLFetcher* source) OVERRIDE;
// NetworkChangeNotifier::IPAddressObserver:
virtual void OnIPAddressChanged() OVERRIDE;
// BrowserContextKeyedService:
virtual void Shutdown() OVERRIDE;
// Registers consumer interest in getting an updated URL from the server.
// Observe chrome::NOTIFICATION_GOOGLE_URL_UPDATED to be notified when the URL
// changes.
void SetNeedToFetch();
// Called when the five second startup sleep has finished. Runs any pending
// fetch.
void FinishSleep();
// Starts the fetch of the up-to-date Google URL if we actually want to fetch
// it and can currently do so.
void StartFetchIfDesirable();
// Called each time the user performs a search. This checks whether we need
// to prompt the user about a domain change, and if so, starts listening for
// the notifications sent when the actual load is triggered.
void SearchCommitted();
// Closes all map entries. If |redo_searches| is true, this also triggers
// each tab with an infobar to re-perform the user's search, but on the new
// Google TLD.
void CloseAllEntries(bool redo_searches);
// Unregisters any listeners for the navigation controller in |map_entry|.
// This sanity-DCHECKs that these are registered (or not) in the specific
// cases we expect. (|must_be_listening_for_commit| is used purely for this
// sanity-checking.) This also unregisters the global navigation pending
// listener if there are no remaining listeners for navigation commits, as we
// no longer need them until another search is committed.
void UnregisterForEntrySpecificNotifications(
const GoogleURLTrackerMapEntry& map_entry,
bool must_be_listening_for_commit);
Profile* profile_;
scoped_ptr<GoogleURLTrackerNavigationHelper> nav_helper_;
// Creates an infobar and adds it to the provided InfoBarService. Returns the
// infobar on success or NULL on failure. The caller does not own the
// returned object, the InfoBarService does.
base::Callback<InfoBar*(InfoBarService*, GoogleURLTracker*, const GURL&)>
infobar_creator_;
GURL google_url_;
GURL fetched_google_url_;
base::WeakPtrFactory<GoogleURLTracker> weak_ptr_factory_;
scoped_ptr<net::URLFetcher> fetcher_;
int fetcher_id_;
bool in_startup_sleep_; // True if we're in the five-second "no fetching"
// period that begins at browser start.
bool already_fetched_; // True if we've already fetched a URL once this run;
// we won't fetch again until after a restart.
bool need_to_fetch_; // True if a consumer actually wants us to fetch an
// updated URL. If this is never set, we won't
// bother to fetch anything.
// Consumers should observe
// chrome::NOTIFICATION_GOOGLE_URL_UPDATED.
bool need_to_prompt_; // True if the last fetched Google URL is not
// matched with current user's default Google URL
// nor the last prompted Google URL.
bool search_committed_; // True when we're expecting a notification of a new
// pending search navigation.
EntryMap entry_map_;
DISALLOW_COPY_AND_ASSIGN(GoogleURLTracker);
};
#endif // CHROME_BROWSER_GOOGLE_GOOGLE_URL_TRACKER_H_