测试自定义钩子使用时,安装程​​序函数返回未定义

时间:2019-11-30 07:36:24

标签: javascript reactjs jestjs enzyme react-testing-library

我有供使用的代码Internet上某处有以前的钩子。 usePrevious的代码如下:

export const usePrevious = (value) => {
  const ref = useRef();
  useEffect(() => {
    ref.current = value;
  }, [value]);
  return ref.current;
};

现在,我正在测试与玩笑和酵素的反应。因此,我尝试测试usePrevious并遇到了一些问题。这是我的测试用例:

import React from 'react';
import { render } from 'enzyme';

import { usePrevious } from './customHooks';

const Component = ({ children, value }) => children(usePrevious(value));
const setup = (value) => {
  let returnVal = '';
  render(
    <Component value={value}>
      {
        (val) => {
          returnVal = val;
          return null;
        }
      }
    </Component>,
  );
  return returnVal;
};

describe('usePrevious', () => {
  it('returns something', () => {
    const test1 = setup('test');
    const test2 = setup(test1);
    expect(test2).toBe('test');
  });
});

测试执行完成后,出现此错误:

Expected: 'test', Received: undefined

任何人都可以让我知道为什么我变得不确定,这是测试react中自定义钩子的正确方法吗?

在来自@Dmitrii G的评论的建议之后,我更改了代码以重新呈现组件(以前我是在重新安装组件)。

这是更新的代码:

import React from 'react';
import PropTypes from 'prop-types';
import { shallow } from 'enzyme';

import { usePrevious } from './customHooks';

const Component = ({ value }) => {
  const hookResult = usePrevious(value);
  return (
    <div>
      <span>{hookResult}</span>
      <span>{value}</span>
    </div>
  );
};

Component.propTypes = {
  value: PropTypes.string,
};

Component.defaultProps = {
  value: '',
};


describe('usePrevious', () => {
  it('returns something', () => {
    const wrapper = shallow(<Component value="test" />);
    console.log('>>>>> first time', wrapper.find('div').childAt(1).text());
    expect(wrapper.find('div').childAt(0).text()).toBe('');

    // Test second render and effect
    wrapper.setProps({ value: 'test2' });
    console.log('>>>>> second time', wrapper.find('div').childAt(1).text());
    expect(wrapper.find('div').childAt(0).text()).toBe('test');
  });
});

但是我仍然遇到相同的错误

Expected: "test", Received: ""

在引入settimeout时测试通过:

import React from 'react';
import PropTypes from 'prop-types';
import { shallow } from 'enzyme';

import { usePrevious } from './customHooks';

const Component = ({ value }) => {
  const hookResult = usePrevious(value);
  return <span>{hookResult}</span>;
};

Component.propTypes = {
  value: PropTypes.string,
};

Component.defaultProps = {
  value: '',
};


describe('usePrevious', () => {
  it('returns empty string when component is rendered first time', () => {
    const wrapper = shallow(<Component value="test" />);
    setTimeout(() => {
      expect(wrapper.find('span').text()).toBe('');
    }, 0);
  });
  it('returns previous value when component is re-rendered', () => {
    const wrapper = shallow(<Component value="test" />);
    wrapper.setProps({ value: 'test2' });
    setTimeout(() => {
      expect(wrapper.find('span').text()).toBe('test');
    }, 0);
  });
});

我不是使用settimeout的忠实拥护者,所以我觉得我可能在做一些错误。如果有人知道不使用settimeout的解决方案,请随时在此处发布。谢谢。

1 个答案:

答案 0 :(得分:1)

引擎盖上的酶利用了React的浅层渲染器。它有issue with running effects。不知道它是否会很快修复。 onnx的解决方法令我感到惊讶,不知道它是否有效。不幸的是,这不是通用方法,因为您需要在案例重新呈现的任何更改中使用该方法。真的很脆弱。

作为解决方案,您可以改用setTimeout

此外,您可以通过模拟每个嵌套组件来模拟mount()的浅层渲染:

mount()