Unreviewed, rolling out r243807 and r243824.
authorcommit-queue@webkit.org <commit-queue@webkit.org@268f45cc-cd09-0410-ab3c-d52691b4dbfc>
Thu, 4 Apr 2019 21:59:02 +0000 (21:59 +0000)
committercommit-queue@webkit.org <commit-queue@webkit.org@268f45cc-cd09-0410-ab3c-d52691b4dbfc>
Thu, 4 Apr 2019 21:59:02 +0000 (21:59 +0000)
https://bugs.webkit.org/show_bug.cgi?id=196611

Test added is a flaky timeout on iOS Simulator, 3 tests
unskipped are flaky failures (Requested by ShawnRoberts on
#webkit).

Reverted changesets:

"Blob type cannot be stored correctly in IDB when
IDBObjectStore has autoIncrement and keyPath options"
https://bugs.webkit.org/show_bug.cgi?id=196128
https://trac.webkit.org/changeset/243807

"Follow up fix for r243807: Use MarkedArgumentBuffer instead
of Vector for JSValue"
https://bugs.webkit.org/show_bug.cgi?id=196547
https://trac.webkit.org/changeset/243824

git-svn-id: https://svn.webkit.org/repository/webkit/trunk@243910 268f45cc-cd09-0410-ab3c-d52691b4dbfc

34 files changed:
LayoutTests/ChangeLog
LayoutTests/TestExpectations
LayoutTests/imported/w3c/ChangeLog
LayoutTests/imported/w3c/web-platform-tests/IndexedDB/nested-cloning-large-expected.txt
LayoutTests/imported/w3c/web-platform-tests/IndexedDB/nested-cloning-large-multiple-expected.txt
LayoutTests/imported/w3c/web-platform-tests/IndexedDB/nested-cloning-small-expected.txt
LayoutTests/storage/indexeddb/modern/objectstore-autoincrement-types-expected.txt [deleted file]
LayoutTests/storage/indexeddb/modern/objectstore-autoincrement-types.html [deleted file]
LayoutTests/storage/indexeddb/modern/resources/objectstore-autoincrement-types.js [deleted file]
Source/WebCore/ChangeLog
Source/WebCore/Modules/indexeddb/IDBCursor.cpp
Source/WebCore/Modules/indexeddb/IDBCursor.h
Source/WebCore/Modules/indexeddb/IDBGetAllResult.cpp
Source/WebCore/Modules/indexeddb/IDBGetAllResult.h
Source/WebCore/Modules/indexeddb/IDBGetResult.cpp
Source/WebCore/Modules/indexeddb/IDBGetResult.h
Source/WebCore/Modules/indexeddb/IDBRequest.cpp
Source/WebCore/Modules/indexeddb/IDBRequest.h
Source/WebCore/Modules/indexeddb/IDBTransaction.cpp
Source/WebCore/Modules/indexeddb/server/MemoryIDBBackingStore.cpp
Source/WebCore/Modules/indexeddb/server/MemoryIndex.cpp
Source/WebCore/Modules/indexeddb/server/MemoryIndexCursor.cpp
Source/WebCore/Modules/indexeddb/server/MemoryObjectStore.cpp
Source/WebCore/Modules/indexeddb/server/MemoryObjectStoreCursor.cpp
Source/WebCore/Modules/indexeddb/server/SQLiteIDBBackingStore.cpp
Source/WebCore/Modules/indexeddb/server/SQLiteIDBCursor.cpp
Source/WebCore/Modules/indexeddb/server/SQLiteIDBCursor.h
Source/WebCore/Modules/indexeddb/server/UniqueIDBDatabase.cpp
Source/WebCore/Modules/indexeddb/shared/IDBResultData.cpp
Source/WebCore/Modules/indexeddb/shared/IDBResultData.h
Source/WebCore/bindings/js/IDBBindingUtilities.cpp
Source/WebCore/bindings/js/IDBBindingUtilities.h
Source/WebCore/bindings/js/JSIDBCursorWithValueCustom.cpp
Source/WebCore/bindings/js/JSIDBRequestCustom.cpp

index 82c27f5..1e54d96 100644 (file)
@@ -1,3 +1,24 @@
+2019-04-04  Commit Queue  <commit-queue@webkit.org>
+
+        Unreviewed, rolling out r243807 and r243824.
+        https://bugs.webkit.org/show_bug.cgi?id=196611
+
+        Test added is a flaky timeout on iOS Simulator, 3 tests
+        unskipped are flaky failures (Requested by ShawnRoberts on
+        #webkit).
+
+        Reverted changesets:
+
+        "Blob type cannot be stored correctly in IDB when
+        IDBObjectStore has autoIncrement and keyPath options"
+        https://bugs.webkit.org/show_bug.cgi?id=196128
+        https://trac.webkit.org/changeset/243807
+
+        "Follow up fix for r243807: Use MarkedArgumentBuffer instead
+        of Vector for JSValue"
+        https://bugs.webkit.org/show_bug.cgi?id=196547
+        https://trac.webkit.org/changeset/243824
+
 2019-04-04  Eric Carlson  <eric.carlson@apple.com>
 
         [MediaStream] Host should be able to mute screen capture and camera/microphone independently
index 9eecd45..3f6b21c 100644 (file)
@@ -358,6 +358,8 @@ webkit.org/b/179608 imported/w3c/web-platform-tests/xhr/access-control-preflight
 webkit.org/b/179608 imported/w3c/web-platform-tests/xhr/send-conditional-cors.htm [ Failure ]
 webkit.org/b/179611 imported/w3c/web-platform-tests/xhr/send-entity-body-document.htm [ Pass Failure ]
 imported/w3c/web-platform-tests/IndexedDB/nested-cloning-large-multiple.html [ Skip ]
+imported/w3c/web-platform-tests/IndexedDB/nested-cloning-large.html [ Skip ]
+imported/w3c/web-platform-tests/IndexedDB/nested-cloning-small.html [ Skip ]
 imported/w3c/web-platform-tests/css/cssom/interfaces.html [ Pass Timeout ]
 [ Debug ] imported/w3c/web-platform-tests/css/cssom-view/interfaces.html [ Skip ]
 webkit.org/b/182292 imported/w3c/web-platform-tests/css/cssom-view/scrollingElement-quirks-dynamic-001.html [ ImageOnlyFailure ]
index 1266bfd..bff1c59 100644 (file)
@@ -1,3 +1,24 @@
+2019-04-04  Commit Queue  <commit-queue@webkit.org>
+
+        Unreviewed, rolling out r243807 and r243824.
+        https://bugs.webkit.org/show_bug.cgi?id=196611
+
+        Test added is a flaky timeout on iOS Simulator, 3 tests
+        unskipped are flaky failures (Requested by ShawnRoberts on
+        #webkit).
+
+        Reverted changesets:
+
+        "Blob type cannot be stored correctly in IDB when
+        IDBObjectStore has autoIncrement and keyPath options"
+        https://bugs.webkit.org/show_bug.cgi?id=196128
+        https://trac.webkit.org/changeset/243807
+
+        "Follow up fix for r243807: Use MarkedArgumentBuffer instead
+        of Vector for JSValue"
+        https://bugs.webkit.org/show_bug.cgi?id=196547
+        https://trac.webkit.org/changeset/243824
+
 2019-04-03  Chris Dumez  <cdumez@apple.com>
 
         HTML fragment serialization should not strip whitespace from URL attribute values
index e0e61e7..2125ef0 100644 (file)
@@ -1,9 +1,9 @@
 
 PASS large typed array 
 PASS blob with large typed array 
-PASS blob with large typed array with key generator 
+FAIL blob with large typed array with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 PASS array of blobs and large typed arrays 
-PASS array of blobs and large typed arrays with key generator 
+FAIL array of blobs and large typed arrays with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 PASS object with blobs and large typed arrays 
-PASS object with blobs and large typed arrays with key generator 
+FAIL object with blobs and large typed arrays with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 
index f8134a4..064bf4a 100644 (file)
@@ -1,4 +1,4 @@
 
 PASS multiple requests of objects with blobs and large typed arrays 
-PASS multiple requests of objects with blobs and large typed arrays with key generator 
+FAIL multiple requests of objects with blobs and large typed arrays with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 
index 468ca0b..66b5fc4 100644 (file)
@@ -2,9 +2,9 @@
 PASS small typed array 
 PASS blob 
 PASS blob with small typed array 
-PASS blob with small typed array with key generator 
+FAIL blob with small typed array with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 PASS blob array 
-PASS blob array with key generator 
+FAIL blob array with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 PASS array of blobs and small typed arrays 
-PASS array of blobs and small typed arrays with key generator 
+FAIL array of blobs and small typed arrays with key generator assert_equals: IndexedDB result class should match put() argument expected "[object Blob]" but got "[object Null]"
 
diff --git a/LayoutTests/storage/indexeddb/modern/objectstore-autoincrement-types-expected.txt b/LayoutTests/storage/indexeddb/modern/objectstore-autoincrement-types-expected.txt
deleted file mode 100644 (file)
index 0ee9564..0000000
+++ /dev/null
@@ -1,27 +0,0 @@
-Test IndexedDB's IDBObjectStore auto-increment feature with more types.
-
-On success, you will see a series of "PASS" messages, followed by "TEST COMPLETE".
-
-
-indexedDB = self.indexedDB || self.webkitIndexedDB || self.mozIndexedDB || self.msIndexedDB || self.OIndexedDB;
-
-indexedDB.deleteDatabase(dbname)
-indexedDB.open(dbname)
-createObjectStore():
-store = db.createObjectStore('Store', {keyPath: 'primaryKey', autoIncrement: true})
-PASS request.result.size is blob.size
-PASS request.result.type is blob.type
-PASS request.result.primaryKey is primaryKey
-PASS request.result.name is file.name
-PASS request.result.lastModified is file.lastModified
-PASS request.result.primaryKey is primaryKey
-PASS request.result.height is imageData.height
-PASS request.result.width is imageData.width
-PASS JSON.stringify(request.result.data) is JSON.stringify(imageData.data)
-PASS request.result.primaryKey is primaryKey
-PASS request.result.length is fileList.length
-PASS request.result.primaryKey is primaryKey
-PASS successfullyParsed is true
-
-TEST COMPLETE
-
diff --git a/LayoutTests/storage/indexeddb/modern/objectstore-autoincrement-types.html b/LayoutTests/storage/indexeddb/modern/objectstore-autoincrement-types.html
deleted file mode 100644 (file)
index 9f4414b..0000000
+++ /dev/null
@@ -1,10 +0,0 @@
-<html>
-<head>
-<script src="../../../resources/js-test.js"></script>
-<script src="../resources/shared.js"></script>
-</head>
-<body>
-<input id="fileInput" multiple type="file">
-<script src="resources/objectstore-autoincrement-types.js"></script>
-</body>
-</html>
\ No newline at end of file
diff --git a/LayoutTests/storage/indexeddb/modern/resources/objectstore-autoincrement-types.js b/LayoutTests/storage/indexeddb/modern/resources/objectstore-autoincrement-types.js
deleted file mode 100644 (file)
index 918fc5b..0000000
+++ /dev/null
@@ -1,95 +0,0 @@
-if (this.importScripts) {
-    importScripts('../../../resources/js-test.js');
-    importScripts('shared.js');
-}
-
-description("Test IndexedDB's IDBObjectStore auto-increment feature with more types.");
-
-indexedDBTest(prepareDatabase);
-
-var db;
-var testGenerator;
-function next()
-{
-    testGenerator.next();
-}
-
-function prepareDatabase()
-{
-    db = event.target.result;
-    event.target.transaction.onabort = unexpectedAbortCallback;
-
-    debug("createObjectStore():");
-    evalAndLog("store = db.createObjectStore('Store', {keyPath: 'primaryKey', autoIncrement: true})");
-    event.target.onsuccess = function() {
-        testGenerator = testSteps();
-        testGenerator.next();
-    };
-}
-
-
-function* testSteps()
-{
-    transaction = db.transaction("Store", "readwrite");
-    objectStore = transaction.objectStore("Store");
-    primaryKey = 1;
-
-    // Blob
-    var imageURL = "";
-    const [typePart, partRest] = imageURL.split(',');
-    const contentType = typePart.split(':')[1].split(';')[0];
-    const raw = atob(partRest);
-    const rawLength = raw.length;
-    const uInt8Array = new Uint8Array(rawLength);
-    for (let i = 0; i < rawLength; ++i)
-        uInt8Array[i] = raw.charCodeAt(i);
-    blob = new Blob([uInt8Array], { type: contentType });
-    objectStore.put(blob).onsuccess = next;
-    yield;
-    request = objectStore.get(primaryKey);
-    request.onsuccess = next;
-    yield;
-    shouldBe("request.result.size", "blob.size");
-    shouldBe("request.result.type", "blob.type");
-    shouldBe("request.result.primaryKey", "primaryKey");
-    ++ primaryKey;
-
-    // File
-    file = new File([blob], "Filename");
-    objectStore.put(file).onsuccess = next;
-    yield;
-    request = objectStore.get(primaryKey);
-    request.onsuccess = next;
-    yield;
-    shouldBe("request.result.name", "file.name");
-    shouldBe("request.result.lastModified", "file.lastModified");
-    shouldBe("request.result.primaryKey", "primaryKey");
-    ++ primaryKey;
-
-    // ImageData
-    canvas = document.createElement('canvas');
-    context = canvas.getContext('2d');
-    imageData = context.createImageData(1, 1);
-    objectStore.put(imageData).onsuccess = next;
-    yield;
-    request = objectStore.get(primaryKey);
-    request.onsuccess = next;
-    yield;
-    shouldBe("request.result.height", "imageData.height");
-    shouldBe("request.result.width", "imageData.width");
-    shouldBe("JSON.stringify(request.result.data)", "JSON.stringify(imageData.data)");
-    shouldBe("request.result.primaryKey", "primaryKey");
-    ++ primaryKey;
-
-    // Filelist
-    fileList = document.getElementById("fileInput").files;
-    objectStore.put(fileList).onsuccess = next;
-    yield;
-    request = objectStore.get(primaryKey);
-    request.onsuccess = next;
-    yield;
-    shouldBe("request.result.length", "fileList.length");
-    shouldBe("request.result.primaryKey", "primaryKey");
-
-    finishJSTest();
-}
\ No newline at end of file
index 3bbf3c0..a189b5d 100644 (file)
@@ -1,3 +1,24 @@
+2019-04-04  Commit Queue  <commit-queue@webkit.org>
+
+        Unreviewed, rolling out r243807 and r243824.
+        https://bugs.webkit.org/show_bug.cgi?id=196611
+
+        Test added is a flaky timeout on iOS Simulator, 3 tests
+        unskipped are flaky failures (Requested by ShawnRoberts on
+        #webkit).
+
+        Reverted changesets:
+
+        "Blob type cannot be stored correctly in IDB when
+        IDBObjectStore has autoIncrement and keyPath options"
+        https://bugs.webkit.org/show_bug.cgi?id=196128
+        https://trac.webkit.org/changeset/243807
+
+        "Follow up fix for r243807: Use MarkedArgumentBuffer instead
+        of Vector for JSValue"
+        https://bugs.webkit.org/show_bug.cgi?id=196547
+        https://trac.webkit.org/changeset/243824
+
 2019-04-04  Simon Fraser  <simon.fraser@apple.com>
 
         Fix rare crash under collectRelatedCoordinatedScrollingNodes()
index 933cba5..5d82ba6 100644 (file)
@@ -344,10 +344,8 @@ bool IDBCursor::setGetResult(IDBRequest& request, const IDBGetResult& getResult)
     m_primaryKeyData = getResult.primaryKeyData();
     m_primaryKey = m_primaryKeyData.maybeCreateIDBKey();
 
-    if (isKeyCursorWithValue()) {
+    if (isKeyCursorWithValue())
         m_value = getResult.value();
-        m_keyPath = getResult.keyPath();
-    }
 
     m_gotValue = true;
     return true;
index 1af95a3..9a881bb 100644 (file)
@@ -30,7 +30,6 @@
 #include "ExceptionOr.h"
 #include "IDBCursorDirection.h"
 #include "IDBCursorInfo.h"
-#include "IDBKeyPath.h"
 #include "IDBValue.h"
 #include "JSValueInWrappedObject.h"
 #include <JavaScriptCore/Strong.h>
@@ -60,7 +59,6 @@ public:
     IDBKey* key() { return m_key.get(); };
     IDBKey* primaryKey() { return m_primaryKey.get(); };
     IDBValue value() { return m_value; };
-    const Optional<IDBKeyPath>& primaryKeyPath() { return m_keyPath; };
     JSValueInWrappedObject& keyWrapper() { return m_keyWrapper; }
     JSValueInWrappedObject& primaryKeyWrapper() { return m_primaryKeyWrapper; }
     JSValueInWrappedObject& valueWrapper() { return m_valueWrapper; }
@@ -107,7 +105,6 @@ private:
     IDBKeyData m_keyData;
     IDBKeyData m_primaryKeyData;
     IDBValue m_value;
-    Optional<IDBKeyPath> m_keyPath;
 
     JSValueInWrappedObject m_keyWrapper;
     JSValueInWrappedObject m_primaryKeyWrapper;
index 5847da8..4d2624d 100644 (file)
 
 #if ENABLE(INDEXED_DATABASE)
 
-#include <wtf/CrossThreadCopier.h>
 #include <wtf/HashSet.h>
 
 namespace WebCore {
 
+template<typename T> void isolatedCopyOfVariant(const WTF::Variant<Vector<IDBKeyData>, Vector<IDBValue>, std::nullptr_t>& source, WTF::Variant<Vector<IDBKeyData>, Vector<IDBValue>, std::nullptr_t>& target)
+{
+    target = Vector<T>();
+    auto& sourceVector = WTF::get<Vector<T>>(source);
+    auto& targetVector = WTF::get<Vector<T>>(target);
+    targetVector.reserveInitialCapacity(sourceVector.size());
+    for (auto& element : sourceVector)
+        targetVector.uncheckedAppend(element.isolatedCopy());
+}
+
 IDBGetAllResult::IDBGetAllResult(const IDBGetAllResult& that, IsolatedCopyTag)
 {
     isolatedCopy(that, *this);
@@ -46,29 +55,46 @@ IDBGetAllResult IDBGetAllResult::isolatedCopy() const
 void IDBGetAllResult::isolatedCopy(const IDBGetAllResult& source, IDBGetAllResult& destination)
 {
     destination.m_type = source.m_type;
-    destination.m_keys = crossThreadCopy(source.m_keys);
-    destination.m_values = crossThreadCopy(source.m_values);
-    destination.m_keyPath = WebCore::isolatedCopy(source.m_keyPath);
+
+    if (WTF::holds_alternative<std::nullptr_t>(source.m_results))
+        return;
+
+    switch (source.m_type) {
+    case IndexedDB::GetAllType::Keys:
+        isolatedCopyOfVariant<IDBKeyData>(source.m_results, destination.m_results);
+        break;
+    case IndexedDB::GetAllType::Values:
+        isolatedCopyOfVariant<IDBValue>(source.m_results, destination.m_results);
+        break;
+    }
 }
 
 void IDBGetAllResult::addKey(IDBKeyData&& key)
 {
-    m_keys.append(WTFMove(key));
+    ASSERT(m_type == IndexedDB::GetAllType::Keys);
+    ASSERT(WTF::holds_alternative<Vector<IDBKeyData>>(m_results));
+    WTF::get<Vector<IDBKeyData>>(m_results).append(WTFMove(key));
 }
 
 void IDBGetAllResult::addValue(IDBValue&& value)
 {
-    m_values.append(WTFMove(value));
+    ASSERT(m_type == IndexedDB::GetAllType::Values);
+    ASSERT(WTF::holds_alternative<Vector<IDBValue>>(m_results));
+    WTF::get<Vector<IDBValue>>(m_results).append(WTFMove(value));
 }
 
 const Vector<IDBKeyData>& IDBGetAllResult::keys() const
 {
-    return m_keys;
+    ASSERT(m_type == IndexedDB::GetAllType::Keys);
+    ASSERT(WTF::holds_alternative<Vector<IDBKeyData>>(m_results));
+    return WTF::get<Vector<IDBKeyData>>(m_results);
 }
 
 const Vector<IDBValue>& IDBGetAllResult::values() const
 {
-    return m_values;
+    ASSERT(m_type == IndexedDB::GetAllType::Values);
+    ASSERT(WTF::holds_alternative<Vector<IDBValue>>(m_results));
+    return WTF::get<Vector<IDBValue>>(m_results);
 }
 
 Vector<String> IDBGetAllResult::allBlobFilePaths() const
@@ -76,7 +102,7 @@ Vector<String> IDBGetAllResult::allBlobFilePaths() const
     ASSERT(m_type == IndexedDB::GetAllType::Values);
 
     HashSet<String> pathSet;
-    for (auto& value : m_values) {
+    for (auto& value : WTF::get<Vector<IDBValue>>(m_results)) {
         for (auto& path : value.blobFilePaths())
             pathSet.add(path);
     }
index 3f97428..baf8043 100644 (file)
@@ -28,7 +28,6 @@
 #if ENABLE(INDEXED_DATABASE)
 
 #include "IDBKeyData.h"
-#include "IDBKeyPath.h"
 #include "IDBValue.h"
 #include "IndexedDB.h"
 
@@ -43,10 +42,17 @@ public:
     {
     }
 
-    IDBGetAllResult(IndexedDB::GetAllType type, const Optional<IDBKeyPath>& keyPath)
+    IDBGetAllResult(IndexedDB::GetAllType type)
         : m_type(type)
-        , m_keyPath(keyPath)
     {
+        switch (m_type) {
+        case IndexedDB::GetAllType::Keys:
+            m_results = Vector<IDBKeyData>();
+            break;
+        case IndexedDB::GetAllType::Values:
+            m_results = Vector<IDBValue>();
+            break;
+        }
     }
 
     enum IsolatedCopyTag { IsolatedCopy };
@@ -54,7 +60,6 @@ public:
     IDBGetAllResult isolatedCopy() const;
 
     IndexedDB::GetAllType type() const { return m_type; }
-    const Optional<IDBKeyPath>& keyPath() const { return m_keyPath; }
     const Vector<IDBKeyData>& keys() const;
     const Vector<IDBValue>& values() const;
 
@@ -70,15 +75,26 @@ private:
     static void isolatedCopy(const IDBGetAllResult& source, IDBGetAllResult& destination);
 
     IndexedDB::GetAllType m_type { IndexedDB::GetAllType::Keys };
-    Vector<IDBKeyData> m_keys;
-    Vector<IDBValue> m_values;
-    Optional<IDBKeyPath> m_keyPath;
+    WTF::Variant<Vector<IDBKeyData>, Vector<IDBValue>, std::nullptr_t> m_results { nullptr };
 };
 
 template<class Encoder>
 void IDBGetAllResult::encode(Encoder& encoder) const
 {
-    encoder << m_type << m_keys << m_values << m_keyPath;
+    encoder << m_type << static_cast<uint64_t>(m_results.index());
+
+    switch (m_results.index()) {
+    case 0:
+        encoder << WTF::get<Vector<IDBKeyData>>(m_results);
+        break;
+    case 1:
+        encoder << WTF::get<Vector<IDBValue>>(m_results);
+        break;
+    case 2:
+        break;
+    default:
+        RELEASE_ASSERT_NOT_REACHED();
+    }
 }
 
 template<class Decoder>
@@ -87,14 +103,32 @@ bool IDBGetAllResult::decode(Decoder& decoder, IDBGetAllResult& result)
     if (!decoder.decode(result.m_type))
         return false;
 
-    if (!decoder.decode(result.m_keys))
+    uint64_t index;
+    if (!decoder.decode(index))
         return false;
 
-    if (!decoder.decode(result.m_values))
-        return false;
-    
-    if (!decoder.decode(result.m_keyPath))
-        return false;
+    switch (index) {
+    case 0: {
+        result.m_results = Vector<IDBKeyData>();
+        if (!decoder.decode(WTF::get<Vector<IDBKeyData>>(result.m_results)))
+            return false;
+        break;
+    }
+    case 1: {
+        result.m_results = Vector<IDBValue>();
+        Optional<Vector<IDBValue>> optional;
+        decoder >> optional;
+        if (!optional)
+            return false;
+        WTF::get<Vector<IDBValue>>(result.m_results) = WTFMove(*optional);
+        break;
+    }
+    case 2:
+        result.m_results = nullptr;
+        break;
+    default:
+        RELEASE_ASSERT_NOT_REACHED();
+    }
 
     return true;
 }
index 949e813..5f8e98a 100644 (file)
@@ -57,11 +57,6 @@ void IDBGetResult::isolatedCopy(const IDBGetResult& source, IDBGetResult& destin
     destination.m_isDefined = source.m_isDefined;
 }
 
-void IDBGetResult::setValue(IDBValue&& value)
-{
-    m_value = WTFMove(value);
-}
-
 } // namespace WebCore
 
 #endif // ENABLE(INDEXED_DATABASE)
index e87c390..b49f9a6 100644 (file)
@@ -43,36 +43,57 @@ public:
     {
     }
 
+    IDBGetResult(const IDBValue& value, const IDBKeyData& currentPrimaryKey)
+        : m_value(value)
+        , m_primaryKeyData(currentPrimaryKey)
+    {
+    }
+
+    IDBGetResult(const ThreadSafeDataBuffer& buffer)
+        : m_value(buffer)
+    {
+    }
+
+    IDBGetResult(IDBValue&& buffer)
+        : m_value(WTFMove(buffer))
+    {
+    }
+
+    IDBGetResult(IDBKey& key)
+        : m_keyData(&key)
+    {
+    }
+
     IDBGetResult(const IDBKeyData& keyData)
         : m_keyData(keyData)
     {
     }
 
-    IDBGetResult(const IDBKeyData& keyData, const IDBKeyData& primaryKeyData)
-        : m_keyData(keyData)
-        , m_primaryKeyData(primaryKeyData)
+    IDBGetResult(SharedBuffer* buffer, IDBKey& key, const IDBKeyPath& path)
+        : m_keyData(&key)
+        , m_keyPath(path)
     {
+        if (buffer)
+            dataFromBuffer(*buffer);
     }
 
-    IDBGetResult(const IDBKeyData& keyData, const ThreadSafeDataBuffer& buffer, const Optional<IDBKeyPath>& keyPath)
-        : m_value(buffer)
-        , m_keyData(keyData)
-        , m_keyPath(keyPath)
+    IDBGetResult(const IDBKeyData& keyData, const IDBKeyData& primaryKeyData)
+        : m_keyData(keyData)
+        , m_primaryKeyData(primaryKeyData)
     {
     }
 
-    IDBGetResult(const IDBKeyData& keyData, IDBValue&& value, const Optional<IDBKeyPath>& keyPath)
+    IDBGetResult(const IDBKeyData& keyData, const IDBKeyData& primaryKeyData, IDBValue&& value)
         : m_value(WTFMove(value))
         , m_keyData(keyData)
-        , m_keyPath(keyPath)
+        , m_primaryKeyData(primaryKeyData)
     {
     }
 
-    IDBGetResult(const IDBKeyData& keyData, const IDBKeyData& primaryKeyData, IDBValue&& value, const Optional<IDBKeyPath>& keyPath)
-        : m_value(WTFMove(value))
+    IDBGetResult(const IDBKeyData& keyData, const IDBKeyData& primaryKeyData, const IDBValue& value)
+        : m_value(value)
         , m_keyData(keyData)
         , m_primaryKeyData(primaryKeyData)
-        , m_keyPath(keyPath)
     {
     }
 
@@ -81,12 +102,10 @@ public:
 
     IDBGetResult isolatedCopy() const;
 
-    void setValue(IDBValue&&);
-
     const IDBValue& value() const { return m_value; }
     const IDBKeyData& keyData() const { return m_keyData; }
     const IDBKeyData& primaryKeyData() const { return m_primaryKeyData; }
-    const Optional<IDBKeyPath>& keyPath() const { return m_keyPath; }
+    const IDBKeyPath& keyPath() const { return m_keyPath; }
     bool isDefined() const { return m_isDefined; }
 
     template<class Encoder> void encode(Encoder&) const;
@@ -100,7 +119,7 @@ private:
     IDBValue m_value;
     IDBKeyData m_keyData;
     IDBKeyData m_primaryKeyData;
-    Optional<IDBKeyPath> m_keyPath;
+    IDBKeyPath m_keyPath;
     bool m_isDefined { true };
 };
 
index aa7543a..21a68ba 100644 (file)
@@ -394,7 +394,7 @@ void IDBRequest::setResult(const Vector<IDBKeyData>& keyDatas)
     m_resultWrapper = { };
 }
 
-void IDBRequest::setResult(const IDBGetAllResult& result)
+void IDBRequest::setResult(const Vector<IDBValue>& values)
 {
     ASSERT(&originThread() == &Thread::current());
 
@@ -404,7 +404,7 @@ void IDBRequest::setResult(const IDBGetAllResult& result)
 
     VM& vm = context->vm();
     JSLockHolder lock(vm);
-    m_result = result;
+    m_result = values;
     m_resultWrapper = { };
 }
 
@@ -422,7 +422,7 @@ void IDBRequest::setResult(uint64_t number)
     m_resultWrapper = { };
 }
 
-void IDBRequest::setResultToStructuredClone(const IDBGetResult& result)
+void IDBRequest::setResultToStructuredClone(const IDBValue& value)
 {
     ASSERT(&originThread() == &Thread::current());
 
@@ -434,7 +434,7 @@ void IDBRequest::setResultToStructuredClone(const IDBGetResult& result)
 
     VM& vm = context->vm();
     JSLockHolder lock(vm);
-    m_result = result;
+    m_result = value;
     m_resultWrapper = { };
 }
 
index 3f66f63..72202e7 100644 (file)
@@ -31,8 +31,6 @@
 #include "ExceptionOr.h"
 #include "IDBActiveDOMObject.h"
 #include "IDBError.h"
-#include "IDBGetAllResult.h"
-#include "IDBGetResult.h"
 #include "IDBKeyData.h"
 #include "IDBResourceIdentifier.h"
 #include "IDBValue.h"
@@ -78,7 +76,7 @@ public:
 
     virtual ~IDBRequest();
 
-    using Result = Variant<RefPtr<IDBCursor>, RefPtr<IDBDatabase>, IDBKeyData, Vector<IDBKeyData>, IDBGetResult, IDBGetAllResult, uint64_t, NullResultType>;
+    using Result = Variant<RefPtr<IDBCursor>, RefPtr<IDBDatabase>, IDBKeyData, Vector<IDBKeyData>, IDBValue, Vector<IDBValue>, uint64_t, NullResultType>;
     ExceptionOr<Result> result() const;
     JSValueInWrappedObject& resultWrapper() { return m_resultWrapper; }
     JSValueInWrappedObject& cursorWrapper() { return m_cursorWrapper; }
@@ -109,9 +107,9 @@ public:
 
     void setResult(const IDBKeyData&);
     void setResult(const Vector<IDBKeyData>&);
-    void setResultToStructuredClone(const IDBGetResult&);
-    void setResult(const IDBGetAllResult&);
+    void setResult(const Vector<IDBValue>&);
     void setResult(uint64_t);
+    void setResultToStructuredClone(const IDBValue&);
     void setResultToUndefined();
 
     void willIterateCursor(IDBCursor&);
index cb0397c..4cd6e52 100644 (file)
@@ -989,7 +989,7 @@ void IDBTransaction::didGetAllRecordsOnServer(IDBRequest& request, const IDBResu
         request.setResult(getAllResult.keys());
         break;
     case IndexedDB::GetAllType::Values:
-        request.setResult(getAllResult);
+        request.setResult(getAllResult.values());
         break;
     }
 
@@ -1093,7 +1093,7 @@ void IDBTransaction::didGetRecordOnServer(IDBRequest& request, const IDBResultDa
             request.setResultToUndefined();
     } else {
         if (resultData.getResult().value().data().data())
-            request.setResultToStructuredClone(resultData.getResult());
+            request.setResultToStructuredClone(resultData.getResult().value());
         else
             request.setResultToUndefined();
     }
index 7d142b0..f6d1658 100644 (file)
@@ -363,11 +363,9 @@ IDBError MemoryIDBBackingStore::getRecord(const IDBResourceIdentifier& transacti
         return IDBError { UnknownError, "No backing store object store found"_s };
 
     switch (type) {
-    case IDBGetRecordDataType::KeyAndValue: {
-        auto key = objectStore->lowestKeyWithRecordInRange(range);
-        outValue = { key, key.isNull() ? ThreadSafeDataBuffer() : objectStore->valueForKey(key), objectStore->info().keyPath() };
+    case IDBGetRecordDataType::KeyAndValue:
+        outValue = objectStore->valueForKeyRange(range);
         break;
-    }
     case IDBGetRecordDataType::KeyOnly:
         outValue = objectStore->lowestKeyWithRecordInRange(range);
         break;
index 49ddca6..22a6ae8 100644 (file)
@@ -126,7 +126,7 @@ IDBGetResult MemoryIndex::getResultForKeyRange(IndexedDB::IndexRecordType type,
     if (!keyValue)
         return { };
 
-    return type == IndexedDB::IndexRecordType::Key ? IDBGetResult(*keyValue) : IDBGetResult(*keyValue, m_objectStore.valueForKeyRange(*keyValue), m_objectStore.info().keyPath());
+    return type == IndexedDB::IndexRecordType::Key ? IDBGetResult(*keyValue) : IDBGetResult(m_objectStore.valueForKeyRange(*keyValue));
 }
 
 uint64_t MemoryIndex::countForKeyRange(const IDBKeyRangeData& inRange)
@@ -156,7 +156,7 @@ void MemoryIndex::getAllRecords(const IDBKeyRangeData& keyRangeData, Optional<ui
 {
     LOG(IndexedDB, "MemoryIndex::getAllRecords");
 
-    result = { type, m_objectStore.info().keyPath() };
+    result = { type };
 
     if (!m_records)
         return;
@@ -179,8 +179,10 @@ void MemoryIndex::getAllRecords(const IDBKeyRangeData& keyRangeData, Optional<ui
 
         auto allValues = m_records->allValuesForKey(key, targetCount - currentCount);
         for (auto& keyValue : allValues) {
-            result.addKey(IDBKeyData(keyValue));
-            if (type == IndexedDB::GetAllType::Values)
+            if (type == IndexedDB::GetAllType::Keys) {
+                IDBKeyData keyCopy { keyValue };
+                result.addKey(WTFMove(keyCopy));
+            } else
                 result.addValue(m_objectStore.valueForKeyRange(keyValue));
         }
 
index 8fbc890..47b0e95 100644 (file)
@@ -75,7 +75,7 @@ void MemoryIndexCursor::currentData(IDBGetResult& getResult)
         getResult = { m_currentKey, m_currentPrimaryKey };
     else {
         IDBValue value = { m_index.objectStore().valueForKey(m_currentPrimaryKey), { }, { }, { } };
-        getResult = { m_currentKey, m_currentPrimaryKey, WTFMove(value), m_index.objectStore().info().keyPath() };
+        getResult = { m_currentKey, m_currentPrimaryKey, WTFMove(value) };
     }
 }
 
index ed6eba9..30ca143 100644 (file)
@@ -310,7 +310,7 @@ IDBError MemoryObjectStore::updateIndexesForPutRecord(const IDBKeyData& key, con
 
     for (auto& index : m_indexesByName.values()) {
         IndexKey indexKey;
-        generateIndexKeyForValue(UniqueIDBDatabase::databaseThreadExecState(), index->info(), jsValue, indexKey, m_info.keyPath(), key);
+        generateIndexKeyForValue(UniqueIDBDatabase::databaseThreadExecState(), index->info(), jsValue, indexKey);
 
         if (indexKey.isNull())
             continue;
@@ -344,7 +344,7 @@ IDBError MemoryObjectStore::populateIndexWithExistingRecords(MemoryIndex& index)
             return IDBError { };
 
         IndexKey indexKey;
-        generateIndexKeyForValue(UniqueIDBDatabase::databaseThreadExecState(), index.info(), jsValue, indexKey, m_info.keyPath(), iterator.key);
+        generateIndexKeyForValue(UniqueIDBDatabase::databaseThreadExecState(), index.info(), jsValue, indexKey);
 
         if (indexKey.isNull())
             continue;
@@ -407,7 +407,7 @@ ThreadSafeDataBuffer MemoryObjectStore::valueForKeyRange(const IDBKeyRangeData&
 
 void MemoryObjectStore::getAllRecords(const IDBKeyRangeData& keyRangeData, Optional<uint32_t> count, IndexedDB::GetAllType type, IDBGetAllResult& result) const
 {
-    result = { type, m_info.keyPath() };
+    result = { type };
 
     uint32_t targetCount;
     if (count && count.value())
@@ -424,9 +424,11 @@ void MemoryObjectStore::getAllRecords(const IDBKeyRangeData& keyRangeData, Optio
 
         range.lowerKey = key;
         range.lowerOpen = true;
-        if (type == IndexedDB::GetAllType::Values)
+
+        if (type == IndexedDB::GetAllType::Keys)
+            result.addKey(WTFMove(key));
+        else
             result.addValue(valueForKey(key));
-        result.addKey(WTFMove(key));
 
         ++currentCount;
     }
index 3703347..f8b23ea 100644 (file)
@@ -194,7 +194,7 @@ void MemoryObjectStoreCursor::currentData(IDBGetResult& data)
         data = { m_currentPositionKey, m_currentPositionKey };
     else {
         IDBValue value = { m_objectStore.valueForKeyRange(m_currentPositionKey), { }, { }, { } };
-        data = { m_currentPositionKey, m_currentPositionKey, WTFMove(value), m_objectStore.info().keyPath() };
+        data = { m_currentPositionKey, m_currentPositionKey, WTFMove(value) };
     }
 }
 
index 489ded4..6249976 100644 (file)
@@ -29,7 +29,6 @@
 #if ENABLE(INDEXED_DATABASE)
 
 #include "IDBBindingUtilities.h"
-#include "IDBCursorInfo.h"
 #include "IDBGetAllRecordsData.h"
 #include "IDBGetAllResult.h"
 #include "IDBGetRecordData.h"
@@ -1753,9 +1752,7 @@ IDBError SQLiteIDBBackingStore::updateOneIndexForAddRecord(const IDBIndexInfo& i
         return IDBError { };
 
     IndexKey indexKey;
-    auto* objectStoreInfo = infoForObjectStore(info.objectStoreIdentifier());
-    ASSERT(objectStoreInfo);
-    generateIndexKeyForValue(*m_globalObject->globalExec(), info, jsValue, indexKey, objectStoreInfo->keyPath(), key);
+    generateIndexKeyForValue(*m_globalObject->globalExec(), info, jsValue, indexKey);
 
     if (indexKey.isNull())
         return IDBError { };
@@ -1775,7 +1772,7 @@ IDBError SQLiteIDBBackingStore::updateAllIndexesForAddRecord(const IDBObjectStor
     bool anyRecordsSucceeded = false;
     for (auto& index : info.indexMap().values()) {
         IndexKey indexKey;
-        generateIndexKeyForValue(*m_globalObject->globalExec(), index, jsValue, indexKey, info.keyPath(), key);
+        generateIndexKeyForValue(*m_globalObject->globalExec(), index, jsValue, indexKey);
 
         if (indexKey.isNull())
             continue;
@@ -2007,12 +2004,12 @@ IDBError SQLiteIDBBackingStore::getRecord(const IDBResourceIdentifier& transacti
     }
 
     int64_t recordID = 0;
-    ThreadSafeDataBuffer keyResultBuffer, valueResultBuffer;
+    ThreadSafeDataBuffer resultBuffer;
     {
-        static const char* const lowerOpenUpperOpen = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
-        static const char* const lowerOpenUpperClosed = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
-        static const char* const lowerClosedUpperOpen = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
-        static const char* const lowerClosedUpperClosed = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
+        static const char* const lowerOpenUpperOpen = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
+        static const char* const lowerOpenUpperClosed = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
+        static const char* const lowerClosedUpperOpen = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
+        static const char* const lowerClosedUpperClosed = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
 
         static const char* const lowerOpenUpperOpenKeyOnly = "SELECT key FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
         static const char* const lowerOpenUpperClosedKeyOnly = "SELECT key FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
@@ -2069,31 +2066,27 @@ IDBError SQLiteIDBBackingStore::getRecord(const IDBResourceIdentifier& transacti
             return IDBError { UnknownError, "Error looking up record in object store by key range"_s };
         }
 
-        Vector<uint8_t> keyBuffer;
-        sql->getColumnBlobAsVector(0, keyBuffer);
-        keyResultBuffer = ThreadSafeDataBuffer::create(WTFMove(keyBuffer));
+        Vector<uint8_t> buffer;
+        sql->getColumnBlobAsVector(0, buffer);
+        resultBuffer = ThreadSafeDataBuffer::create(WTFMove(buffer));
 
-        if (type == IDBGetRecordDataType::KeyAndValue) {
-            Vector<uint8_t> valueBuffer;
-            sql->getColumnBlobAsVector(1, valueBuffer);
-            valueResultBuffer = ThreadSafeDataBuffer::create(WTFMove(valueBuffer));
-            recordID = sql->getColumnInt64(2);
-        }
-    }
-
-    auto* keyVector = keyResultBuffer.data();
-    if (!keyVector) {
-        LOG_ERROR("Unable to deserialize key data from database for IDBObjectStore");
-        return IDBError { UnknownError, "Error extracting key data from database executing IDBObjectStore get"_s };
-    }
-    
-    IDBKeyData keyData;
-    if (!deserializeIDBKeyData(keyVector->data(), keyVector->size(), keyData)) {
-        LOG_ERROR("Unable to deserialize key data from database for IDBObjectStore");
-        return IDBError { UnknownError, "Error extracting key data from database executing IDBObjectStore get"_s };
+        if (type == IDBGetRecordDataType::KeyAndValue)
+            recordID = sql->getColumnInt64(1);
     }
 
     if (type == IDBGetRecordDataType::KeyOnly) {
+        auto* vector = resultBuffer.data();
+        if (!vector) {
+            LOG_ERROR("Unable to deserialize key data from database for IDBObjectStore.getKey()");
+            return IDBError { UnknownError, "Error extracting key data from database executing IDBObjectStore.getKey()"_s };
+        }
+
+        IDBKeyData keyData;
+        if (!deserializeIDBKeyData(vector->data(), vector->size(), keyData)) {
+            LOG_ERROR("Unable to deserialize key data from database for IDBObjectStore.getKey()");
+            return IDBError { UnknownError, "Error extracting key data from database executing IDBObjectStore.getKey()"_s };
+        }
+
         resultValue = { keyData };
         return IDBError { };
     }
@@ -2107,9 +2100,7 @@ IDBError SQLiteIDBBackingStore::getRecord(const IDBResourceIdentifier& transacti
     if (!error.isNull())
         return error;
 
-    auto* objectStoreInfo = infoForObjectStore(objectStoreID);
-    ASSERT(objectStoreInfo);
-    resultValue = { keyData, { valueResultBuffer, WTFMove(blobURLs), sessionID, WTFMove(blobFilePaths) }, objectStoreInfo->keyPath()};
+    resultValue = { { resultBuffer, WTFMove(blobURLs), sessionID, WTFMove(blobFilePaths) } };
     return IDBError { };
 }
 
@@ -2124,10 +2115,10 @@ SQLiteStatement* SQLiteIDBBackingStore::cachedStatementForGetAllObjectStoreRecor
     static const char* const lowerOpenUpperClosedKey = "SELECT key FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
     static const char* const lowerClosedUpperOpenKey = "SELECT key FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
     static const char* const lowerClosedUpperClosedKey = "SELECT key FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
-    static const char* const lowerOpenUpperOpenValue = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
-    static const char* const lowerOpenUpperClosedValue = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
-    static const char* const lowerClosedUpperOpenValue = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
-    static const char* const lowerClosedUpperClosedValue = "SELECT key, value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
+    static const char* const lowerOpenUpperOpenValue = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
+    static const char* const lowerOpenUpperClosedValue = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key > CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
+    static const char* const lowerClosedUpperOpenValue = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key < CAST(? AS TEXT) ORDER BY key;";
+    static const char* const lowerClosedUpperClosedValue = "SELECT value, ROWID FROM Records WHERE objectStoreID = ? AND key >= CAST(? AS TEXT) AND key <= CAST(? AS TEXT) ORDER BY key;";
 
     if (getAllRecordsData.getAllType == IndexedDB::GetAllType::Keys) {
         if (getAllRecordsData.keyRangeData.lowerOpen) {
@@ -2192,9 +2183,7 @@ IDBError SQLiteIDBBackingStore::getAllObjectStoreRecords(const IDBResourceIdenti
         return IDBError { UnknownError, "Failed to look up record in object store by key range"_s };
     }
 
-    auto* objectStoreInfo = infoForObjectStore(getAllRecordsData.objectStoreIdentifier);
-    ASSERT(objectStoreInfo);
-    result = { getAllRecordsData.getAllType, objectStoreInfo->keyPath() };
+    result = { getAllRecordsData.getAllType };
 
     uint32_t targetResults;
     if (getAllRecordsData.count && getAllRecordsData.count.value())
@@ -2206,21 +2195,12 @@ IDBError SQLiteIDBBackingStore::getAllObjectStoreRecords(const IDBResourceIdenti
     uint32_t returnedResults = 0;
 
     while (sqlResult == SQLITE_ROW && returnedResults < targetResults) {
-        Vector<uint8_t> keyBuffer;
-        IDBKeyData keyData;
-        sql->getColumnBlobAsVector(0, keyBuffer);
-        if (!deserializeIDBKeyData(keyBuffer.data(), keyBuffer.size(), keyData)) {
-            LOG_ERROR("Unable to deserialize key data from database while getting all records");
-            return IDBError { UnknownError, "Unable to deserialize key data while getting all records"_s };
-        }
-        result.addKey(WTFMove(keyData));
-
         if (getAllRecordsData.getAllType == IndexedDB::GetAllType::Values) {
-            Vector<uint8_t> valueBuffer;
-            sql->getColumnBlobAsVector(1, valueBuffer);
-            ThreadSafeDataBuffer valueResultBuffer = ThreadSafeDataBuffer::create(WTFMove(valueBuffer));
+            Vector<uint8_t> buffer;
+            sql->getColumnBlobAsVector(0, buffer);
+            ThreadSafeDataBuffer resultBuffer = ThreadSafeDataBuffer::create(WTFMove(buffer));
 
-            auto recordID = sql->getColumnInt64(2);
+            auto recordID = sql->getColumnInt64(1);
 
             ASSERT(recordID);
             Vector<String> blobURLs, blobFilePaths;
@@ -2231,7 +2211,18 @@ IDBError SQLiteIDBBackingStore::getAllObjectStoreRecords(const IDBResourceIdenti
             if (!error.isNull())
                 return error;
 
-            result.addValue({ valueResultBuffer, WTFMove(blobURLs), sessionID, WTFMove(blobFilePaths) });
+            result.addValue({ resultBuffer, WTFMove(blobURLs), sessionID, WTFMove(blobFilePaths) });
+        } else {
+            Vector<uint8_t> keyData;
+            IDBKeyData key;
+            sql->getColumnBlobAsVector(0, keyData);
+
+            if (!deserializeIDBKeyData(keyData.data(), keyData.size(), key)) {
+                LOG_ERROR("Unable to deserialize key data from database while getting all key records");
+                return IDBError { UnknownError, "Unable to deserialize key data while getting all key records"_s };
+            }
+
+            result.addKey(WTFMove(key));
         }
 
         ++returnedResults;
@@ -2272,18 +2263,16 @@ IDBError SQLiteIDBBackingStore::getAllIndexRecords(const IDBResourceIdentifier&
         return IDBError { UnknownError, "Cursor failed while looking up index records in database"_s };
     }
 
-    auto* objectStoreInfo = infoForObjectStore(getAllRecordsData.objectStoreIdentifier);
-    ASSERT(objectStoreInfo);
-    result = { getAllRecordsData.getAllType, objectStoreInfo->keyPath() };
-
+    result = { getAllRecordsData.getAllType };
     uint32_t currentCount = 0;
     uint32_t targetCount = getAllRecordsData.count ? getAllRecordsData.count.value() : 0;
     if (!targetCount)
         targetCount = std::numeric_limits<uint32_t>::max();
     while (!cursor->didComplete() && !cursor->didError() && currentCount < targetCount) {
-        IDBKeyData keyCopy = cursor->currentPrimaryKey();
-        result.addKey(WTFMove(keyCopy));
-        if (getAllRecordsData.getAllType == IndexedDB::GetAllType::Values)
+        if (getAllRecordsData.getAllType == IndexedDB::GetAllType::Keys) {
+            IDBKeyData keyCopy = cursor->currentPrimaryKey();
+            result.addKey(WTFMove(keyCopy));
+        } else
             result.addValue(cursor->currentValue() ? *cursor->currentValue() : IDBValue());
 
         ++currentCount;
@@ -2330,11 +2319,8 @@ IDBError SQLiteIDBBackingStore::getIndexRecord(const IDBResourceIdentifier& tran
     else {
         if (type == IndexedDB::IndexRecordType::Key)
             getResult = { cursor->currentPrimaryKey() };
-        else {
-            auto* objectStoreInfo = infoForObjectStore(objectStoreID);
-            ASSERT(objectStoreInfo);
-            getResult = { cursor->currentPrimaryKey(), cursor->currentPrimaryKey(), cursor->currentValue() ? *cursor->currentValue() : IDBValue(), objectStoreInfo->keyPath() };
-        }
+        else
+            getResult = { cursor->currentValue() ? *cursor->currentValue() : IDBValue(), cursor->currentPrimaryKey() };
     }
 
     return IDBError { };
@@ -2385,8 +2371,7 @@ IDBError SQLiteIDBBackingStore::uncheckedGetIndexRecordForOneKey(int64_t indexID
         return IDBError { };
     }
 
-    Vector<uint8_t> valueVector;
-    sql->getColumnBlobAsVector(1, valueVector);
+    sql->getColumnBlobAsVector(1, keyVector);
 
     int64_t recordID = sql->getColumnInt64(2);
     Vector<String> blobURLs, blobFilePaths;
@@ -2397,9 +2382,7 @@ IDBError SQLiteIDBBackingStore::uncheckedGetIndexRecordForOneKey(int64_t indexID
     if (!error.isNull())
         return error;
 
-    auto* objectStoreInfo = infoForObjectStore(objectStoreID);
-    ASSERT(objectStoreInfo);
-    getResult = { objectStoreKey, objectStoreKey, { ThreadSafeDataBuffer::create(WTFMove(valueVector)), WTFMove(blobURLs), sessionID, WTFMove(blobFilePaths) }, objectStoreInfo->keyPath() };
+    getResult = { { ThreadSafeDataBuffer::create(WTFMove(keyVector)), WTFMove(blobURLs), sessionID, WTFMove(blobFilePaths) }, objectStoreKey };
     return IDBError { };
 }
 
@@ -2564,9 +2547,7 @@ IDBError SQLiteIDBBackingStore::openCursor(const IDBResourceIdentifier& transact
 
     m_cursors.set(cursor->identifier(), cursor);
 
-    auto* objectStoreInfo = infoForObjectStore(info.objectStoreIdentifier());
-    ASSERT(objectStoreInfo);
-    cursor->currentData(result, objectStoreInfo->keyPath());
+    cursor->currentData(result);
     return IDBError { };
 }
 
@@ -2609,9 +2590,7 @@ IDBError SQLiteIDBBackingStore::iterateCursor(const IDBResourceIdentifier& trans
         }
     }
 
-    auto* objectStoreInfo = infoForObjectStore(cursor->objectStoreID());
-    ASSERT(objectStoreInfo);
-    cursor->currentData(result, objectStoreInfo->keyPath());
+    cursor->currentData(result);
     return IDBError { };
 }
 
index caeacf2..878c8bd 100644 (file)
@@ -101,7 +101,7 @@ SQLiteIDBCursor::~SQLiteIDBCursor()
         m_transaction->closeCursor(*this);
 }
 
-void SQLiteIDBCursor::currentData(IDBGetResult& result, const Optional<IDBKeyPath>& keyPath)
+void SQLiteIDBCursor::currentData(IDBGetResult& result)
 {
     ASSERT(!m_fetchedRecords.isEmpty());
 
@@ -112,7 +112,7 @@ void SQLiteIDBCursor::currentData(IDBGetResult& result, const Optional<IDBKeyPat
         return;
     }
 
-    result = { currentRecord.record.key, currentRecord.record.primaryKey, currentRecord.record.value ? *currentRecord.record.value : IDBValue(), keyPath};
+    result = { currentRecord.record.key, currentRecord.record.primaryKey, currentRecord.record.value ? *currentRecord.record.value : IDBValue() };
 }
 
 static String buildIndexStatement(const IDBKeyRangeData& keyRange, IndexedDB::CursorDirection cursorDirection)
index 8cc2ab9..f78ee76 100644 (file)
@@ -79,7 +79,7 @@ public:
 
     void objectStoreRecordsChanged();
 
-    void currentData(IDBGetResult&, const Optional<IDBKeyPath>&);
+    void currentData(IDBGetResult&);
 
 private:
     bool establishStatement();
index 68ea83c..8332503 100644 (file)
@@ -1251,6 +1251,35 @@ void UniqueIDBDatabase::performPutOrAdd(uint64_t callbackIdentifier, const IDBRe
         }
     }
 
+    // 3.4.1.2 Object Store Storage Operation
+    // If ObjectStore has a key path and the key is autogenerated, then inject the key into the value
+    // using steps to assign a key to a value using a key path.
+    ThreadSafeDataBuffer injectedRecordValue;
+    if (usedKeyIsGenerated && objectStoreInfo->keyPath()) {
+        VM& vm = databaseThreadVM();
+        JSLockHolder locker(vm);
+        auto scope = DECLARE_THROW_SCOPE(vm);
+
+        auto value = deserializeIDBValueToJSValue(databaseThreadExecState(), originalRecordValue.data());
+        if (value.isUndefined()) {
+            postDatabaseTaskReply(createCrossThreadTask(*this, &UniqueIDBDatabase::didPerformPutOrAdd, callbackIdentifier, IDBError(ConstraintError, "Unable to deserialize record value for record key injection"_s), usedKey));
+            return;
+        }
+
+        if (!injectIDBKeyIntoScriptValue(databaseThreadExecState(), usedKey, value, objectStoreInfo->keyPath().value())) {
+            postDatabaseTaskReply(createCrossThreadTask(*this, &UniqueIDBDatabase::didPerformPutOrAdd, callbackIdentifier, IDBError(ConstraintError, "Unable to inject record key into record value"_s), usedKey));
+            return;
+        }
+
+        auto serializedValue = SerializedScriptValue::create(databaseThreadExecState(), value);
+        if (UNLIKELY(scope.exception())) {
+            postDatabaseTaskReply(createCrossThreadTask(*this, &UniqueIDBDatabase::didPerformPutOrAdd, callbackIdentifier, IDBError(ConstraintError, "Unable to serialize record value after injecting record key"_s), usedKey));
+            return;
+        }
+
+        injectedRecordValue = ThreadSafeDataBuffer::copyVector(serializedValue->data());
+    }
+
     // 3.4.1 Object Store Storage Operation
     // ...If a record already exists in store ...
     // then remove the record from store using the steps for deleting records from an object store...
@@ -1261,7 +1290,10 @@ void UniqueIDBDatabase::performPutOrAdd(uint64_t callbackIdentifier, const IDBRe
         return;
     }
 
-    error = m_backingStore->addRecord(transactionIdentifier, *objectStoreInfo, usedKey, originalRecordValue);
+    if (injectedRecordValue.data())
+        error = m_backingStore->addRecord(transactionIdentifier, *objectStoreInfo, usedKey, { injectedRecordValue, originalRecordValue.blobURLs(), originalRecordValue.sessionID(), originalRecordValue.blobFilePaths() });
+    else
+        error = m_backingStore->addRecord(transactionIdentifier, *objectStoreInfo, usedKey, originalRecordValue);
 
     if (!error.isNull()) {
         postDatabaseTaskReply(createCrossThreadTask(*this, &UniqueIDBDatabase::didPerformPutOrAdd, callbackIdentifier, error, usedKey));
index a89454e..7ff08ee 100644 (file)
@@ -233,12 +233,6 @@ const IDBGetResult& IDBResultData::getResult() const
     return *m_getResult;
 }
 
-IDBGetResult& IDBResultData::getResultRef()
-{
-    RELEASE_ASSERT(m_getResult);
-    return *m_getResult;
-}
-
 const IDBGetAllResult& IDBResultData::getAllResult() const
 {
     RELEASE_ASSERT(m_getAllResult);
index 717b3e2..80d8ce9 100644 (file)
@@ -107,7 +107,6 @@ public:
     uint64_t resultInteger() const { return m_resultInteger; }
 
     WEBCORE_EXPORT const IDBGetResult& getResult() const;
-    WEBCORE_EXPORT IDBGetResult& getResultRef();
     WEBCORE_EXPORT const IDBGetAllResult& getAllResult() const;
 
     WEBCORE_EXPORT IDBResultData();
index ad9234c..da106ec 100644 (file)
@@ -31,7 +31,6 @@
 
 #include "IDBBindingUtilities.h"
 
-#include "ExceptionCode.h"
 #include "IDBIndexInfo.h"
 #include "IDBKey.h"
 #include "IDBKeyData.h"
@@ -42,7 +41,6 @@
 #include "JSDOMBinding.h"
 #include "JSDOMConvertDate.h"
 #include "JSDOMConvertNullable.h"
-#include "JSDOMExceptionHandling.h"
 #include "JSFile.h"
 #include "Logging.h"
 #include "MessagePort.h"
@@ -326,12 +324,6 @@ bool injectIDBKeyIntoScriptValue(ExecState& exec, const IDBKeyData& keyData, JSV
     if (!key)
         return false;
 
-    // Do not set if object already has the correct property value.
-    auto jsKey = toJS(exec, *exec.lexicalGlobalObject(), key.get());
-    JSValue existingKey;
-    if (get(exec, parent, keyPathElements.last(), existingKey) && existingKey == jsKey)
-        return true;
-
     if (!set(exec, parent, keyPathElements.last(), toJS(exec, *exec.lexicalGlobalObject(), key.get())))
         return false;
 
@@ -419,13 +411,9 @@ JSC::JSValue toJS(JSC::ExecState* state, JSDOMGlobalObject* globalObject, const
     return toJS(*state, *globalObject, keyData.maybeCreateIDBKey().get());
 }
 
-static Vector<IDBKeyData> createKeyPathArray(ExecState& exec, JSValue value, const IDBIndexInfo& info, Optional<IDBKeyPath> objectStoreKeyPath, const IDBKeyData& objectStoreKey)
+static Vector<IDBKeyData> createKeyPathArray(ExecState& exec, JSValue value, const IDBIndexInfo& info)
 {
     auto visitor = WTF::makeVisitor([&](const String& string) -> Vector<IDBKeyData> {
-        // Value doesn't contain auto-generated key, so we need to manually add key if it is possibly auto-generated.
-        if (objectStoreKeyPath && WTF::holds_alternative<String>(objectStoreKeyPath.value()) && IDBKeyPath(string) == objectStoreKeyPath.value())
-            return { objectStoreKey };
-
         auto idbKey = internalCreateIDBKeyFromScriptValueAndKeyPath(exec, value, string);
         if (!idbKey)
             return { };
@@ -440,14 +428,10 @@ static Vector<IDBKeyData> createKeyPathArray(ExecState& exec, JSValue value, con
     }, [&](const Vector<String>& vector) -> Vector<IDBKeyData> {
         Vector<IDBKeyData> keys;
         for (auto& entry : vector) {
-            if (objectStoreKeyPath && WTF::holds_alternative<String>(objectStoreKeyPath.value()) && IDBKeyPath(entry) == objectStoreKeyPath.value())
-                keys.append(objectStoreKey);
-            else {
-                auto key = internalCreateIDBKeyFromScriptValueAndKeyPath(exec, value, entry);
-                if (!key || !key->isValid())
-                    return { };
-                keys.append(key.get());
-            }
+            auto key = internalCreateIDBKeyFromScriptValueAndKeyPath(exec, value, entry);
+            if (!key || !key->isValid())
+                return { };
+            keys.append(key.get());
         }
         return keys;
     });
@@ -455,31 +439,16 @@ static Vector<IDBKeyData> createKeyPathArray(ExecState& exec, JSValue value, con
     return WTF::visit(visitor, info.keyPath());
 }
 
-void generateIndexKeyForValue(ExecState& exec, const IDBIndexInfo& info, JSValue value, IndexKey& outKey, const Optional<IDBKeyPath>& objectStoreKeyPath, const IDBKeyData& objectStoreKey)
+void generateIndexKeyForValue(ExecState& exec, const IDBIndexInfo& info, JSValue value, IndexKey& outKey)
 {
-    auto keyDatas = createKeyPathArray(exec, value, info, objectStoreKeyPath, objectStoreKey);
+    auto keyDatas = createKeyPathArray(exec, value, info);
+
     if (keyDatas.isEmpty())
         return;
 
     outKey = IndexKey(WTFMove(keyDatas));
 }
 
-Optional<JSC::JSValue> deserializeIDBValueWithKeyInjection(ExecState& state, const IDBValue& value, const IDBKeyData& key, const Optional<IDBKeyPath>& keyPath)
-{
-    auto jsValue = deserializeIDBValueToJSValue(state, value);
-    if (jsValue.isUndefined() || !keyPath || !WTF::holds_alternative<String>(keyPath.value()) || !isIDBKeyPathValid(keyPath.value()))
-        return jsValue;
-
-    JSLockHolder locker(state.vm());
-    if (!injectIDBKeyIntoScriptValue(state, key, jsValue, keyPath.value())) {
-        auto throwScope = DECLARE_THROW_SCOPE(state.vm());
-        propagateException(state, throwScope, Exception(UnknownError, "Cannot inject key into script value"_s));
-        return WTF::nullopt;
-    }
-
-    return jsValue;
-}
-
 } // namespace WebCore
 
 #endif // ENABLE(INDEXED_DATABASE)
index f067cb0..c491fee 100644 (file)
@@ -50,17 +50,15 @@ RefPtr<IDBKey> maybeCreateIDBKeyFromScriptValueAndKeyPath(JSC::ExecState&, const
 bool canInjectIDBKeyIntoScriptValue(JSC::ExecState&, const JSC::JSValue&, const IDBKeyPath&);
 bool injectIDBKeyIntoScriptValue(JSC::ExecState&, const IDBKeyData&, JSC::JSValue, const IDBKeyPath&);
 
-void generateIndexKeyForValue(JSC::ExecState&, const IDBIndexInfo&, JSC::JSValue, IndexKey& outKey, const Optional<IDBKeyPath>&, const IDBKeyData&);
+void generateIndexKeyForValue(JSC::ExecState&, const IDBIndexInfo&, JSC::JSValue, IndexKey& outKey);
 
 Ref<IDBKey> scriptValueToIDBKey(JSC::ExecState&, const JSC::JSValue&);
 
-JSC::JSValue deserializeIDBValueToJSValue(JSC::ExecState&, const IDBValue&, Vector<std::pair<String, String>>&);
 JSC::JSValue deserializeIDBValueToJSValue(JSC::ExecState&, const IDBValue&);
 JSC::JSValue toJS(JSC::ExecState*, JSDOMGlobalObject*, const IDBValue&);
 JSC::JSValue toJS(JSC::ExecState&, JSC::JSGlobalObject&, IDBKey*);
 JSC::JSValue toJS(JSC::ExecState*, JSDOMGlobalObject*, const IDBKeyData&);
 
-Optional<JSC::JSValue> deserializeIDBValueWithKeyInjection(JSC::ExecState&, const IDBValue&, const IDBKeyData&, const Optional<IDBKeyPath>&);
 }
 
 #endif // ENABLE(INDEXED_DATABASE)
index a84690b..f43501c 100644 (file)
@@ -38,8 +38,7 @@ using namespace JSC;
 JSC::JSValue JSIDBCursorWithValue::value(JSC::ExecState& state) const
 {
     return cachedPropertyValue(state, *this, wrapped().valueWrapper(), [&] {
-        auto result = deserializeIDBValueWithKeyInjection(state, wrapped().value(), wrapped().primaryKey(), wrapped().primaryKeyPath());
-        return result ? result.value() : jsNull();
+        return deserializeIDBValueToJSValue(state, wrapped().value());
     });
 }
 
index 51902e2..5ebb7cc 100644 (file)
@@ -57,26 +57,10 @@ JSC::JSValue JSIDBRequest::result(JSC::ExecState& state) const
             return toJS<IDLIDBKeyData>(state, *jsCast<JSDOMGlobalObject*>(state.lexicalGlobalObject()), keyData);
         }, [&state] (Vector<IDBKeyData> keyDatas) {
             return toJS<IDLSequence<IDLIDBKeyData>>(state, *jsCast<JSDOMGlobalObject*>(state.lexicalGlobalObject()), keyDatas);
-        }, [&state] (IDBGetResult getResult) {
-            auto result = deserializeIDBValueWithKeyInjection(state, getResult.value(), getResult.keyData(), getResult.keyPath());
-            return result ? result.value() : jsNull();
-        }, [&state] (IDBGetAllResult getAllResult) {
-            auto& keys = getAllResult.keys();
-            auto& values = getAllResult.values();
-            auto& keyPath = getAllResult.keyPath();
-            auto scope = DECLARE_THROW_SCOPE(state.vm());
-            JSC::MarkedArgumentBuffer list;
-            for (unsigned i = 0; i < values.size(); i ++) {
-                auto result = deserializeIDBValueWithKeyInjection(state, values[i], keys[i], keyPath);
-                if (!result)
-                    return jsNull();
-                list.append(result.value());
-                if (UNLIKELY(list.hasOverflowed())) {
-                    propagateException(state, scope, Exception(UnknownError));
-                    return jsNull();
-                }
-            }
-            return JSValue(JSC::constructArray(&state, nullptr, state.lexicalGlobalObject(), list));
+        }, [&state] (IDBValue value) {
+            return toJS<IDLIDBValue>(state, *jsCast<JSDOMGlobalObject*>(state.lexicalGlobalObject()), value);
+        }, [&state] (Vector<IDBValue> values) {
+            return toJS<IDLSequence<IDLIDBValue>>(state, *jsCast<JSDOMGlobalObject*>(state.lexicalGlobalObject()), values);
         }, [] (uint64_t number) {
             return toJS<IDLUnsignedLongLong>(number);
         }, [] (IDBRequest::NullResultType other) {