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
23 using content::BrowserThread;
24
25 namespace {
26
27 // TODO(shishir): This should move to a more generic name.
28 const base::FilePath::CharType kPredictorDatabaseName[] =
29 FILE_PATH_LITERAL("Network Action Predictor");
30
31 } // namespace
32
33 namespace 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.
37 class 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
PredictorDatabaseInternal(Profile * profile)69 PredictorDatabaseInternal::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
~PredictorDatabaseInternal()81 PredictorDatabaseInternal::~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
Initialize()87 void 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
SetCancelled()104 void 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
LogDatabaseStats()113 void 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
PredictorDatabase(Profile * profile)129 PredictorDatabase::PredictorDatabase(Profile* profile)
130 : db_(new PredictorDatabaseInternal(profile)) {
131 BrowserThread::PostTask(BrowserThread::DB, FROM_HERE,
132 base::Bind(&PredictorDatabaseInternal::Initialize, db_));
133 }
134
~PredictorDatabase()135 PredictorDatabase::~PredictorDatabase() {
136 }
137
Shutdown()138 void PredictorDatabase::Shutdown() {
139 db_->SetCancelled();
140 }
141
142 scoped_refptr<AutocompleteActionPredictorTable>
autocomplete_table()143 PredictorDatabase::autocomplete_table() {
144 return db_->autocomplete_table_;
145 }
146
147 scoped_refptr<LoggedInPredictorTable>
logged_in_table()148 PredictorDatabase::logged_in_table() {
149 return db_->logged_in_table_;
150 }
151
152 scoped_refptr<ResourcePrefetchPredictorTables>
resource_prefetch_tables()153 PredictorDatabase::resource_prefetch_tables() {
154 return db_->resource_prefetch_tables_;
155 }
156
GetDatabase()157 sql::Connection* PredictorDatabase::GetDatabase() {
158 return db_->db_.get();
159 }
160
161 } // namespace predictors
162