// 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 <algorithm>
#include <string>
#include "base/strings/string_util.h"
#include "chrome/browser/history/archived_database.h"
#include "sql/transaction.h"
namespace history {
namespace {
static const int kCurrentVersionNumber = 4;
static const int kCompatibleVersionNumber = 2;
} // namespace
ArchivedDatabase::ArchivedDatabase() {
}
ArchivedDatabase::~ArchivedDatabase() {
}
bool ArchivedDatabase::Init(const base::FilePath& file_name) {
// Set the database page size to something a little larger to give us
// better performance (we're typically seek rather than bandwidth limited).
// This only has an effect before any tables have been created, otherwise
// this is a NOP. Must be a power of 2 and a max of 8192.
db_.set_page_size(4096);
// Don't use very much memory caching this database. We seldom use it for
// anything important.
db_.set_cache_size(64);
// Run the database in exclusive mode. Nobody else should be accessing the
// database while we're running, and this will give somewhat improved perf.
db_.set_exclusive_locking();
if (!db_.Open(file_name))
return false;
if (!InitTables()) {
db_.Close();
return false;
}
return true;
}
bool ArchivedDatabase::InitTables() {
sql::Transaction transaction(&db_);
if (!transaction.Begin())
return false;
// Version check.
if (!meta_table_.Init(&db_, kCurrentVersionNumber,
kCompatibleVersionNumber))
return false;
// Create the tables.
if (!CreateURLTable(false) || !InitVisitTable() ||
!InitKeywordSearchTermsTable())
return false;
CreateMainURLIndex();
CreateKeywordSearchTermsIndices();
if (EnsureCurrentVersion() != sql::INIT_OK)
return false;
return transaction.Commit();
}
void ArchivedDatabase::TrimMemory(bool aggressively) {
db_.TrimMemory(aggressively);
}
void ArchivedDatabase::BeginTransaction() {
db_.BeginTransaction();
}
void ArchivedDatabase::CommitTransaction() {
db_.CommitTransaction();
}
sql::Connection& ArchivedDatabase::GetDB() {
return db_;
}
// static
int ArchivedDatabase::GetCurrentVersion() {
return kCurrentVersionNumber;
}
// Migration -------------------------------------------------------------------
sql::InitStatus ArchivedDatabase::EnsureCurrentVersion() {
// We can't read databases newer than we were designed for.
if (meta_table_.GetCompatibleVersionNumber() > kCurrentVersionNumber) {
LOG(WARNING) << "Archived database is too new.";
return sql::INIT_TOO_NEW;
}
// NOTICE: If you are changing structures for things shared with the archived
// history file like URLs, visits, or downloads, that will need migration as
// well. Instead of putting such migration code in this class, it should be
// in the corresponding file (url_database.cc, etc.) and called from here and
// from the archived_database.cc.
int cur_version = meta_table_.GetVersionNumber();
if (cur_version == 1) {
if (!DropStarredIDFromURLs()) {
LOG(WARNING) << "Unable to update archived database to version 2.";
return sql::INIT_FAILURE;
}
++cur_version;
meta_table_.SetVersionNumber(cur_version);
meta_table_.SetCompatibleVersionNumber(
std::min(cur_version, kCompatibleVersionNumber));
}
if (cur_version == 2) {
// This is the version prior to adding visit_source table.
++cur_version;
meta_table_.SetVersionNumber(cur_version);
}
if (cur_version == 3) {
// This is the version prior to adding the visit_duration field in visits
// database. We need to migrate the database.
if (!MigrateVisitsWithoutDuration()) {
LOG(WARNING) << "Unable to update archived database to version 4.";
return sql::INIT_FAILURE;
}
++cur_version;
meta_table_.SetVersionNumber(cur_version);
}
// Put future migration cases here.
// When the version is too old, we just try to continue anyway, there should
// not be a released product that makes a database too old for us to handle.
LOG_IF(WARNING, cur_version < kCurrentVersionNumber) <<
"Archived database version " << cur_version << " is too old to handle.";
return sql::INIT_OK;
}
} // namespace history