|  | // META: title=IndexedDB: index renaming support | 
|  | // META: global=window,worker | 
|  | // META: script=resources/support-promises.js | 
|  | // META: timeout=long | 
|  |  | 
|  | // Spec: https://w3c.github.io/IndexedDB/#dom-idbindex-name | 
|  |  | 
|  | 'use strict'; | 
|  |  | 
|  | promise_test(testCase => { | 
|  | let authorIndex = null; | 
|  | let authorIndex2 = null; | 
|  | let renamedAuthorIndex = null; | 
|  | let renamedAuthorIndex2 = null; | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | const store = createBooksStore(testCase, database); | 
|  | authorIndex = store.index('by_author'); | 
|  | }) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_author', 'by_title'], | 
|  | 'Test setup should have created two indexes'); | 
|  | authorIndex2 = store.index('by_author'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, authorIndex2, | 
|  | 'The index should have the expected contents before any renaming') | 
|  | .then(() => database.close()); | 
|  | }) | 
|  | .then( | 
|  | () => migrateDatabase( | 
|  | testCase, 2, | 
|  | (database, transaction) => { | 
|  | const store = transaction.objectStore('books'); | 
|  | renamedAuthorIndex = store.index('by_author'); | 
|  | renamedAuthorIndex.name = 'renamed_by_author'; | 
|  |  | 
|  | assert_equals( | 
|  | renamedAuthorIndex.name, 'renamed_by_author', | 
|  | 'IDBIndex name should change immediately after a rename'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title', 'renamed_by_author'], | 
|  | 'IDBObjectStore.indexNames should immediately reflect the rename'); | 
|  | assert_equals( | 
|  | store.index('renamed_by_author'), renamedAuthorIndex, | 
|  | 'IDBObjectStore.index should return the renamed index store when ' + | 
|  | 'queried using the new name immediately after the rename'); | 
|  | assert_throws_dom( | 
|  | 'NotFoundError', () => store.index('by_author'), | 
|  | 'IDBObjectStore.index should throw when queried using the ' + | 
|  | 'renamed index\'s old name immediately after the rename'); | 
|  | })) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title', 'renamed_by_author'], | 
|  | 'IDBObjectStore.indexNames should still reflect the rename after ' + | 
|  | 'the versionchange transaction commits'); | 
|  | renamedAuthorIndex2 = store.index('renamed_by_author'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, renamedAuthorIndex2, | 
|  | 'Renaming an index should not change its contents') | 
|  | .then(() => database.close()); | 
|  | }) | 
|  | .then(() => { | 
|  | assert_equals( | 
|  | authorIndex.name, 'by_author', | 
|  | 'IDBIndex obtained before the rename transaction should not ' + | 
|  | 'reflect the rename'); | 
|  | assert_equals( | 
|  | authorIndex2.name, 'by_author', | 
|  | 'IDBIndex obtained before the rename transaction should not ' + | 
|  | 'reflect the rename'); | 
|  | assert_equals( | 
|  | renamedAuthorIndex.name, 'renamed_by_author', | 
|  | 'IDBIndex used in the rename transaction should keep reflecting ' + | 
|  | 'the new name after the transaction is committed'); | 
|  | assert_equals( | 
|  | renamedAuthorIndex2.name, 'renamed_by_author', | 
|  | 'IDBIndex obtained after the rename transaction should reflect ' + | 
|  | 'the new name'); | 
|  | }); | 
|  | }, 'IndexedDB index rename in new transaction'); | 
|  |  | 
|  | promise_test(testCase => { | 
|  | let renamedAuthorIndex = null; | 
|  | let renamedAuthorIndex2 = null; | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | const store = createBooksStore(testCase, database); | 
|  | renamedAuthorIndex = store.index('by_author'); | 
|  | renamedAuthorIndex.name = 'renamed_by_author'; | 
|  |  | 
|  | assert_equals( | 
|  | renamedAuthorIndex.name, 'renamed_by_author', | 
|  | 'IDBIndex name should change immediately after a rename'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title', 'renamed_by_author'], | 
|  | 'IDBObjectStore.indexNames should immediately reflect the rename'); | 
|  | assert_equals( | 
|  | store.index('renamed_by_author'), renamedAuthorIndex, | 
|  | 'IDBObjectStore.index should return the renamed index store when ' + | 
|  | 'queried using the new name immediately after the rename'); | 
|  | assert_throws_dom( | 
|  | 'NotFoundError', () => store.index('by_author'), | 
|  | 'IDBObjectStore.index should throw when queried using the ' + | 
|  | 'renamed index\'s old name immediately after the rename'); | 
|  | }) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title', 'renamed_by_author'], | 
|  | 'IDBObjectStore.indexNames should still reflect the rename after ' + | 
|  | 'the versionchange transaction commits'); | 
|  | renamedAuthorIndex2 = store.index('renamed_by_author'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, renamedAuthorIndex2, | 
|  | 'Renaming an index should not change its contents') | 
|  | .then(() => database.close()); | 
|  | }) | 
|  | .then(() => { | 
|  | assert_equals( | 
|  | renamedAuthorIndex.name, 'renamed_by_author', | 
|  | 'IDBIndex used in the rename transaction should keep reflecting ' + | 
|  | 'the new name after the transaction is committed'); | 
|  | assert_equals( | 
|  | renamedAuthorIndex2.name, 'renamed_by_author', | 
|  | 'IDBIndex obtained after the rename transaction should reflect ' + | 
|  | 'the new name'); | 
|  | }); | 
|  | }, 'IndexedDB index rename in the transaction where it is created'); | 
|  |  | 
|  | promise_test(testCase => { | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | createBooksStore(testCase, database); | 
|  | }) | 
|  | .then(database => { | 
|  | database.close(); | 
|  | }) | 
|  | .then( | 
|  | () => migrateDatabase( | 
|  | testCase, 2, | 
|  | (database, transaction) => { | 
|  | const store = transaction.objectStore('books'); | 
|  | const index = store.index('by_author'); | 
|  | index.name = 'by_author'; | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_author', 'by_title'], | 
|  | 'Renaming an index to the same name should not change the ' + | 
|  | 'index\'s IDBObjectStore.indexNames'); | 
|  | })) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_author', 'by_title'], | 
|  | 'Committing a transaction that renames a store to the same name ' + | 
|  | 'should not change the index\'s IDBObjectStore.indexNames'); | 
|  | const index = store.index('by_author'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, index, | 
|  | 'Committing a transaction that renames an index to the same name ' + | 
|  | 'should not change the index\'s contents') | 
|  | .then(() => database.close()); | 
|  | }); | 
|  | }, 'IndexedDB index rename to the same name succeeds'); | 
|  |  | 
|  | promise_test(testCase => { | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | createBooksStore(testCase, database); | 
|  | }) | 
|  | .then(database => { | 
|  | database.close(); | 
|  | }) | 
|  | .then( | 
|  | () => migrateDatabase( | 
|  | testCase, 2, | 
|  | (database, transaction) => { | 
|  | const store = transaction.objectStore('books'); | 
|  | const index = store.index('by_author'); | 
|  | store.deleteIndex('by_title'); | 
|  | index.name = 'by_title'; | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title'], | 
|  | 'IDBObjectStore.indexNames should immediately reflect the rename'); | 
|  | })) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title'], | 
|  | 'IDBObjectStore.indexNames should still reflect the rename after ' + | 
|  | 'the versionchange transaction commits'); | 
|  | const index = store.index('by_title'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, index, | 
|  | 'Renaming an index should not change its contents') | 
|  | .then(() => database.close()); | 
|  | }); | 
|  | }, 'IndexedDB index rename to the name of a deleted index succeeds'); | 
|  |  | 
|  | promise_test(testCase => { | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | createBooksStore(testCase, database); | 
|  | }) | 
|  | .then(database => { | 
|  | database.close(); | 
|  | }) | 
|  | .then( | 
|  | () => migrateDatabase( | 
|  | testCase, 2, | 
|  | (database, transaction) => { | 
|  | const store = transaction.objectStore('books'); | 
|  | store.index('by_author').name = 'tmp'; | 
|  | store.index('by_title').name = 'by_author'; | 
|  | store.index('tmp').name = 'by_title'; | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_author', 'by_title'], | 
|  | 'IDBObjectStore.indexNames should reflect the swap immediately ' + | 
|  | 'after the renames'); | 
|  | return checkTitleIndexContents( | 
|  | testCase, store.index('by_author'), | 
|  | 'Renaming an index should not change its contents'); | 
|  | })) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_author', 'by_title'], | 
|  | 'IDBObjectStore.indexNames should still reflect the swap after ' + | 
|  | 'the versionchange transaction commits'); | 
|  | const index = store.index('by_title'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, index, | 
|  | 'Renaming an index should not change its contents') | 
|  | .then(() => database.close()); | 
|  | }); | 
|  | }, 'IndexedDB index swapping via renames succeeds'); | 
|  |  | 
|  | promise_test(testCase => { | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | createBooksStore(testCase, database); | 
|  | }) | 
|  | .then(database => { | 
|  | database.close(); | 
|  | }) | 
|  | .then( | 
|  | () => migrateDatabase( | 
|  | testCase, 2, | 
|  | (database, transaction) => { | 
|  | const store = transaction.objectStore('books'); | 
|  | const index = store.index('by_author'); | 
|  |  | 
|  | index.name = 42; | 
|  | assert_equals( | 
|  | index.name, '42', | 
|  | 'IDBIndex name should change immediately after a rename to a ' + | 
|  | 'number'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['42', 'by_title'], | 
|  | 'IDBObjectStore.indexNames should immediately reflect the ' + | 
|  | 'stringifying rename'); | 
|  |  | 
|  | index.name = true; | 
|  | assert_equals( | 
|  | index.name, 'true', | 
|  | 'IDBIndex name should change immediately after a rename to a ' + | 
|  | 'boolean'); | 
|  |  | 
|  | index.name = {}; | 
|  | assert_equals( | 
|  | index.name, '[object Object]', | 
|  | 'IDBIndex name should change immediately after a rename to an ' + | 
|  | 'object'); | 
|  |  | 
|  | index.name = () => null; | 
|  | assert_equals( | 
|  | index.name, '() => null', | 
|  | 'IDBIndex name should change immediately after a rename to a ' + | 
|  | 'function'); | 
|  |  | 
|  | index.name = undefined; | 
|  | assert_equals( | 
|  | index.name, 'undefined', | 
|  | 'IDBIndex name should change immediately after a rename to ' + | 
|  | 'undefined'); | 
|  | })) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, ['by_title', 'undefined'], | 
|  | 'IDBObjectStore.indexNames should reflect the last rename ' + | 
|  | 'after the versionchange transaction commits'); | 
|  | const index = store.index('undefined'); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, index, | 
|  | 'Renaming an index should not change its contents') | 
|  | .then(() => database.close()); | 
|  | }); | 
|  | }, 'IndexedDB index rename stringifies non-string names'); | 
|  |  | 
|  | for (let escapedName of ['', '\\u0000', '\\uDC00\\uD800']) | 
|  | ((escapedName) => { | 
|  | const name = JSON.parse('"' + escapedName + '"'); | 
|  | promise_test(testCase => { | 
|  | return createDatabase( | 
|  | testCase, | 
|  | (database, transaction) => { | 
|  | createBooksStore(testCase, database); | 
|  | }) | 
|  | .then(database => { | 
|  | database.close(); | 
|  | }) | 
|  | .then( | 
|  | () => migrateDatabase( | 
|  | testCase, 2, | 
|  | (database, transaction) => { | 
|  | const store = transaction.objectStore('books'); | 
|  | const index = store.index('by_author'); | 
|  |  | 
|  | index.name = name; | 
|  | assert_equals( | 
|  | index.name, name, | 
|  | 'IDBIndex name should change immediately after the rename'); | 
|  | assert_array_equals( | 
|  | store.indexNames, [name, 'by_title'].sort(), | 
|  | 'IDBObjectStore.indexNames should immediately reflect the rename'); | 
|  | })) | 
|  | .then(database => { | 
|  | const transaction = database.transaction('books', 'readonly'); | 
|  | const store = transaction.objectStore('books'); | 
|  | assert_array_equals( | 
|  | store.indexNames, [name, 'by_title'].sort(), | 
|  | 'IDBObjectStore.indexNames should reflect the rename ' + | 
|  | 'after the versionchange transaction commits'); | 
|  | const index = store.index(name); | 
|  | return checkAuthorIndexContents( | 
|  | testCase, index, | 
|  | 'Renaming an index should not change its contents') | 
|  | .then(() => database.close()); | 
|  | }); | 
|  | }, 'IndexedDB index can be renamed to "' + escapedName + '"'); | 
|  | })(escapedName); |