gpt4 book ai didi

javascript - 单元测试:Q.js 和 Jasmine.js

转载 作者:行者123 更新时间:2023-11-29 22:11:40 25 4
gpt4 key购买 nike

我将我的库从 jQuery 移植到 Q.js,同时将我的单元测试从 QUnit 重写到 Jasmine.js,现在面临一些问题:

  • 因为所有 promise 对象都是异步解析的 (setTimeout( func, 0 )),所以我不得不用“runs”和“waitsFor”编写不太好的单元测试。
  • “jasmine.Clock.useMock”方法无效(两周前租用)

所以我的问题是如何测试使用 Q.js 和 Jasmine.js 的库?

更新 1:你可以找到我的单元测试 here .请参阅测试“‘工具包的“检索”方法允许加载单个记录’”该库用于 Dynamics CRM 2011 上下文中的 CRUD 操作。

更新 2:似乎 setTimeout 不是我唯一的“问题”。 Q.js 将使用“setImmediate”或“MessageChannel”或“setTimeout”来解决异步操作。

最佳答案

如果您正在进行单元测试,则不必担心时钟。

假设我们有这样的事情:

var obj = {
functionToTest: function () {
callServer().then(function () {
//success
}, function () {
//error
})
}
}

要测试它,我们需要修改时钟。避免它的一种选择是修改超时以同步解析函数。

var aux = window.setTimeout; //save to restore later
window.setTimeout = function(func){
func();
};

大多数时候使用假超时会产生相同的结果,但有时它可能会失败:

var obj = {
functionToTest: function () {
var foo = 'bar';
callServer().then(function () {
alert(foo);
}, function () {
//error
})
foo = 'buz';
}
}

当正常使用时,它会提醒 buz,但如果超时,它会提醒 bar。请谨慎使用此选项。

当您使用 Q promise 时,您指定了一个成功回调和一个错误回调,它们不需要在测试函数中。

var obj = {
functionToTest: function () {
callServer().then(this.success, this.error)
},

success: function () {
//success
}
error: function () {
//error
}
}

现在,使用假超时,您可以测试调用了成功函数和错误函数,然后测试成功函数和错误函数。

如果您需要回调中测试函数的一些数据,您可以使用闭包:

var obj = {
functionToTest: function () {
var data = 'foo';
callServer().then(this.success(data), this.error)
},

success: function (data) {
var that = this;
return function () {
//we have access to data and the object
}
}
error: function () {
//error
}
}

在进行单元测试时,您会检查指定输入的函数输出。如果被测试的函数正在调用一个返回 promise 的函数,这意味着被测试函数的输出是对该函数的调用,所以它应该被测试。稍后您应该测试回调。

在示例中,我将测试我是否使用预期数据调用 callServer,然后测试回调。

如果您需要使用“runs”和“waitsFor”(或其他类似选项),这意味着您不是在进行单元测试,而是在进行功能或集成测试。

单元测试确保项目的每个部分都正常工作,而集成测试确保所有部分都能正常工作。

如果单元测试太复杂,则意味着您应该修改正在编写的程序以使其可测试。

编辑:

对于你评论我会做的测试:

var aux;
beforeEach(function () {
aux = window.setTimeout;

window.setTimeout = function(func){
func();
};
});

afterEach(function () {
window.setTimeout = aux;
});


it('should return a single object', function () {
CrmRestKit.Retrieve(entitySchemaName, fakeid, columns).then(function (data) {
expect(data.d).not.toBeArray();
});
});

it('will return the fake-account', function () {
CrmRestKit.Retrieve(entitySchemaName, fakeid, columns).then(function (data) {
expect(data.d).toBe(fakeAccount);
});
});

编辑 2:

我提供的 setTimeout 解决方案是一种快速解决方案,但不是最纯粹的“单元测试”解决方案,而且很多时候它都能工作并保持测试简单,但这次不行。

通过您的测试,除了您的代码之外,您还在测试 Q 和 ajax 实现,这就是使测试变得复杂的原因。

快速解决方案改进了:

让我们从等式中删除 promise 实现,并考虑它会完成它的工作。

您目前期望从 promise 中获得您从 ajax 调用中收到的数据。为什么不测试 promise 的“解析”函数是否使用预期数据调用?

单元测试解决方案:

不要测试第 3 方实现并认为他们会完成他们的工作。

我会做 3 个测试:第一个检查 Retrieve 是否返回 promise ,第二个检查它是否使用服务器需要的数据调用 ajax,最后一个检查回调函数使用预期数据调用解析(请参阅我对回调测试的第一个答案)。

关于javascript - 单元测试:Q.js 和 Jasmine.js,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17830033/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com