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/predictors/predictor_database.h" 6 7#include "base/bind.h" 8#include "base/file_util.h" 9#include "base/files/file_path.h" 10#include "base/logging.h" 11#include "base/metrics/histogram.h" 12#include "base/strings/stringprintf.h" 13#include "chrome/browser/predictors/autocomplete_action_predictor_table.h" 14#include "chrome/browser/predictors/logged_in_predictor_table.h" 15#include "chrome/browser/predictors/resource_prefetch_predictor.h" 16#include "chrome/browser/predictors/resource_prefetch_predictor_tables.h" 17#include "chrome/browser/prerender/prerender_field_trial.h" 18#include "chrome/browser/profiles/profile.h" 19#include "content/public/browser/browser_thread.h" 20#include "sql/connection.h" 21#include "sql/statement.h" 22 23using content::BrowserThread; 24 25namespace { 26 27// TODO(shishir): This should move to a more generic name. 28const base::FilePath::CharType kPredictorDatabaseName[] = 29 FILE_PATH_LITERAL("Network Action Predictor"); 30 31} // namespace 32 33namespace predictors { 34 35// Refcounted as it is created, initialized and destroyed on a different thread 36// to the DB thread that is required for all methods performing database access. 37class PredictorDatabaseInternal 38 : public base::RefCountedThreadSafe<PredictorDatabaseInternal> { 39 private: 40 friend class base::RefCountedThreadSafe<PredictorDatabaseInternal>; 41 friend class PredictorDatabase; 42 43 explicit PredictorDatabaseInternal(Profile* profile); 44 virtual ~PredictorDatabaseInternal(); 45 46 // Opens the database file from the profile path. Separated from the 47 // constructor to ease construction/destruction of this object on one thread 48 // but database access on the DB thread. 49 void Initialize(); 50 void LogDatabaseStats(); // DB Thread. 51 52 // Cancels pending DB transactions. Should only be called on the UI thread. 53 void SetCancelled(); 54 55 bool is_resource_prefetch_predictor_enabled_; 56 base::FilePath db_path_; 57 scoped_ptr<sql::Connection> db_; 58 59 // TODO(shishir): These tables may not need to be refcounted. Maybe move them 60 // to using a WeakPtr instead. 61 scoped_refptr<AutocompleteActionPredictorTable> autocomplete_table_; 62 scoped_refptr<LoggedInPredictorTable> logged_in_table_; 63 scoped_refptr<ResourcePrefetchPredictorTables> resource_prefetch_tables_; 64 65 DISALLOW_COPY_AND_ASSIGN(PredictorDatabaseInternal); 66}; 67 68 69PredictorDatabaseInternal::PredictorDatabaseInternal(Profile* profile) 70 : db_path_(profile->GetPath().Append(kPredictorDatabaseName)), 71 db_(new sql::Connection()), 72 autocomplete_table_(new AutocompleteActionPredictorTable()), 73 logged_in_table_(new LoggedInPredictorTable()), 74 resource_prefetch_tables_(new ResourcePrefetchPredictorTables()) { 75 db_->set_histogram_tag("Predictor"); 76 ResourcePrefetchPredictorConfig config; 77 is_resource_prefetch_predictor_enabled_ = 78 IsSpeculativeResourcePrefetchingEnabled(profile, &config); 79} 80 81PredictorDatabaseInternal::~PredictorDatabaseInternal() { 82 // The connection pointer needs to be deleted on the DB thread since there 83 // might be a task in progress on the DB thread which uses this connection. 84 BrowserThread::DeleteSoon(BrowserThread::DB, FROM_HERE, db_.release()); 85} 86 87void PredictorDatabaseInternal::Initialize() { 88 CHECK(BrowserThread::CurrentlyOn(BrowserThread::DB) || 89 !BrowserThread::IsMessageLoopValid(BrowserThread::DB)); 90 // TODO(tburkard): figure out if we need this. 91 // db_->set_exclusive_locking(); 92 bool success = db_->Open(db_path_); 93 94 if (!success) 95 return; 96 97 autocomplete_table_->Initialize(db_.get()); 98 logged_in_table_->Initialize(db_.get()); 99 resource_prefetch_tables_->Initialize(db_.get()); 100 101 LogDatabaseStats(); 102} 103 104void PredictorDatabaseInternal::SetCancelled() { 105 CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI) || 106 !BrowserThread::IsMessageLoopValid(BrowserThread::UI)); 107 108 autocomplete_table_->SetCancelled(); 109 logged_in_table_->SetCancelled(); 110 resource_prefetch_tables_->SetCancelled(); 111} 112 113void PredictorDatabaseInternal::LogDatabaseStats() { 114 CHECK(BrowserThread::CurrentlyOn(BrowserThread::DB) || 115 !BrowserThread::IsMessageLoopValid(BrowserThread::DB)); 116 117 int64 db_size; 118 bool success = base::GetFileSize(db_path_, &db_size); 119 DCHECK(success) << "Failed to get file size for " << db_path_.value(); 120 UMA_HISTOGRAM_MEMORY_KB("PredictorDatabase.DatabaseSizeKB", 121 static_cast<int>(db_size / 1024)); 122 123 autocomplete_table_->LogDatabaseStats(); 124 logged_in_table_->LogDatabaseStats(); 125 if (is_resource_prefetch_predictor_enabled_) 126 resource_prefetch_tables_->LogDatabaseStats(); 127} 128 129PredictorDatabase::PredictorDatabase(Profile* profile) 130 : db_(new PredictorDatabaseInternal(profile)) { 131 BrowserThread::PostTask(BrowserThread::DB, FROM_HERE, 132 base::Bind(&PredictorDatabaseInternal::Initialize, db_)); 133} 134 135PredictorDatabase::~PredictorDatabase() { 136} 137 138void PredictorDatabase::Shutdown() { 139 db_->SetCancelled(); 140} 141 142scoped_refptr<AutocompleteActionPredictorTable> 143 PredictorDatabase::autocomplete_table() { 144 return db_->autocomplete_table_; 145} 146 147scoped_refptr<LoggedInPredictorTable> 148 PredictorDatabase::logged_in_table() { 149 return db_->logged_in_table_; 150} 151 152scoped_refptr<ResourcePrefetchPredictorTables> 153 PredictorDatabase::resource_prefetch_tables() { 154 return db_->resource_prefetch_tables_; 155} 156 157sql::Connection* PredictorDatabase::GetDatabase() { 158 return db_->db_.get(); 159} 160 161} // namespace predictors 162