1 // Copyright 2013 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.
5 #include "content/browser/media/webrtc_identity_store_backend.h"
7 #include "base/file_util.h"
8 #include "base/files/file_path.h"
9 #include "base/memory/scoped_vector.h"
10 #include "base/strings/string_util.h"
11 #include "content/public/browser/browser_thread.h"
12 #include "net/base/net_errors.h"
13 #include "sql/error_delegate_util.h"
14 #include "sql/statement.h"
15 #include "sql/transaction.h"
17 #include "webkit/browser/quota/special_storage_policy.h"
21 static const char* kWebRTCIdentityStoreDBName
= "webrtc_identity_store";
23 static const base::FilePath::CharType kWebRTCIdentityStoreDirectory
[] =
24 FILE_PATH_LITERAL("WebRTCIdentityStore");
26 // Initializes the identity table, returning true on success.
27 static bool InitDB(sql::Connection
* db
) {
28 if (db
->DoesTableExist(kWebRTCIdentityStoreDBName
)) {
29 if (db
->DoesColumnExist(kWebRTCIdentityStoreDBName
, "origin") &&
30 db
->DoesColumnExist(kWebRTCIdentityStoreDBName
, "identity_name") &&
31 db
->DoesColumnExist(kWebRTCIdentityStoreDBName
, "common_name") &&
32 db
->DoesColumnExist(kWebRTCIdentityStoreDBName
, "certificate") &&
33 db
->DoesColumnExist(kWebRTCIdentityStoreDBName
, "private_key") &&
34 db
->DoesColumnExist(kWebRTCIdentityStoreDBName
, "creation_time"))
37 if (!db
->Execute("DROP TABLE webrtc_identity_store"))
42 "CREATE TABLE webrtc_identity_store"
44 "origin TEXT NOT NULL,"
45 "identity_name TEXT NOT NULL,"
46 "common_name TEXT NOT NULL,"
47 "certificate BLOB NOT NULL,"
48 "private_key BLOB NOT NULL,"
49 "creation_time INTEGER)");
52 struct WebRTCIdentityStoreBackend::IdentityKey
{
53 IdentityKey(const GURL
& origin
, const std::string
& identity_name
)
54 : origin(origin
), identity_name(identity_name
) {}
56 bool operator<(const IdentityKey
& other
) const {
57 return origin
< other
.origin
||
58 (origin
== other
.origin
&& identity_name
< other
.identity_name
);
62 std::string identity_name
;
65 struct WebRTCIdentityStoreBackend::Identity
{
66 Identity(const std::string
& common_name
,
67 const std::string
& certificate
,
68 const std::string
& private_key
)
69 : common_name(common_name
),
70 certificate(certificate
),
71 private_key(private_key
),
72 creation_time(base::Time::Now().ToInternalValue()) {}
74 Identity(const std::string
& common_name
,
75 const std::string
& certificate
,
76 const std::string
& private_key
,
78 : common_name(common_name
),
79 certificate(certificate
),
80 private_key(private_key
),
81 creation_time(creation_time
) {}
83 std::string common_name
;
84 std::string certificate
;
85 std::string private_key
;
89 struct WebRTCIdentityStoreBackend::PendingFindRequest
{
90 PendingFindRequest(const GURL
& origin
,
91 const std::string
& identity_name
,
92 const std::string
& common_name
,
93 const FindIdentityCallback
& callback
)
95 identity_name(identity_name
),
96 common_name(common_name
),
99 ~PendingFindRequest() {}
102 std::string identity_name
;
103 std::string common_name
;
104 FindIdentityCallback callback
;
107 // The class encapsulates the database operations. All members except ctor and
108 // dtor should be accessed on the DB thread.
109 // It can be created/destroyed on any thread.
110 class WebRTCIdentityStoreBackend::SqlLiteStorage
111 : public base::RefCountedThreadSafe
<SqlLiteStorage
> {
113 SqlLiteStorage(base::TimeDelta validity_period
,
114 const base::FilePath
& path
,
115 quota::SpecialStoragePolicy
* policy
)
116 : validity_period_(validity_period
), special_storage_policy_(policy
) {
118 path_
= path
.Append(kWebRTCIdentityStoreDirectory
);
121 void Load(IdentityMap
* out_map
);
123 void AddIdentity(const GURL
& origin
,
124 const std::string
& identity_name
,
125 const Identity
& identity
);
126 void DeleteIdentity(const GURL
& origin
,
127 const std::string
& identity_name
,
128 const Identity
& identity
);
129 void DeleteBetween(base::Time delete_begin
, base::Time delete_end
);
131 void SetValidityPeriodForTesting(base::TimeDelta validity_period
) {
132 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
134 validity_period_
= validity_period
;
138 friend class base::RefCountedThreadSafe
<SqlLiteStorage
>;
144 struct PendingOperation
{
145 PendingOperation(OperationType type
,
147 const std::string
& identity_name
,
148 const Identity
& identity
)
151 identity_name(identity_name
),
152 identity(identity
) {}
156 std::string identity_name
;
159 typedef ScopedVector
<PendingOperation
> PendingOperationList
;
161 virtual ~SqlLiteStorage() {}
162 void OnDatabaseError(int error
, sql::Statement
* stmt
);
163 void BatchOperation(OperationType type
,
165 const std::string
& identity_name
,
166 const Identity
& identity
);
169 base::TimeDelta validity_period_
;
170 // The file path of the DB. Empty if temporary.
171 base::FilePath path_
;
172 scoped_refptr
<quota::SpecialStoragePolicy
> special_storage_policy_
;
173 scoped_ptr
<sql::Connection
> db_
;
174 // Batched DB operations pending to commit.
175 PendingOperationList pending_operations_
;
177 DISALLOW_COPY_AND_ASSIGN(SqlLiteStorage
);
180 WebRTCIdentityStoreBackend::WebRTCIdentityStoreBackend(
181 const base::FilePath
& path
,
182 quota::SpecialStoragePolicy
* policy
,
183 base::TimeDelta validity_period
)
184 : validity_period_(validity_period
),
186 sql_lite_storage_(new SqlLiteStorage(validity_period
, path
, policy
)) {}
188 bool WebRTCIdentityStoreBackend::FindIdentity(
190 const std::string
& identity_name
,
191 const std::string
& common_name
,
192 const FindIdentityCallback
& callback
) {
193 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO
));
194 if (state_
== CLOSED
)
197 if (state_
!= LOADED
) {
198 // Queues the request to wait for the DB to load.
199 pending_find_requests_
.push_back(
200 new PendingFindRequest(origin
, identity_name
, common_name
, callback
));
201 if (state_
== LOADING
)
204 DCHECK_EQ(state_
, NOT_STARTED
);
206 // Kick off loading the DB.
207 scoped_ptr
<IdentityMap
> out_map(new IdentityMap());
209 base::Bind(&SqlLiteStorage::Load
, sql_lite_storage_
, out_map
.get()));
210 // |out_map| will be NULL after this call.
211 if (BrowserThread::PostTaskAndReply(
215 base::Bind(&WebRTCIdentityStoreBackend::OnLoaded
,
217 base::Passed(&out_map
)))) {
221 // If it fails to post task, falls back to ERR_FILE_NOT_FOUND.
224 IdentityKey
key(origin
, identity_name
);
225 IdentityMap::iterator iter
= identities_
.find(key
);
226 if (iter
!= identities_
.end() && iter
->second
.common_name
== common_name
) {
227 base::TimeDelta age
= base::Time::Now() - base::Time::FromInternalValue(
228 iter
->second
.creation_time
);
229 if (age
< validity_period_
) {
231 return BrowserThread::PostTask(BrowserThread::IO
,
235 iter
->second
.certificate
,
236 iter
->second
.private_key
));
238 // Removes the expired identity from the in-memory cache. The copy in the
239 // database will be removed on the next load.
240 identities_
.erase(iter
);
243 return BrowserThread::PostTask(
246 base::Bind(callback
, net::ERR_FILE_NOT_FOUND
, "", ""));
249 void WebRTCIdentityStoreBackend::AddIdentity(const GURL
& origin
,
250 const std::string
& identity_name
,
251 const std::string
& common_name
,
252 const std::string
& certificate
,
253 const std::string
& private_key
) {
254 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO
));
255 if (state_
== CLOSED
)
258 // If there is an existing identity for the same origin and identity_name,
260 IdentityKey
key(origin
, identity_name
);
261 Identity
identity(common_name
, certificate
, private_key
);
263 if (identities_
.find(key
) != identities_
.end()) {
264 if (!BrowserThread::PostTask(BrowserThread::DB
,
266 base::Bind(&SqlLiteStorage::DeleteIdentity
,
270 identities_
.find(key
)->second
)))
273 identities_
.insert(std::pair
<IdentityKey
, Identity
>(key
, identity
));
275 BrowserThread::PostTask(BrowserThread::DB
,
277 base::Bind(&SqlLiteStorage::AddIdentity
,
284 void WebRTCIdentityStoreBackend::Close() {
285 if (!BrowserThread::CurrentlyOn(BrowserThread::IO
)) {
286 BrowserThread::PostTask(
289 base::Bind(&WebRTCIdentityStoreBackend::Close
, this));
293 if (state_
== CLOSED
)
297 BrowserThread::PostTask(
300 base::Bind(&SqlLiteStorage::Close
, sql_lite_storage_
));
303 void WebRTCIdentityStoreBackend::DeleteBetween(base::Time delete_begin
,
304 base::Time delete_end
,
305 const base::Closure
& callback
) {
306 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO
));
307 if (state_
== CLOSED
)
310 // Delete the in-memory cache.
311 IdentityMap::iterator it
= identities_
.begin();
312 while (it
!= identities_
.end()) {
313 if (it
->second
.creation_time
>= delete_begin
.ToInternalValue() &&
314 it
->second
.creation_time
<= delete_end
.ToInternalValue()) {
315 identities_
.erase(it
++);
320 BrowserThread::PostTaskAndReply(BrowserThread::DB
,
322 base::Bind(&SqlLiteStorage::DeleteBetween
,
329 void WebRTCIdentityStoreBackend::SetValidityPeriodForTesting(
330 base::TimeDelta validity_period
) {
331 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO
));
332 validity_period_
= validity_period
;
333 BrowserThread::PostTask(
336 base::Bind(&SqlLiteStorage::SetValidityPeriodForTesting
,
341 WebRTCIdentityStoreBackend::~WebRTCIdentityStoreBackend() {}
343 void WebRTCIdentityStoreBackend::OnLoaded(scoped_ptr
<IdentityMap
> out_map
) {
344 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO
));
346 if (state_
!= LOADING
)
349 DVLOG(3) << "WebRTC identity store has loaded.";
352 identities_
.swap(*out_map
);
354 for (size_t i
= 0; i
< pending_find_requests_
.size(); ++i
) {
355 FindIdentity(pending_find_requests_
[i
]->origin
,
356 pending_find_requests_
[i
]->identity_name
,
357 pending_find_requests_
[i
]->common_name
,
358 pending_find_requests_
[i
]->callback
);
359 delete pending_find_requests_
[i
];
361 pending_find_requests_
.clear();
365 // Implementation of SqlLiteStorage.
368 void WebRTCIdentityStoreBackend::SqlLiteStorage::Load(IdentityMap
* out_map
) {
369 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
372 // Ensure the parent directory for storing certs is created before reading
374 const base::FilePath dir
= path_
.DirName();
375 if (!base::PathExists(dir
) && !base::CreateDirectory(dir
)) {
376 DVLOG(2) << "Unable to open DB file path.";
380 db_
.reset(new sql::Connection());
382 db_
->set_error_callback(base::Bind(&SqlLiteStorage::OnDatabaseError
, this));
384 if (!db_
->Open(path_
)) {
385 DVLOG(2) << "Unable to open DB.";
390 if (!InitDB(db_
.get())) {
391 DVLOG(2) << "Unable to init DB.";
398 // Delete expired identities.
399 DeleteBetween(base::Time(), base::Time::Now() - validity_period_
);
401 // Slurp all the identities into the out_map.
402 sql::Statement
stmt(db_
->GetUniqueStatement(
403 "SELECT origin, identity_name, common_name, "
404 "certificate, private_key, creation_time "
405 "FROM webrtc_identity_store"));
406 CHECK(stmt
.is_valid());
408 while (stmt
.Step()) {
409 IdentityKey
key(GURL(stmt
.ColumnString(0)), stmt
.ColumnString(1));
410 std::string
common_name(stmt
.ColumnString(2));
411 std::string cert
, private_key
;
412 stmt
.ColumnBlobAsString(3, &cert
);
413 stmt
.ColumnBlobAsString(4, &private_key
);
414 int64 creation_time
= stmt
.ColumnInt64(5);
415 std::pair
<IdentityMap::iterator
, bool> result
=
416 out_map
->insert(std::pair
<IdentityKey
, Identity
>(
417 key
, Identity(common_name
, cert
, private_key
, creation_time
)));
418 DCHECK(result
.second
);
422 void WebRTCIdentityStoreBackend::SqlLiteStorage::Close() {
423 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
428 void WebRTCIdentityStoreBackend::SqlLiteStorage::AddIdentity(
430 const std::string
& identity_name
,
431 const Identity
& identity
) {
432 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
436 // Do not add for session only origins.
437 if (special_storage_policy_
.get() &&
438 !special_storage_policy_
->IsStorageProtected(origin
) &&
439 special_storage_policy_
->IsStorageSessionOnly(origin
)) {
442 BatchOperation(ADD_IDENTITY
, origin
, identity_name
, identity
);
445 void WebRTCIdentityStoreBackend::SqlLiteStorage::DeleteIdentity(
447 const std::string
& identity_name
,
448 const Identity
& identity
) {
449 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
452 BatchOperation(DELETE_IDENTITY
, origin
, identity_name
, identity
);
455 void WebRTCIdentityStoreBackend::SqlLiteStorage::DeleteBetween(
456 base::Time delete_begin
,
457 base::Time delete_end
) {
458 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
462 // Commit pending operations first.
465 sql::Statement
del_stmt(db_
->GetCachedStatement(
467 "DELETE FROM webrtc_identity_store"
468 " WHERE creation_time >= ? AND creation_time <= ?"));
469 CHECK(del_stmt
.is_valid());
471 del_stmt
.BindInt64(0, delete_begin
.ToInternalValue());
472 del_stmt
.BindInt64(1, delete_end
.ToInternalValue());
474 sql::Transaction
transaction(db_
.get());
475 if (!transaction
.Begin()) {
476 DVLOG(2) << "Failed to begin the transaction.";
480 if (!del_stmt
.Run()) {
481 DVLOG(2) << "Failed to run the delete statement.";
485 if (!transaction
.Commit())
486 DVLOG(2) << "Failed to commit the transaction.";
489 void WebRTCIdentityStoreBackend::SqlLiteStorage::OnDatabaseError(
491 sql::Statement
* stmt
) {
492 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
495 // It's not safe to reset |db_| here.
498 void WebRTCIdentityStoreBackend::SqlLiteStorage::BatchOperation(
501 const std::string
& identity_name
,
502 const Identity
& identity
) {
503 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
504 // Commit every 30 seconds.
505 static const base::TimeDelta
kCommitInterval(
506 base::TimeDelta::FromSeconds(30));
507 // Commit right away if we have more than 512 outstanding operations.
508 static const size_t kCommitAfterBatchSize
= 512;
510 // We do a full copy of the cert here, and hopefully just here.
511 scoped_ptr
<PendingOperation
> operation(
512 new PendingOperation(type
, origin
, identity_name
, identity
));
514 pending_operations_
.push_back(operation
.release());
516 if (pending_operations_
.size() == 1) {
517 // We've gotten our first entry for this batch, fire off the timer.
518 BrowserThread::PostDelayedTask(BrowserThread::DB
,
520 base::Bind(&SqlLiteStorage::Commit
, this),
522 } else if (pending_operations_
.size() >= kCommitAfterBatchSize
) {
523 // We've reached a big enough batch, fire off a commit now.
524 BrowserThread::PostTask(BrowserThread::DB
,
526 base::Bind(&SqlLiteStorage::Commit
, this));
530 void WebRTCIdentityStoreBackend::SqlLiteStorage::Commit() {
531 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB
));
532 // Maybe an old timer fired or we are already Close()'ed.
533 if (!db_
.get() || pending_operations_
.empty())
536 sql::Statement
add_stmt(db_
->GetCachedStatement(
538 "INSERT INTO webrtc_identity_store "
539 "(origin, identity_name, common_name, certificate,"
540 " private_key, creation_time) VALUES"
543 CHECK(add_stmt
.is_valid());
545 sql::Statement
del_stmt(db_
->GetCachedStatement(
547 "DELETE FROM webrtc_identity_store WHERE origin=? AND identity_name=?"));
549 CHECK(del_stmt
.is_valid());
551 sql::Transaction
transaction(db_
.get());
552 if (!transaction
.Begin()) {
553 DVLOG(2) << "Failed to begin the transaction.";
557 // Swaps |pending_operations_| into a temporary list to make sure
558 // |pending_operations_| is always cleared in case of DB errors.
559 PendingOperationList pending_operations_copy
;
560 pending_operations_
.swap(pending_operations_copy
);
562 for (PendingOperationList::const_iterator it
=
563 pending_operations_copy
.begin();
564 it
!= pending_operations_copy
.end();
566 switch ((*it
)->type
) {
568 add_stmt
.Reset(true);
569 add_stmt
.BindString(0, (*it
)->origin
.spec());
570 add_stmt
.BindString(1, (*it
)->identity_name
);
571 add_stmt
.BindString(2, (*it
)->identity
.common_name
);
572 const std::string
& cert
= (*it
)->identity
.certificate
;
573 add_stmt
.BindBlob(3, cert
.data(), cert
.size());
574 const std::string
& private_key
= (*it
)->identity
.private_key
;
575 add_stmt
.BindBlob(4, private_key
.data(), private_key
.size());
576 add_stmt
.BindInt64(5, (*it
)->identity
.creation_time
);
577 if (!add_stmt
.Run()) {
578 DVLOG(2) << "Failed to add the identity to DB.";
583 case DELETE_IDENTITY
:
584 del_stmt
.Reset(true);
585 del_stmt
.BindString(0, (*it
)->origin
.spec());
586 del_stmt
.BindString(1, (*it
)->identity_name
);
587 if (!del_stmt
.Run()) {
588 DVLOG(2) << "Failed to delete the identity from DB.";
599 if (!transaction
.Commit())
600 DVLOG(2) << "Failed to commit the transaction.";
603 } // namespace content