blob: 530c21528b4ea10e570121e640a5f3ca7d159dc8 [file] [log] [blame]
// Copyright (c) 2006-2008 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_SYNC_GLUE_HTTP_BRIDGE_H_
#define CHROME_BROWSER_SYNC_GLUE_HTTP_BRIDGE_H_
#pragma once
#include <string>
#include "base/ref_counted.h"
#include "base/waitable_event.h"
#include "chrome/browser/sync/engine/syncapi.h"
#include "chrome/common/net/url_fetcher.h"
#include "chrome/common/net/url_request_context_getter.h"
#include "googleurl/src/gurl.h"
#include "net/url_request/url_request_context.h"
#include "testing/gtest/include/gtest/gtest_prod.h"
class MessageLoop;
class HttpBridgeTest;
namespace browser_sync {
// 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 HttpBridge : public base::RefCountedThreadSafe<HttpBridge>,
public sync_api::HttpPostProviderInterface,
public URLFetcher::Delegate {
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.
// TODO(timsteele): We subclass here instead of add a factory method on
// ChromeURLRequestContext because:
// 1) we want the ability to set_user_agent
// 2) avoids ifdefs for now
// 3) not sure we want to strictly follow settings for cookie policy,
// accept lang/charset, since changing these could break syncing.
class RequestContext : public URLRequestContext {
public:
// |baseline_context| is used to obtain the accept-language,
// accept-charsets, and proxy service information for bridged requests.
// Typically |baseline_context| should be the URLRequestContext of the
// currently active profile.
explicit RequestContext(URLRequestContext* baseline_context);
// Set the user agent for requests using this context. The default is
// the browser's UA string.
void set_user_agent(const std::string& ua) { user_agent_ = ua; }
virtual const std::string& GetUserAgent(const GURL& url) const {
// If the user agent is set explicitly return that, otherwise call the
// base class method to return default value.
return user_agent_.empty() ?
URLRequestContext::GetUserAgent(url) : user_agent_;
}
virtual bool AllowSendingCookies(const net::URLRequest* request) const {
return false;
}
private:
// The destructor MUST be called on the IO thread.
~RequestContext();
std::string user_agent_;
URLRequestContext* baseline_context_;
DISALLOW_COPY_AND_ASSIGN(RequestContext);
};
// Lazy-getter for RequestContext objects.
class RequestContextGetter : public URLRequestContextGetter {
public:
explicit RequestContextGetter(
URLRequestContextGetter* baseline_context_getter);
void set_user_agent(const std::string& ua) { user_agent_ = ua; }
bool is_user_agent_set() const { return !user_agent_.empty(); }
// URLRequestContextGetter implementation.
virtual URLRequestContext* GetURLRequestContext();
virtual scoped_refptr<base::MessageLoopProxy> GetIOMessageLoopProxy() const;
private:
~RequestContextGetter() {}
// User agent to apply to the URLRequestContext.
std::string user_agent_;
scoped_refptr<URLRequestContextGetter> baseline_context_getter_;
// Lazily initialized by GetURLRequestContext().
scoped_refptr<RequestContext> context_;
DISALLOW_COPY_AND_ASSIGN(RequestContextGetter);
};
explicit HttpBridge(RequestContextGetter* context);
// sync_api::HttpPostProvider implementation.
virtual void SetUserAgent(const char* user_agent);
virtual void SetExtraRequestHeaders(const char* headers);
virtual void SetURL(const char* url, int port);
virtual void SetPostPayload(const char* content_type, int content_length,
const char* content);
virtual bool MakeSynchronousPost(int* os_error_code, int* response_code);
// 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;
virtual const char* GetResponseContent() const;
virtual const std::string GetResponseHeaderValue(
const std::string& name) const;
// URLFetcher::Delegate implementation.
virtual void OnURLFetchComplete(const URLFetcher* source, const GURL& url,
const URLRequestStatus& status,
int response_code,
const ResponseCookies& cookies,
const std::string& data);
#if defined(UNIT_TEST)
URLRequestContextGetter* GetRequestContextGetter() const {
return context_getter_for_request_;
}
#endif
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 ::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(); }
// Gets a customized URLRequestContext for bridged requests. See
// RequestContext definition for details.
scoped_refptr<RequestContextGetter> context_getter_for_request_;
// 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.
URLFetcher* url_poster_;
// 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.
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_;
scoped_refptr<net::HttpResponseHeaders> response_headers_;
// Cached response data.
bool request_completed_;
bool request_succeeded_;
int http_response_code_;
int os_error_code_;
std::string response_content_;
// 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_;
DISALLOW_COPY_AND_ASSIGN(HttpBridge);
};
class HttpBridgeFactory
: public sync_api::HttpPostProviderFactory {
public:
explicit HttpBridgeFactory(URLRequestContextGetter* baseline_context_getter);
virtual ~HttpBridgeFactory();
virtual sync_api::HttpPostProviderInterface* Create();
virtual void Destroy(sync_api::HttpPostProviderInterface* http);
private:
// This request context is built on top of the baseline context and shares
// common components.
HttpBridge::RequestContextGetter* GetRequestContextGetter();
scoped_refptr<HttpBridge::RequestContextGetter> request_context_getter_;
DISALLOW_COPY_AND_ASSIGN(HttpBridgeFactory);
};
} // namespace browser_sync
#endif // CHROME_BROWSER_SYNC_GLUE_HTTP_BRIDGE_H_