Подтвердить что ты не робот

ComponentDidMount setState, не отражающийся в тесте Enzyme

компонент:

import React from 'react'
import request from 'superagent'

export default React.createClass({
    getInitialState() {
        return {cats: []}
    },

    componentDidMount() {
        request('/api', (err, res) => {
            if (err) return;
            this.setState({
                cats: res.body.results
            })
        })
    },

    render() {
        let cats = this.state.cats
        let catsList = (
            <ul>
                {cats.map((c) => <li key={c.id}>cat</li>)}
            </ul>
        )
        return (
            <div>
                hello world
                {cats.length ? catsList : null}
            </div>
        )
    }
})

Тест:

jest.unmock('../app.js')
jest.unmock('superagent')

import React from 'react'
import {mount} from 'enzyme'
import nock from 'nock'
import App from '../app.js'

describe('App', () => {
    let ajaxFn
    beforeEach(() => {
        ajaxFn = nock('http://localhost')
            .get('/api')
            .reply(200, {
                results: [{id: 1}, {id: 2}, {id: 3}]
            })
    })

    it('says hello world', () => {
        let wrapper = mount(<App/>)
        expect(wrapper.text()).toContain('hello worl')
    })

    it('makes ajax request', () => {
        let wrapper = mount(<App/>)
        expect(ajaxFn.isDone()).toBe(true)
    })

    it('renders correct number of cats', () => {
        let wrapper = mount(<App />)
        expect(wrapper.state('cats').length).toBe(3)
    })
})

Первые два теста проходят, но окончательный - нет, т.е. wrapper.state('cats'). length == 0

Я понимаю, что setState не гарантирует немедленное обновление состояния, однако, если я зарегистрирую "кошки" в компоненте, я могу увидеть его обновление.

4b9b3361

Ответ 1

Если вы закончите настройку состояния вашего компонента в каком-то контексте, о котором фермент не знает, вам придется вручную вызвать .update() на обертке, чтобы получить обновленную версию дерева рендеринга.

it('renders correct number of cats', () => {
    let wrapper = mount(<App />)
    expect(wrapper.update().state('cats').length).toBe(3)
})

Ответ 2

У меня была аналогичная проблема, и было необходимо возвратить обещание от обратного вызова it и проверить ожидание в методе then обещания.

В вашем случае (предполагая, что ajaxFn было обещанием, или вы могли превратить его в один), я думаю, что это будет примерно:

it('renders correct number of cats', () => {
    let wrapper = mount(<App />) 
    return ajaxFn.then(() => {
        expect(wrapper.state('cats').length).toBe(3);
    }
})