blob: ca6c336ac6303e62d177ba383b90a759de04c159 [file] [log] [blame]
/*
* Copyright (C) 2016 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.
*/
#pragma once
#if ENABLE(INDEXED_DATABASE)
#include "IDBBackingStore.h"
#include "IDBDatabaseIdentifier.h"
#include "IDBDatabaseInfo.h"
#include "IDBResourceIdentifier.h"
#include "SQLiteIDBTransaction.h"
#include <JavaScriptCore/Strong.h>
#include <pal/SessionID.h>
#include <wtf/HashMap.h>
namespace WebCore {
class IndexKey;
class SQLiteDatabase;
class SQLiteStatement;
namespace IDBServer {
class IDBSerializationContext;
class SQLiteIDBCursor;
class SQLiteIDBBackingStore : public IDBBackingStore {
WTF_MAKE_FAST_ALLOCATED;
public:
SQLiteIDBBackingStore(PAL::SessionID, const IDBDatabaseIdentifier&, const String& databaseRootDirectory, IDBBackingStoreTemporaryFileHandler&);
~SQLiteIDBBackingStore() final;
IDBError getOrEstablishDatabaseInfo(IDBDatabaseInfo&) final;
IDBError beginTransaction(const IDBTransactionInfo&) final;
IDBError abortTransaction(const IDBResourceIdentifier& transactionIdentifier) final;
IDBError commitTransaction(const IDBResourceIdentifier& transactionIdentifier) final;
IDBError createObjectStore(const IDBResourceIdentifier& transactionIdentifier, const IDBObjectStoreInfo&) final;
IDBError deleteObjectStore(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier) final;
IDBError renameObjectStore(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, const String& newName) final;
IDBError clearObjectStore(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier) final;
IDBError createIndex(const IDBResourceIdentifier& transactionIdentifier, const IDBIndexInfo&) final;
IDBError deleteIndex(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, uint64_t indexIdentifier) final;
IDBError renameIndex(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, uint64_t indexIdentifier, const String& newName) final;
IDBError keyExistsInObjectStore(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, const IDBKeyData&, bool& keyExists) final;
IDBError deleteRange(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, const IDBKeyRangeData&) final;
IDBError addRecord(const IDBResourceIdentifier& transactionIdentifier, const IDBObjectStoreInfo&, const IDBKeyData&, const IDBValue&) final;
IDBError getRecord(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, const IDBKeyRangeData&, IDBGetRecordDataType, IDBGetResult& outValue) final;
IDBError getAllRecords(const IDBResourceIdentifier& transactionIdentifier, const IDBGetAllRecordsData&, IDBGetAllResult& outValue) final;
IDBError getIndexRecord(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, uint64_t indexIdentifier, IndexedDB::IndexRecordType, const IDBKeyRangeData&, IDBGetResult& outValue) final;
IDBError getCount(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, uint64_t indexIdentifier, const IDBKeyRangeData&, uint64_t& outCount) final;
IDBError generateKeyNumber(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, uint64_t& keyNumber) final;
IDBError revertGeneratedKeyNumber(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, uint64_t keyNumber) final;
IDBError maybeUpdateKeyGeneratorNumber(const IDBResourceIdentifier& transactionIdentifier, uint64_t objectStoreIdentifier, double newKeyNumber) final;
IDBError openCursor(const IDBResourceIdentifier& transactionIdentifier, const IDBCursorInfo&, IDBGetResult& outResult) final;
IDBError iterateCursor(const IDBResourceIdentifier& transactionIdentifier, const IDBResourceIdentifier& cursorIdentifier, const IDBIterateCursorData&, IDBGetResult& outResult) final;
bool prefetchCursor(const IDBResourceIdentifier&, const IDBResourceIdentifier&) final;
IDBObjectStoreInfo* infoForObjectStore(uint64_t objectStoreIdentifier) final;
void deleteBackingStore() final;
bool supportsSimultaneousTransactions() final { return false; }
bool isEphemeral() final { return false; }
void unregisterCursor(SQLiteIDBCursor&);
IDBBackingStoreTemporaryFileHandler& temporaryFileHandler() const { return m_temporaryFileHandler; }
IDBError getBlobRecordsForObjectStoreRecord(int64_t objectStoreRecord, Vector<String>& blobURLs, Vector<String>& blobFilePaths);
static String databaseNameFromEncodedFilename(const String&);
static uint64_t databasesSizeForDirectory(const String& directory);
String databaseDirectory() const { return m_databaseDirectory; };
static String fullDatabasePathForDirectory(const String&);
static String databaseNameFromFile(const String&);
bool hasTransaction(const IDBResourceIdentifier&) const final;
PAL::SessionID sessionID() const { return m_sessionID; }
private:
String filenameForDatabaseName() const;
String fullDatabasePath() const;
String fullDatabaseDirectoryWithUpgrade();
String databaseRootDirectoryIsolatedCopy() const { return m_databaseRootDirectory.isolatedCopy(); }
bool ensureValidRecordsTable();
bool ensureValidIndexRecordsTable();
bool ensureValidIndexRecordsIndex();
bool ensureValidBlobTables();
std::unique_ptr<IDBDatabaseInfo> createAndPopulateInitialDatabaseInfo();
std::unique_ptr<IDBDatabaseInfo> extractExistingDatabaseInfo();
IDBError deleteRecord(SQLiteIDBTransaction&, int64_t objectStoreID, const IDBKeyData&);
IDBError uncheckedGetKeyGeneratorValue(int64_t objectStoreID, uint64_t& outValue);
IDBError uncheckedSetKeyGeneratorValue(int64_t objectStoreID, uint64_t value);
IDBError updateAllIndexesForAddRecord(const IDBObjectStoreInfo&, const IDBKeyData&, const ThreadSafeDataBuffer& value, int64_t recordID);
IDBError updateOneIndexForAddRecord(const IDBIndexInfo&, const IDBKeyData&, const ThreadSafeDataBuffer& value, int64_t recordID);
IDBError uncheckedPutIndexKey(const IDBIndexInfo&, const IDBKeyData& keyValue, const IndexKey&, int64_t recordID);
IDBError uncheckedPutIndexRecord(int64_t objectStoreID, int64_t indexID, const IDBKeyData& keyValue, const IDBKeyData& indexKey, int64_t recordID);
IDBError uncheckedHasIndexRecord(const IDBIndexInfo&, const IDBKeyData&, bool& hasRecord);
IDBError uncheckedGetIndexRecordForOneKey(int64_t indexeID, int64_t objectStoreID, IndexedDB::IndexRecordType, const IDBKeyData&, IDBGetResult&);
IDBError deleteUnusedBlobFileRecords(SQLiteIDBTransaction&);
IDBError getAllObjectStoreRecords(const IDBResourceIdentifier& transactionIdentifier, const IDBGetAllRecordsData&, IDBGetAllResult& outValue);
IDBError getAllIndexRecords(const IDBResourceIdentifier& transactionIdentifier, const IDBGetAllRecordsData&, IDBGetAllResult& outValue);
void closeSQLiteDB();
void close() final;
uint64_t databaseSize() const final;
enum class SQL : size_t {
CreateObjectStoreInfo,
CreateObjectStoreKeyGenerator,
DeleteObjectStoreInfo,
DeleteObjectStoreKeyGenerator,
DeleteObjectStoreRecords,
DeleteObjectStoreIndexInfo,
DeleteObjectStoreIndexRecords,
DeleteObjectStoreBlobRecords,
RenameObjectStore,
ClearObjectStoreRecords,
ClearObjectStoreIndexRecords,
CreateIndexInfo,
DeleteIndexInfo,
HasIndexRecord,
PutIndexRecord,
GetIndexRecordForOneKey,
DeleteIndexRecords,
RenameIndex,
KeyExistsInObjectStore,
GetUnusedBlobFilenames,
DeleteUnusedBlobs,
GetObjectStoreRecordID,
DeleteBlobRecord,
DeleteObjectStoreRecord,
DeleteObjectStoreIndexRecord,
AddObjectStoreRecord,
AddBlobRecord,
BlobFilenameForBlobURL,
AddBlobFilename,
GetBlobURL,
GetKeyGeneratorValue,
SetKeyGeneratorValue,
GetAllKeyRecordsLowerOpenUpperOpen,
GetAllKeyRecordsLowerOpenUpperClosed,
GetAllKeyRecordsLowerClosedUpperOpen,
GetAllKeyRecordsLowerClosedUpperClosed,
GetValueRecordsLowerOpenUpperOpen,
GetValueRecordsLowerOpenUpperClosed,
GetValueRecordsLowerClosedUpperOpen,
GetValueRecordsLowerClosedUpperClosed,
GetKeyRecordsLowerOpenUpperOpen,
GetKeyRecordsLowerOpenUpperClosed,
GetKeyRecordsLowerClosedUpperOpen,
GetKeyRecordsLowerClosedUpperClosed,
CountRecordsLowerOpenUpperOpen,
CountRecordsLowerOpenUpperClosed,
CountRecordsLowerClosedUpperOpen,
CountRecordsLowerClosedUpperClosed,
CountIndexRecordsLowerOpenUpperOpen,
CountIndexRecordsLowerOpenUpperClosed,
CountIndexRecordsLowerClosedUpperOpen,
CountIndexRecordsLowerClosedUpperClosed,
Invalid,
};
SQLiteStatement* cachedStatement(SQL, const char*);
SQLiteStatement* cachedStatementForGetAllObjectStoreRecords(const IDBGetAllRecordsData&);
std::unique_ptr<SQLiteStatement> m_cachedStatements[static_cast<int>(SQL::Invalid)];
PAL::SessionID m_sessionID;
IDBDatabaseIdentifier m_identifier;
std::unique_ptr<IDBDatabaseInfo> m_databaseInfo;
std::unique_ptr<IDBDatabaseInfo> m_originalDatabaseInfoBeforeVersionChange;
std::unique_ptr<SQLiteDatabase> m_sqliteDB;
HashMap<IDBResourceIdentifier, std::unique_ptr<SQLiteIDBTransaction>> m_transactions;
HashMap<IDBResourceIdentifier, SQLiteIDBCursor*> m_cursors;
String m_databaseRootDirectory;
String m_databaseDirectory;
IDBBackingStoreTemporaryFileHandler& m_temporaryFileHandler;
Ref<IDBSerializationContext> m_serializationContext;
};
} // namespace IDBServer
} // namespace WebCore
#endif // ENABLE(INDEXED_DATABASE)