blob: 18387137f92813bee00dcc228121064a397e9bb6 [file] [log] [blame]
// Copyright (c) 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.
#include "chrome/browser/webdata/web_data_service_factory.h"
#include "base/bind.h"
#include "base/files/file_path.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/profiles/incognito_helpers.h"
#include "chrome/browser/sync/glue/sync_start_util.h"
#include "chrome/browser/ui/profile_error_dialog.h"
#include "chrome/browser/webdata/autocomplete_syncable_service.h"
#include "chrome/browser/webdata/autofill_profile_syncable_service.h"
#include "chrome/browser/webdata/keyword_table.h"
#include "chrome/browser/webdata/logins_table.h"
#include "chrome/browser/webdata/token_service_table.h"
#include "chrome/browser/webdata/token_web_data.h"
#include "chrome/browser/webdata/web_apps_table.h"
#include "chrome/browser/webdata/web_data_service.h"
#include "chrome/browser/webdata/web_intents_table.h"
#include "components/autofill/core/browser/autofill_country.h"
#include "components/autofill/core/browser/webdata/autofill_table.h"
#include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
#include "components/browser_context_keyed_service/browser_context_dependency_manager.h"
#include "components/webdata/common/webdata_constants.h"
#include "content/public/browser/browser_thread.h"
#include "grit/chromium_strings.h"
#include "grit/generated_resources.h"
using autofill::AutofillWebDataService;
using content::BrowserThread;
namespace {
// Callback to show error dialog on profile load error.
void ProfileErrorCallback(sql::InitStatus status) {
ShowProfileErrorDialog(
(status == sql::INIT_FAILURE) ?
IDS_COULDNT_OPEN_PROFILE_ERROR : IDS_PROFILE_TOO_NEW_ERROR);
}
void InitSyncableServicesOnDBThread(
scoped_refptr<AutofillWebDataService> autofill_web_data,
const base::FilePath& profile_path,
const std::string& app_locale,
autofill::AutofillWebDataBackend* autofill_backend) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB));
// Currently only Autocomplete and Autofill profiles use the new Sync API, but
// all the database data should migrate to this API over time.
AutocompleteSyncableService::CreateForWebDataServiceAndBackend(
autofill_web_data.get(), autofill_backend);
AutocompleteSyncableService::FromWebDataService(autofill_web_data.get())
->InjectStartSyncFlare(
sync_start_util::GetFlareForSyncableService(profile_path));
AutofillProfileSyncableService::CreateForWebDataServiceAndBackend(
autofill_web_data.get(), autofill_backend, app_locale);
AutofillProfileSyncableService::FromWebDataService(autofill_web_data.get())
->InjectStartSyncFlare(
sync_start_util::GetFlareForSyncableService(profile_path));
}
} // namespace
WebDataServiceWrapper::WebDataServiceWrapper() {}
WebDataServiceWrapper::WebDataServiceWrapper(Profile* profile) {
base::FilePath profile_path = profile->GetPath();
base::FilePath path = profile_path.Append(kWebDataFilename);
web_database_ = new WebDatabaseService(path,
BrowserThread::GetMessageLoopProxyForThread(BrowserThread::UI),
BrowserThread::GetMessageLoopProxyForThread(BrowserThread::DB));
// All tables objects that participate in managing the database must
// be added here.
web_database_->AddTable(
scoped_ptr<WebDatabaseTable>(new autofill::AutofillTable(
g_browser_process->GetApplicationLocale())));
web_database_->AddTable(
scoped_ptr<WebDatabaseTable>(new KeywordTable()));
// TODO(mdm): We only really need the LoginsTable on Windows for IE7 password
// access, but for now, we still create it on all platforms since it deletes
// the old logins table. We can remove this after a while, e.g. in M22 or so.
web_database_->AddTable(
scoped_ptr<WebDatabaseTable>(new LoginsTable()));
web_database_->AddTable(
scoped_ptr<WebDatabaseTable>(new TokenServiceTable()));
web_database_->AddTable(
scoped_ptr<WebDatabaseTable>(new WebAppsTable()));
// TODO(thakis): Add a migration to delete the SQL table used by
// WebIntentsTable, then remove this.
web_database_->AddTable(
scoped_ptr<WebDatabaseTable>(new WebIntentsTable()));
web_database_->LoadDatabase();
autofill_web_data_ = new AutofillWebDataService(
web_database_, base::Bind(&ProfileErrorCallback));
autofill_web_data_->Init();
token_web_data_ = new TokenWebData(
web_database_, base::Bind(&ProfileErrorCallback));
token_web_data_->Init();
web_data_ = new WebDataService(
web_database_, base::Bind(&ProfileErrorCallback));
web_data_->Init();
autofill_web_data_->GetAutofillBackend(
base::Bind(&InitSyncableServicesOnDBThread,
autofill_web_data_,
profile_path,
g_browser_process->GetApplicationLocale()));
}
WebDataServiceWrapper::~WebDataServiceWrapper() {
}
void WebDataServiceWrapper::Shutdown() {
autofill_web_data_->ShutdownOnUIThread();
token_web_data_->ShutdownOnUIThread();
web_data_->ShutdownOnUIThread();
web_database_->ShutdownDatabase();
}
scoped_refptr<AutofillWebDataService>
WebDataServiceWrapper::GetAutofillWebData() {
return autofill_web_data_.get();
}
scoped_refptr<WebDataService> WebDataServiceWrapper::GetWebData() {
return web_data_.get();
}
scoped_refptr<TokenWebData> WebDataServiceWrapper::GetTokenWebData() {
return token_web_data_.get();
}
// static
scoped_refptr<AutofillWebDataService>
AutofillWebDataService::FromBrowserContext(content::BrowserContext* context) {
// For this service, the implicit/explicit distinction doesn't
// really matter; it's just used for a DCHECK. So we currently
// cheat and always say EXPLICIT_ACCESS.
WebDataServiceWrapper* wrapper =
WebDataServiceFactory::GetForProfile(
static_cast<Profile*>(context), Profile::EXPLICIT_ACCESS);
if (wrapper)
return wrapper->GetAutofillWebData();
// |wrapper| can be NULL in Incognito mode.
return scoped_refptr<AutofillWebDataService>(NULL);
}
// static
scoped_refptr<TokenWebData> TokenWebData::FromBrowserContext(
content::BrowserContext* context) {
// For this service, the implicit/explicit distinction doesn't
// really matter; it's just used for a DCHECK. So we currently
// cheat and always say EXPLICIT_ACCESS.
WebDataServiceWrapper* wrapper =
WebDataServiceFactory::GetForProfile(
static_cast<Profile*>(context), Profile::EXPLICIT_ACCESS);
if (wrapper)
return wrapper->GetTokenWebData();
// |wrapper| can be NULL in Incognito mode.
return scoped_refptr<TokenWebData>(NULL);
}
// static
scoped_refptr<WebDataService> WebDataService::FromBrowserContext(
content::BrowserContext* context) {
// For this service, the implicit/explicit distinction doesn't
// really matter; it's just used for a DCHECK. So we currently
// cheat and always say EXPLICIT_ACCESS.
WebDataServiceWrapper* wrapper =
WebDataServiceFactory::GetForProfile(
static_cast<Profile*>(context), Profile::EXPLICIT_ACCESS);
if (wrapper)
return wrapper->GetWebData();
// |wrapper| can be NULL in Incognito mode.
return scoped_refptr<WebDataService>(NULL);
}
WebDataServiceFactory::WebDataServiceFactory()
: BrowserContextKeyedServiceFactory(
"WebDataService",
BrowserContextDependencyManager::GetInstance()) {
// WebDataServiceFactory has no dependecies.
}
WebDataServiceFactory::~WebDataServiceFactory() {}
// static
WebDataServiceWrapper* WebDataServiceFactory::GetForProfile(
Profile* profile, Profile::ServiceAccessType access_type) {
// If |access_type| starts being used for anything other than this
// DCHECK, we need to start taking it as a parameter to
// AutofillWebDataService::FromBrowserContext (see above).
DCHECK(access_type != Profile::IMPLICIT_ACCESS || !profile->IsOffTheRecord());
return static_cast<WebDataServiceWrapper*>(
GetInstance()->GetServiceForBrowserContext(profile, true));
}
// static
WebDataServiceWrapper* WebDataServiceFactory::GetForProfileIfExists(
Profile* profile, Profile::ServiceAccessType access_type) {
// If |access_type| starts being used for anything other than this
// DCHECK, we need to start taking it as a parameter to
// AutofillWebDataService::FromBrowserContext (see above).
DCHECK(access_type != Profile::IMPLICIT_ACCESS || !profile->IsOffTheRecord());
return static_cast<WebDataServiceWrapper*>(
GetInstance()->GetServiceForBrowserContext(profile, false));
}
// static
WebDataServiceFactory* WebDataServiceFactory::GetInstance() {
return Singleton<WebDataServiceFactory>::get();
}
content::BrowserContext* WebDataServiceFactory::GetBrowserContextToUse(
content::BrowserContext* context) const {
return chrome::GetBrowserContextRedirectedInIncognito(context);
}
BrowserContextKeyedService* WebDataServiceFactory::BuildServiceInstanceFor(
content::BrowserContext* profile) const {
return new WebDataServiceWrapper(static_cast<Profile*>(profile));
}
bool WebDataServiceFactory::ServiceIsNULLWhileTesting() const {
return true;
}