|  | // 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 <stddef.h> | 
|  | #include <stdint.h> | 
|  |  | 
|  | #include <list> | 
|  | #include <map> | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <tuple> | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/callback.h" | 
|  | #include "base/containers/queue.h" | 
|  | #include "base/gtest_prod_util.h" | 
|  | #include "base/macros.h" | 
|  | #include "base/memory/ref_counted.h" | 
|  | #include "base/memory/weak_ptr.h" | 
|  | #include "components/services/storage/indexed_db/scopes/scopes_lock_manager.h" | 
|  | #include "content/browser/indexed_db/indexed_db.h" | 
|  | #include "content/browser/indexed_db/indexed_db_backing_store.h" | 
|  | #include "content/browser/indexed_db/indexed_db_callbacks.h" | 
|  | #include "content/browser/indexed_db/indexed_db_connection_coordinator.h" | 
|  | #include "content/browser/indexed_db/indexed_db_origin_state_handle.h" | 
|  | #include "content/browser/indexed_db/indexed_db_pending_connection.h" | 
|  | #include "content/browser/indexed_db/indexed_db_task_helper.h" | 
|  | #include "content/browser/indexed_db/indexed_db_value.h" | 
|  | #include "content/browser/indexed_db/list_set.h" | 
|  | #include "content/common/content_export.h" | 
|  | #include "third_party/blink/public/common/indexeddb/indexeddb_key.h" | 
|  | #include "third_party/blink/public/common/indexeddb/web_idb_types.h" | 
|  | #include "third_party/blink/public/mojom/indexeddb/indexeddb.mojom-forward.h" | 
|  |  | 
|  | namespace blink { | 
|  | class IndexedDBKeyPath; | 
|  | class IndexedDBKeyRange; | 
|  | struct IndexedDBDatabaseMetadata; | 
|  | struct IndexedDBIndexMetadata; | 
|  | struct IndexedDBObjectStoreMetadata; | 
|  | }  // namespace blink | 
|  |  | 
|  | namespace url { | 
|  | class Origin; | 
|  | } | 
|  |  | 
|  | namespace content { | 
|  | class IndexedDBClassFactory; | 
|  | class IndexedDBConnection; | 
|  | class IndexedDBDatabaseCallbacks; | 
|  | class IndexedDBFactory; | 
|  | class IndexedDBMetadataCoding; | 
|  | class IndexedDBOriginStateHandle; | 
|  | class IndexedDBTransaction; | 
|  | struct IndexedDBValue; | 
|  |  | 
|  | class CONTENT_EXPORT IndexedDBDatabase { | 
|  | public: | 
|  | // Identifier is pair of (origin, database name). | 
|  | using Identifier = std::pair<url::Origin, std::u16string>; | 
|  | // Used to report irrecoverable backend errors. The second argument can be | 
|  | // null. | 
|  | using ErrorCallback = | 
|  | base::RepeatingCallback<void(leveldb::Status, const char*)>; | 
|  |  | 
|  | static const int64_t kInvalidId = 0; | 
|  | static const int64_t kMinimumIndexId = 30; | 
|  |  | 
|  | virtual ~IndexedDBDatabase(); | 
|  |  | 
|  | const Identifier& identifier() const { return identifier_; } | 
|  | IndexedDBBackingStore* backing_store() { return backing_store_; } | 
|  |  | 
|  | int64_t id() const { return metadata_.id; } | 
|  | const std::u16string& name() const { return metadata_.name; } | 
|  | const url::Origin& origin() const { return identifier_.first; } | 
|  | const blink::IndexedDBDatabaseMetadata& metadata() const { return metadata_; } | 
|  |  | 
|  | ScopesLockManager* transaction_lock_manager() { return lock_manager_; } | 
|  | const ScopesLockManager* transaction_lock_manager() const { | 
|  | return lock_manager_; | 
|  | } | 
|  |  | 
|  | const list_set<IndexedDBConnection*>& connections() const { | 
|  | return connections_; | 
|  | } | 
|  | TasksAvailableCallback tasks_available_callback() { | 
|  | return tasks_available_callback_; | 
|  | } | 
|  |  | 
|  | // TODO(dmurph): Remove this method and have transactions be directly | 
|  | // scheduled using the lock manager. | 
|  |  | 
|  | enum class RunTasksResult { kDone, kError, kCanBeDestroyed }; | 
|  | std::tuple<RunTasksResult, leveldb::Status> RunTasks(); | 
|  | void RegisterAndScheduleTransaction(IndexedDBTransaction* transaction); | 
|  |  | 
|  | // The database object (this object) must be kept alive for the duration of | 
|  | // this call. This means the caller should own an IndexedDBOriginStateHandle | 
|  | // while caling this methods. | 
|  | leveldb::Status ForceCloseAndRunTasks(); | 
|  |  | 
|  | void Commit(IndexedDBTransaction* transaction); | 
|  |  | 
|  | void TransactionCreated(); | 
|  | void TransactionFinished(blink::mojom::IDBTransactionMode mode, | 
|  | bool committed); | 
|  |  | 
|  | void ScheduleOpenConnection( | 
|  | IndexedDBOriginStateHandle origin_state_handle, | 
|  | std::unique_ptr<IndexedDBPendingConnection> connection); | 
|  | void ScheduleDeleteDatabase(IndexedDBOriginStateHandle origin_state_handle, | 
|  | scoped_refptr<IndexedDBCallbacks> callbacks, | 
|  | base::OnceClosure on_deletion_complete); | 
|  |  | 
|  | void AddObjectStoreToMetadata(blink::IndexedDBObjectStoreMetadata metadata, | 
|  | int64_t new_max_object_store_id); | 
|  | blink::IndexedDBObjectStoreMetadata RemoveObjectStoreFromMetadata( | 
|  | int64_t object_store_id); | 
|  | void AddIndexToMetadata(int64_t object_store_id, | 
|  | blink::IndexedDBIndexMetadata metadata, | 
|  | int64_t new_max_index_id); | 
|  | blink::IndexedDBIndexMetadata RemoveIndexFromMetadata(int64_t object_store_id, | 
|  | int64_t index_id); | 
|  |  | 
|  | // The following methods all schedule a task on the transaction & modify the | 
|  | // database: | 
|  |  | 
|  | // Number of connections that have progressed passed initial open call. | 
|  | size_t ConnectionCount() const { return connections_.size(); } | 
|  |  | 
|  | // Number of active open/delete calls (running or blocked on other | 
|  | // connections). | 
|  | size_t ActiveOpenDeleteCount() const { | 
|  | return connection_coordinator_.ActiveOpenDeleteCount(); | 
|  | } | 
|  |  | 
|  | // Number of open/delete calls that are waiting their turn. | 
|  | size_t PendingOpenDeleteCount() const { | 
|  | return connection_coordinator_.PendingOpenDeleteCount(); | 
|  | } | 
|  |  | 
|  | // The following methods are all of the ones actually scheduled asynchronously | 
|  | // within transctions: | 
|  |  | 
|  | leveldb::Status CreateObjectStoreOperation( | 
|  | int64_t object_store_id, | 
|  | const std::u16string& name, | 
|  | const blink::IndexedDBKeyPath& key_path, | 
|  | bool auto_increment, | 
|  | IndexedDBTransaction* transaction); | 
|  | void CreateObjectStoreAbortOperation(int64_t object_store_id); | 
|  |  | 
|  | leveldb::Status DeleteObjectStoreOperation(int64_t object_store_id, | 
|  | IndexedDBTransaction* transaction); | 
|  | void DeleteObjectStoreAbortOperation( | 
|  | blink::IndexedDBObjectStoreMetadata object_store_metadata); | 
|  |  | 
|  | leveldb::Status RenameObjectStoreOperation(int64_t object_store_id, | 
|  | const std::u16string& new_name, | 
|  | IndexedDBTransaction* transaction); | 
|  | void RenameObjectStoreAbortOperation(int64_t object_store_id, | 
|  | std::u16string old_name); | 
|  |  | 
|  | leveldb::Status VersionChangeOperation( | 
|  | int64_t version, | 
|  | scoped_refptr<IndexedDBCallbacks> callbacks, | 
|  | IndexedDBTransaction* transaction); | 
|  | void VersionChangeAbortOperation(int64_t previous_version); | 
|  |  | 
|  | leveldb::Status CreateIndexOperation(int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | const std::u16string& name, | 
|  | const blink::IndexedDBKeyPath& key_path, | 
|  | bool unique, | 
|  | bool multi_entry, | 
|  | IndexedDBTransaction* transaction); | 
|  | void CreateIndexAbortOperation(int64_t object_store_id, int64_t index_id); | 
|  |  | 
|  | leveldb::Status DeleteIndexOperation(int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | IndexedDBTransaction* transaction); | 
|  | void DeleteIndexAbortOperation(int64_t object_store_id, | 
|  | blink::IndexedDBIndexMetadata index_metadata); | 
|  |  | 
|  | leveldb::Status RenameIndexOperation(int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | const std::u16string& new_name, | 
|  | IndexedDBTransaction* transaction); | 
|  | void RenameIndexAbortOperation(int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | std::u16string old_name); | 
|  |  | 
|  | leveldb::Status GetOperation( | 
|  | base::WeakPtr<IndexedDBDispatcherHost> dispatcher_host, | 
|  | int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | std::unique_ptr<blink::IndexedDBKeyRange> key_range, | 
|  | indexed_db::CursorType cursor_type, | 
|  | blink::mojom::IDBDatabase::GetCallback callback, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status GetAllOperation( | 
|  | base::WeakPtr<IndexedDBDispatcherHost> dispatcher_host, | 
|  | int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | std::unique_ptr<blink::IndexedDBKeyRange> key_range, | 
|  | indexed_db::CursorType cursor_type, | 
|  | int64_t max_count, | 
|  | blink::mojom::IDBDatabase::GetAllCallback callback, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | struct CONTENT_EXPORT PutOperationParams { | 
|  | PutOperationParams(); | 
|  | ~PutOperationParams(); | 
|  | int64_t object_store_id; | 
|  | IndexedDBValue value; | 
|  | std::unique_ptr<blink::IndexedDBKey> key; | 
|  | blink::mojom::IDBPutMode put_mode; | 
|  | blink::mojom::IDBTransaction::PutCallback callback; | 
|  | std::vector<blink::IndexedDBIndexKeys> index_keys; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(PutOperationParams); | 
|  | }; | 
|  | leveldb::Status PutOperation(std::unique_ptr<PutOperationParams> params, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | struct CONTENT_EXPORT PutAllOperationParams { | 
|  | PutAllOperationParams(); | 
|  | ~PutAllOperationParams(); | 
|  | IndexedDBValue value; | 
|  | std::unique_ptr<blink::IndexedDBKey> key; | 
|  | std::vector<blink::IndexedDBIndexKeys> index_keys; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(PutAllOperationParams); | 
|  | }; | 
|  | leveldb::Status PutAllOperation( | 
|  | int64_t object_store_id, | 
|  | std::vector<std::unique_ptr<PutAllOperationParams>> params, | 
|  | blink::mojom::IDBTransaction::PutAllCallback callback, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status SetIndexKeysOperation( | 
|  | int64_t object_store_id, | 
|  | std::unique_ptr<blink::IndexedDBKey> primary_key, | 
|  | const std::vector<blink::IndexedDBIndexKeys>& index_keys, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status SetIndexesReadyOperation(size_t index_count, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | struct OpenCursorOperationParams { | 
|  | OpenCursorOperationParams(); | 
|  | ~OpenCursorOperationParams(); | 
|  | int64_t object_store_id; | 
|  | int64_t index_id; | 
|  | std::unique_ptr<blink::IndexedDBKeyRange> key_range; | 
|  | blink::mojom::IDBCursorDirection direction; | 
|  | indexed_db::CursorType cursor_type; | 
|  | blink::mojom::IDBTaskType task_type; | 
|  | blink::mojom::IDBDatabase::OpenCursorCallback callback; | 
|  |  | 
|  | private: | 
|  | DISALLOW_COPY_AND_ASSIGN(OpenCursorOperationParams); | 
|  | }; | 
|  | leveldb::Status OpenCursorOperation( | 
|  | std::unique_ptr<OpenCursorOperationParams> params, | 
|  | const url::Origin& origin, | 
|  | base::WeakPtr<IndexedDBDispatcherHost> dispatcher_host, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status CountOperation( | 
|  | int64_t object_store_id, | 
|  | int64_t index_id, | 
|  | std::unique_ptr<blink::IndexedDBKeyRange> key_range, | 
|  | scoped_refptr<IndexedDBCallbacks> callbacks, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status DeleteRangeOperation( | 
|  | int64_t object_store_id, | 
|  | std::unique_ptr<blink::IndexedDBKeyRange> key_range, | 
|  | scoped_refptr<IndexedDBCallbacks> callbacks, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status GetKeyGeneratorCurrentNumberOperation( | 
|  | int64_t object_store_id, | 
|  | scoped_refptr<IndexedDBCallbacks> callbacks, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | leveldb::Status ClearOperation(int64_t object_store_id, | 
|  | scoped_refptr<IndexedDBCallbacks> callbacks, | 
|  | IndexedDBTransaction* transaction); | 
|  |  | 
|  | bool IsObjectStoreIdInMetadata(int64_t object_store_id) const; | 
|  | bool IsObjectStoreIdAndIndexIdInMetadata(int64_t object_store_id, | 
|  | int64_t index_id) const; | 
|  | bool IsObjectStoreIdAndMaybeIndexIdInMetadata(int64_t object_store_id, | 
|  | int64_t index_id) const; | 
|  | bool IsObjectStoreIdInMetadataAndIndexNotInMetadata(int64_t object_store_id, | 
|  | int64_t index_id) const; | 
|  |  | 
|  | base::WeakPtr<IndexedDBDatabase> AsWeakPtr() { | 
|  | return weak_factory_.GetWeakPtr(); | 
|  | } | 
|  |  | 
|  | void AddConnectionForTesting(IndexedDBConnection* connection) { | 
|  | connections_.insert(connection); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | friend class IndexedDBTransaction; | 
|  | friend class IndexedDBConnectionCoordinator; | 
|  | friend class IndexedDBConnectionCoordinator::ConnectionRequest; | 
|  | friend class IndexedDBConnectionCoordinator::OpenRequest; | 
|  | friend class IndexedDBConnectionCoordinator::DeleteRequest; | 
|  |  | 
|  | IndexedDBDatabase(const std::u16string& name, | 
|  | IndexedDBBackingStore* backing_store, | 
|  | IndexedDBFactory* factory, | 
|  | IndexedDBClassFactory* class_factory, | 
|  | TasksAvailableCallback tasks_available_callback, | 
|  | std::unique_ptr<IndexedDBMetadataCoding> metadata_coding, | 
|  | const Identifier& unique_identifier, | 
|  | ScopesLockManager* transaction_lock_manager); | 
|  |  | 
|  | // May be overridden in tests. | 
|  | virtual size_t GetUsableMessageSizeInBytes() const; | 
|  |  | 
|  | private: | 
|  | friend class MockBrowserTestIndexedDBClassFactory; | 
|  | friend class IndexedDBClassFactory; | 
|  |  | 
|  | FRIEND_TEST_ALL_PREFIXES(IndexedDBDatabaseTest, OpenDeleteClear); | 
|  |  | 
|  | void CallUpgradeTransactionStartedForTesting(int64_t old_version); | 
|  |  | 
|  | class ConnectionRequest; | 
|  | class OpenRequest; | 
|  | class DeleteRequest; | 
|  |  | 
|  | leveldb::Status OpenInternal(); | 
|  |  | 
|  | // If there is no active request, grab a new one from the pending queue and | 
|  | // start it. Afterwards, possibly release the database by calling | 
|  | // MaybeReleaseDatabase(). | 
|  | void ProcessRequestQueueAndMaybeRelease(); | 
|  |  | 
|  | // If there are no connections, pending requests, or an active request, then | 
|  | // this function will call |destroy_me_|, which can destruct this object. | 
|  | void MaybeReleaseDatabase(); | 
|  |  | 
|  | std::unique_ptr<IndexedDBConnection> CreateConnection( | 
|  | IndexedDBOriginStateHandle origin_state_handle, | 
|  | scoped_refptr<IndexedDBDatabaseCallbacks> database_callbacks); | 
|  |  | 
|  | // Ack that one of the connections notified with a "versionchange" event did | 
|  | // not promptly close. Therefore a "blocked" event should be fired at the | 
|  | // pending connection. | 
|  | void VersionChangeIgnored(); | 
|  |  | 
|  | bool HasNoConnections() const; | 
|  |  | 
|  | void SendVersionChangeToAllConnections(int64_t old_version, | 
|  | int64_t new_version); | 
|  |  | 
|  | // This can only be called when the given connection is closed and no longer | 
|  | // has any transaction objects. | 
|  | void ConnectionClosed(IndexedDBConnection* connection); | 
|  |  | 
|  | bool CanBeDestroyed(); | 
|  |  | 
|  | // Safe because the IndexedDBBackingStore is owned by the same object which | 
|  | // owns us, the IndexedDBPerOriginFactory. | 
|  | IndexedDBBackingStore* backing_store_; | 
|  | blink::IndexedDBDatabaseMetadata metadata_; | 
|  |  | 
|  | const Identifier identifier_; | 
|  | // TODO(dmurph): Remove the need for this to be here (and then remove it). | 
|  | IndexedDBFactory* factory_; | 
|  | IndexedDBClassFactory* const class_factory_; | 
|  | std::unique_ptr<IndexedDBMetadataCoding> metadata_coding_; | 
|  |  | 
|  | ScopesLockManager* lock_manager_; | 
|  | int64_t transaction_count_ = 0; | 
|  |  | 
|  | list_set<IndexedDBConnection*> connections_; | 
|  |  | 
|  | TasksAvailableCallback tasks_available_callback_; | 
|  |  | 
|  | bool force_closing_ = false; | 
|  |  | 
|  | IndexedDBConnectionCoordinator connection_coordinator_; | 
|  |  | 
|  | // |weak_factory_| is used for all callback uses. | 
|  | base::WeakPtrFactory<IndexedDBDatabase> weak_factory_{this}; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(IndexedDBDatabase); | 
|  | }; | 
|  |  | 
|  | }  // namespace content | 
|  |  | 
|  | #endif  // CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_DATABASE_H_ |