off_the_record_profile_impl.cc revision 6d86b77056ed63eb6871182f42a9fd5f07550f90
1// Copyright (c) 2012 The Chromium Authors. All rights reserved. 2// Use of this source code is governed by a BSD-style license that can be 3// found in the LICENSE file. 4 5#include "chrome/browser/profiles/off_the_record_profile_impl.h" 6 7#include "base/bind.h" 8#include "base/command_line.h" 9#include "base/compiler_specific.h" 10#include "base/files/file_path.h" 11#include "base/memory/scoped_ptr.h" 12#include "base/path_service.h" 13#include "base/prefs/json_pref_store.h" 14#include "base/strings/string_number_conversions.h" 15#include "base/strings/string_util.h" 16#include "build/build_config.h" 17#include "chrome/browser/background/background_contents_service_factory.h" 18#include "chrome/browser/browser_process.h" 19#include "chrome/browser/content_settings/host_content_settings_map.h" 20#include "chrome/browser/download/chrome_download_manager_delegate.h" 21#include "chrome/browser/download/download_service.h" 22#include "chrome/browser/download/download_service_factory.h" 23#include "chrome/browser/extensions/extension_service.h" 24#include "chrome/browser/extensions/extension_special_storage_policy.h" 25#include "chrome/browser/io_thread.h" 26#include "chrome/browser/net/pref_proxy_config_tracker.h" 27#include "chrome/browser/net/proxy_service_factory.h" 28#include "chrome/browser/plugins/chrome_plugin_service_filter.h" 29#include "chrome/browser/plugins/plugin_prefs.h" 30#include "chrome/browser/prefs/incognito_mode_prefs.h" 31#include "chrome/browser/prefs/pref_service_syncable.h" 32#include "chrome/browser/themes/theme_service.h" 33#include "chrome/browser/ui/webui/extensions/extension_icon_source.h" 34#include "chrome/common/chrome_constants.h" 35#include "chrome/common/chrome_paths.h" 36#include "chrome/common/chrome_switches.h" 37#include "chrome/common/pref_names.h" 38#include "chrome/common/render_messages.h" 39#include "components/keyed_service/content/browser_context_dependency_manager.h" 40#include "components/user_prefs/user_prefs.h" 41#include "content/public/browser/browser_thread.h" 42#include "content/public/browser/host_zoom_map.h" 43#include "content/public/browser/render_process_host.h" 44#include "content/public/browser/storage_partition.h" 45#include "content/public/browser/url_data_source.h" 46#include "content/public/browser/web_contents.h" 47#include "extensions/browser/extension_system.h" 48#include "extensions/common/extension.h" 49#include "net/http/http_server_properties.h" 50#include "net/http/transport_security_state.h" 51#include "webkit/browser/database/database_tracker.h" 52 53#if defined(OS_ANDROID) 54#include "chrome/browser/media/protected_media_identifier_permission_context.h" 55#include "chrome/browser/media/protected_media_identifier_permission_context_factory.h" 56#endif // defined(OS_ANDROID) 57 58#if defined(OS_ANDROID) || defined(OS_IOS) 59#include "base/prefs/scoped_user_pref_update.h" 60#include "chrome/browser/prefs/proxy_prefs.h" 61#endif // defined(OS_ANDROID) || defined(OS_IOS) 62 63#if defined(OS_CHROMEOS) 64#include "chrome/browser/chromeos/preferences.h" 65#include "chrome/browser/chromeos/profiles/profile_helper.h" 66#endif 67 68#if defined(ENABLE_CONFIGURATION_POLICY) && !defined(OS_CHROMEOS) 69#include "chrome/browser/policy/cloud/user_cloud_policy_manager_factory.h" 70#endif 71 72#if defined(ENABLE_EXTENSIONS) 73#include "chrome/browser/guest_view/guest_view_manager.h" 74#endif 75 76#if defined(ENABLE_EXTENSIONS) 77#include "chrome/browser/extensions/api/web_request/web_request_api.h" 78#endif 79 80using content::BrowserThread; 81using content::DownloadManagerDelegate; 82using content::HostZoomMap; 83 84#if defined(ENABLE_EXTENSIONS) 85namespace { 86 87void NotifyOTRProfileCreatedOnIOThread(void* original_profile, 88 void* otr_profile) { 89 ExtensionWebRequestEventRouter::GetInstance()->OnOTRProfileCreated( 90 original_profile, otr_profile); 91} 92 93void NotifyOTRProfileDestroyedOnIOThread(void* original_profile, 94 void* otr_profile) { 95 ExtensionWebRequestEventRouter::GetInstance()->OnOTRProfileDestroyed( 96 original_profile, otr_profile); 97} 98 99} // namespace 100#endif 101 102OffTheRecordProfileImpl::OffTheRecordProfileImpl(Profile* real_profile) 103 : profile_(real_profile), 104 prefs_(PrefServiceSyncable::IncognitoFromProfile(real_profile)), 105 start_time_(Time::Now()) { 106 // Register on BrowserContext. 107 user_prefs::UserPrefs::Set(this, prefs_); 108} 109 110void OffTheRecordProfileImpl::Init() { 111 // The construction of OffTheRecordProfileIOData::Handle needs the profile 112 // type returned by this->GetProfileType(). Since GetProfileType() is a 113 // virtual member function, we cannot call the function defined in the most 114 // derived class (e.g. GuestSessionProfile) until a ctor finishes. Thus, 115 // we have to instantiate OffTheRecordProfileIOData::Handle here after a ctor. 116 InitIoData(); 117 118#if defined(ENABLE_CONFIGURATION_POLICY) && !defined(OS_CHROMEOS) 119 // Because UserCloudPolicyManager is in a component, it cannot access 120 // GetOriginalProfile. Instead, we have to inject this relation here. 121 policy::UserCloudPolicyManagerFactory::RegisterForOffTheRecordBrowserContext( 122 this->GetOriginalProfile(), this); 123#endif 124 125 BrowserContextDependencyManager::GetInstance()->CreateBrowserContextServices( 126 this); 127 128 DCHECK_NE(IncognitoModePrefs::DISABLED, 129 IncognitoModePrefs::GetAvailability(profile_->GetPrefs())); 130 131#if defined(OS_ANDROID) || defined(OS_IOS) 132 UseSystemProxy(); 133#endif // defined(OS_ANDROID) || defined(OS_IOS) 134 135 // TODO(oshima): Remove the need to eagerly initialize the request context 136 // getter. chromeos::OnlineAttempt is illegally trying to access this 137 // Profile member from a thread other than the UI thread, so we need to 138 // prevent a race. 139#if defined(OS_CHROMEOS) 140 GetRequestContext(); 141#endif // defined(OS_CHROMEOS) 142 143 InitHostZoomMap(); 144 145 // Make the chrome//extension-icon/ resource available. 146 extensions::ExtensionIconSource* icon_source = 147 new extensions::ExtensionIconSource(profile_); 148 content::URLDataSource::Add(this, icon_source); 149 150#if defined(ENABLE_PLUGINS) 151 ChromePluginServiceFilter::GetInstance()->RegisterResourceContext( 152 PluginPrefs::GetForProfile(this).get(), 153 io_data_->GetResourceContextNoInit()); 154#endif 155 156#if defined(ENABLE_EXTENSIONS) 157 BrowserThread::PostTask( 158 BrowserThread::IO, FROM_HERE, 159 base::Bind(&NotifyOTRProfileCreatedOnIOThread, profile_, this)); 160#endif 161} 162 163OffTheRecordProfileImpl::~OffTheRecordProfileImpl() { 164 MaybeSendDestroyedNotification(); 165 166#if defined(ENABLE_PLUGINS) 167 ChromePluginServiceFilter::GetInstance()->UnregisterResourceContext( 168 io_data_->GetResourceContextNoInit()); 169#endif 170 171 BrowserContextDependencyManager::GetInstance()->DestroyBrowserContextServices( 172 this); 173 174#if defined(ENABLE_EXTENSIONS) 175 BrowserThread::PostTask( 176 BrowserThread::IO, FROM_HERE, 177 base::Bind(&NotifyOTRProfileDestroyedOnIOThread, profile_, this)); 178#endif 179 180 if (host_content_settings_map_.get()) 181 host_content_settings_map_->ShutdownOnUIThread(); 182 183 if (pref_proxy_config_tracker_) 184 pref_proxy_config_tracker_->DetachFromPrefService(); 185 186 // Clears any data the network stack contains that may be related to the 187 // OTR session. 188 g_browser_process->io_thread()->ChangedToOnTheRecord(); 189} 190 191void OffTheRecordProfileImpl::InitIoData() { 192 io_data_.reset(new OffTheRecordProfileIOData::Handle(this)); 193} 194 195void OffTheRecordProfileImpl::InitHostZoomMap() { 196 HostZoomMap* host_zoom_map = HostZoomMap::GetForBrowserContext(this); 197 HostZoomMap* parent_host_zoom_map = 198 HostZoomMap::GetForBrowserContext(profile_); 199 host_zoom_map->CopyFrom(parent_host_zoom_map); 200 // Observe parent's HZM change for propagating change of parent's 201 // change to this HZM. 202 zoom_subscription_ = parent_host_zoom_map->AddZoomLevelChangedCallback( 203 base::Bind(&OffTheRecordProfileImpl::OnZoomLevelChanged, 204 base::Unretained(this))); 205} 206 207#if defined(OS_ANDROID) || defined(OS_IOS) 208void OffTheRecordProfileImpl::UseSystemProxy() { 209 // Force the use of the system-assigned proxy when off the record. 210 const char kProxyMode[] = "mode"; 211 const char kProxyServer[] = "server"; 212 const char kProxyBypassList[] = "bypass_list"; 213 const char kProxyPacUrl[] = "pac_url"; 214 DictionaryPrefUpdate update(prefs_, prefs::kProxy); 215 base::DictionaryValue* dict = update.Get(); 216 dict->SetString(kProxyMode, ProxyModeToString(ProxyPrefs::MODE_SYSTEM)); 217 dict->SetString(kProxyPacUrl, ""); 218 dict->SetString(kProxyServer, ""); 219 dict->SetString(kProxyBypassList, ""); 220} 221#endif // defined(OS_ANDROID) || defined(OS_IOS) 222 223std::string OffTheRecordProfileImpl::GetProfileName() { 224 // Incognito profile should not return the profile name. 225 return std::string(); 226} 227 228Profile::ProfileType OffTheRecordProfileImpl::GetProfileType() const { 229 return INCOGNITO_PROFILE; 230} 231 232base::FilePath OffTheRecordProfileImpl::GetPath() const { 233 return profile_->GetPath(); 234} 235 236scoped_refptr<base::SequencedTaskRunner> 237OffTheRecordProfileImpl::GetIOTaskRunner() { 238 return profile_->GetIOTaskRunner(); 239} 240 241bool OffTheRecordProfileImpl::IsOffTheRecord() const { 242 return true; 243} 244 245Profile* OffTheRecordProfileImpl::GetOffTheRecordProfile() { 246 return this; 247} 248 249void OffTheRecordProfileImpl::DestroyOffTheRecordProfile() { 250 // Suicide is bad! 251 NOTREACHED(); 252} 253 254bool OffTheRecordProfileImpl::HasOffTheRecordProfile() { 255 return true; 256} 257 258Profile* OffTheRecordProfileImpl::GetOriginalProfile() { 259 return profile_; 260} 261 262ExtensionService* OffTheRecordProfileImpl::GetExtensionService() { 263 return extensions::ExtensionSystem::Get(this)->extension_service(); 264} 265 266ExtensionSpecialStoragePolicy* 267 OffTheRecordProfileImpl::GetExtensionSpecialStoragePolicy() { 268 return GetOriginalProfile()->GetExtensionSpecialStoragePolicy(); 269} 270 271bool OffTheRecordProfileImpl::IsSupervised() { 272 return GetOriginalProfile()->IsSupervised(); 273} 274 275PrefService* OffTheRecordProfileImpl::GetPrefs() { 276 return prefs_; 277} 278 279PrefService* OffTheRecordProfileImpl::GetOffTheRecordPrefs() { 280 return prefs_; 281} 282 283DownloadManagerDelegate* OffTheRecordProfileImpl::GetDownloadManagerDelegate() { 284 return DownloadServiceFactory::GetForBrowserContext(this)-> 285 GetDownloadManagerDelegate(); 286} 287 288net::URLRequestContextGetter* OffTheRecordProfileImpl::GetRequestContext() { 289 return GetDefaultStoragePartition(this)->GetURLRequestContext(); 290} 291 292net::URLRequestContextGetter* OffTheRecordProfileImpl::CreateRequestContext( 293 content::ProtocolHandlerMap* protocol_handlers, 294 content::URLRequestInterceptorScopedVector request_interceptors) { 295 return io_data_->CreateMainRequestContextGetter( 296 protocol_handlers, request_interceptors.Pass()).get(); 297} 298 299net::URLRequestContextGetter* 300 OffTheRecordProfileImpl::GetRequestContextForRenderProcess( 301 int renderer_child_id) { 302 content::RenderProcessHost* rph = content::RenderProcessHost::FromID( 303 renderer_child_id); 304 return rph->GetStoragePartition()->GetURLRequestContext(); 305} 306 307net::URLRequestContextGetter* 308 OffTheRecordProfileImpl::GetMediaRequestContext() { 309 // In OTR mode, media request context is the same as the original one. 310 return GetRequestContext(); 311} 312 313net::URLRequestContextGetter* 314 OffTheRecordProfileImpl::GetMediaRequestContextForRenderProcess( 315 int renderer_child_id) { 316 // In OTR mode, media request context is the same as the original one. 317 return GetRequestContextForRenderProcess(renderer_child_id); 318} 319 320net::URLRequestContextGetter* 321OffTheRecordProfileImpl::GetMediaRequestContextForStoragePartition( 322 const base::FilePath& partition_path, 323 bool in_memory) { 324 return io_data_->GetIsolatedAppRequestContextGetter(partition_path, in_memory) 325 .get(); 326} 327 328net::URLRequestContextGetter* 329 OffTheRecordProfileImpl::GetRequestContextForExtensions() { 330 return io_data_->GetExtensionsRequestContextGetter().get(); 331} 332 333net::URLRequestContextGetter* 334OffTheRecordProfileImpl::CreateRequestContextForStoragePartition( 335 const base::FilePath& partition_path, 336 bool in_memory, 337 content::ProtocolHandlerMap* protocol_handlers, 338 content::URLRequestInterceptorScopedVector request_interceptors) { 339 return io_data_->CreateIsolatedAppRequestContextGetter( 340 partition_path, 341 in_memory, 342 protocol_handlers, 343 request_interceptors.Pass()).get(); 344} 345 346content::ResourceContext* OffTheRecordProfileImpl::GetResourceContext() { 347 return io_data_->GetResourceContext(); 348} 349 350net::SSLConfigService* OffTheRecordProfileImpl::GetSSLConfigService() { 351 return profile_->GetSSLConfigService(); 352} 353 354HostContentSettingsMap* OffTheRecordProfileImpl::GetHostContentSettingsMap() { 355 // Retrieve the host content settings map of the parent profile in order to 356 // ensure the preferences have been migrated. 357 profile_->GetHostContentSettingsMap(); 358 if (!host_content_settings_map_.get()) { 359 host_content_settings_map_ = new HostContentSettingsMap(GetPrefs(), true); 360#if defined(ENABLE_EXTENSIONS) 361 ExtensionService* extension_service = GetExtensionService(); 362 if (extension_service) 363 host_content_settings_map_->RegisterExtensionService(extension_service); 364#endif 365 } 366 return host_content_settings_map_.get(); 367} 368 369content::BrowserPluginGuestManager* 370 OffTheRecordProfileImpl::GetGuestManager() { 371#if defined(ENABLE_EXTENSIONS) 372 return GuestViewManager::FromBrowserContext(this); 373#else 374 return NULL; 375#endif 376} 377 378quota::SpecialStoragePolicy* 379 OffTheRecordProfileImpl::GetSpecialStoragePolicy() { 380 return GetExtensionSpecialStoragePolicy(); 381} 382 383content::PushMessagingService* 384OffTheRecordProfileImpl::GetPushMessagingService() { 385 // TODO(johnme): Support push messaging in incognito if possible. 386 return NULL; 387} 388 389bool OffTheRecordProfileImpl::IsSameProfile(Profile* profile) { 390 return (profile == this) || (profile == profile_); 391} 392 393Time OffTheRecordProfileImpl::GetStartTime() const { 394 return start_time_; 395} 396 397history::TopSites* OffTheRecordProfileImpl::GetTopSitesWithoutCreating() { 398 return NULL; 399} 400 401history::TopSites* OffTheRecordProfileImpl::GetTopSites() { 402 return NULL; 403} 404 405void OffTheRecordProfileImpl::SetExitType(ExitType exit_type) { 406} 407 408base::FilePath OffTheRecordProfileImpl::last_selected_directory() { 409 const base::FilePath& directory = last_selected_directory_; 410 if (directory.empty()) { 411 return profile_->last_selected_directory(); 412 } 413 return directory; 414} 415 416void OffTheRecordProfileImpl::set_last_selected_directory( 417 const base::FilePath& path) { 418 last_selected_directory_ = path; 419} 420 421bool OffTheRecordProfileImpl::WasCreatedByVersionOrLater( 422 const std::string& version) { 423 return profile_->WasCreatedByVersionOrLater(version); 424} 425 426Profile::ExitType OffTheRecordProfileImpl::GetLastSessionExitType() { 427 return profile_->GetLastSessionExitType(); 428} 429 430#if defined(OS_CHROMEOS) 431void OffTheRecordProfileImpl::ChangeAppLocale(const std::string& locale, 432 AppLocaleChangedVia) { 433} 434 435void OffTheRecordProfileImpl::OnLogin() { 436} 437 438void OffTheRecordProfileImpl::InitChromeOSPreferences() { 439 // The incognito profile shouldn't have Chrome OS's preferences. 440 // The preferences are associated with the regular user profile. 441} 442#endif // defined(OS_CHROMEOS) 443 444PrefProxyConfigTracker* OffTheRecordProfileImpl::GetProxyConfigTracker() { 445 if (!pref_proxy_config_tracker_) 446 pref_proxy_config_tracker_.reset(CreateProxyConfigTracker()); 447 return pref_proxy_config_tracker_.get(); 448} 449 450chrome_browser_net::Predictor* OffTheRecordProfileImpl::GetNetworkPredictor() { 451 // We do not store information about websites visited in OTR profiles which 452 // is necessary for a Predictor, so we do not have a Predictor at all. 453 return NULL; 454} 455 456DevToolsNetworkController* 457OffTheRecordProfileImpl::GetDevToolsNetworkController() { 458 return io_data_->GetDevToolsNetworkController(); 459} 460 461void OffTheRecordProfileImpl::ClearNetworkingHistorySince( 462 base::Time time, 463 const base::Closure& completion) { 464 // Nothing to do here, our transport security state is read-only. 465 // Still, fire the callback to indicate we have finished, otherwise the 466 // BrowsingDataRemover will never be destroyed and the dialog will never be 467 // closed. We must do this asynchronously in order to avoid reentrancy issues. 468 if (!completion.is_null()) { 469 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, completion); 470 } 471} 472 473void OffTheRecordProfileImpl::ClearDomainReliabilityMonitor( 474 domain_reliability::DomainReliabilityClearMode mode, 475 const base::Closure& completion) { 476 // Incognito profiles don't have Domain Reliability Monitors, so there's 477 // nothing to do here. 478 if (!completion.is_null()) { 479 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, completion); 480 } 481} 482 483GURL OffTheRecordProfileImpl::GetHomePage() { 484 return profile_->GetHomePage(); 485} 486 487#if defined(OS_CHROMEOS) 488// Special case of the OffTheRecordProfileImpl which is used while Guest 489// session in CrOS. 490class GuestSessionProfile : public OffTheRecordProfileImpl { 491 public: 492 explicit GuestSessionProfile(Profile* real_profile) 493 : OffTheRecordProfileImpl(real_profile) { 494 } 495 496 virtual ProfileType GetProfileType() const OVERRIDE { 497 return GUEST_PROFILE; 498 } 499 500 virtual void InitChromeOSPreferences() OVERRIDE { 501 chromeos_preferences_.reset(new chromeos::Preferences()); 502 chromeos_preferences_->Init(static_cast<PrefServiceSyncable*>(GetPrefs()), 503 chromeos::UserManager::Get()->GetActiveUser()); 504 } 505 506 private: 507 // The guest user should be able to customize Chrome OS preferences. 508 scoped_ptr<chromeos::Preferences> chromeos_preferences_; 509}; 510#endif 511 512Profile* Profile::CreateOffTheRecordProfile() { 513 OffTheRecordProfileImpl* profile = NULL; 514#if defined(OS_CHROMEOS) 515 if (IsGuestSession()) 516 profile = new GuestSessionProfile(this); 517#endif 518 if (!profile) 519 profile = new OffTheRecordProfileImpl(this); 520 profile->Init(); 521 return profile; 522} 523 524void OffTheRecordProfileImpl::OnZoomLevelChanged( 525 const HostZoomMap::ZoomLevelChange& change) { 526 HostZoomMap* host_zoom_map = HostZoomMap::GetForBrowserContext(this); 527 switch (change.mode) { 528 case HostZoomMap::ZOOM_CHANGED_TEMPORARY_ZOOM: 529 return; 530 case HostZoomMap::ZOOM_CHANGED_FOR_HOST: 531 host_zoom_map->SetZoomLevelForHost(change.host, change.zoom_level); 532 return; 533 case HostZoomMap::ZOOM_CHANGED_FOR_SCHEME_AND_HOST: 534 host_zoom_map->SetZoomLevelForHostAndScheme(change.scheme, 535 change.host, 536 change.zoom_level); 537 return; 538 } 539} 540 541PrefProxyConfigTracker* OffTheRecordProfileImpl::CreateProxyConfigTracker() { 542#if defined(OS_CHROMEOS) 543 if (chromeos::ProfileHelper::IsSigninProfile(this)) { 544 return ProxyServiceFactory::CreatePrefProxyConfigTrackerOfLocalState( 545 g_browser_process->local_state()); 546 } 547#endif // defined(OS_CHROMEOS) 548 return ProxyServiceFactory::CreatePrefProxyConfigTrackerOfProfile( 549 GetPrefs(), g_browser_process->local_state()); 550} 551