// Copyright (c) 2013 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. #ifndef CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_DATABASE_H_ #define CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_DATABASE_H_ #include #include #include #include #include "base/basictypes.h" #include "base/memory/ref_counted.h" #include "content/browser/indexed_db/indexed_db.h" #include "content/browser/indexed_db/indexed_db_callbacks.h" #include "content/browser/indexed_db/indexed_db_metadata.h" #include "content/browser/indexed_db/indexed_db_transaction_coordinator.h" #include "content/browser/indexed_db/list_set.h" namespace content { class IndexedDBConnection; class IndexedDBDatabaseCallbacks; class IndexedDBBackingStore; class IndexedDBFactory; class IndexedDBKey; class IndexedDBKeyPath; class IndexedDBKeyRange; class IndexedDBTransaction; class CONTENT_EXPORT IndexedDBDatabase : NON_EXPORTED_BASE(public base::RefCounted) { public: enum TaskType { NORMAL_TASK = 0, PREEMPTIVE_TASK }; enum PutMode { ADD_OR_UPDATE, ADD_ONLY, CURSOR_UPDATE }; typedef std::vector IndexKeys; // Identifier is pair of (origin identifier, database name). typedef std::pair Identifier; static const int64 kInvalidId = 0; static const int64 kMinimumIndexId = 30; static scoped_refptr Create( const string16& name, IndexedDBBackingStore* database, IndexedDBFactory* factory, const Identifier& unique_identifier); scoped_refptr BackingStore() const; int64 id() const { return metadata_.id; } const base::string16& name() const { return metadata_.name; } void AddObjectStore(const IndexedDBObjectStoreMetadata& metadata, int64 new_max_object_store_id); void RemoveObjectStore(int64 object_store_id); void AddIndex(int64 object_store_id, const IndexedDBIndexMetadata& metadata, int64 new_max_index_id); void RemoveIndex(int64 object_store_id, int64 index_id); void OpenConnection( scoped_refptr callbacks, scoped_refptr database_callbacks, int64 transaction_id, int64 version); void OpenConnection( scoped_refptr callbacks, scoped_refptr database_callbacks, int64 transaction_id, int64 version, WebKit::WebIDBCallbacks::DataLoss data_loss); void DeleteDatabase(scoped_refptr callbacks); const IndexedDBDatabaseMetadata& metadata() const { return metadata_; } void CreateObjectStore(int64 transaction_id, int64 object_store_id, const string16& name, const IndexedDBKeyPath& key_path, bool auto_increment); void DeleteObjectStore(int64 transaction_id, int64 object_store_id); void CreateTransaction(int64 transaction_id, IndexedDBConnection* connection, const std::vector& object_store_ids, uint16 mode); void Close(IndexedDBConnection* connection); void Commit(int64 transaction_id); void Abort(int64 transaction_id); void Abort(int64 transaction_id, const IndexedDBDatabaseError& error); void CreateIndex(int64 transaction_id, int64 object_store_id, int64 index_id, const string16& name, const IndexedDBKeyPath& key_path, bool unique, bool multi_entry); void DeleteIndex(int64 transaction_id, int64 object_store_id, int64 index_id); IndexedDBTransactionCoordinator& transaction_coordinator() { return transaction_coordinator_; } const IndexedDBTransactionCoordinator& transaction_coordinator() const { return transaction_coordinator_; } void TransactionStarted(IndexedDBTransaction* transaction); void TransactionFinished(IndexedDBTransaction* transaction); void TransactionFinishedAndCompleteFired(IndexedDBTransaction* transaction); void TransactionFinishedAndAbortFired(IndexedDBTransaction* transaction); void Get(int64 transaction_id, int64 object_store_id, int64 index_id, scoped_ptr key_range, bool key_only, scoped_refptr callbacks); void Put(int64 transaction_id, int64 object_store_id, std::string* value, scoped_ptr key, PutMode mode, scoped_refptr callbacks, const std::vector& index_ids, const std::vector& index_keys); void SetIndexKeys(int64 transaction_id, int64 object_store_id, scoped_ptr primary_key, const std::vector& index_ids, const std::vector& index_keys); void SetIndexesReady(int64 transaction_id, int64 object_store_id, const std::vector& index_ids); void OpenCursor(int64 transaction_id, int64 object_store_id, int64 index_id, scoped_ptr key_range, indexed_db::CursorDirection, bool key_only, TaskType task_type, scoped_refptr callbacks); void Count(int64 transaction_id, int64 object_store_id, int64 index_id, scoped_ptr key_range, scoped_refptr callbacks); void DeleteRange(int64 transaction_id, int64 object_store_id, scoped_ptr key_range, scoped_refptr callbacks); void Clear(int64 transaction_id, int64 object_store_id, scoped_refptr callbacks); // Number of connections that have progressed passed initial open call. size_t ConnectionCount() const; // Number of open calls that are blocked on other connections. size_t PendingOpenCount() const; // Number of pending upgrades (0 or 1). Also included in ConnectionCount(). size_t PendingUpgradeCount() const; // Number of running upgrades (0 or 1). Also included in ConnectionCount(). size_t RunningUpgradeCount() const; // Number of pending deletes, blocked on other connections. size_t PendingDeleteCount() const; private: friend class base::RefCounted; IndexedDBDatabase( const string16& name, IndexedDBBackingStore* database, IndexedDBFactory* factory, const Identifier& unique_identifier); ~IndexedDBDatabase(); bool IsOpenConnectionBlocked() const; bool OpenInternal(); void RunVersionChangeTransaction(scoped_refptr callbacks, scoped_ptr connection, int64 transaction_id, int64 requested_version, WebKit::WebIDBCallbacks::DataLoss data_loss); void RunVersionChangeTransactionFinal( scoped_refptr callbacks, scoped_ptr connection, int64 transaction_id, int64 requested_version); void RunVersionChangeTransactionFinal( scoped_refptr callbacks, scoped_ptr connection, int64 transaction_id, int64 requested_version, WebKit::WebIDBCallbacks::DataLoss data_loss); void ProcessPendingCalls(); bool IsDeleteDatabaseBlocked() const; void DeleteDatabaseFinal(scoped_refptr callbacks); IndexedDBTransaction* GetTransaction(int64 transaction_id) const; bool ValidateObjectStoreId(int64 object_store_id) const; bool ValidateObjectStoreIdAndIndexId(int64 object_store_id, int64 index_id) const; bool ValidateObjectStoreIdAndOptionalIndexId(int64 object_store_id, int64 index_id) const; bool ValidateObjectStoreIdAndNewIndexId(int64 object_store_id, int64 index_id) const; class VersionChangeOperation; // When a "versionchange" transaction aborts, these restore the back-end // object hierarchy. class VersionChangeAbortOperation; scoped_refptr backing_store_; IndexedDBDatabaseMetadata metadata_; const Identifier identifier_; // This might not need to be a scoped_refptr since the factory's lifetime is // that of the page group, but it's better to be conservitive than sorry. scoped_refptr factory_; IndexedDBTransactionCoordinator transaction_coordinator_; IndexedDBTransaction* running_version_change_transaction_; typedef std::map TransactionMap; TransactionMap transactions_; class PendingOpenCall; typedef std::list PendingOpenCallList; PendingOpenCallList pending_open_calls_; class PendingUpgradeCall; scoped_ptr pending_run_version_change_transaction_call_; class PendingSuccessCall; scoped_ptr pending_second_half_open_; class PendingDeleteCall; typedef std::list PendingDeleteCallList; PendingDeleteCallList pending_delete_calls_; typedef list_set ConnectionSet; ConnectionSet connections_; bool closing_connection_; }; } // namespace content #endif // CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_DATABASE_H_