[GTK] Add initial database process support
[WebKit-https.git] / Source / WebKit2 / DatabaseProcess / IndexedDB / DatabaseProcessIDBConnection.cpp
1 /*
2  * Copyright (C) 2013 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "DatabaseProcessIDBConnection.h"
28
29 #if ENABLE(INDEXED_DATABASE) && ENABLE(DATABASE_PROCESS)
30
31 #include "DataReference.h"
32 #include "DatabaseProcess.h"
33 #include "DatabaseToWebProcessConnection.h"
34 #include "IDBIdentifier.h"
35 #include "Logging.h"
36 #include "UniqueIDBDatabase.h"
37 #include "WebCoreArgumentCoders.h"
38 #include "WebIDBServerConnectionMessages.h"
39 #include <WebCore/IDBDatabaseMetadata.h>
40 #include <WebCore/IDBIndexMetadata.h>
41 #include <WebCore/IDBObjectStoreMetadata.h>
42 #include <WebCore/IDBServerConnection.h>
43 #include <WebCore/IndexedDB.h>
44
45 using namespace WebCore;
46
47 namespace WebKit {
48
49 DatabaseProcessIDBConnection::DatabaseProcessIDBConnection(DatabaseToWebProcessConnection& connection, uint64_t serverConnectionIdentifier)
50     : m_connection(connection)
51     , m_serverConnectionIdentifier(serverConnectionIdentifier)
52 {
53 }
54
55 DatabaseProcessIDBConnection::~DatabaseProcessIDBConnection()
56 {
57     ASSERT(!m_uniqueIDBDatabase);
58 }
59
60 void DatabaseProcessIDBConnection::disconnectedFromWebProcess()
61 {
62     // It's possible that the m_uniqueIDBDatabase pointer has already been cleared
63     // if the represented database was deleted, or if it was closed between delete
64     // and the delete callback
65     if (!m_uniqueIDBDatabase)
66         return;
67
68     m_uniqueIDBDatabase->unregisterConnection(*this);
69     m_uniqueIDBDatabase = nullptr;
70 }
71
72 void DatabaseProcessIDBConnection::establishConnection(const String& databaseName, const SecurityOriginData& openingOrigin, const SecurityOriginData& mainFrameOrigin)
73 {
74     m_uniqueIDBDatabase = DatabaseProcess::shared().getOrCreateUniqueIDBDatabase(UniqueIDBDatabaseIdentifier(databaseName, openingOrigin, mainFrameOrigin));
75     m_uniqueIDBDatabase->registerConnection(*this);
76 }
77
78 void DatabaseProcessIDBConnection::getOrEstablishIDBDatabaseMetadata(uint64_t requestID)
79 {
80     ASSERT(m_uniqueIDBDatabase);
81
82     LOG(IDB, "DatabaseProcess getOrEstablishIDBDatabaseMetadata request ID %llu (connection %p)", requestID, this);
83
84     RefPtr<DatabaseProcessIDBConnection> connection(this);
85     m_uniqueIDBDatabase->getOrEstablishIDBDatabaseMetadata([connection, requestID](bool success, const IDBDatabaseMetadata& metadata) {
86         connection->send(Messages::WebIDBServerConnection::DidGetOrEstablishIDBDatabaseMetadata(requestID, success, metadata));
87     });
88 }
89
90 void DatabaseProcessIDBConnection::deleteDatabase(uint64_t requestID, const String& databaseName)
91 {
92     ASSERT(m_uniqueIDBDatabase);
93
94     LOG(IDB, "DatabaseProcess deleteDatabase request ID %llu (connection %p)", requestID, this);
95
96     if (databaseName != m_uniqueIDBDatabase->identifier().databaseName()) {
97         LOG_ERROR("Request to delete database name that doesn't match with this database connection's database name");
98         send(Messages::WebIDBServerConnection::DidDeleteDatabase(requestID, false));
99     }
100
101     RefPtr<DatabaseProcessIDBConnection> connection(this);
102     m_uniqueIDBDatabase->deleteDatabase([connection, this, requestID](bool success) {
103         if (success)
104             disconnectedFromWebProcess();
105
106         connection->send(Messages::WebIDBServerConnection::DidDeleteDatabase(requestID, success));
107     });
108 }
109
110 void DatabaseProcessIDBConnection::openTransaction(uint64_t requestID, int64_t transactionID, const Vector<int64_t>& objectStoreIDs, uint64_t intMode)
111 {
112     ASSERT(m_uniqueIDBDatabase);
113
114 #ifndef NDEBUG
115     const char* modeString = nullptr;
116     switch (static_cast<IndexedDB::TransactionMode>(intMode)) {
117     case IndexedDB::TransactionMode::ReadOnly:
118         modeString = "readonly";
119         break;
120     case IndexedDB::TransactionMode::ReadWrite:
121         modeString = "readwrite";
122         break;
123     case IndexedDB::TransactionMode::VersionChange:
124         modeString = "versionchange";
125         break;
126     }
127
128     LOG(IDB, "DatabaseProcess openTransaction id %llu in mode '%s', request ID %llu", transactionID, modeString, requestID);
129 #endif
130
131     if (intMode > IndexedDB::TransactionModeMaximum) {
132         send(Messages::WebIDBServerConnection::DidOpenTransaction(requestID, false));
133         return;
134     }
135
136     IndexedDB::TransactionMode mode = static_cast<IndexedDB::TransactionMode>(intMode);
137     RefPtr<DatabaseProcessIDBConnection> connection(this);
138     m_uniqueIDBDatabase->openTransaction(IDBIdentifier(*this, transactionID), objectStoreIDs, mode, [connection, requestID](bool success) {
139         connection->send(Messages::WebIDBServerConnection::DidOpenTransaction(requestID, success));
140     });
141 }
142
143 void DatabaseProcessIDBConnection::beginTransaction(uint64_t requestID, int64_t transactionID)
144 {
145     ASSERT(m_uniqueIDBDatabase);
146
147     LOG(IDB, "DatabaseProcess beginTransaction request ID %llu", requestID);
148
149     RefPtr<DatabaseProcessIDBConnection> connection(this);
150     m_uniqueIDBDatabase->beginTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) {
151         connection->send(Messages::WebIDBServerConnection::DidBeginTransaction(requestID, success));
152     });
153 }
154
155 void DatabaseProcessIDBConnection::commitTransaction(uint64_t requestID, int64_t transactionID)
156 {
157     ASSERT(m_uniqueIDBDatabase);
158
159     LOG(IDB, "DatabaseProcess commitTransaction request ID %llu", requestID);
160
161     RefPtr<DatabaseProcessIDBConnection> connection(this);
162     m_uniqueIDBDatabase->commitTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) {
163         connection->send(Messages::WebIDBServerConnection::DidCommitTransaction(requestID, success));
164     });
165 }
166
167 void DatabaseProcessIDBConnection::resetTransaction(uint64_t requestID, int64_t transactionID)
168 {
169     ASSERT(m_uniqueIDBDatabase);
170
171     LOG(IDB, "DatabaseProcess resetTransaction request ID %llu", requestID);
172
173     RefPtr<DatabaseProcessIDBConnection> connection(this);
174     m_uniqueIDBDatabase->resetTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) {
175         connection->send(Messages::WebIDBServerConnection::DidResetTransaction(requestID, success));
176     });
177 }
178
179 void DatabaseProcessIDBConnection::rollbackTransaction(uint64_t requestID, int64_t transactionID)
180 {
181     ASSERT(m_uniqueIDBDatabase);
182
183     LOG(IDB, "DatabaseProcess rollbackTransaction request ID %llu", requestID);
184
185     RefPtr<DatabaseProcessIDBConnection> connection(this);
186     m_uniqueIDBDatabase->rollbackTransaction(IDBIdentifier(*this, transactionID), [connection, requestID](bool success) {
187         connection->send(Messages::WebIDBServerConnection::DidRollbackTransaction(requestID, success));
188     });
189 }
190
191 void DatabaseProcessIDBConnection::resetTransactionSync(int64_t transactionID, PassRefPtr<Messages::DatabaseProcessIDBConnection::ResetTransactionSync::DelayedReply> prpReply)
192 {
193     RefPtr<Messages::DatabaseProcessIDBConnection::ResetTransactionSync::DelayedReply> reply(prpReply);
194     RefPtr<DatabaseProcessIDBConnection> connection(this);
195     m_uniqueIDBDatabase->resetTransaction(IDBIdentifier(*this, transactionID), [connection, reply](bool success) {
196         reply->send(success);
197     });
198 }
199
200 void DatabaseProcessIDBConnection::rollbackTransactionSync(int64_t transactionID, PassRefPtr<Messages::DatabaseProcessIDBConnection::RollbackTransactionSync::DelayedReply> prpReply)
201 {
202     RefPtr<Messages::DatabaseProcessIDBConnection::RollbackTransactionSync::DelayedReply> reply(prpReply);
203     RefPtr<DatabaseProcessIDBConnection> connection(this);
204     m_uniqueIDBDatabase->rollbackTransaction(IDBIdentifier(*this, transactionID), [connection, reply](bool success) {
205         reply->send(success);
206     });
207 }
208
209 void DatabaseProcessIDBConnection::changeDatabaseVersion(uint64_t requestID, int64_t transactionID, uint64_t newVersion)
210 {
211     ASSERT(m_uniqueIDBDatabase);
212
213     LOG(IDB, "DatabaseProcess changeDatabaseVersion request ID %llu, new version %llu", requestID, newVersion);
214
215     RefPtr<DatabaseProcessIDBConnection> connection(this);
216     m_uniqueIDBDatabase->changeDatabaseVersion(IDBIdentifier(*this, transactionID), newVersion, [connection, requestID](bool success) {
217         connection->send(Messages::WebIDBServerConnection::DidChangeDatabaseVersion(requestID, success));
218     });
219 }
220
221 void DatabaseProcessIDBConnection::createObjectStore(uint64_t requestID, int64_t transactionID, IDBObjectStoreMetadata metadata)
222 {
223     ASSERT(m_uniqueIDBDatabase);
224
225     LOG(IDB, "DatabaseProcess createObjectStore request ID %llu, object store name '%s'", requestID, metadata.name.utf8().data());
226     RefPtr<DatabaseProcessIDBConnection> connection(this);
227     m_uniqueIDBDatabase->createObjectStore(IDBIdentifier(*this, transactionID), metadata, [connection, requestID](bool success) {
228         connection->send(Messages::WebIDBServerConnection::DidCreateObjectStore(requestID, success));
229     });
230 }
231
232 void DatabaseProcessIDBConnection::deleteObjectStore(uint64_t requestID, int64_t transactionID, int64_t objectStoreID)
233 {
234     ASSERT(m_uniqueIDBDatabase);
235
236     LOG(IDB, "DatabaseProcess deleteObjectStore request ID %llu, object store id %lli", requestID, objectStoreID);
237     RefPtr<DatabaseProcessIDBConnection> connection(this);
238     m_uniqueIDBDatabase->deleteObjectStore(IDBIdentifier(*this, transactionID), objectStoreID, [connection, requestID](bool success) {
239         connection->send(Messages::WebIDBServerConnection::DidDeleteObjectStore(requestID, success));
240     });
241 }
242
243 void DatabaseProcessIDBConnection::clearObjectStore(uint64_t requestID, int64_t transactionID, int64_t objectStoreID)
244 {
245     ASSERT(m_uniqueIDBDatabase);
246
247     LOG(IDB, "DatabaseProcess clearObjectStore request ID %llu, object store id %lli", requestID, objectStoreID);
248     RefPtr<DatabaseProcessIDBConnection> connection(this);
249     m_uniqueIDBDatabase->clearObjectStore(IDBIdentifier(*this, transactionID), objectStoreID, [connection, requestID](bool success) {
250         connection->send(Messages::WebIDBServerConnection::DidClearObjectStore(requestID, success));
251     });
252 }
253
254 void DatabaseProcessIDBConnection::createIndex(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, const IDBIndexMetadata& metadata)
255 {
256     ASSERT(m_uniqueIDBDatabase);
257
258     LOG(IDB, "DatabaseProcess createIndex request ID %llu, object store id %lli", requestID, objectStoreID);
259     RefPtr<DatabaseProcessIDBConnection> connection(this);
260     m_uniqueIDBDatabase->createIndex(IDBIdentifier(*this, transactionID), objectStoreID, metadata, [connection, requestID](bool success) {
261         connection->send(Messages::WebIDBServerConnection::DidCreateIndex(requestID, success));
262     });
263 }
264
265 void DatabaseProcessIDBConnection::deleteIndex(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID)
266 {
267     ASSERT(m_uniqueIDBDatabase);
268
269     LOG(IDB, "DatabaseProcess deleteIndex request ID %llu, object store id %lli", requestID, objectStoreID);
270     RefPtr<DatabaseProcessIDBConnection> connection(this);
271     m_uniqueIDBDatabase->deleteIndex(IDBIdentifier(*this, transactionID), objectStoreID, indexID, [connection, requestID](bool success) {
272         connection->send(Messages::WebIDBServerConnection::DidDeleteIndex(requestID, success));
273     });
274 }
275
276 void DatabaseProcessIDBConnection::putRecord(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, const IDBKeyData& key, const IPC::DataReference& value, int64_t putMode, const Vector<int64_t>& indexIDs, const Vector<Vector<IDBKeyData>>& indexKeys)
277 {
278     ASSERT(m_uniqueIDBDatabase);
279
280     LOG(IDB, "DatabaseProcess putRecord request ID %llu, object store id %lli", requestID, objectStoreID);
281     RefPtr<DatabaseProcessIDBConnection> connection(this);
282     m_uniqueIDBDatabase->putRecord(IDBIdentifier(*this, transactionID), objectStoreID, key, value, putMode, indexIDs, indexKeys, [connection, requestID](const IDBKeyData& keyData, uint32_t errorCode, const String& errorMessage) {
283         connection->send(Messages::WebIDBServerConnection::DidPutRecord(requestID, keyData, errorCode, errorMessage));
284     });
285 }
286
287 void DatabaseProcessIDBConnection::getRecord(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID, const IDBKeyRangeData& keyRange, int64_t cursorType)
288 {
289     ASSERT(m_uniqueIDBDatabase);
290
291     LOG(IDB, "DatabaseProcess getRecord request ID %llu, object store id %lli", requestID, objectStoreID);
292     RefPtr<DatabaseProcessIDBConnection> connection(this);
293     m_uniqueIDBDatabase->getRecord(IDBIdentifier(*this, transactionID), objectStoreID, indexID, keyRange, static_cast<IndexedDB::CursorType>(cursorType), [connection, requestID](const IDBGetResult& getResult, uint32_t errorCode, const String& errorMessage) {
294         connection->send(Messages::WebIDBServerConnection::DidGetRecord(requestID, getResult, errorCode, errorMessage));
295     });
296 }
297
298 void DatabaseProcessIDBConnection::count(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, int64_t indexID, const IDBKeyRangeData& keyRangeData)
299 {
300     ASSERT(m_uniqueIDBDatabase);
301
302     LOG(IDB, "DatabaseProcess count request ID %llu, object store id %lli", requestID, objectStoreID);
303
304     RefPtr<DatabaseProcessIDBConnection> connection(this);
305     m_uniqueIDBDatabase->count(IDBIdentifier(*this, transactionID), objectStoreID, indexID, keyRangeData, [connection, requestID](int64_t count, uint32_t errorCode, const String& errorMessage) {
306         connection->send(Messages::WebIDBServerConnection::DidCount(requestID, count, errorCode, errorMessage));
307     });
308 }
309
310 void DatabaseProcessIDBConnection::deleteRange(uint64_t requestID, int64_t transactionID, int64_t objectStoreID, const IDBKeyRangeData& keyRangeData)
311 {
312     ASSERT(m_uniqueIDBDatabase);
313
314     LOG(IDB, "DatabaseProcess deleteRange request ID %llu, object store id %lli", requestID, objectStoreID);
315
316     RefPtr<DatabaseProcessIDBConnection> connection(this);
317     m_uniqueIDBDatabase->deleteRange(IDBIdentifier(*this, transactionID), objectStoreID, keyRangeData, [connection, requestID](uint32_t errorCode, const String& errorMessage) {
318         connection->send(Messages::WebIDBServerConnection::DidDeleteRange(requestID, errorCode, errorMessage));
319     });
320 }
321
322 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)
323 {
324     ASSERT(m_uniqueIDBDatabase);
325
326     LOG(IDB, "DatabaseProcess openCursor request ID %llu, object store id %lli", requestID, objectStoreID);
327     RefPtr<DatabaseProcessIDBConnection> connection(this);
328     m_uniqueIDBDatabase->openCursor(IDBIdentifier(*this, transactionID), objectStoreID, indexID, static_cast<IndexedDB::CursorDirection>(cursorDirection), static_cast<IndexedDB::CursorType>(cursorType), static_cast<IDBDatabaseBackend::TaskType>(taskType), keyRangeData, [connection, requestID](int64_t cursorID, const IDBKeyData& resultKey, const IDBKeyData& primaryKey, PassRefPtr<SharedBuffer> value, uint32_t errorCode, const String& errorMessage) {
329         IPC::DataReference data = value ? IPC::DataReference(reinterpret_cast<const uint8_t*>(value->data()), value->size()) : IPC::DataReference();
330         connection->send(Messages::WebIDBServerConnection::DidOpenCursor(requestID, cursorID, resultKey, primaryKey, data, errorCode, errorMessage));
331     });
332 }
333
334 void DatabaseProcessIDBConnection::cursorAdvance(uint64_t requestID, int64_t cursorID, uint64_t count)
335 {
336     ASSERT(m_uniqueIDBDatabase);
337
338     LOG(IDB, "DatabaseProcess cursorAdvance request ID %llu, cursor id %lli", requestID, cursorID);
339     RefPtr<DatabaseProcessIDBConnection> connection(this);
340     m_uniqueIDBDatabase->cursorAdvance(IDBIdentifier(*this, cursorID), count, [connection, requestID](const IDBKeyData& resultKey, const IDBKeyData& primaryKey, PassRefPtr<SharedBuffer> value, uint32_t errorCode, const String& errorMessage) {
341         IPC::DataReference data = value ? IPC::DataReference(reinterpret_cast<const uint8_t*>(value->data()), value->size()) : IPC::DataReference();
342         connection->send(Messages::WebIDBServerConnection::DidAdvanceCursor(requestID, resultKey, primaryKey, data, errorCode, errorMessage));
343     });
344 }
345
346 void DatabaseProcessIDBConnection::cursorIterate(uint64_t requestID, int64_t cursorID, const IDBKeyData& key)
347 {
348     ASSERT(m_uniqueIDBDatabase);
349
350     LOG(IDB, "DatabaseProcess cursorIterate request ID %llu, cursor id %lli", requestID, cursorID);
351     RefPtr<DatabaseProcessIDBConnection> connection(this);
352     m_uniqueIDBDatabase->cursorIterate(IDBIdentifier(*this, cursorID), key, [connection, requestID](const IDBKeyData& resultKey, const IDBKeyData& primaryKey, PassRefPtr<SharedBuffer> value, uint32_t errorCode, const String& errorMessage) {
353         IPC::DataReference data = value ? IPC::DataReference(reinterpret_cast<const uint8_t*>(value->data()), value->size()) : IPC::DataReference();
354         connection->send(Messages::WebIDBServerConnection::DidIterateCursor(requestID, resultKey, primaryKey, data, errorCode, errorMessage));
355     });
356 }
357
358 void DatabaseProcessIDBConnection::close()
359 {
360     LOG(IDB, "DatabaseProcessIDBConnection close");
361
362     disconnectedFromWebProcess();
363 }
364
365 IPC::Connection* DatabaseProcessIDBConnection::messageSenderConnection()
366 {
367     return m_connection->connection();
368 }
369
370 } // namespace WebKit
371
372 #endif // ENABLE(INDEXED_DATABASE) && ENABLE(DATABASE_PROCESS)