// Copyright 2014 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 COMPONENTS_SUGGESTIONS_SUGGESTIONS_SERVICE_H_ #define COMPONENTS_SUGGESTIONS_SUGGESTIONS_SERVICE_H_ #include #include #include "base/basictypes.h" #include "base/callback.h" #include "base/cancelable_callback.h" #include "base/gtest_prod_util.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/threading/thread_checker.h" #include "base/time/time.h" #include "components/keyed_service/core/keyed_service.h" #include "components/suggestions/image_manager.h" #include "components/suggestions/proto/suggestions.pb.h" #include "components/suggestions/suggestions_utils.h" #include "net/url_request/url_fetcher_delegate.h" #include "ui/gfx/image/image_skia.h" #include "url/gurl.h" namespace net { class URLRequestContextGetter; } // namespace net namespace user_prefs { class PrefRegistrySyncable; } // namespace user_prefs namespace suggestions { class BlacklistStore; class SuggestionsStore; extern const char kSuggestionsFieldTrialName[]; extern const char kSuggestionsFieldTrialControlParam[]; extern const char kSuggestionsFieldTrialStateEnabled[]; extern const char kSuggestionsFieldTrialStateParam[]; extern const char kSuggestionsURL[]; extern const char kSuggestionsBlacklistURLPrefix[]; extern const char kSuggestionsBlacklistURLParam[]; extern const int64 kDefaultExpiryUsec; // An interface to fetch server suggestions asynchronously. class SuggestionsService : public KeyedService, public net::URLFetcherDelegate { public: typedef base::Callback ResponseCallback; // Class taking ownership of |suggestions_store|, |thumbnail_manager| and // |blacklist_store|. SuggestionsService( net::URLRequestContextGetter* url_request_context, scoped_ptr suggestions_store, scoped_ptr thumbnail_manager, scoped_ptr blacklist_store); ~SuggestionsService() override; // Whether this service is enabled. static bool IsEnabled(); // Whether the user is part of a control group. static bool IsControlGroup(); // Request suggestions data, which will be passed to |callback|. |sync_state| // will influence the behavior of this function (see SyncState definition). // // |sync_state| must be specified based on the current state of the system // (see suggestions::GetSyncState). Callers should call this function again if // sync state changes. // // If state allows for a network request, it is initiated unless a pending one // exists, to fill the cache for next time. void FetchSuggestionsData(SyncState sync_state, ResponseCallback callback); // Retrieves stored thumbnail for website |url| asynchronously. Calls // |callback| with Bitmap pointer if found, and NULL otherwise. void GetPageThumbnail( const GURL& url, base::Callback callback); // Adds a URL to the blacklist cache, invoking |callback| on success or // |fail_callback| otherwise. The URL will eventually be uploaded to the // server. void BlacklistURL(const GURL& candidate_url, const ResponseCallback& callback, const base::Closure& fail_callback); // Removes a URL from the local blacklist, then invokes |callback|. If the URL // cannot be removed, the |fail_callback| is called. void UndoBlacklistURL(const GURL& url, const ResponseCallback& callback, const base::Closure& fail_callback); // Determines which URL a blacklist request was for, irrespective of the // request's status. Returns false if |request| is not a blacklist request. static bool GetBlacklistedUrl(const net::URLFetcher& request, GURL* url); // Register SuggestionsService related prefs in the Profile prefs. static void RegisterProfilePrefs(user_prefs::PrefRegistrySyncable* registry); // Sets default timestamp for suggestions which do not have expiry timestamp. void SetDefaultExpiryTimestamp(SuggestionsProfile* suggestions, int64 timestamp_usec); // Issue a network request if there isn't already one happening. Visible for // testing. void IssueRequestIfNoneOngoing(const GURL& url); private: friend class SuggestionsServiceTest; FRIEND_TEST_ALL_PREFIXES(SuggestionsServiceTest, BlacklistURL); FRIEND_TEST_ALL_PREFIXES(SuggestionsServiceTest, BlacklistURLRequestFails); FRIEND_TEST_ALL_PREFIXES(SuggestionsServiceTest, UndoBlacklistURL); FRIEND_TEST_ALL_PREFIXES(SuggestionsServiceTest, UndoBlacklistURLFailsHelper); FRIEND_TEST_ALL_PREFIXES(SuggestionsServiceTest, UpdateBlacklistDelay); // Creates a request to the suggestions service, properly setting headers. net::URLFetcher* CreateSuggestionsRequest(const GURL& url); // net::URLFetcherDelegate implementation. // Called when fetch request completes. Parses the received suggestions data, // and dispatches them to callbacks stored in queue. void OnURLFetchComplete(const net::URLFetcher* source) override; // KeyedService implementation. void Shutdown() override; // Loads the cached suggestions (or empty suggestions if no cache) and serves // the requestors with them. void ServeFromCache(); // Applies the local blacklist to |suggestions|, then serves the requestors. void FilterAndServe(SuggestionsProfile* suggestions); // Schedules a blacklisting request if the local blacklist isn't empty. void ScheduleBlacklistUpload(); // If the local blacklist isn't empty, picks a URL from it and issues a // blacklist request for it. void UploadOneFromBlacklist(); // Updates |scheduling_delay_| based on the success of the last request. void UpdateBlacklistDelay(bool last_request_successful); // Test seams. base::TimeDelta blacklist_delay() const { return scheduling_delay_; } void set_blacklist_delay(base::TimeDelta delay) { scheduling_delay_ = delay; } base::ThreadChecker thread_checker_; net::URLRequestContextGetter* url_request_context_; // The cache for the suggestions. scoped_ptr suggestions_store_; // Used to obtain server thumbnails, if available. scoped_ptr thumbnail_manager_; // The local cache for temporary blacklist, until uploaded to the server. scoped_ptr blacklist_store_; // Delay used when scheduling a blacklisting task. base::TimeDelta scheduling_delay_; // Contains the current suggestions fetch request. Will only have a value // while a request is pending, and will be reset by |OnURLFetchComplete| or // if cancelled. scoped_ptr pending_request_; // The start time of the previous suggestions request. This is used to measure // the latency of requests. Initially zero. base::TimeTicks last_request_started_time_; // The URL to fetch suggestions data from. GURL suggestions_url_; // Prefix for building the blacklisting URL. std::string blacklist_url_prefix_; // Queue of callbacks. These are flushed when fetch request completes. std::vector waiting_requestors_; // For callbacks may be run after destruction. base::WeakPtrFactory weak_ptr_factory_; DISALLOW_COPY_AND_ASSIGN(SuggestionsService); }; } // namespace suggestions #endif // COMPONENTS_SUGGESTIONS_SUGGESTIONS_SERVICE_H_