// Copyright 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 SYNC_INTERNAL_API_PUBLIC_HTTP_BRIDGE_H_ #define SYNC_INTERNAL_API_PUBLIC_HTTP_BRIDGE_H_ #include <string> #include "base/basictypes.h" #include "base/compiler_specific.h" #include "base/gtest_prod_util.h" #include "base/memory/ref_counted.h" #include "base/synchronization/lock.h" #include "base/synchronization/waitable_event.h" #include "net/url_request/url_fetcher_delegate.h" #include "net/url_request/url_request_context.h" #include "net/url_request/url_request_context_getter.h" #include "sync/base/sync_export.h" #include "sync/internal_api/public/base/cancelation_observer.h" #include "sync/internal_api/public/http_post_provider_factory.h" #include "sync/internal_api/public/http_post_provider_interface.h" #include "sync/internal_api/public/network_time_update_callback.h" #include "url/gurl.h" class HttpBridgeTest; namespace base { class MessageLoop; } namespace net { class HttpResponseHeaders; class HttpUserAgentSettings; class URLFetcher; class URLRequestJobFactory; } namespace syncer { class CancelationSignal; // A bridge between the syncer and Chromium HTTP layers. // Provides a way for the sync backend to use Chromium directly for HTTP // requests rather than depending on a third party provider (e.g libcurl). // This is a one-time use bridge. Create one for each request you want to make. // It is RefCountedThreadSafe because it can PostTask to the io loop, and thus // needs to stick around across context switches, etc. class SYNC_EXPORT_PRIVATE HttpBridge : public base::RefCountedThreadSafe<HttpBridge>, public HttpPostProviderInterface, public net::URLFetcherDelegate { public: // A request context used for HTTP requests bridged from the sync backend. // A bridged RequestContext has a dedicated in-memory cookie store and does // not use a cache. Thus the same type can be used for incognito mode. class RequestContext : public net::URLRequestContext { public: // |baseline_context| is used to obtain the accept-language // and proxy service information for bridged requests. // Typically |baseline_context| should be the net::URLRequestContext of the // currently active profile. RequestContext( net::URLRequestContext* baseline_context, const scoped_refptr<base::SingleThreadTaskRunner>& network_task_runner, const std::string& user_agent); // The destructor MUST be called on the IO thread. virtual ~RequestContext(); private: net::URLRequestContext* const baseline_context_; const scoped_refptr<base::SingleThreadTaskRunner> network_task_runner_; scoped_ptr<net::HttpUserAgentSettings> http_user_agent_settings_; scoped_ptr<net::URLRequestJobFactory> job_factory_; DISALLOW_COPY_AND_ASSIGN(RequestContext); }; // Lazy-getter for RequestContext objects. class SYNC_EXPORT_PRIVATE RequestContextGetter : public net::URLRequestContextGetter { public: RequestContextGetter( net::URLRequestContextGetter* baseline_context_getter, const std::string& user_agent); // net::URLRequestContextGetter implementation. virtual net::URLRequestContext* GetURLRequestContext() OVERRIDE; virtual scoped_refptr<base::SingleThreadTaskRunner> GetNetworkTaskRunner() const OVERRIDE; protected: virtual ~RequestContextGetter(); private: scoped_refptr<net::URLRequestContextGetter> baseline_context_getter_; const scoped_refptr<base::SingleThreadTaskRunner> network_task_runner_; // User agent to apply to the net::URLRequestContext. const std::string user_agent_; // Lazily initialized by GetURLRequestContext(). scoped_ptr<RequestContext> context_; DISALLOW_COPY_AND_ASSIGN(RequestContextGetter); }; HttpBridge(RequestContextGetter* context, const NetworkTimeUpdateCallback& network_time_update_callback); // HttpPostProvider implementation. virtual void SetExtraRequestHeaders(const char* headers) OVERRIDE; virtual void SetURL(const char* url, int port) OVERRIDE; virtual void SetPostPayload(const char* content_type, int content_length, const char* content) OVERRIDE; virtual bool MakeSynchronousPost(int* error_code, int* response_code) OVERRIDE; virtual void Abort() OVERRIDE; // WARNING: these response content methods are used to extract plain old data // and not null terminated strings, so you should make sure you have read // GetResponseContentLength() characters when using GetResponseContent. e.g // string r(b->GetResponseContent(), b->GetResponseContentLength()). virtual int GetResponseContentLength() const OVERRIDE; virtual const char* GetResponseContent() const OVERRIDE; virtual const std::string GetResponseHeaderValue( const std::string& name) const OVERRIDE; // net::URLFetcherDelegate implementation. virtual void OnURLFetchComplete(const net::URLFetcher* source) OVERRIDE; net::URLRequestContextGetter* GetRequestContextGetterForTest() const; protected: friend class base::RefCountedThreadSafe<HttpBridge>; virtual ~HttpBridge(); // Protected virtual so the unit test can override to shunt network requests. virtual void MakeAsynchronousPost(); private: friend class SyncHttpBridgeTest; friend class ::HttpBridgeTest; // Called on the IO loop to issue the network request. The extra level // of indirection is so that the unit test can override this behavior but we // still have a function to statically pass to PostTask. void CallMakeAsynchronousPost() { MakeAsynchronousPost(); } // Used to destroy a fetcher when the bridge is Abort()ed, to ensure that // a reference to |this| is held while flushing any pending fetch completion // callbacks coming from the IO thread en route to finally destroying the // fetcher. void DestroyURLFetcherOnIOThread(net::URLFetcher* fetcher); void UpdateNetworkTime(); // The message loop of the thread we were created on. This is the thread that // will block on MakeSynchronousPost while the IO thread fetches data from // the network. // This should be the main syncer thread (SyncerThread) which is what blocks // on network IO through curl_easy_perform. base::MessageLoop* const created_on_loop_; // The URL to POST to. GURL url_for_request_; // POST payload information. std::string content_type_; std::string request_content_; std::string extra_headers_; // A waitable event we use to provide blocking semantics to // MakeSynchronousPost. We block created_on_loop_ while the IO loop fetches // network request. base::WaitableEvent http_post_completed_; struct URLFetchState { URLFetchState(); ~URLFetchState(); // Our hook into the network layer is a URLFetcher. USED ONLY ON THE IO // LOOP, so we can block created_on_loop_ while the fetch is in progress. // NOTE: This is not a scoped_ptr for a reason. It must be deleted on the // same thread that created it, which isn't the same thread |this| gets // deleted on. We must manually delete url_poster_ on the IO loop. net::URLFetcher* url_poster; // Start and finish time of request. Set immediately before sending // request and after receiving response. base::Time start_time; base::Time end_time; // Used to support 'Abort' functionality. bool aborted; // Cached response data. bool request_completed; bool request_succeeded; int http_response_code; int error_code; std::string response_content; scoped_refptr<net::HttpResponseHeaders> response_headers; }; // This lock synchronizes use of state involved in the flow to fetch a URL // using URLFetcher, including |fetch_state_| and // |context_getter_for_request_| on any thread, for example, this flow needs // to be synchronized to gracefully clean up URLFetcher and return // appropriate values in |error_code|. mutable base::Lock fetch_state_lock_; URLFetchState fetch_state_; // Gets a customized net::URLRequestContext for bridged requests. See // RequestContext definition for details. scoped_refptr<RequestContextGetter> context_getter_for_request_; const scoped_refptr<base::SingleThreadTaskRunner> network_task_runner_; // Callback for updating network time. NetworkTimeUpdateCallback network_time_update_callback_; DISALLOW_COPY_AND_ASSIGN(HttpBridge); }; class SYNC_EXPORT HttpBridgeFactory : public HttpPostProviderFactory, public CancelationObserver { public: HttpBridgeFactory( net::URLRequestContextGetter* baseline_context_getter, const NetworkTimeUpdateCallback& network_time_update_callback, CancelationSignal* cancelation_signal); virtual ~HttpBridgeFactory(); // HttpPostProviderFactory: virtual void Init(const std::string& user_agent) OVERRIDE; virtual HttpPostProviderInterface* Create() OVERRIDE; virtual void Destroy(HttpPostProviderInterface* http) OVERRIDE; // CancelationObserver implementation: virtual void OnSignalReceived() OVERRIDE; private: // Protects |request_context_getter_| and |baseline_request_context_getter_|. base::Lock context_getter_lock_; // This request context is the starting point for the request_context_getter_ // that we eventually use to make requests. During shutdown we must drop all // references to it before the ProfileSyncService's Shutdown() call is // complete. scoped_refptr<net::URLRequestContextGetter> baseline_request_context_getter_; // This request context is built on top of the baseline context and shares // common components. Takes a reference to the // baseline_request_context_getter_. It's mostly used on sync thread when // creating connection but is released as soon as possible during shutdown. // Protected by |context_getter_lock_|. scoped_refptr<HttpBridge::RequestContextGetter> request_context_getter_; NetworkTimeUpdateCallback network_time_update_callback_; CancelationSignal* const cancelation_signal_; DISALLOW_COPY_AND_ASSIGN(HttpBridgeFactory); }; } // namespace syncer #endif // SYNC_INTERNAL_API_PUBLIC_HTTP_BRIDGE_H_