RESOLVED FIXED 216769
Indexed DB transactions outdated immediately after it just created
https://bugs.webkit.org/show_bug.cgi?id=216769
Summary Indexed DB transactions outdated immediately after it just created
wards.afar.0j
Reported 2020-09-21 03:00:31 PDT
Reproduction: Note: idb is a famous library that wrap IndexedDB API into Promise style to make it easier to use. It does not contains high level abstractions so it is likely not the bug of idb itself. For reference, this issue is also tracked in the https://github.com/jakearchibald/idb/issues/201 Expected result on Chrome and Firefox: https://user-images.githubusercontent.com/5390719/93753730-358a2900-fc33-11ea-80a5-b7d4f3eb731e.png Bad result on Safari: https://user-images.githubusercontent.com/5390719/93753660-12f81000-fc33-11ea-94c5-7771dab160c2.png ```js import('https://cdn.skypack.dev/idb@5.0.6/with-async-ittr').then(async (idb) => { await idb.deleteDB('test') const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms)) const openDB = (() => { /** @type {import('idb').IDBPDatabase<unknown>} */ let db = undefined return async () => { if (db) return db if (!db) db = await idb.openDB('test', 1, { upgrade(db, oldVersion, newVersion, transaction) { db.createObjectStore('store') }, }) db.addEventListener('close', () => (db = undefined)) return db } })() /** @type {import('idb').IDBPTransaction<unknown, ["store"]>} */ let transaction = undefined const beforeTx = async () => { try { await transaction.objectStore('store').openCursor(IDBKeyRange.only('a')) console.log('The transaction is alive!') } catch (e) { console.log('Transaction outdated', e, 'creating new one') transaction = (await openDB()).transaction('store', 'readwrite') } } await beforeTx() await transaction.store.add({ a: 1 }, 'a') console.log('added a') await beforeTx() await transaction.store.add({ a: 1 }, 'b') console.log('added b') await sleep(120) await beforeTx() await transaction.store.add({ a: 1 }, 'c') console.log('added c') await sleep(120) console.log('before call beforeTx') await beforeTx() console.log('after call beforeTx') const cursor = await transaction.store.openCursor(IDBKeyRange.only('a')) console.log(cursor.value) const next = await cursor.continue() console.log(next) }) ```
Attachments
Patch (10.34 KB, patch)
2020-10-01 11:26 PDT, Sihui Liu
ews-feeder: commit-queue-
Patch (6.55 KB, patch)
2020-10-01 11:41 PDT, Sihui Liu
no flags
Patch (10.33 KB, patch)
2020-10-01 11:47 PDT, Sihui Liu
no flags
Patch (11.07 KB, patch)
2021-03-09 08:45 PST, Sihui Liu
no flags
Patch (10.92 KB, patch)
2021-03-10 10:20 PST, Sihui Liu
no flags
Patch for landing (10.12 KB, patch)
2021-03-10 23:57 PST, Sihui Liu
no flags
wards.afar.0j
Comment 1 2020-09-21 03:01:47 PDT
This issues happens on Safari 12 to 14.0 (15610.1.28.1.9, 15610)
Radar WebKit Bug Importer
Comment 2 2020-09-21 12:48:04 PDT
Sihui Liu
Comment 3 2020-09-22 16:57:41 PDT
(In reply to zjwpeter from comment #1) > This issues happens on Safari 12 to 14.0 (15610.1.28.1.9, 15610) Hi, Are you referring to the exception thrown after sleep(120)? The behavior seems correct according to spec: https://w3c.github.io/IndexedDB/#transaction-finished, "automatic committing". Transaction is supposed to be short-lived, and it will be automatically if there is no pending request and no request is to be created synchronously.
wards.afar.0j
Comment 4 2020-09-22 17:23:28 PDT
Hi I understand this idea of transaction auto commits. You can see I use a function to check if the transaction has been outdated every time before I start a transaction operation. My check do work, if the transaction has been outdated, it will create a new transaction immediately. After the new transaction created there is no any synchronous operation, on the chrome and Firefox, this work well. But on the safari it seems like it is possible that the transaction is outdated immediately just after it has been created. Please carefully check my example again, I didn't include a wrong usage of indexed DB transaction.
Sihui Liu
Comment 5 2020-09-22 19:18:30 PDT
(In reply to zjwpeter from comment #4) > Hi I understand this idea of transaction auto commits. > You can see I use a function to check if the transaction has been outdated > every time before I start a transaction operation. > My check do work, if the transaction has been outdated, it will create a new > transaction immediately. > > After the new transaction created there is no any synchronous operation, on > the chrome and Firefox, this work well. But on the safari it seems like it > is possible that the transaction is outdated immediately just after it has > been created. > > Please carefully check my example again, I didn't include a wrong usage of > indexed DB transaction. ah sorry, I misunderstood your question. You meant await transaction.store.add({ a: 1 }, 'c') fails as newly created transaction by beforeTx() is inactive. This seems a real bug. Specifically, we are deactivating transaction when vm is idle: vm.whenIdle([protectedThis = makeRef(*this)]() { protectedThis->deactivate(); }); I don't know why we need this, changelog does not provide explanation either: https://trac.webkit.org/changeset/198762/webkit. Maybe we should remove this.
Sihui Liu
Comment 6 2020-09-23 09:59:03 PDT
(In reply to Sihui Liu from comment #5) > (In reply to zjwpeter from comment #4) > > Hi I understand this idea of transaction auto commits. > > You can see I use a function to check if the transaction has been outdated > > every time before I start a transaction operation. > > My check do work, if the transaction has been outdated, it will create a new > > transaction immediately. > > > > After the new transaction created there is no any synchronous operation, on > > the chrome and Firefox, this work well. But on the safari it seems like it > > is possible that the transaction is outdated immediately just after it has > > been created. > > > > Please carefully check my example again, I didn't include a wrong usage of > > indexed DB transaction. > > ah sorry, I misunderstood your question. You meant > await transaction.store.add({ a: 1 }, 'c') > fails as newly created transaction by beforeTx() is inactive. This seems a > real bug. > > Specifically, we are deactivating transaction when vm is idle: > vm.whenIdle([protectedThis = makeRef(*this)]() { > protectedThis->deactivate(); > }); > > I don't know why we need this, changelog does not provide explanation > either: https://trac.webkit.org/changeset/198762/webkit. Maybe we should > remove this. Another look at the spec today, https://w3c.github.io/IndexedDB/#transaction-inactive says transaction is inactive when control returns to event loop. For this test case, IIUC, transaction being and records being added are two tasks, i.e. execute JS -> JS VM idle(deactivating transaction) -> execute JS... in this perspective, it seems right to deactivate the transaction :|. @Brady may know more as he initially added this.
wards.afar.0j
Comment 7 2020-09-23 17:57:48 PDT
> Another look at the spec today, https://w3c.github.io/IndexedDB/#transaction-inactive > says transaction is inactive when control returns to event loop. For this test case, IIUC, transaction being and records being added are two tasks, i.e. execute JS -> JS VM idle(deactivating transaction) -> execute JS... in this perspective, it seems right to deactivate the transaction :|. @Brady may know more as he initially added this. It says "A transaction is in this state(inactive) after control returns to the event loop after its creation". But Promises and async function are not join the event loop, they are microtasks. > A microtask is a short function which is executed after the function or program which created it exits and only if the JavaScript execution stack is empty, but before returning control to the event loop being used by the user agent to drive the script's execution environment (https://developer.mozilla.org/en-US/docs/Web/API/HTML_DOM_API/Microtask_guide)
Sihui Liu
Comment 8 2020-09-23 21:40:08 PDT
(In reply to zjwpeter from comment #7) > > Another look at the spec today, https://w3c.github.io/IndexedDB/#transaction-inactive > > says transaction is inactive when control returns to event loop. For this test case, IIUC, transaction being and records being added are two tasks, i.e. execute JS -> JS VM idle(deactivating transaction) -> execute JS... in this perspective, it seems right to deactivate the transaction :|. @Brady may know more as he initially added this. > > > It says "A transaction is in this state(inactive) after control returns to > the event loop after its creation". But Promises and async function are not > join the event loop, they are microtasks. > > > A microtask is a short function which is executed after the function or program which created it exits and only if the JavaScript execution stack is empty, but before returning control to the event loop being used by the user agent to drive the script's execution environment > > (https://developer.mozilla.org/en-US/docs/Web/API/HTML_DOM_API/ > Microtask_guide) Thanks for the info. I confirmed transaction being created is in the same runloop task with record being added, so VM::whenIdle is definitely not what should be used here, as it does not tell us when control is returned.
Sihui Liu
Comment 9 2020-10-01 11:26:55 PDT
Sihui Liu
Comment 10 2020-10-01 11:41:29 PDT
Sihui Liu
Comment 11 2020-10-01 11:47:47 PDT
Ryosuke Niwa
Comment 12 2020-10-01 20:19:40 PDT
Comment on attachment 410252 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=410252&action=review > Source/WebCore/dom/EventLoop.cpp:71 > +void EventLoop::queueIndexedDBTask(TaskFunction&& function) I wouldn't call this IndexedDBTask. Rather, I'd call this queueWorkAtEndOfCurrentTask or something. > Source/WebCore/dom/EventLoop.cpp:73 > + m_indexedDBTaskFunctions.append(WTFMove(function)); It's possible for a task to be executing outside of EventLoop::run so we'd need to schedule it here. Just call scheduleToRunIfNeeded > Source/WebCore/dom/EventLoop.cpp:79 > + auto functions = std::exchange(m_indexedDBTaskFunctions, { }); > + for (auto& function : functions) This isn't right. Some of these functions may be associated with a page that has gotten into a page cache or otherwise stopped. > Source/WebCore/dom/EventLoop.cpp:180 > + m_eventLoop->queueIndexedDBTask(WTFMove(function)); We need to create EventLoopTask here and associate that task with this group.
Ryosuke Niwa
Comment 13 2020-11-10 11:30:20 PST
Comment on attachment 410252 [details] Patch Don't set r- on your own patch. The right thing to do is to remove r?.
Sihui Liu
Comment 14 2021-03-09 08:45:28 PST
Ryosuke Niwa
Comment 15 2021-03-09 15:39:10 PST
Comment on attachment 422706 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=422706&action=review > Source/WebCore/Modules/indexeddb/IDBTransaction.cpp:101 > + context->eventLoop().completeAfterMicrotaskCheckpoint([protectedThis = makeRef(*this)] { I'd call this runAtEndOfMicrotaskCheckpoint instead. > Source/WebCore/dom/EventLoop.cpp:189 > + m_functionsAfterMicrotaskCheckpoint.append(WTFMove(function)); > + microtaskQueue().addCheckpointListener(makeWeakPtr(this)); I'd preferred if we just had this mapping in EventLoop itself so that we preserve ordering between functions coming from different groups. Otherwise, we may inadvertently introduce a race condition depending on whether this task group had already been scheduled on the event loop or not.
Sihui Liu
Comment 16 2021-03-10 10:20:51 PST
Ryosuke Niwa
Comment 17 2021-03-10 14:24:07 PST
Comment on attachment 422842 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=422842&action=review > LayoutTests/storage/indexeddb/transaction-state-active-after-creation.html:7 > +<script src="resources/transaction-state-active-after-creation.js"></script> Can we just inline the script? It's old style of js-test to split the main testing script into a separate file.
Sihui Liu
Comment 18 2021-03-10 23:57:40 PST
Created attachment 422907 [details] Patch for landing
EWS
Comment 19 2021-03-11 00:41:28 PST
Committed r274269: <https://commits.webkit.org/r274269> All reviewed patches have been landed. Closing bug and clearing flags on attachment 422907 [details].
Sihui Liu
Comment 20 2021-03-11 10:30:09 PST
*** Bug 222746 has been marked as a duplicate of this bug. ***
Note You need to log in before you can comment on or make changes to this bug.