Skip to content

Commit

Permalink
fix: Stop calling waitFor callback after timeout (#1271)
Browse files Browse the repository at this point in the history
* prevent waitFor callback racing condition

* change logic

* adapt the waitFor resolve test to make it more readable

* surface `timeout` + waitFor `callback` calls

* adding comment

* adding reproductible test

* robuster assertion for not calling callback again

---------

Co-authored-by: Kevin Bon <[email protected]>
Co-authored-by: Sebastian Silbermann <[email protected]>
  • Loading branch information
3 people authored Jan 8, 2024
1 parent a7b7252 commit 9aaf715
Show file tree
Hide file tree
Showing 2 changed files with 109 additions and 20 deletions.
121 changes: 105 additions & 16 deletions src/__tests__/wait-for.js
Original file line number Diff line number Diff line change
Expand Up @@ -292,12 +292,11 @@ test('the real timers => fake timers error shows the original stack trace when c

test('does not work after it resolves', async () => {
jest.useFakeTimers('modern')
let context = 'initial'
const contextStack = []
configure({
// @testing-library/react usage to ensure `IS_REACT_ACT_ENVIRONMENT` is set when acting.
unstable_advanceTimersWrapper: callback => {
const originalContext = context
context = 'act'
contextStack.push('act:start')
try {
const result = callback()
// eslint-disable-next-line jest/no-if, jest/no-conditional-in-test -- false-positive
Expand All @@ -307,54 +306,144 @@ test('does not work after it resolves', async () => {
then: (resolve, reject) => {
thenable.then(
returnValue => {
context = originalContext
contextStack.push('act:end')
resolve(returnValue)
},
error => {
context = originalContext
contextStack.push('act:end')
reject(error)
},
)
},
}
} else {
context = originalContext
contextStack.push('act:end')
return undefined
}
} catch {
context = originalContext
contextStack.push('act:end')
return undefined
}
},
asyncWrapper: async callback => {
const originalContext = context
context = 'no-act'
contextStack.push('no-act:start')
try {
await callback()
} finally {
context = originalContext
contextStack.push('no-act:end')
}
},
})

let data = null
let timeoutResolved = false
setTimeout(() => {
data = 'resolved'
contextStack.push('timeout')
timeoutResolved = true
}, 100)

contextStack.push('waitFor:start')
await waitFor(
() => {
contextStack.push('callback')
// eslint-disable-next-line jest/no-conditional-in-test -- false-positive
if (data === null) {
if (!timeoutResolved) {
throw new Error('not found')
}
},
{interval: 50},
)

expect(context).toEqual('initial')
contextStack.push('waitFor:end')

expect(contextStack).toMatchInlineSnapshot(`
[
waitFor:start,
no-act:start,
callback,
act:start,
act:end,
callback,
act:start,
timeout,
act:end,
callback,
no-act:end,
waitFor:end,
]
`)

await Promise.resolve()

expect(context).toEqual('initial')
// The context call stack should not change
expect(contextStack).toMatchInlineSnapshot(`
[
waitFor:start,
no-act:start,
callback,
act:start,
act:end,
callback,
act:start,
timeout,
act:end,
callback,
no-act:end,
waitFor:end,
]
`)
})

test(`when fake timer is installed, on waitFor timeout, it doesn't call the callback afterward`, async () => {
jest.useFakeTimers('modern')

configure({
// @testing-library/react usage to ensure `IS_REACT_ACT_ENVIRONMENT` is set when acting.
unstable_advanceTimersWrapper: callback => {
try {
const result = callback()
// eslint-disable-next-line jest/no-if, jest/no-conditional-in-test -- false-positive
if (typeof result?.then === 'function') {
const thenable = result
return {
then: (resolve, reject) => {
thenable.then(
returnValue => {
resolve(returnValue)
},
error => {
reject(error)
},
)
},
}
} else {
return undefined
}
} catch {
return undefined
}
},
asyncWrapper: async callback => {
try {
await callback()
} finally {
/* eslint no-empty: "off" */
}
},
})

const callback = jest.fn(() => {
throw new Error('We want to timeout')
})
const interval = 50

await expect(() =>
waitFor(callback, {interval, timeout: 100}),
).rejects.toThrow()
expect(callback).toHaveBeenCalledWith()

callback.mockClear()

await jest.advanceTimersByTimeAsync(interval)

expect(callback).not.toHaveBeenCalledWith()
})
8 changes: 4 additions & 4 deletions src/wait-for.js
Original file line number Diff line number Diff line change
Expand Up @@ -81,15 +81,15 @@ function waitFor(
jest.advanceTimersByTime(interval)
})

// Could have timed-out
if (finished) {
break
}
// It's really important that checkCallback is run *before* we flush
// in-flight promises. To be honest, I'm not sure why, and I can't quite
// think of a way to reproduce the problem in a test, but I spent
// an entire day banging my head against a wall on this.
checkCallback()

if (finished) {
break
}
}
} else {
try {
Expand Down

0 comments on commit 9aaf715

Please sign in to comment.