反应如何测试FileReader onload函数中调用的函数

时间:2017-12-24 18:29:16

标签: javascript reactjs tdd jestjs enzyme

我在组件中有以下代码。我想测试表单的onSubmit,它在reader中调用this.props.onUpload方法。 我怎么测试呢? 我的期望测试不起作用,我猜是因为this.props.onUpload在reader.onload函数里面?

UploadForm.js

handleSubmit(e) {
    e.preventDefault();

    var inputData = '';
    var file = this.state.file;
    if (file) {
        var reader = new FileReader();
        reader.onload = (function(file) {
            return function(e) {
                inputData = e.target.result;
                this.props.onUpload(inputData);
            };
        })(file).bind(this);
        reader.readAsText(file);            
    }

}

render() {
    return(
        <form onSubmit={this.handleSubmit}>
            <label> Enter File: <br/>
                <input type="file" id="fileinput" onChange={this.handleChange}/>    
            </label>
            <input type="submit" value="Submit" className="btn-upload" />
        </form>
    );
}

UploadForm.test.js

const mockOnUpload = jest.fn();
const file = new File([""], "filename");
const form = shallow(<UploadForm onUpload={mockOnUpload}/>);
const event = {
          preventDefault: jest.fn(),
          target: {files : [file]}
        };

describe('when clicking `upload-file` button', () => {
    beforeEach(() => {
      form.find('#fileinput').simulate('change', event);
      form.find('form').simulate('submit', event);
    });

    it('calls the handleSubmit CallBack', () => {
      expect(mockOnUpload).toHaveBeenCalledWith(input);
    });

});

1 个答案:

答案 0 :(得分:1)

超级好的开始,模拟上传作为道具传递并创建假事件!

我总是喜欢在我自己的工作中遇到这样的测试问题,因为它告诉我我有一个代码味道:如果它不容易测试,可能意味着它更难预测,更难调试,更难解释, &安培; C。

我建议将你的职能分解为更加单一的责任。就目前而言,您的handleSubmit所做的不仅仅是处理提交。它还向FileReader实例添加onload函数,并在该实例上调用readAsText

您的IIFE

function(file) {
  return function(e) {
    inputData = e.target.result;
    this.props.onUpload(inputData);
  };
})(file).bind(this);
可以将

拉出到组件上的箭头函数(处理bind):

readerOnLoad = file => (e) => {
  this.props.onUpload(e.target.result);
}

(此外,这里需要file作为参数吗?似乎没有使用。)

然后handleSubmit can interact with readerOnLoad`喜欢;

reader.onload = this.readOnLoad(file);

此时,您可以测试:

  1. handleSubmit如果状态存在,则使用文件参数调用readerOnLoad
  2. 使用文件参数和事件参数调用的
  3. readerOnLoad使用事件目标this.props.onLoad值调用result
  4. 如果这两个测试都通过,您可以确信您的代码可以使用真实事件,文件和FileReader实例。

    看起来你已经理解了如何传入与本机/浏览器对象匹配的鸭子类型参数(如event s),所以只需将它们组合在一起,就可以享受精心协作的功能!