Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Testing async componentDidMount() with react-test-renderer

I have a component which does some SQLite loads in it's componentDidMount() function

async componentDidMount() {
    try {
        const user = await this.userDao.getUserData();
        const setupNeeded = user === null;
        if( setupNeeded ) {
            this.setState({
                status : 'setup'
            });
        }
        else {
            this.setState({
                status : 'ready',
                seed: user.seed
            })
        }
    } catch (e) {
        logger.error("Error during db query", e);
        this.setState({
            status: 'corrupted'
        });
    }
}

And I would like to test the rendered result after the call to getUserData() has been resolved, and the state has been set accordingly. Now in my test, I have the actual call to the database mocked away, so the Promise should be resolved immediately, yet testing like this does not work as expected:

Initially, I tried it like this:

test('Should render SetNewPasswordScreen', async () => {
    const tree = await renderer.create(<IndexScreen/>);
    const json = tree.toJSON();
    // expect stuff
});

However the json in this case contains the data of the initial render() call.`

Doing it like this will work:

 test('Should render SetNewPasswordScreen', (done) => {
     const tree = renderer.create(<IndexScreen/>);
     setTimeout(() => {
         const json = tree.toJSON();
         // expect stuff
     }, 5000);
 });

But this is not ideal, because I am just guessing that after 5 seconds everything will be done, but I don't know. Also, it's less then suitable if a test takes 5 seconds to finish. (I used 5 seconds arbitrarily, probably it will also work with much less since the async call is mocked anyway, but I can never really know)

My question is if anybody has a better idea of how to solve this issue?

like image 233
Jakob Abfalter Avatar asked Oct 29 '25 15:10

Jakob Abfalter


1 Answers

Little late for the party but faced the issues as well. Here is a handy shortcut function I use with the adviced usage of act:

import renderer from "react-test-renderer"
import { ReactElement } from "react"
import { act } from "@testing-library/react-native"

export const expectToMatchSnapshot = async (component: ReactElement) => {
   let tree

   await act(async () => {
      tree = renderer.create(component)
   })

   expect(tree.toJSON()).toMatchSnapshot()
}

In your case you should only then add:

test("should render ***", async () => {
   await expectToMatchSnapshot(<IndexScreen/>)
}
like image 185
Kia Kaha Avatar answered Oct 31 '25 04:10

Kia Kaha



Donate For Us

If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!