| OLD | NEW |
| (Empty) |
| 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 "content/common_child/web_database_observer_impl.h" | |
| 6 | |
| 7 #include "base/metrics/histogram.h" | |
| 8 #include "base/string16.h" | |
| 9 #include "content/common/database_messages.h" | |
| 10 #include "third_party/WebKit/public/platform/WebString.h" | |
| 11 #include "third_party/WebKit/Source/WebKit/chromium/public/WebDatabase.h" | |
| 12 #include "third_party/sqlite/sqlite3.h" | |
| 13 | |
| 14 using WebKit::WebDatabase; | |
| 15 | |
| 16 namespace content { | |
| 17 namespace { | |
| 18 | |
| 19 const int kResultHistogramSize = 50; | |
| 20 const int kCallsiteHistogramSize = 10; | |
| 21 | |
| 22 int DetermineHistogramResult(int websql_error, int sqlite_error) { | |
| 23 // If we have a sqlite error, log it after trimming the extended bits. | |
| 24 // There are 26 possible values, but we leave room for some new ones. | |
| 25 if (sqlite_error) | |
| 26 return std::min(sqlite_error & 0xff, 30); | |
| 27 | |
| 28 // Otherwise, websql_error may be an SQLExceptionCode, SQLErrorCode | |
| 29 // or a DOMExceptionCode, or -1 for success. | |
| 30 if (websql_error == -1) | |
| 31 return 0; // no error | |
| 32 | |
| 33 // SQLExceptionCode starts at 1000 | |
| 34 if (websql_error >= 1000) | |
| 35 websql_error -= 1000; | |
| 36 | |
| 37 return std::min(websql_error + 30, kResultHistogramSize - 1); | |
| 38 } | |
| 39 | |
| 40 #define HISTOGRAM_WEBSQL_RESULT(name, database, callsite, \ | |
| 41 websql_error, sqlite_error) \ | |
| 42 do { \ | |
| 43 DCHECK(callsite < kCallsiteHistogramSize); \ | |
| 44 int result = DetermineHistogramResult(websql_error, sqlite_error); \ | |
| 45 if (database.isSyncDatabase()) { \ | |
| 46 UMA_HISTOGRAM_ENUMERATION("websql.Sync." name, \ | |
| 47 result, kResultHistogramSize); \ | |
| 48 if (result) { \ | |
| 49 UMA_HISTOGRAM_ENUMERATION("websql.Sync." name ".ErrorSite", \ | |
| 50 callsite, kCallsiteHistogramSize); \ | |
| 51 } \ | |
| 52 } else { \ | |
| 53 UMA_HISTOGRAM_ENUMERATION("websql.Async." name, \ | |
| 54 result, kResultHistogramSize); \ | |
| 55 if (result) { \ | |
| 56 UMA_HISTOGRAM_ENUMERATION("websql.Async." name ".ErrorSite", \ | |
| 57 callsite, kCallsiteHistogramSize); \ | |
| 58 } \ | |
| 59 } \ | |
| 60 } while (0) | |
| 61 | |
| 62 } // namespace | |
| 63 | |
| 64 WebDatabaseObserverImpl::WebDatabaseObserverImpl( | |
| 65 IPC::SyncMessageFilter* sender) | |
| 66 : sender_(sender), | |
| 67 open_connections_(new webkit_database::DatabaseConnectionsWrapper) { | |
| 68 DCHECK(sender); | |
| 69 } | |
| 70 | |
| 71 WebDatabaseObserverImpl::~WebDatabaseObserverImpl() { | |
| 72 } | |
| 73 | |
| 74 void WebDatabaseObserverImpl::databaseOpened( | |
| 75 const WebDatabase& database) { | |
| 76 string16 origin_identifier = database.securityOrigin().databaseIdentifier(); | |
| 77 string16 database_name = database.name(); | |
| 78 open_connections_->AddOpenConnection(origin_identifier, database_name); | |
| 79 sender_->Send(new DatabaseHostMsg_Opened( | |
| 80 origin_identifier, database_name, | |
| 81 database.displayName(), database.estimatedSize())); | |
| 82 } | |
| 83 | |
| 84 void WebDatabaseObserverImpl::databaseModified( | |
| 85 const WebDatabase& database) { | |
| 86 sender_->Send(new DatabaseHostMsg_Modified( | |
| 87 database.securityOrigin().databaseIdentifier(), database.name())); | |
| 88 } | |
| 89 | |
| 90 void WebDatabaseObserverImpl::databaseClosed( | |
| 91 const WebDatabase& database) { | |
| 92 string16 origin_identifier = database.securityOrigin().databaseIdentifier(); | |
| 93 string16 database_name = database.name(); | |
| 94 sender_->Send(new DatabaseHostMsg_Closed( | |
| 95 origin_identifier, database_name)); | |
| 96 open_connections_->RemoveOpenConnection(origin_identifier, database_name); | |
| 97 } | |
| 98 | |
| 99 void WebDatabaseObserverImpl::reportOpenDatabaseResult( | |
| 100 const WebDatabase& database, int callsite, int websql_error, | |
| 101 int sqlite_error) { | |
| 102 HISTOGRAM_WEBSQL_RESULT("OpenResult", database, callsite, | |
| 103 websql_error, sqlite_error); | |
| 104 HandleSqliteError(database, sqlite_error); | |
| 105 } | |
| 106 | |
| 107 void WebDatabaseObserverImpl::reportChangeVersionResult( | |
| 108 const WebDatabase& database, int callsite, int websql_error, | |
| 109 int sqlite_error) { | |
| 110 HISTOGRAM_WEBSQL_RESULT("ChangeVersionResult", database, callsite, | |
| 111 websql_error, sqlite_error); | |
| 112 HandleSqliteError(database, sqlite_error); | |
| 113 } | |
| 114 | |
| 115 void WebDatabaseObserverImpl::reportStartTransactionResult( | |
| 116 const WebDatabase& database, int callsite, int websql_error, | |
| 117 int sqlite_error) { | |
| 118 HISTOGRAM_WEBSQL_RESULT("BeginResult", database, callsite, | |
| 119 websql_error, sqlite_error); | |
| 120 HandleSqliteError(database, sqlite_error); | |
| 121 } | |
| 122 | |
| 123 void WebDatabaseObserverImpl::reportCommitTransactionResult( | |
| 124 const WebDatabase& database, int callsite, int websql_error, | |
| 125 int sqlite_error) { | |
| 126 HISTOGRAM_WEBSQL_RESULT("CommitResult", database, callsite, | |
| 127 websql_error, sqlite_error); | |
| 128 HandleSqliteError(database, sqlite_error); | |
| 129 } | |
| 130 | |
| 131 void WebDatabaseObserverImpl::reportExecuteStatementResult( | |
| 132 const WebDatabase& database, int callsite, int websql_error, | |
| 133 int sqlite_error) { | |
| 134 HISTOGRAM_WEBSQL_RESULT("StatementResult", database, callsite, | |
| 135 websql_error, sqlite_error); | |
| 136 HandleSqliteError(database, sqlite_error); | |
| 137 } | |
| 138 | |
| 139 void WebDatabaseObserverImpl::reportVacuumDatabaseResult( | |
| 140 const WebDatabase& database, int sqlite_error) { | |
| 141 int result = DetermineHistogramResult(-1, sqlite_error); | |
| 142 if (database.isSyncDatabase()) { | |
| 143 UMA_HISTOGRAM_ENUMERATION("websql.Sync.VacuumResult", | |
| 144 result, kResultHistogramSize); | |
| 145 } else { | |
| 146 UMA_HISTOGRAM_ENUMERATION("websql.Async.VacuumResult", | |
| 147 result, kResultHistogramSize); | |
| 148 } | |
| 149 HandleSqliteError(database, sqlite_error); | |
| 150 } | |
| 151 | |
| 152 void WebDatabaseObserverImpl::WaitForAllDatabasesToClose() { | |
| 153 open_connections_->WaitForAllDatabasesToClose(); | |
| 154 } | |
| 155 | |
| 156 void WebDatabaseObserverImpl::HandleSqliteError( | |
| 157 const WebDatabase& database, int error) { | |
| 158 // We filter out errors which the backend doesn't act on to avoid | |
| 159 // a unnecessary ipc traffic, this method can get called at a fairly | |
| 160 // high frequency (per-sqlstatement). | |
| 161 if (error == SQLITE_CORRUPT || error == SQLITE_NOTADB) { | |
| 162 sender_->Send(new DatabaseHostMsg_HandleSqliteError( | |
| 163 database.securityOrigin().databaseIdentifier(), | |
| 164 database.name(), | |
| 165 error)); | |
| 166 } | |
| 167 } | |
| 168 | |
| 169 } // namespace content | |
| OLD | NEW |