如何使用Jest测试带嵌套函数的Redux动作的高阶函数

时间:2020-03-31 22:45:11

标签: reactjs testing redux jestjs higher-order-functions

我正在使用Jest测试Redux动作函数fn1。 fn1是包装fn2的高阶函数。我的测试只是确保执行fn1时调用了fn2。似乎不起作用。我正在考虑使用jest.spyOn,但这似乎没有任何意义。

myActions.js:

export const fn1 = obj => {
  return strInput => {
    fn2(strInput, obj);
  };
};

export const fn2 = (strInput, obj) => ({name:strInput, obj});

myAction.test.js:

import {fn1, fn2} from myAction.test.js

it("should call fn2", () => {
    fn1({test:"test"})("David")
    expect(fn2).toHaveBeenCalled();
  });

1 个答案:

答案 0 :(得分:1)

从某种意义上讲,我觉得您要尝试的是测试实现细节而不是功能的API,但是在单元测试中,您确实希望基于以下内容测试或声明正确的输出:指定的输入,即f(x) = y,测试输入x会产生输出y

假设您的fn2将具有自己的单元测试,因此可以假定它已在其他功能(例如fn1)中使用时进行了测试并正确。

这就是我测试fn1的方式:

it("should compute composed value", () => {
  expect(fn1({ test: "test" })("David")).toEqual({
    name: "David",
    obj: { test: "test" }
  });
});

Edit interesting-shtern-uqofl

我要说的是监视或断言函数调用的典型用例是回调。回调不是函数实现的一部分,而通常是外部副作用。

const fn3 = (value, callback) => {
  // bunch of code logic
  callback(value);
  // more code logic
  return true;
};

it("should callback function", () => {
  const cb = jest.fn();
  fn3(3, cb);
  expect(cb).toHaveBeenCalledWith(3);
});

不是答案的一部分,但内容丰富

在这一点上,我想指出一个命名错误:高阶函数与Currying

Higher Order Function是将一个函数作为输入并返回一个新函数的函数。示例包括.map.filter,功能组成

const plus3 = x => x + 3; // a first order function
const double = (fn, v) => fn(v) * 2; // a higher order function

const plus3AndDouble = x => double(plus3, x); // a (decorated) first order function

console.log(plus3AndDouble(0)); // (0 + 3) * 2 = 6
console.log(plus3AndDouble(1)); // (1 + 3) * 2 = 8

您实际上完成的是 一个名为currying的概念,其中您采用了一个具有多个输入的函数,并将其转换为每个均具有单个输入的函数序列。

const foo = (a, b, c) => a + b * c;
const bar = a => b => c => foo(a, b, c);

console.log(foo(1, 2, 3) === bar(1)(2)(3)); // true