Index: third_party/WebKit/LayoutTests/storage/indexeddb/upgrade-transaction-lifecycle-committed.html |
diff --git a/third_party/WebKit/LayoutTests/storage/indexeddb/upgrade-transaction-lifecycle-committed.html b/third_party/WebKit/LayoutTests/storage/indexeddb/upgrade-transaction-lifecycle-committed.html |
new file mode 100644 |
index 0000000000000000000000000000000000000000..4b8831c5f4db00e9b9d197ede98d3b833b7efd40 |
--- /dev/null |
+++ b/third_party/WebKit/LayoutTests/storage/indexeddb/upgrade-transaction-lifecycle-committed.html |
@@ -0,0 +1,144 @@ |
+<!doctype html> |
+<title>IndexedDB: committed versionchange transaction lifecycle</title> |
+<link rel="help" |
+ href="https://w3c.github.io/IndexedDB/#upgrade-transaction-steps"> |
+<link rel="help" |
+ href="https://w3c.github.io/IndexedDB/#dom-idbdatabase-createobjectstore"> |
+<link rel="help" |
+ href="https://w3c.github.io/IndexedDB/#dom-idbdatabase-deleteobjectstore"> |
+<link rel="author" href="pwnall@chromium.org" title="Victor Costan"> |
+<script src="../../resources/testharness.js"></script> |
+<script src="../../resources/testharnessreport.js"></script> |
+<script src="resources/support-promises.js"></script> |
+<script> |
+'use strict'; |
+ |
+promise_test(t => { |
+ let timeoutPromise = null, completePromise = null; |
+ return createDatabase(t, database => { |
+ createBooksStore(t, database); |
+ }).then(database => { |
+ database.close(); |
+ }).then(() => migrateDatabase(t, 2, (database, transaction, request) => { |
+ // Queue I/O operations to make sure the transaction does not complete |
+ // before setTimeout's callback is called. |
+ createNotBooksStore(t, database); |
+ const store = transaction.objectStore('books'); |
+ for (let i = 0; i < 1000; ++i) |
+ store.put({ title: 'Bedrock Nights ' + i, author: 'Barney' }); |
+ |
+ let completeFired = false; |
+ completePromise = new Promise((resolve, reject) => { |
+ transaction.addEventListener('complete', () => { |
+ completeFired = true; |
+ resolve(); |
+ }, false); |
+ }); |
+ |
+ timeoutPromise = new Promise((resolve, reject) => { |
+ setTimeout(() => { |
+ const testStepPromise = new Promise((resolve, reject) => { |
+ assert_false( |
+ completeFired, |
+ 'The complete event should fire after setTimeout expires'); |
+ assert_equals( |
+ request.transaction, transaction, |
+ "The open request's transaction should be reset after " + |
+ 'oncomplete'); |
+ assert_throws( |
+ 'TransactionInactiveError', |
+ () => { database.createObjectStore('books2'); }, |
+ 'createObjectStore exception should reflect that the ' + |
+ 'transaction is still running'); |
+ assert_throws( |
+ 'TransactionInactiveError', |
+ () => { database.deleteObjectStore('books'); }, |
+ 'deleteObjectStore exception should reflect that the ' + |
+ 'transaction is still running'); |
+ resolve(); |
+ }); |
+ resolve(testStepPromise); |
+ }, 0); |
+ }); |
+ })).then(database => { |
+ database.close(); |
+ return Promise.all([timeoutPromise, completePromise]); |
+ }); |
+}, 'in a setTimeout(0) callback after the transaction requests are performed'); |
+ |
+promise_test(t => { |
+ let completePromise = null; |
+ return createDatabase(t, database => { |
+ createBooksStore(t, database); |
+ }).then(database => { |
+ database.close(); |
+ }).then(() => migrateDatabase(t, 2, (database, transaction, request) => { |
+ completePromise = new Promise((resolve, reject) => { |
+ transaction.addEventListener('complete', () => { |
+ const testStepPromise = new Promise((resolve, reject) => { |
+ assert_equals( |
+ request.transaction, transaction, |
+ "The open request's transaction should be reset after " + |
+ 'oncomplete'); |
+ assert_throws( |
+ 'InvalidStateError', |
+ () => { database.createObjectStore('books2'); }, |
+ 'createObjectStore exception should reflect that the ' + |
+ 'transaction is no longer running'); |
+ assert_throws( |
+ 'InvalidStateError', |
+ () => { database.deleteObjectStore('books'); }, |
+ 'deleteObjectStore exception should reflect that the ' + |
+ 'transaction is no longer running'); |
+ resolve(); |
+ }); |
+ resolve(testStepPromise); |
+ }, false); |
+ }); |
+ })).then(database => { |
+ database.close(); |
+ return completePromise; |
+ }); |
+}, 'while the complete event is fired'); |
+ |
+promise_test(t => { |
+ let completePromise = null; |
+ return createDatabase(t, database => { |
+ createBooksStore(t, database); |
+ }).then(database => { |
+ database.close(); |
+ }).then(() => migrateDatabase(t, 2, (database, transaction, request) => { |
+ completePromise = new Promise((resolve, reject) => { |
+ transaction.addEventListener('complete', () => { |
+ const timeoutPromise = new Promise((resolve, reject) => { |
+ setTimeout(() => { |
+ const testStepPromise = new Promise((resolve, reject) => { |
+ assert_equals( |
+ request.transaction, null, |
+ "The open request's transaction should be reset after " + |
+ 'oncomplete microtasks'); |
+ assert_throws( |
+ 'InvalidStateError', |
+ () => { database.createObjectStore('books2'); }, |
+ 'createObjectStore exception should reflect that the ' + |
+ 'transaction is no longer running'); |
+ assert_throws( |
+ 'InvalidStateError', |
+ () => { database.deleteObjectStore('books'); }, |
+ 'deleteObjectStore exception should reflect that the ' + |
+ 'transaction is no longer running'); |
+ resolve(); |
+ }); |
+ resolve(testStepPromise); |
+ }, 0); |
+ }); |
+ resolve(timeoutPromise); |
+ }, false); |
+ }); |
+ })).then(database => { |
+ database.close(); |
+ return completePromise; |
+ }); |
+}, 'in a setTimeout(0) callback after the complete event is fired'); |
+ |
+</script> |