summaryrefslogtreecommitdiffstats
path: root/chrome/browser/profile.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chrome/browser/profile.cc')
-rw-r--r--chrome/browser/profile.cc304
1 files changed, 28 insertions, 276 deletions
diff --git a/chrome/browser/profile.cc b/chrome/browser/profile.cc
index d8f2f64..9014424 100644
--- a/chrome/browser/profile.cc
+++ b/chrome/browser/profile.cc
@@ -21,6 +21,7 @@
#include "chrome/browser/greasemonkey_master.h"
#include "chrome/browser/history/history.h"
#include "chrome/browser/navigation_controller.h"
+#include "chrome/browser/net/chrome_url_request_context.h"
#include "chrome/browser/profile_manager.h"
#include "chrome/browser/render_process_host.h"
#include "chrome/browser/sessions/session_service.h"
@@ -76,272 +77,6 @@ URLRequestContext* Profile::GetDefaultRequestContext() {
}
-// Sets up proxy info if it was specified, otherwise returns NULL. The
-// returned pointer MUST be deleted by the caller if non-NULL.
-static net::ProxyInfo* CreateProxyInfo(const CommandLine& command_line) {
- net::ProxyInfo* proxy_info = NULL;
-
- if (command_line.HasSwitch(switches::kProxyServer)) {
- proxy_info = new net::ProxyInfo();
- const std::wstring& proxy_server =
- command_line.GetSwitchValue(switches::kProxyServer);
- proxy_info->UseNamedProxy(WideToASCII(proxy_server));
- }
-
- return proxy_info;
-}
-
-// Releases the URLRequestContext and sends out a notification about it.
-// Note: runs on IO thread.
-static void ReleaseURLRequestContext(URLRequestContext* context) {
- NotificationService::current()->Notify(NOTIFY_URL_REQUEST_CONTEXT_RELEASED,
- Source<URLRequestContext>(context),
- NotificationService::NoDetails());
- context->Release();
-}
-
-// A context for URLRequests issued relative to this profile.
-class ProfileImpl::RequestContext : public URLRequestContext,
- public NotificationObserver {
- public:
- // |cookie_store_path| is the local disk path at which the cookie store
- // is persisted.
- RequestContext(const std::wstring& cookie_store_path,
- const std::wstring& disk_cache_path,
- PrefService* prefs)
- : prefs_(prefs) {
- cookie_store_ = NULL;
-
- // setup user agent
- user_agent_ = webkit_glue::GetUserAgent();
- // set up Accept-Language and Accept-Charset header values
- // TODO(jungshik) : This may slow down http requests. Perhaps,
- // we have to come up with a better way to set up these values.
- accept_language_ = WideToASCII(prefs_->GetString(prefs::kAcceptLanguages));
- accept_charset_ = WideToASCII(prefs_->GetString(prefs::kDefaultCharset));
- accept_charset_ += ",*,utf-8";
-
- CommandLine command_line;
-
- scoped_ptr<net::ProxyInfo> proxy_info(CreateProxyInfo(command_line));
- proxy_service_ = net::ProxyService::Create(proxy_info.get());
-
- net::HttpCache* cache =
- new net::HttpCache(proxy_service_, disk_cache_path, 0);
-
- bool record_mode = chrome::kRecordModeEnabled &&
- CommandLine().HasSwitch(switches::kRecordMode);
- bool playback_mode = CommandLine().HasSwitch(switches::kPlaybackMode);
-
- if (record_mode || playback_mode) {
- // Don't use existing cookies and use an in-memory store.
- cookie_store_ = new net::CookieMonster();
- cache->set_mode(
- record_mode ? net::HttpCache::RECORD : net::HttpCache::PLAYBACK);
- }
- http_transaction_factory_ = cache;
-
- // setup cookie store
- if (!cookie_store_) {
- DCHECK(!cookie_store_path.empty());
- cookie_db_.reset(new SQLitePersistentCookieStore(
- cookie_store_path, g_browser_process->db_thread()->message_loop()));
- cookie_store_ = new net::CookieMonster(cookie_db_.get());
- }
-
- cookie_policy_.SetType(net::CookiePolicy::FromInt(
- prefs_->GetInteger(prefs::kCookieBehavior)));
-
- // The first request context to be created is the one for the default
- // profile - at least until we support multiple profiles.
- if (!default_request_context_)
- default_request_context_ = this;
- NotificationService::current()->Notify(
- NOTIFY_DEFAULT_REQUEST_CONTEXT_AVAILABLE,
- NotificationService::AllSources(), NotificationService::NoDetails());
-
- // Register for notifications about prefs.
- prefs_->AddPrefObserver(prefs::kAcceptLanguages, this);
- prefs_->AddPrefObserver(prefs::kCookieBehavior, this);
- }
-
- // NotificationObserver implementation.
- virtual void Observe(NotificationType type,
- const NotificationSource& source,
- const NotificationDetails& details) {
- if (NOTIFY_PREF_CHANGED == type) {
- std::wstring* pref_name_in = Details<std::wstring>(details).ptr();
- PrefService* prefs = Source<PrefService>(source).ptr();
- DCHECK(pref_name_in && prefs);
- if (*pref_name_in == prefs::kAcceptLanguages) {
- std::string accept_language =
- WideToASCII(prefs->GetString(prefs::kAcceptLanguages));
- g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
- NewRunnableMethod(this,
- &RequestContext::OnAcceptLanguageChange,
- accept_language));
- } else if (*pref_name_in == prefs::kCookieBehavior) {
- net::CookiePolicy::Type type = net::CookiePolicy::FromInt(
- prefs_->GetInteger(prefs::kCookieBehavior));
- g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
- NewRunnableMethod(this,
- &RequestContext::OnCookiePolicyChange,
- type));
- }
- }
- }
-
- // Since ProfileImpl::RequestContext will be destroyed on IO thread, but all
- // PrefService observers are needed to clear in before destroying ProfileImpl.
- // So we use to CleanupBeforeDestroy to do this thing. This function need to
- // be called on destructor of ProfileImpl.
- void CleanupBeforeDestroy() {
- // Unregister for pref notifications.
- prefs_->RemovePrefObserver(prefs::kAcceptLanguages, this);
- prefs_->RemovePrefObserver(prefs::kCookieBehavior, this);
- prefs_ = NULL;
- }
-
- void OnAcceptLanguageChange(std::string accept_language) {
- DCHECK(MessageLoop::current() ==
- ChromeThread::GetMessageLoop(ChromeThread::IO));
- accept_language_ = accept_language;
- }
-
- void OnCookiePolicyChange(net::CookiePolicy::Type type) {
- DCHECK(MessageLoop::current() ==
- ChromeThread::GetMessageLoop(ChromeThread::IO));
- cookie_policy_.SetType(type);
- }
-
- virtual ~RequestContext() {
- DCHECK(NULL == prefs_);
- delete cookie_store_;
- delete http_transaction_factory_;
- delete proxy_service_;
-
- if (default_request_context_ == this)
- default_request_context_ = NULL;
- }
-
- private:
- scoped_ptr<SQLitePersistentCookieStore> cookie_db_;
- PrefService* prefs_;
-};
-
-////////////////////////////////////////////////////////////////////////////////
-//
-// An URLRequestContext proxy for OffTheRecord. This request context is
-// really a proxy to the original profile's request context but set
-// is_off_the_record_ to true.
-//
-// TODO(ACW). Do we need to share the FtpAuthCache with the real request context
-// see bug 974328
-//
-// TODO(jackson): http://b/issue?id=1197350 Remove duplicated code from above.
-//
-////////////////////////////////////////////////////////////////////////////////
-class OffTheRecordRequestContext : public URLRequestContext,
- public NotificationObserver {
- public:
- explicit OffTheRecordRequestContext(Profile* profile) {
- DCHECK(!profile->IsOffTheRecord());
- prefs_ = profile->GetPrefs();
- DCHECK(prefs_);
-
- // The OffTheRecordRequestContext is owned by the OffTheRecordProfileImpl
- // which is itself owned by the original profile. We reference the original
- // context to make sure it doesn't go away when we delete the object graph.
- original_context_ = profile->GetRequestContext();
-
- // Share the same proxy service as the original profile. This proxy
- // service's lifespan is dependent on the lifespan of the original profile,
- // which we reference (see above).
- proxy_service_ = original_context_->proxy_service();
-
- http_transaction_factory_ = new net::HttpCache(proxy_service_, 0);
- cookie_store_ = new net::CookieMonster;
- cookie_policy_.SetType(net::CookiePolicy::FromInt(
- prefs_->GetInteger(prefs::kCookieBehavior)));
- user_agent_ = original_context_->user_agent();
- accept_language_ = original_context_->accept_language();
- accept_charset_ = original_context_->accept_charset();
- is_off_the_record_ = true;
-
- // Register for notifications about prefs.
- prefs_->AddPrefObserver(prefs::kAcceptLanguages, this);
- prefs_->AddPrefObserver(prefs::kCookieBehavior, this);
- }
-
- // Since OffTheRecordProfileImpl maybe be destroyed after destroying
- // PrefService, but all PrefService observers are needed to clear in
- // before destroying PrefService. So we use to CleanupBeforeDestroy
- // to do this thing. This function need to be called on destructor
- // of ProfileImpl.
- void CleanupBeforeDestroy() {
- // Unregister for pref notifications.
- prefs_->RemovePrefObserver(prefs::kAcceptLanguages, this);
- prefs_->RemovePrefObserver(prefs::kCookieBehavior, this);
- prefs_ = NULL;
- }
-
- // NotificationObserver implementation.
- virtual void Observe(NotificationType type,
- const NotificationSource& source,
- const NotificationDetails& details) {
- if (NOTIFY_PREF_CHANGED == type) {
- std::wstring* pref_name_in = Details<std::wstring>(details).ptr();
- PrefService* prefs = Source<PrefService>(source).ptr();
- DCHECK(pref_name_in && prefs);
- if (*pref_name_in == prefs::kAcceptLanguages) {
- std::string accept_language =
- WideToASCII(prefs->GetString(prefs::kAcceptLanguages));
- g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
- NewRunnableMethod(
- this,
- &OffTheRecordRequestContext::OnAcceptLanguageChange,
- accept_language));
- } else if (*pref_name_in == prefs::kCookieBehavior) {
- net::CookiePolicy::Type type = net::CookiePolicy::FromInt(
- prefs_->GetInteger(prefs::kCookieBehavior));
- g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
- NewRunnableMethod(this,
- &OffTheRecordRequestContext::OnCookiePolicyChange,
- type));
- }
- }
- }
-
- void OnAcceptLanguageChange(std::string accept_language) {
- DCHECK(MessageLoop::current() ==
- ChromeThread::GetMessageLoop(ChromeThread::IO));
- accept_language_ = accept_language;
- }
-
- void OnCookiePolicyChange(net::CookiePolicy::Type type) {
- DCHECK(MessageLoop::current() ==
- ChromeThread::GetMessageLoop(ChromeThread::IO));
- cookie_policy_.SetType(type);
- }
-
- virtual ~OffTheRecordRequestContext() {
- DCHECK(NULL == prefs_);
- delete cookie_store_;
- delete http_transaction_factory_;
- // NOTE: do not delete |proxy_service_| as is owned by the original profile.
-
- // The OffTheRecordRequestContext simply act as a proxy to the real context.
- // There is nothing else to delete.
- }
-
- private:
- // The original (non off the record) URLRequestContext.
- scoped_refptr<URLRequestContext> original_context_;
- PrefService* prefs_;
-
- DISALLOW_EVIL_CONSTRUCTORS(OffTheRecordRequestContext);
-};
-
////////////////////////////////////////////////////////////////////////////////
//
// OffTheRecordProfileImpl is a profile subclass that wraps an existing profile
@@ -354,7 +89,7 @@ class OffTheRecordProfileImpl : public Profile,
explicit OffTheRecordProfileImpl(Profile* real_profile)
: profile_(real_profile),
start_time_(Time::Now()) {
- request_context_ = new OffTheRecordRequestContext(real_profile);
+ request_context_ = ChromeURLRequestContext::CreateOffTheRecord(this);
request_context_->AddRef();
// Register for browser close notifications so we can detect when the last
// off-the-record window is closed, in which case we can clean our states
@@ -365,10 +100,12 @@ class OffTheRecordProfileImpl : public Profile,
virtual ~OffTheRecordProfileImpl() {
if (request_context_) {
- request_context_->CleanupBeforeDestroy();
+ request_context_->CleanupOnUIThread();
+
// Clean up request context on IO thread.
g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
- NewRunnableFunction(&ReleaseURLRequestContext, request_context_));
+ NewRunnableMethod(request_context_,
+ &base::RefCountedThreadSafe<URLRequestContext>::Release));
request_context_ = NULL;
}
NotificationService::current()->RemoveObserver(
@@ -545,8 +282,8 @@ class OffTheRecordProfileImpl : public Profile,
// The real underlying profile.
Profile* profile_;
- // A proxy to the real request context.
- OffTheRecordRequestContext* request_context_;
+ // The context to use for requests made from this OTR session.
+ ChromeURLRequestContext* request_context_;
// The download manager that only stores downloaded items in memory.
scoped_refptr<DownloadManager> download_manager_;
@@ -628,10 +365,15 @@ ProfileImpl::~ProfileImpl() {
}
if (request_context_) {
- request_context_->CleanupBeforeDestroy();
+ request_context_->CleanupOnUIThread();
+
+ if (default_request_context_ == request_context_)
+ default_request_context_ = NULL;
+
// Clean up request context on IO thread.
- io_thread->message_loop()->PostTask(FROM_HERE,
- NewRunnableFunction(&ReleaseURLRequestContext, request_context_));
+ g_browser_process->io_thread()->message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(request_context_,
+ &base::RefCountedThreadSafe<URLRequestContext>::Release));
request_context_ = NULL;
}
@@ -753,10 +495,20 @@ URLRequestContext* ProfileImpl::GetRequestContext() {
file_util::AppendToPath(&cookie_path, chrome::kCookieFilename);
std::wstring cache_path = GetPath();
file_util::AppendToPath(&cache_path, chrome::kCacheDirname);
- request_context_ =
- new ProfileImpl::RequestContext(cookie_path, cache_path, GetPrefs());
+ request_context_ = ChromeURLRequestContext::CreateOriginal(
+ this, cookie_path, cache_path);
request_context_->AddRef();
+ // The first request context is always a normal (non-OTR) request context.
+ // Even when Chromium is started in OTR mode, a normal profile is always
+ // created first.
+ if (!default_request_context_) {
+ default_request_context_ = request_context_;
+ NotificationService::current()->Notify(
+ NOTIFY_DEFAULT_REQUEST_CONTEXT_AVAILABLE,
+ NotificationService::AllSources(), NotificationService::NoDetails());
+ }
+
DCHECK(request_context_->cookie_store());
}