diff --git a/src/__tests__/act-compat.js b/src/__tests__/act-compat.js
new file mode 100644
index 00000000..89aacbc9
--- /dev/null
+++ b/src/__tests__/act-compat.js
@@ -0,0 +1,99 @@
+import * as React from 'react'
+import {render, fireEvent, screen} from '../'
+import {actIfEnabled} from '../act-compat'
+
+beforeEach(() => {
+ global.IS_REACT_ACT_ENVIRONMENT = true
+})
+
+test('render calls useEffect immediately', async () => {
+ const effectCb = jest.fn()
+ function MyUselessComponent() {
+ React.useEffect(effectCb)
+ return null
+ }
+ await render()
+ expect(effectCb).toHaveBeenCalledTimes(1)
+})
+
+test('findByTestId returns the element', async () => {
+ const ref = React.createRef()
+ await render(
)
+ expect(await screen.findByTestId('foo')).toBe(ref.current)
+})
+
+test('fireEvent triggers useEffect calls', async () => {
+ const effectCb = jest.fn()
+ function Counter() {
+ React.useEffect(effectCb)
+ const [count, setCount] = React.useState(0)
+ return
+ }
+ const {
+ container: {firstChild: buttonNode},
+ } = await render()
+
+ effectCb.mockClear()
+ // eslint-disable-next-line testing-library/no-await-sync-events -- TODO: Remove lint rule.
+ await fireEvent.click(buttonNode)
+ expect(buttonNode).toHaveTextContent('1')
+ expect(effectCb).toHaveBeenCalledTimes(1)
+})
+
+test('calls to hydrate will run useEffects', async () => {
+ const effectCb = jest.fn()
+ function MyUselessComponent() {
+ React.useEffect(effectCb)
+ return null
+ }
+ await render(, {hydrate: true})
+ expect(effectCb).toHaveBeenCalledTimes(1)
+})
+
+test('cleans up IS_REACT_ACT_ENVIRONMENT if its callback throws', async () => {
+ global.IS_REACT_ACT_ENVIRONMENT = false
+
+ await expect(() =>
+ actIfEnabled(() => {
+ throw new Error('threw')
+ }),
+ ).rejects.toThrow('threw')
+
+ expect(global.IS_REACT_ACT_ENVIRONMENT).toEqual(false)
+})
+
+test('cleans up IS_REACT_ACT_ENVIRONMENT if its async callback throws', async () => {
+ global.IS_REACT_ACT_ENVIRONMENT = false
+
+ await expect(() =>
+ actIfEnabled(async () => {
+ throw new Error('thenable threw')
+ }),
+ ).rejects.toThrow('thenable threw')
+
+ expect(global.IS_REACT_ACT_ENVIRONMENT).toEqual(false)
+})
+
+test('state update from microtask does not trigger "missing act" warning', async () => {
+ let triggerStateUpdateFromMicrotask
+ function App() {
+ const [state, setState] = React.useState(0)
+ triggerStateUpdateFromMicrotask = () => setState(1)
+ React.useEffect(() => {
+ // eslint-disable-next-line jest/no-conditional-in-test
+ if (state === 1) {
+ Promise.resolve().then(() => {
+ setState(2)
+ })
+ }
+ }, [state])
+ return state
+ }
+ const {container} = await render()
+
+ await actIfEnabled(() => {
+ triggerStateUpdateFromMicrotask()
+ })
+
+ expect(container).toHaveTextContent('2')
+})
diff --git a/src/__tests__/act.js b/src/__tests__/act.js
index b4485f3c..4d40baf0 100644
--- a/src/__tests__/act.js
+++ b/src/__tests__/act.js
@@ -1,98 +1,26 @@
import * as React from 'react'
-import {act, render, fireEvent, screen} from '../'
+import {act, render} from '../'
beforeEach(() => {
global.IS_REACT_ACT_ENVIRONMENT = true
})
-test('render calls useEffect immediately', async () => {
- const effectCb = jest.fn()
- function MyUselessComponent() {
- React.useEffect(effectCb)
- return null
- }
- await render()
- expect(effectCb).toHaveBeenCalledTimes(1)
-})
-
-test('findByTestId returns the element', async () => {
- const ref = React.createRef()
- await render()
- expect(await screen.findByTestId('foo')).toBe(ref.current)
-})
-
-test('fireEvent triggers useEffect calls', async () => {
- const effectCb = jest.fn()
- function Counter() {
- React.useEffect(effectCb)
- const [count, setCount] = React.useState(0)
- return
- }
- const {
- container: {firstChild: buttonNode},
- } = await render()
-
- effectCb.mockClear()
- // eslint-disable-next-line testing-library/no-await-sync-events -- TODO: Remove lint rule.
- await fireEvent.click(buttonNode)
- expect(buttonNode).toHaveTextContent('1')
- expect(effectCb).toHaveBeenCalledTimes(1)
-})
-
-test('calls to hydrate will run useEffects', async () => {
- const effectCb = jest.fn()
- function MyUselessComponent() {
- React.useEffect(effectCb)
- return null
- }
- await render(, {hydrate: true})
- expect(effectCb).toHaveBeenCalledTimes(1)
-})
-
-test('cleans up IS_REACT_ACT_ENVIRONMENT if its callback throws', async () => {
- global.IS_REACT_ACT_ENVIRONMENT = false
-
- await expect(() =>
- act(() => {
- throw new Error('threw')
- }),
- ).rejects.toThrow('threw')
-
- expect(global.IS_REACT_ACT_ENVIRONMENT).toEqual(false)
-})
-
-test('cleans up IS_REACT_ACT_ENVIRONMENT if its async callback throws', async () => {
- global.IS_REACT_ACT_ENVIRONMENT = false
-
- await expect(() =>
- act(async () => {
- throw new Error('thenable threw')
- }),
- ).rejects.toThrow('thenable threw')
-
- expect(global.IS_REACT_ACT_ENVIRONMENT).toEqual(false)
-})
-
-test('state update from microtask does not trigger "missing act" warning', async () => {
- let triggerStateUpdateFromMicrotask
- function App() {
- const [state, setState] = React.useState(0)
- triggerStateUpdateFromMicrotask = () => setState(1)
- React.useEffect(() => {
- // eslint-disable-next-line jest/no-conditional-in-test
- if (state === 1) {
- Promise.resolve().then(() => {
- setState(2)
- })
- }
- }, [state])
+test('does not work outside IS_REACT_ENVIRONMENT like React.act', async () => {
+ let setState
+ function Component() {
+ const [state, _setState] = React.useState(0)
+ setState = _setState
return state
}
- const {container} = await render()
-
- await act(() => {
- triggerStateUpdateFromMicrotask()
- })
+ await render()
- expect(container).toHaveTextContent('2')
+ global.IS_REACT_ACT_ENVIRONMENT = false
+ await expect(async () => {
+ await act(() => {
+ setState(1)
+ })
+ }).toErrorDev(
+ 'Warning: The current testing environment is not configured to support act(...)',
+ {withoutStack: true},
+ )
})
diff --git a/src/__tests__/auto-cleanup-skip.js b/src/__tests__/auto-cleanup-skip.js
index b1c88d26..89a99e9b 100644
--- a/src/__tests__/auto-cleanup-skip.js
+++ b/src/__tests__/auto-cleanup-skip.js
@@ -3,6 +3,7 @@ import * as React from 'react'
let render
beforeAll(() => {
process.env.RTL_SKIP_AUTO_CLEANUP = 'true'
+ globalThis.IS_REACT_ACT_ENVIRONMENT = true
const rtl = require('../')
render = rtl.render
})
diff --git a/src/__tests__/end-to-end.js b/src/__tests__/end-to-end.js
index da3b2640..0c93b7f8 100644
--- a/src/__tests__/end-to-end.js
+++ b/src/__tests__/end-to-end.js
@@ -1,5 +1,4 @@
-import * as React from 'react'
-import {render, waitForElementToBeRemoved, screen, waitFor} from '../'
+let React, cleanup, render, screen, waitFor, waitForElementToBeRemoved
describe.each([
['real timers', () => jest.useRealTimers()],
@@ -9,10 +8,25 @@ describe.each([
'it waits for the data to be loaded in a macrotask using %s',
(label, useTimers) => {
beforeEach(() => {
+ jest.resetModules()
+ global.IS_REACT_ACT_ENVIRONMENT = true
+ process.env.RTL_SKIP_AUTO_CLEANUP = '0'
+
useTimers()
+
+ React = require('react')
+ ;({
+ cleanup,
+ render,
+ screen,
+ waitFor,
+ waitForElementToBeRemoved,
+ } = require('..'))
})
- afterEach(() => {
+ afterEach(async () => {
+ await cleanup()
+ global.IS_REACT_ACT_ENVIRONMENT = false
jest.useRealTimers()
})
@@ -83,10 +97,25 @@ describe.each([
'it waits for the data to be loaded in many microtask using %s',
(label, useTimers) => {
beforeEach(() => {
+ jest.resetModules()
+ global.IS_REACT_ACT_ENVIRONMENT = true
+ process.env.RTL_SKIP_AUTO_CLEANUP = '0'
+
useTimers()
+
+ React = require('react')
+ ;({
+ cleanup,
+ render,
+ screen,
+ waitFor,
+ waitForElementToBeRemoved,
+ } = require('..'))
})
- afterEach(() => {
+ afterEach(async () => {
+ await cleanup()
+ global.IS_REACT_ACT_ENVIRONMENT = false
jest.useRealTimers()
})
@@ -167,10 +196,25 @@ describe.each([
'it waits for the data to be loaded in a microtask using %s',
(label, useTimers) => {
beforeEach(() => {
+ jest.resetModules()
+ global.IS_REACT_ACT_ENVIRONMENT = true
+ process.env.RTL_SKIP_AUTO_CLEANUP = '0'
+
useTimers()
+
+ React = require('react')
+ ;({
+ cleanup,
+ render,
+ screen,
+ waitFor,
+ waitForElementToBeRemoved,
+ } = require('..'))
})
- afterEach(() => {
+ afterEach(async () => {
+ await cleanup()
+ global.IS_REACT_ACT_ENVIRONMENT = false
jest.useRealTimers()
})
@@ -218,3 +262,84 @@ describe.each([
})
},
)
+
+describe.each([
+ // ['real timers', () => jest.useRealTimers()],
+ ['fake legacy timers', () => jest.useFakeTimers('legacy')],
+ // ['fake modern timers', () => jest.useFakeTimers('modern')],
+])('testing intermediate states using %s', (label, useTimers) => {
+ beforeEach(() => {
+ jest.resetModules()
+ global.IS_REACT_ACT_ENVIRONMENT = false
+ process.env.RTL_SKIP_AUTO_CLEANUP = '0'
+
+ useTimers()
+
+ React = require('react')
+ ;({
+ cleanup,
+ render,
+ screen,
+ waitFor,
+ waitForElementToBeRemoved,
+ } = require('..'))
+ })
+
+ afterEach(async () => {
+ await cleanup()
+ jest.useRealTimers()
+ global.IS_REACT_ACT_ENVIRONMENT = true
+ })
+
+ const fetchAMessageInAMicrotask = () =>
+ Promise.resolve({
+ status: 200,
+ json: () => Promise.resolve({title: 'Hello World'}),
+ })
+
+ function ComponentWithMicrotaskLoader() {
+ const [fetchState, setFetchState] = React.useState({fetching: true})
+
+ React.useEffect(() => {
+ if (fetchState.fetching) {
+ fetchAMessageInAMicrotask().then(res => {
+ return res.json().then(data => {
+ setFetchState({todo: data.title, fetching: false})
+ })
+ })
+ }
+ }, [fetchState])
+
+ if (fetchState.fetching) {
+ return Loading..
+ }
+
+ return (
+ Loaded this message: {fetchState.todo}
+ )
+ }
+
+ test('waitFor', async () => {
+ await render()
+
+ // TODO: How to assert on the intermediate state?
+ await expect(
+ waitFor(() => {
+ expect(screen.getByText('Loading..')).toBeInTheDocument()
+ }),
+ ).rejects.toThrowError(/Unable to find an element/)
+
+ expect(screen.getByTestId('message')).toHaveTextContent(/Hello World/)
+ })
+
+ test('findBy', async () => {
+ await render()
+
+ // TODO: How to assert on the intermediate state?
+ await expect(screen.findByText('Loading..')).rejects.toThrowError(
+ /Unable to find an element/,
+ )
+
+ expect(screen.getByTestId('message')).toHaveTextContent(/Hello World/)
+ })
+})
diff --git a/src/__tests__/new-act.js b/src/__tests__/new-act.js
index 0464ad24..2bb671b6 100644
--- a/src/__tests__/new-act.js
+++ b/src/__tests__/new-act.js
@@ -1,4 +1,4 @@
-let asyncAct
+let actIfEnabled
jest.mock('react', () => {
return {
@@ -11,7 +11,7 @@ jest.mock('react', () => {
beforeEach(() => {
jest.resetModules()
- asyncAct = require('../act-compat').default
+ actIfEnabled = require('../act-compat').actIfEnabled
jest.spyOn(console, 'error').mockImplementation(() => {})
})
@@ -21,7 +21,7 @@ afterEach(() => {
test('async act works when it does not exist (older versions of react)', async () => {
const callback = jest.fn()
- await asyncAct(async () => {
+ await actIfEnabled(async () => {
await Promise.resolve()
await callback()
})
@@ -31,7 +31,7 @@ test('async act works when it does not exist (older versions of react)', async (
callback.mockClear()
console.error.mockClear()
- await asyncAct(async () => {
+ await actIfEnabled(async () => {
await Promise.resolve()
await callback()
})
@@ -41,7 +41,7 @@ test('async act works when it does not exist (older versions of react)', async (
test('async act recovers from errors', async () => {
try {
- await asyncAct(async () => {
+ await actIfEnabled(async () => {
await null
throw new Error('test error')
})
@@ -60,7 +60,7 @@ test('async act recovers from errors', async () => {
test('async act recovers from sync errors', async () => {
try {
- await asyncAct(() => {
+ await actIfEnabled(() => {
throw new Error('test error')
})
} catch (err) {
diff --git a/src/__tests__/renderHook.js b/src/__tests__/renderHook.js
index 52a8a4a8..44313e4d 100644
--- a/src/__tests__/renderHook.js
+++ b/src/__tests__/renderHook.js
@@ -1,5 +1,5 @@
import React from 'react'
-import {renderHook} from '../pure'
+import {renderHook} from '../'
const isReact18 = React.version.startsWith('18.')
const isReact19 = React.version.startsWith('19.')
diff --git a/src/act-compat.js b/src/act-compat.js
index 4302c1b8..f209659b 100644
--- a/src/act-compat.js
+++ b/src/act-compat.js
@@ -33,26 +33,21 @@ function getIsReactActEnvironment() {
return getGlobalThis().IS_REACT_ACT_ENVIRONMENT
}
-async function act(scope) {
- const previousActEnvironment = getIsReactActEnvironment()
- setIsReactActEnvironment(true)
- try {
+async function actIfEnabled(scope) {
+ if (getIsReactActEnvironment()) {
// scope passed to domAct needs to be `async` until React.act treats every scope as async.
// We already enforce `await act()` (regardless of scope) to flush microtasks
// inside the act scope.
- const result = await reactAct(async () => {
+ return reactAct(async () => {
return scope()
})
- return result
- } finally {
- setIsReactActEnvironment(previousActEnvironment)
+ } else {
+ // We wrap everything in act internally.
+ // But a userspace call might not want that so we respect global config here.
+ return scope()
}
}
-export default act
-export {
- setIsReactActEnvironment as setReactActEnvironment,
- getIsReactActEnvironment,
-}
+export {actIfEnabled, setIsReactActEnvironment, getIsReactActEnvironment}
/* eslint no-console:0 */
diff --git a/src/index.js b/src/index.js
index 4637ebf6..ac8b36f9 100644
--- a/src/index.js
+++ b/src/index.js
@@ -1,4 +1,4 @@
-import {getIsReactActEnvironment, setReactActEnvironment} from './act-compat'
+import {getIsReactActEnvironment, setIsReactActEnvironment} from './act-compat'
import {cleanup} from './pure'
// if we're running in a test runner that supports afterEach
@@ -29,11 +29,11 @@ if (typeof process === 'undefined' || !process.env?.RTL_SKIP_AUTO_CLEANUP) {
let previousIsReactActEnvironment = getIsReactActEnvironment()
beforeAll(() => {
previousIsReactActEnvironment = getIsReactActEnvironment()
- setReactActEnvironment(true)
+ setIsReactActEnvironment(true)
})
afterAll(() => {
- setReactActEnvironment(previousIsReactActEnvironment)
+ setIsReactActEnvironment(previousIsReactActEnvironment)
})
}
}
diff --git a/src/pure.js b/src/pure.js
index 4a18b20e..390dcf55 100644
--- a/src/pure.js
+++ b/src/pure.js
@@ -1,43 +1,58 @@
import * as React from 'react'
import ReactDOM from 'react-dom'
+import {act as domAct} from 'react-dom/test-utils'
import * as ReactDOMClient from 'react-dom/client'
import {
getQueriesForElement,
prettyDOM,
configure as configureDTL,
} from '@testing-library/dom'
-import act, {
+import {
+ actIfEnabled,
getIsReactActEnvironment,
- setReactActEnvironment,
+ setIsReactActEnvironment,
} from './act-compat'
import {fireEvent} from './fire-event'
import {getConfig, configure} from './config'
+function enqueueTask(task) {
+ const channel = new MessageChannel()
+ channel.port1.onmessage = () => {
+ channel.port1.close()
+ task()
+ }
+ channel.port2.postMessage(undefined)
+}
+
+async function waitForMicrotasks() {
+ return new Promise(resolve => {
+ enqueueTask(() => resolve())
+ })
+}
+
configureDTL({
- unstable_advanceTimersWrapper: cb => {
- // Only needed to support test environments that enable fake timers after modules are loaded.
- // React's scheduler will detect fake timers when it's initialized and use them.
- // So if we change the timers after that, we need to re-initialize the scheduler.
- // But not every test runner supports module reset.
- // It's not even clear how modules should be reset in ESM.
- // So for this brief period we go back to using the act queue.
- return act(cb)
+ unstable_advanceTimersWrapper: async scope => {
+ if (getIsReactActEnvironment()) {
+ return actIfEnabled(scope)
+ } else {
+ const result = await scope()
+ await waitForMicrotasks()
+ return result
+ }
},
// We just want to run `waitFor` without IS_REACT_ACT_ENVIRONMENT
// But that's not necessarily how `asyncWrapper` is used since it's a public method.
// Let's just hope nobody else is using it.
asyncWrapper: async cb => {
const previousActEnvironment = getIsReactActEnvironment()
- setReactActEnvironment(false)
+ setIsReactActEnvironment(false)
try {
return await cb()
} finally {
- setReactActEnvironment(previousActEnvironment)
+ setIsReactActEnvironment(previousActEnvironment)
}
},
- eventWrapper: cb => {
- return act(cb)
- },
+ eventWrapper: actIfEnabled,
})
// Ideally we'd just use a WeakMap where containers are keys and roots are values.
@@ -69,7 +84,7 @@ async function createConcurrentRoot(
) {
let root
if (hydrate) {
- await act(() => {
+ await actIfEnabled(() => {
root = ReactDOMClient.hydrateRoot(
container,
strictModeIfNeeded(wrapUiIfNeeded(ui, WrapperComponent)),
@@ -90,12 +105,12 @@ async function createConcurrentRoot(
// Nothing to do since hydration happens when creating the root object.
},
render(element) {
- return act(() => {
+ return actIfEnabled(() => {
root.render(element)
})
},
unmount() {
- return act(() => {
+ return actIfEnabled(() => {
root.unmount()
})
},
@@ -105,17 +120,17 @@ async function createConcurrentRoot(
async function createLegacyRoot(container) {
return {
hydrate(element) {
- return act(() => {
+ return actIfEnabled(() => {
ReactDOM.hydrate(element, container)
})
},
render(element) {
- return act(() => {
+ return actIfEnabled(() => {
ReactDOM.render(element, container)
})
},
unmount() {
- return act(() => {
+ return actIfEnabled(() => {
ReactDOM.unmountComponentAtNode(container)
})
},
@@ -290,8 +305,25 @@ async function renderHook(renderCallback, options = {}) {
return {result, rerender, unmount}
}
+function compatAct(scope) {
+ // scope passed to domAct needs to be `async` until React.act treats every scope as async.
+ // We already enforce `await act()` (regardless of scope) to flush microtasks
+ // inside the act scope.
+ return domAct(async () => {
+ return scope()
+ })
+}
+
// just re-export everything from dom-testing-library
export * from '@testing-library/dom'
-export {render, renderHook, cleanup, act, fireEvent, getConfig, configure}
+export {
+ render,
+ renderHook,
+ cleanup,
+ compatAct as act,
+ fireEvent,
+ getConfig,
+ configure,
+}
/* eslint func-name-matching:0 */