/* * Copyright (C) 2015 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 "UniqueIDBDatabaseConnection.h" #if ENABLE(INDEXED_DATABASE) #include "IDBConnectionToClient.h" #include "IDBServer.h" #include "IDBTransactionInfo.h" #include "Logging.h" #include "UniqueIDBDatabase.h" namespace WebCore { namespace IDBServer { static uint64_t nextDatabaseConnectionIdentifier() { static uint64_t nextIdentifier = 0; return ++nextIdentifier; } Ref UniqueIDBDatabaseConnection::create(UniqueIDBDatabase& database, IDBConnectionToClient& connection) { return adoptRef(*new UniqueIDBDatabaseConnection(database, connection)); } UniqueIDBDatabaseConnection::UniqueIDBDatabaseConnection(UniqueIDBDatabase& database, IDBConnectionToClient& connection) : m_identifier(nextDatabaseConnectionIdentifier()) , m_database(database) , m_connectionToClient(connection) { m_database.server().registerDatabaseConnection(*this); } UniqueIDBDatabaseConnection::~UniqueIDBDatabaseConnection() { m_database.server().unregisterDatabaseConnection(*this); } bool UniqueIDBDatabaseConnection::hasNonFinishedTransactions() const { return !m_transactionMap.isEmpty(); } void UniqueIDBDatabaseConnection::abortTransactionWithoutCallback(UniqueIDBDatabaseTransaction& transaction) { ASSERT(m_transactionMap.contains(transaction.info().identifier())); auto takenTransaction = m_transactionMap.take(transaction.info().identifier()); m_database.abortTransaction(*takenTransaction, [](const IDBError&) { }); } void UniqueIDBDatabaseConnection::connectionClosedFromClient() { LOG(IndexedDB, "UniqueIDBDatabaseConnection::connectionClosedFromClient - %" PRIu64, m_identifier); m_closePending = true; m_database.connectionClosedFromClient(*this); } void UniqueIDBDatabaseConnection::didFireVersionChangeEvent(const IDBResourceIdentifier& requestIdentifier) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didFireVersionChangeEvent"); m_database.didFireVersionChangeEvent(*this, requestIdentifier); } void UniqueIDBDatabaseConnection::fireVersionChangeEvent(const IDBResourceIdentifier& requestIdentifier, uint64_t requestedVersion) { ASSERT(!m_closePending); m_connectionToClient.fireVersionChangeEvent(*this, requestIdentifier, requestedVersion); } UniqueIDBDatabaseTransaction& UniqueIDBDatabaseConnection::createVersionChangeTransaction(uint64_t newVersion) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::createVersionChangeTransaction"); ASSERT(!m_closePending); IDBTransactionInfo info = IDBTransactionInfo::versionChange(m_connectionToClient, m_database.info(), newVersion); Ref transaction = UniqueIDBDatabaseTransaction::create(*this, info); m_transactionMap.set(transaction->info().identifier(), &transaction.get()); return transaction.get(); } void UniqueIDBDatabaseConnection::establishTransaction(const IDBTransactionInfo& info) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::establishTransaction"); ASSERT(info.mode() != IndexedDB::TransactionMode::VersionChange); // No transactions should ever come from the client after the client has already told us // the connection is closing. ASSERT(!m_closePending); Ref transaction = UniqueIDBDatabaseTransaction::create(*this, info); m_transactionMap.set(transaction->info().identifier(), &transaction.get()); m_database.enqueueTransaction(WTFMove(transaction)); } void UniqueIDBDatabaseConnection::didAbortTransaction(UniqueIDBDatabaseTransaction& transaction, const IDBError& error) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didAbortTransaction"); auto transactionIdentifier = transaction.info().identifier(); ASSERT(m_transactionMap.contains(transactionIdentifier)); m_transactionMap.remove(transactionIdentifier); m_connectionToClient.didAbortTransaction(transactionIdentifier, error); } void UniqueIDBDatabaseConnection::didCommitTransaction(UniqueIDBDatabaseTransaction& transaction, const IDBError& error) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didCommitTransaction"); auto transactionIdentifier = transaction.info().identifier(); ASSERT(m_transactionMap.contains(transactionIdentifier)); m_transactionMap.remove(transactionIdentifier); m_connectionToClient.didCommitTransaction(transactionIdentifier, error); } void UniqueIDBDatabaseConnection::didCreateObjectStore(const IDBResultData& resultData) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didCreateObjectStore"); m_connectionToClient.didCreateObjectStore(resultData); } void UniqueIDBDatabaseConnection::didDeleteObjectStore(const IDBResultData& resultData) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didDeleteObjectStore"); m_connectionToClient.didDeleteObjectStore(resultData); } void UniqueIDBDatabaseConnection::didClearObjectStore(const IDBResultData& resultData) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didClearObjectStore"); m_connectionToClient.didClearObjectStore(resultData); } void UniqueIDBDatabaseConnection::didCreateIndex(const IDBResultData& resultData) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didCreateIndex"); m_connectionToClient.didCreateIndex(resultData); } void UniqueIDBDatabaseConnection::didDeleteIndex(const IDBResultData& resultData) { LOG(IndexedDB, "UniqueIDBDatabaseConnection::didDeleteIndex"); m_connectionToClient.didDeleteIndex(resultData); } } // namespace IDBServer } // namespace WebCore #endif // ENABLE(INDEXED_DATABASE)