/* * Copyright (C) 2013 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "DatabaseProcessIDBConnection.h" #if ENABLE(INDEXED_DATABASE) && ENABLE(DATABASE_PROCESS) #include "DataReference.h" #include "DatabaseProcess.h" #include "DatabaseToWebProcessConnection.h" #include "IDBIdentifier.h" #include "Logging.h" #include "UniqueIDBDatabase.h" #include "WebCoreArgumentCoders.h" #include "WebIDBServerConnectionMessages.h" #include #include #include using namespace WebCore; namespace WebKit { DatabaseProcessIDBConnection::DatabaseProcessIDBConnection(DatabaseToWebProcessConnection& connection, uint64_t serverConnectionIdentifier) : m_connection(connection) , m_serverConnectionIdentifier(serverConnectionIdentifier) { } DatabaseProcessIDBConnection::~DatabaseProcessIDBConnection() { ASSERT(!m_uniqueIDBDatabase); } void DatabaseProcessIDBConnection::disconnectedFromWebProcess() { m_uniqueIDBDatabase->unregisterConnection(*this); m_uniqueIDBDatabase.clear(); } void DatabaseProcessIDBConnection::establishConnection(const String& databaseName, const SecurityOriginData& openingOrigin, const SecurityOriginData& mainFrameOrigin) { m_uniqueIDBDatabase = DatabaseProcess::shared().getOrCreateUniqueIDBDatabase(UniqueIDBDatabaseIdentifier(databaseName, openingOrigin, mainFrameOrigin)); m_uniqueIDBDatabase->registerConnection(*this); } void DatabaseProcessIDBConnection::getOrEstablishIDBDatabaseMetadata(uint64_t requestID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess getOrEstablishIDBDatabaseMetadata request ID %llu", requestID); RefPtr connection(this); m_uniqueIDBDatabase->getOrEstablishIDBDatabaseMetadata([connection, requestID](bool success, const IDBDatabaseMetadata& metadata) { connection->send(Messages::WebIDBServerConnection::DidGetOrEstablishIDBDatabaseMetadata(requestID, success, metadata)); }); } void DatabaseProcessIDBConnection::deleteDatabase(uint64_t requestID, const String& databaseName) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess deleteDatabase request ID %llu", requestID); if (databaseName != m_uniqueIDBDatabase->identifier().databaseName()) { LOG_ERROR("Request to delete database name that doesn't match with this database connection's database name"); send(Messages::WebIDBServerConnection::DidDeleteDatabase(requestID, false)); } RefPtr connection(this); m_uniqueIDBDatabase->deleteDatabase([connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidDeleteDatabase(requestID, success)); }); } void DatabaseProcessIDBConnection::openTransaction(uint64_t requestID, int64_t transactionID, const Vector& objectStoreIDs, uint64_t intMode) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess openTransaction request ID %llu", requestID); if (intMode > IndexedDB::TransactionModeMaximum) { send(Messages::WebIDBServerConnection::DidOpenTransaction(requestID, false)); return; } IndexedDB::TransactionMode mode = static_cast(intMode); RefPtr connection(this); m_uniqueIDBDatabase->openTransaction(IDBIdentifier(*this, transactionID), objectStoreIDs, mode, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidOpenTransaction(requestID, success)); }); } void DatabaseProcessIDBConnection::beginTransaction(uint64_t requestID, int64_t transactionID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess beginTransaction request ID %llu", requestID); RefPtr connection(this); m_uniqueIDBDatabase->beginTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidBeginTransaction(requestID, success)); }); } void DatabaseProcessIDBConnection::commitTransaction(uint64_t requestID, int64_t transactionID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess commitTransaction request ID %llu", requestID); RefPtr connection(this); m_uniqueIDBDatabase->commitTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidCommitTransaction(requestID, success)); }); } void DatabaseProcessIDBConnection::resetTransaction(uint64_t requestID, int64_t transactionID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess resetTransaction request ID %llu", requestID); RefPtr connection(this); m_uniqueIDBDatabase->resetTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidResetTransaction(requestID, success)); }); } void DatabaseProcessIDBConnection::rollbackTransaction(uint64_t requestID, int64_t transactionID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess rollbackTransaction request ID %llu", requestID); RefPtr connection(this); m_uniqueIDBDatabase->rollbackTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidRollbackTransaction(requestID, success)); }); } void DatabaseProcessIDBConnection::changeDatabaseVersion(uint64_t requestID, int64_t transactionID, uint64_t newVersion) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess changeDatabaseVersion request ID %llu, new version %llu", requestID, newVersion); RefPtr connection(this); m_uniqueIDBDatabase->changeDatabaseVersion(IDBIdentifier(*this, transactionID), newVersion, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidChangeDatabaseVersion(requestID, success)); }); } void DatabaseProcessIDBConnection::createObjectStore(uint64_t requestID, int64_t transactionID, IDBObjectStoreMetadata metadata) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess createObjectStore request ID %llu, object store name '%s'", requestID, metadata.name.utf8().data()); RefPtr connection(this); m_uniqueIDBDatabase->createObjectStore(IDBIdentifier(*this, transactionID), metadata, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidCreateObjectStore(requestID, success)); }); } void DatabaseProcessIDBConnection::deleteObjectStore(uint64_t requestID, int64_t transactionID, int64_t objectStoreID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess deleteObjectStore request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->deleteObjectStore(IDBIdentifier(*this, transactionID), objectStoreID, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidDeleteObjectStore(requestID, success)); }); } void DatabaseProcessIDBConnection::clearObjectStore(uint64_t requestID, int64_t transactionID, int64_t objectStoreID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess clearObjectStore request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->clearObjectStore(IDBIdentifier(*this, transactionID), objectStoreID, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidClearObjectStore(requestID, success)); }); } void DatabaseProcessIDBConnection::createIndex(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, const IDBIndexMetadata& metadata) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess createIndex request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->createIndex(IDBIdentifier(*this, transactionID), objectStoreID, metadata, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidCreateIndex(requestID, success)); }); } void DatabaseProcessIDBConnection::deleteIndex(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess deleteIndex request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->deleteIndex(IDBIdentifier(*this, transactionID), objectStoreID, indexID, [connection, requestID](bool success) { connection->send(Messages::WebIDBServerConnection::DidDeleteIndex(requestID, success)); }); } void DatabaseProcessIDBConnection::putRecord(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, const IDBKeyData& key, const IPC::DataReference& value, int64_t putMode, const Vector& indexIDs, const Vector>& indexKeys) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess putRecord request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->putRecord(IDBIdentifier(*this, transactionID), objectStoreID, key, value, putMode, indexIDs, indexKeys, [connection, requestID](const IDBKeyData& keyData, uint32_t errorCode, const String& errorMessage) { connection->send(Messages::WebIDBServerConnection::DidPutRecord(requestID, keyData, errorCode, errorMessage)); }); } void DatabaseProcessIDBConnection::getRecord(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID, const IDBKeyRangeData& keyRange, int64_t cursorType) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess getRecord request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->getRecord(IDBIdentifier(*this, transactionID), objectStoreID, indexID, keyRange, static_cast(cursorType), [connection, requestID](const IDBGetResult& getResult, uint32_t errorCode, const String& errorMessage) { connection->send(Messages::WebIDBServerConnection::DidGetRecord(requestID, getResult, errorCode, errorMessage)); }); } void DatabaseProcessIDBConnection::count(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID, const IDBKeyRangeData& keyRangeData) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess count request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->count(IDBIdentifier(*this, transactionID), objectStoreID, indexID, keyRangeData, [connection, requestID](int64_t count, uint32_t errorCode, const String& errorMessage) { connection->send(Messages::WebIDBServerConnection::DidCount(requestID, count, errorCode, errorMessage)); }); } void DatabaseProcessIDBConnection::deleteRange(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, const IDBKeyRangeData& keyRangeData) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess deleteRange request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->deleteRange(IDBIdentifier(*this, transactionID), objectStoreID, keyRangeData, [connection, requestID](uint32_t errorCode, const String& errorMessage) { connection->send(Messages::WebIDBServerConnection::DidDeleteRange(requestID, errorCode, errorMessage)); }); } void DatabaseProcessIDBConnection::openCursor(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID, int64_t cursorDirection, int64_t cursorType, int64_t taskType, const IDBKeyRangeData& keyRangeData) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess openCursor request ID %llu, object store id %lli", requestID, objectStoreID); RefPtr connection(this); m_uniqueIDBDatabase->openCursor(IDBIdentifier(*this, transactionID), objectStoreID, indexID, static_cast(cursorDirection), static_cast(cursorType), static_cast(taskType), keyRangeData, [connection, requestID](int64_t cursorID, const IDBKeyData& resultKey, const IDBKeyData& primaryKey, PassRefPtr value, const IDBKeyData& valueKey, uint32_t errorCode, const String& errorMessage) { IPC::DataReference data = value ? IPC::DataReference(reinterpret_cast(value->data()), value->size()) : IPC::DataReference(); connection->send(Messages::WebIDBServerConnection::DidOpenCursor(requestID, cursorID, resultKey, primaryKey, data, valueKey, errorCode, errorMessage)); }); } void DatabaseProcessIDBConnection::cursorAdvance(uint64_t requestID, int64_t cursorID, uint64_t count) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess cursorAdvance request ID %llu, cursor id %lli", requestID, cursorID); RefPtr connection(this); m_uniqueIDBDatabase->cursorAdvance(IDBIdentifier(*this, cursorID), count, [connection, requestID](const IDBKeyData& resultKey, const IDBKeyData& primaryKey, PassRefPtr value, const IDBKeyData& valueKey, uint32_t errorCode, const String& errorMessage) { IPC::DataReference data = value ? IPC::DataReference(reinterpret_cast(value->data()), value->size()) : IPC::DataReference(); connection->send(Messages::WebIDBServerConnection::DidAdvanceCursor(requestID, resultKey, primaryKey, data, valueKey, errorCode, errorMessage)); }); } void DatabaseProcessIDBConnection::cursorIterate(uint64_t requestID, int64_t cursorID, const IDBKeyData& key) { ASSERT(m_uniqueIDBDatabase); LOG(IDB, "DatabaseProcess cursorIterate request ID %llu, cursor id %lli", requestID, cursorID); RefPtr connection(this); m_uniqueIDBDatabase->cursorIterate(IDBIdentifier(*this, cursorID), key, [connection, requestID](const IDBKeyData& resultKey, const IDBKeyData& primaryKey, PassRefPtr value, const IDBKeyData& valueKey, uint32_t errorCode, const String& errorMessage) { IPC::DataReference data = value ? IPC::DataReference(reinterpret_cast(value->data()), value->size()) : IPC::DataReference(); connection->send(Messages::WebIDBServerConnection::DidIterateCursor(requestID, resultKey, primaryKey, data, valueKey, errorCode, errorMessage)); }); } IPC::Connection* DatabaseProcessIDBConnection::messageSenderConnection() { return m_connection->connection(); } } // namespace WebKit #endif // ENABLE(INDEXED_DATABASE) && ENABLE(DATABASE_PROCESS)