在React TestUtils中模拟文件输入

23

我有一个组件,其渲染函数如下:

  render: function() {
   <input
    type="file"
    name: this.props.name,
    className={this.props.className}
    onChange={this.props.handleChange}
    accept={this.props.accept}/>
 }

状态由容器管理,该容器使用jQuery AJAX调用在服务器端上传文件:

 getInitialState: function() {
   return {
     uploaded: false
  };
}

handleChange: function(event) {
event.preventDefault();

var file = event.target.files[0];
if (!file) {
  return;
}

var reader = new FileReader();
reader.readAsText(file, 'UTF-8');
var self = this;

reader.onload = function(e) {
  var content = e.target.result;

 var a =  $.ajax({
    type: 'PUT',
    url: 'http://localhost:8080/upload',
    contentType: 'application/json',
    dataType: "json",
    data: JSON.stringify({
      "input": content
    })
  })
  .fail(function(jqXHR, textStatus, errorThrown) {
    console.log("ERROR WHEN UPLOADING");
  });

 $.when(a).done(function() {
      self.setState({
      uploaded: true,
      });
  });

  } 
} 

这在服务器运行时完美地工作。 但是,我想进行无需调用服务器的测试。 这是我目前编写的Mocha测试:

var React = require('react');
var assert = require('chai').assert;
var TestUtils = require('react-addons-test-utils');
var nock = require("nock");
var MyContainer = require('../containers/MyContainer');

describe('assert upload', function () {
  it("user action", function () {

    var api = nock("http://localhost:8080")
        .put("/upload", {input: "input"})
        .reply(200, {
        });

  var renderedComponent = TestUtils.renderIntoDocument(
          <MyContainer />
  );

  var fileInput = TestUtils.findAllInRenderedTree(renderedComponent,
           function(comp) { 
                           return(comp.type == "file");
                          });

 var fs = require('fs') ;
 var filename = "upload_test.txt"; 
 var fakeF = fs.readFile(filename, 'utf8', function(err, data) {
    if (err) throw err;
 });

  TestUtils.Simulate.change(fileInput, { target: { value: fakeF } });

  assert(renderedComponent.state.uploaded === true);
  });
});

收到这个错误信息:

TypeError: Cannot read property '__reactInternalInstance$sn5kvzyx2f39pb9' of undefined

我可以通过简单执行const node = document.getElementsByClassName("close"); TestUtils.Simulate.click(node);来重现相同的错误,其中我有一个带有class="close"的按钮,它是唯一具有该类的元素/对象,并且这两行代码位于我正在使用的另一个按钮事件处理程序内。一直在寻找解决方案。使用Chrome版本51.0.2704.103,React版本15.3.1。 - vapcguy
我通过执行 const node = document.getElementById("btnRequest") 解决了我的问题,这是一个打开(或关闭)模态窗口的按钮,而不是尝试寻找模态窗口上的关闭按钮。您可以尝试逐个查找,而不是执行 var fileInput = TestUtils.findAllInRenderedTree... 吗? - vapcguy
1个回答

1

你需要一个“真实”的文件吗?我认为你的问题在于如何管理你的“虚假”文件以及你的文件输入的更改事件。你可以使用File Api来“创建”一个虚拟文件,然后只需正确设置你的文件输入即可。它不使用value,而是使用files

// similar example from KCD https://github.com/testing-library/react-testing-library/issues/93#issuecomment-392126991
const file = new File(['(⌐□_□)'], 'chucknorris.png', { type: 'image/png' });
TestUtils.Simulate.change(fileInput, { target: { files: [ file ] } });

如果你确实需要这个“文件”(upload_text.txt),那就会更加棘手。Node的fs.readFile是异步的,数据在回调之外不可用,而且你需要将数据转换为一个实际的文件对象才能提供给输入。
// rough pass at this, you'll have to play with it
import fs from 'fs';
const filename = 'upload_text.txt';
fs.readFile(filename, 'utf8', (err, data) => {
  if (err) throw err;
  const file = new File(data, filename, { type: 'text/plain' });
  TestUtils.Simulate.change(fileInput, { target: { files: [ file ] } });
  // whatever comes next
});

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接