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