【发布时间】:2020-07-16 03:21:02
【问题描述】:
我有一个基于它读取的文件具有可变功能的函数,该函数通过它保存在内存中的Map 进行控制:
file1.ts
function f1(x: number): number {
// do far-reaching things
return 1;
}
function f2(x: number): number {
// do different far-reaching things
return 2;
}
function f3(x: number): number {
// do still-different far-reaching things
return 3;
}
const myMap: Map<string, (number) => number> = new Map<string, () => void>([
['key1', f1],
['key2', f2],
['key3', f3],
]
export function doThing(filename: string): number {
// open file, make some database calls, and figure out the name of a key
// ...
let fileToExecute = myMap.get(key);
return fileToExecute(someValueDerivedFromFile);
}
随着代码的发展和用例的不断发展,可能需要调用任意数量的函数,具体取决于不断扩展的输入集。 doThing() 很复杂,它的信息来自许多不同的来源,包括给定文件的内容和数据库,这有助于它选择要执行的文件。从客户的角度来看,doThing() 是它唯一关心的功能。因此,它是该文件唯一exported。
我正在尝试测试doThing() 中的机制,以确定它应该使用什么key。我不想专门模拟f1、f2 和f3 - 我想展示更多选项,这些选项是我为doThing() 模拟的其他内容所指出的。但是,要检查它是否调用了 正确 假方法,我需要弄清楚它调用的是哪个假方法。我尝试的解决方案使用类型转换来尝试将私有 myMap 从文件中提取出来,然后监视其 get() 方法:
file1.spec.ts
import * as file1 from '../src/file1'
...
it("calls the correct fake method", () => {
// lots of other mocks
let spies = [
jasmine.createSpy('f1spy').and.returnValue(4),
jasmine.createSpy('f2spy').and.returnValue(5),
jasmine.createSpy('f3spy').and.returnValue(6),
...
]
let mockMap = spyOn((file1 as any).myMap, 'get').and.callFake((key) => { // this fails
var spy;
switch(key) {
case 'key1': spy = spies[0]; break;
case 'key2': spy = spies[1]; break;
case 'key3': spy = spies[2]; break;
...
}
return spy;
}
result = file1.doThing(...);
expect(spies[0]).not.toHaveBeenCalled();
expect(spies[1]).toHaveBeenCalledWith(7);
expect(spies[2]).not.toHaveBeenCalled();
});
但是,我在上面的注释行中遇到错误:Error: <spyOn> : could not find an object to spy upon for get()。经过进一步调查(即分步调试器),我发现我导入的file1 对象只有doThing(),并且没有任何其他私有变量。
我如何在这里成功地模拟键值转换 - 这意味着,在这种情况下,监视私有变量的属性,以便我可以在正确的位置找到我的间谍?如果可以的话,要么完全替换 myMap,要么替换 myMap.get()。
【问题讨论】:
-
我不确定这是否好用,但我相信如果你这样做
spyOn(((file1 as any).myMap as any), 'get'),它应该可以工作并消除private障碍。 -
@AliF50 不工作,很遗憾。有问题的
file1是一个完整的模块,而不是一个类,所以我认为对象上甚至没有非导出的属性供我通过类型转换获得。
标签: typescript unit-testing mocking jasmine private