为什么我的 Sinon 间谍函数在 promise then 子句中调用时不起作用?
Why isn't my Sinon spy function working when called in a promise then clause?
我正在为基于 Promise 的函数编写测试。具体来说,它是一个 React 组件,我正在测试以确保正确调用 onChange 处理程序。
我的组件如下所示:
class TextInput extends React.Component {
constructor(props) {
super(props);
this.state = {
value: props.value || '',
};
this.onChange = this.onChange.bind(this);
}
updateState(values) {
return new Promise(
(resolve) => {
this.setState(values, () => { resolve(this.state); });
}
);
}
onChange(event) {
this.updateState({ value: event.target.value })
// then fire the onChange handler (if necessary)
//
.then((state) => {
if (this.props.onChange) {
// console.log(this.props.onChange) shows that this IS a
// Sinon spy function
this.props.onChange(state.value);
}
})
.catch((err) => { console.log('-----------', err); });
}
render() {
// render the component (omitted to keep this short)
}
}
我的测试是这样的:
import React from 'react';
import { mount } from 'enzyme';
import chai from 'chai';
import sinon from 'sinon';
import TextInput from '../../../../client/modules/components/TextInput';
const expect = chai.expect;
describe('TextInput component editing', () => {
it('calls the onChange handler', () => {
const onchange = sinon.spy();
const value = '';
const editedValue = 'something';
const component = mount(<TextInput value={value} onChange={onchange} />);
// change the value
//
component.find('input').simulate('change', {
target: { value: editedValue }
});
expect(component.find('input').prop('value')).to.equal(editedValue);
expect(onchange.calledOnce).to.equal(true);
expect(onchange.calledWith(editedValue)).to.equal(true);
});
});
最后两次 expect
调用测试失败。
如果我将 sinon spy 替换为普通的旧函数,则会调用该函数。例如,
// instead of this...
// const onchange = sinon.spy();
// do this...
const onchange = (value) => { console.log(`VALUE = ${value}`); };
如果我直接使用 setState
方法的回调,它会起作用。例如,
// instead of...
// this.updateState(values).then(...)
// do this...
this.setState(values, () => {
// call the onChange handler...
});
我可以这样做,但我想避免它,因为我要为这个组件添加更多功能,我不想陷入 pyramid of doom.
起初我认为这可能与 updateState
方法范围内的 this
或该方法中的回调函数之一的问题有关,但添加 console.log
声明在所有适当的地方都表明 this
指的是 TextInput
的一个实例。
添加 console.log
语句以在触发 onChange
处理程序之前转储它表明 this.props.onChange
实际上是一个 Sinon 间谍。
我查看了其他包,例如 sinon-as-promised,但我认为该包并没有真正解决我正在尝试做的事情 - 我只是想确保我的回调在promise then
子句。 sinon-as-promised
是一个用来完成整个承诺的包。
我可能忽略了一些直截了当的东西,但不管它是什么,我都没有看到。
您的同步测试似乎在执行对状态的异步调用之前完成。我不会评论您是否应该同时设置状态和调用更改方法以及何时。但我认为您当前的简单答案是通过传入 done
参数来使用异步测试。 (很明显,那时你甚至不需要间谍,但我把它留在里面只是为了表明不是间谍本身不起作用:
describe('TextInput component editing', () => {
it('calls the onChange handler', done => {
const fakeOnChange = stuff => {
expect(spyOnChange.calledOnce).to.equal(true);
expect(editedValue).to.equal(stuff);
expect(component.find('input').prop('value')).to.equal(editedValue);
done();
}
const spyOnChange = sinon.spy(fakeOnChange);
const value = '';
const editedValue = 'something';
const component = mount(<TextInput value={value} onChange={spyOnChange} />);
component.find('input').simulate('change', {
target: { value: editedValue }
});
});
});
我正在为基于 Promise 的函数编写测试。具体来说,它是一个 React 组件,我正在测试以确保正确调用 onChange 处理程序。
我的组件如下所示:
class TextInput extends React.Component {
constructor(props) {
super(props);
this.state = {
value: props.value || '',
};
this.onChange = this.onChange.bind(this);
}
updateState(values) {
return new Promise(
(resolve) => {
this.setState(values, () => { resolve(this.state); });
}
);
}
onChange(event) {
this.updateState({ value: event.target.value })
// then fire the onChange handler (if necessary)
//
.then((state) => {
if (this.props.onChange) {
// console.log(this.props.onChange) shows that this IS a
// Sinon spy function
this.props.onChange(state.value);
}
})
.catch((err) => { console.log('-----------', err); });
}
render() {
// render the component (omitted to keep this short)
}
}
我的测试是这样的:
import React from 'react';
import { mount } from 'enzyme';
import chai from 'chai';
import sinon from 'sinon';
import TextInput from '../../../../client/modules/components/TextInput';
const expect = chai.expect;
describe('TextInput component editing', () => {
it('calls the onChange handler', () => {
const onchange = sinon.spy();
const value = '';
const editedValue = 'something';
const component = mount(<TextInput value={value} onChange={onchange} />);
// change the value
//
component.find('input').simulate('change', {
target: { value: editedValue }
});
expect(component.find('input').prop('value')).to.equal(editedValue);
expect(onchange.calledOnce).to.equal(true);
expect(onchange.calledWith(editedValue)).to.equal(true);
});
});
最后两次 expect
调用测试失败。
如果我将 sinon spy 替换为普通的旧函数,则会调用该函数。例如,
// instead of this...
// const onchange = sinon.spy();
// do this...
const onchange = (value) => { console.log(`VALUE = ${value}`); };
如果我直接使用 setState
方法的回调,它会起作用。例如,
// instead of...
// this.updateState(values).then(...)
// do this...
this.setState(values, () => {
// call the onChange handler...
});
我可以这样做,但我想避免它,因为我要为这个组件添加更多功能,我不想陷入 pyramid of doom.
起初我认为这可能与 updateState
方法范围内的 this
或该方法中的回调函数之一的问题有关,但添加 console.log
声明在所有适当的地方都表明 this
指的是 TextInput
的一个实例。
添加 console.log
语句以在触发 onChange
处理程序之前转储它表明 this.props.onChange
实际上是一个 Sinon 间谍。
我查看了其他包,例如 sinon-as-promised,但我认为该包并没有真正解决我正在尝试做的事情 - 我只是想确保我的回调在promise then
子句。 sinon-as-promised
是一个用来完成整个承诺的包。
我可能忽略了一些直截了当的东西,但不管它是什么,我都没有看到。
您的同步测试似乎在执行对状态的异步调用之前完成。我不会评论您是否应该同时设置状态和调用更改方法以及何时。但我认为您当前的简单答案是通过传入 done
参数来使用异步测试。 (很明显,那时你甚至不需要间谍,但我把它留在里面只是为了表明不是间谍本身不起作用:
describe('TextInput component editing', () => {
it('calls the onChange handler', done => {
const fakeOnChange = stuff => {
expect(spyOnChange.calledOnce).to.equal(true);
expect(editedValue).to.equal(stuff);
expect(component.find('input').prop('value')).to.equal(editedValue);
done();
}
const spyOnChange = sinon.spy(fakeOnChange);
const value = '';
const editedValue = 'something';
const component = mount(<TextInput value={value} onChange={spyOnChange} />);
component.find('input').simulate('change', {
target: { value: editedValue }
});
});
});