测试 AngularJS 的异步服务

最近,在做项目时掉进了 AngularJS 异步调用 $q 测试的坑中,直接躺枪了。折腾了许久日子,终于想通了其中的道道,但并不确定是最佳的解决方案,最后还是决定总结成文以求能与其它的园友共同分享以求找到更好的解决方案。

首先,我的测试环境是 [Karma|http://karma-runner.github.io/0.12/index.html] + [Jasmine|http://jasmine.github.io/] ,这属于 AngularJS的其中一种配置,也是AngularJS官方所推荐的框架,Jasmine 用起来也确实很不错。

很多的spec都没有什么大问题,只是当我为其中的几重要的异步处理服务编写测试时就出事了,代码在实际运行环境中是能正常运行的,但在测试中却不能通过,这肯定是测试没写好。在网上google了多天,也对各种方案进行尝试一直也没有找到解决方法,然而这种问题不会是特例,而是经常会遇到的,那就是在Angular服务中返回的 promise 很难进行测试。

从代码入手会更容易了解问题的始末:

噩梦的开始

jasmine 的异步测试模式是实现一种简单的超时机制,通过等待 done() 方法对计时器重置,当在超时限制内(默认5s) done() 没有被调用则会引发测试失败的异常。在 1.3 之前是采用 runswaitsFor 方法进行处理,在2.0后这两个方法被简化去除掉了,只能用 done,这里就以我们最经常会用到的 FileAPI 中的 FileReader 来做实例,FileReader 对文件对象(Blob)的读取是一个异步方法,那么将这个实现逻辑直接写在 jasmine 中应该是这样的:

  1. describe '异步调用测试', ->
  2. beforeEach module 'tdd'
  3. it 'Blob内的数据应该被读取为文本', (done)->
  4. expected_text = chance.sentence() # 用 chance 产生随机的字符串
  5. blob = new Blob([expected_text])
  6. reader = new FileReader()
  7. reader.onloadend = (e)->
  8. expect(e.target.result).toBe expected_text
  9. done()
  10. reader.readAsText blob

测试结果是 pass , 这只是为了试用一下 jasmine 中 done 的效果。当然在项目中这样做是完全没有意义的,这只是一个引子,我会分三步来完整这个测试。

接下来是将这个实现逻辑封装成为 AngularJS的 service。由于是异步处理所以这个 service 应该是返回一个 promise 对象。 为了更具体地说明这个问题,这里只建立一个空白的 fileReader 服务,此服务只为了测试 then 的触发时机:

  1. 'use strict'
  2. fileReader=($q)->
  3. (_blob)->
  4. deferred=$q.defer()
  5. deferred.resolve('马上返回')
  6. deferred.promise
  7. angular.module('tdd').service 'fileReader', fileReader

那么前文的测试就应该修改为:

  1. describe '异步调用测试' ,->
  2. beforeEach module 'tdd'
  3. _fileReader={}
  4. it '应该通过fileReader 服务从Blob 对象中读出文本', (done)->
  5. expected_text = '马上返回'
  6. blob = new Blob([expected_text])
  7. fileReader(blob).then (actual_text)->
  8. expect(expected_text).toEqual actual_text
  9. done()

问题开始来了,这个测试运行的结果是 Fail! 并且得到以下的提示:

  1. Error: Timeout - Async callback was not invoked within timeout specified by jasmine.DEFAULT_TIMEOUT_INTERVAL.

超时!也就是说 由于then 并没有被调用,所以超时返回方法 done()没有被执行而直接出现这个测试错误。然而讽刺的是,fileReader 这个服务在浏览器内是可以直接运行而不会产生任何的错误的。

问题出在哪里 ?找了许久,最终发现,由于在 jasmine 中的环境是被 mock 出来的,是由 ngMock 对 angular 的对象和brower对象内的服务进行了重新的模拟,这个会与实际的运行有些许的差异,由其要使 then 方法被正确调用那需要在返回then之后调用 $rootScope.$apply() (这个内容可以直接参考:https://docs.angularjs.org/api/ng/service/\(q) 也就是说,我们并不需要直接使用 jasmine 提供的“阻塞”模拟,而是直接用 `\)rootScope.$apply()` 让异步方法直接返回。

  1. describe ' 异步调用测试' , ->
  2. describe module 'tdd'
  3. it '应该通过fileReader 服务从Blob 对象中读出文本',$inject (fileReader,$rootScope)->
  4. expected_text='马上返回'
  5. blob=new Blob([expected_text])
  6. actual_text=''
  7. fileReader(blob).then (data)->
  8. actual=data
  9. $rootScope.$apply()
  10. expect(expected_text) .toEqual actual_text

这一次测试 pass 了。既然测试写好了,那么我们就回到最初那里,将实现逻辑真正地加入到 fileReader 服务中:

  1. 'use strict'
  2. fileReader=($q)->
  3. (_blob)->
  4. deferred=$q.defer()
  5. reader = new FileReader()
  6. reader.onloadend=(e)->
  7. # 注意:事件触发实质上等同于异步回调
  8. deferred.resolve e.target.result
  9. deferred.promise
  10. angular.module('tdd').service 'fileReader', fileReader

同样地,运行刚才写好的异步测试程序。 当然在运行前要修改一下 expected_textexpected_text = chance.sentence()。然而,运行结果是让人失望的:

  1. Chrome 39.0.2171 (Mac OS X 10.10.2) 异步调用测试 应该通过fileReader 服务从Blob 对象中读出文本 FAILED
  2. Expected 'Gipik ejfim renma fanibi nub otu nihwojtat il bepu koufo dibe ohepusaw monumba.' to equal ''.
  3. Error: Expected 'Gipik ejfim renma fanibi nub otu nihwojtat il bepu koufo dibe ohepusaw monumba.' to equal ''.
  4. at Object.<anonymous> (/Users/Ray/code/tdd/test/spec/file_reader.service.spec.js:40:36 <- file_reader.service.spec.coffee:48:28)
  5. at Object.invoke (/Users/Ray/code/tdd/bower_components/angular/angular.js:4182:17)
  6. at Object.workFn (/Users/Ray/code/tdd/bower_components/angular-mocks/angular-mocks.js:2350:20)

很明显,then 方法又无法触发了,deferred.resolve 并没有如我们预期那般在文件读取时调用,而且 \(rootScope.\)apply() 也貌似失效了。在此时真正的问题才开始显现:

resolve 是在其它的(非Angular Mock)异步调用中返回时 $rootScope.$apply() 是无法正确触发 then 的。

我的实际项目比这个要更为复杂,因为我的实际的服务是操控 indexedDB的,众所周知 indexedDB 里一切都是异步的,所以他们在测试中无一通过!

就是为了这个问题我折腾了好久,最好还是将视线落在 ngMock 上。

曙光

[ngMock|https://docs.angularjs.org/api/ngMock] 这个模块上只提供了最简单的三种异步服务 $httpBackend$tiimeout$interval ,正是因为他们的存在,在我们的测试中可以正常调用 $http$resource 等的常规异步服务。 然而, FileAPI, IndexedDB等这些 HTML5内的高等服务并没有提供 mock。当时我的测试初衷并不想mock而是期往能实际地调用,而然这种想法貌似不太容易实现,加之,自我看了 ["Mocking Dependencies AngularJS Tests"|http://www.sitepoint.com/mocking-dependencies-angularjs-tests/] 一文后,更加确定了我的想法。

我的结论是,如果在自定义的 Angular服务中返回的 promise 是在 Angular的 scope内调用 resolve 那么我们直接使用前面第二种测试方式就可以了,但如果 service 是包装了其它的依赖服务,如FileReader 、IndexedDB、WebSQL或其它的以异步方式为主的服务那么就只能通过 mock 来解决测试的问题,要不就不使用 $q而采用 callback 方式将回调方法直接传递给第三方依赖服务(我现在的IndexedDB服务就是这种方式)。

以本文中所提及的 FileReader 为例的话,要测试通过那可以自己写一个 mockFileReader,通过 jasmine 的 spyOn 方法截取方法调用:

  1. beforeEach(function () {
  2. // Mock FileReader
  3. MockFileReader = {
  4. readAsDataURL: function (file) {
  5. if (file === 'file') {
  6. this.result = 'readedFile';
  7. this.onload();
  8. } else if (file === 'progress') {
  9. this.onprogress({total: 70, loaded: 30});
  10. } else {
  11. this.result = 'fileError';
  12. this.onerror();
  13. }
  14. },
  15. readAsText: function (file, encoding) {
  16. if (file === 'file') {
  17. this.result = 'readedFile';
  18. this.onload();
  19. } else if (file === 'progress') {
  20. this.onprogress({total: 70, loaded: 30});
  21. } else {
  22. this.result = 'fileError';
  23. this.onerror();
  24. }
  25. }
  26. };
  27. spyOn(MockFileReader, 'readAsDataURL').and.callThrough();
  28. spyOn(MockFileReader, 'readAsText').and.callThrough();
  29. // 将 MockFileReader 挂到 window 中
  30. $window = {
  31. FileReader: jasmine.createSpy('FileReader').and.returnValue(MockFileReader)
  32. };

笨一点的做法就是对有使用的第三方依赖都编写一个 Mock 加以取代,更快捷的方法就是看看谁已经将这个“轮子”发明了而不用我们重新造一次。

参考:

AngularJS 的异步服务测试与Mocking的更多相关文章

  1. grpc使用记录(三)简单异步服务实例

    目录 grpc使用记录(三)简单异步服务实例 1.编写proto文件,定义服务 2.编译proto文件,生成代码 3.编写服务端代码 async_service.cpp async_service2. ...

  2. AngularJs之六(服务)

    服务:AngularJS 中,服务是一个函数或对象,可在你的 AngularJS 应用中使用.AngularJS 内建了30 多个服务. 最常用的服务:$location  服务,  $http 服务 ...

  3. 让AngularJS的$http 服务像jQuery.ajax()一样工作

    让AngularJS的$http 服务像jQuery.ajax()一样工作 $http的post . 请求默认的content-Type=application/json . 提交的是json对象的字 ...

  4. C# Socket基础(一)之启动异步服务监听

    本文主要是以代码为主..NET技术交流群 199281001 .欢迎加入. //通知一个或多个正在等待的线程已发生事件. ManualResetEvent manager = new ManualRe ...

  5. vs自带服务测试工具

    在vs安装目录有一个vs自带的服务测试工具,地址为: "C:\Program Files (x86)\Microsoft Visual Studio 11.0\Common7\IDE\Wcf ...

  6. Ⅳ.AngularJS的点点滴滴-- 服务

    服务(Angularjs很多方法都是服务组成的) 1.使用service方法创建的单例服务 <html> <script src="http://ajax.googleap ...

  7. 12个强大的Web服务测试工具

    在过去的几年中,web服务或API的普及和使用有所增加. web服务或API是程序或软件组件的集合,可以帮助应用程序进行交互或通过形成其他应用程序或服务器之间的连接执行一些进程/事务处理.基本上有两种 ...

  8. 怎么理解angularjs中的服务?

    AngularJS中的服务其实就是提供一种方式抽取共用类库 比如说一些工具类方法,我们传统的做法就是自己写个 utility 类,把相关的工具方法填充到utility里面去,最后把utility类放到 ...

  9. AngularJS之使用服务封装

    AngularJS之使用服务封装可复用代码   创建服务组件 在AngularJS中创建一个服务组件很简单,只需要定义一个具有$get方法的构造函数, 然后使用模块的provider方法进行登记: / ...

随机推荐

  1. Oracle EBS AP 创建贷项通知单并核销到相应发票

    --1.0 生成与发票一样的贷项通知单 created by jenrry 20170423 DECLARE L_CUSTOMER_TRX_ID NUMBER; L_INVOICE_NUMBER VA ...

  2. oracle中insert 多条数据方法

    oracle中的insert 和 mysql添加多条数据的 方式不太一样 用到的语法: insert all into 表名(需要添加的表字段)values(添加的字段数据一定要对应字段顺序) int ...

  3. 异常System.BadImageFormatException

    [问题描述] Server Error in '/' Application. Could not load file or assembly 'WebDemo' or one of its depe ...

  4. 解决VS调试Web应用无问题而部署在IIS上报500和403的问题

    [问题:报500]不能在此路径中使用此配置节.如果在父级别上锁定了该节,便会出现这种情况.锁定是默认设置的(overrideModeDefault="Deny" [解决方案] 运行 ...

  5. MySQL索引选择不正确并详细解析OPTIMIZER_TRACE格式

    一 表结构如下: CREATE TABLE t_audit_operate_log (  Fid bigint(16) AUTO_INCREMENT,  Fcreate_time int(10) un ...

  6. .Net 环境

    更多系统版本下载:https://www.microsoft.com/net/download VSCode :https://code.visualstudio.com/

  7. JList动态添加元素

    JList动态添加元素   http://www.cnblogs.com/tianguook/archive/2012/01/31/2333992.html https://zhuanlan.zhih ...

  8. (转)glew的安装

    http://blog.sina.com.cn/s/blog_858820890100vbys.html 下载链接: https://sourceforge.net/project/downloadi ...

  9. windows的一些好用命令-自己总结:

    在win+R运行框中:     cmd:进入命令行界面     msconfig:可以查看“系统配置”     msinfo32:查看系统信息     services.msc打开"服务&q ...

  10. 如何在SAE搭建属于自己的黑盒xss安全测试平台

    Author:雪碧 http://weibo.com/520613815 此篇文章技术含量不高,大牛不喜勿喷,Thx!写这篇文章主要是为了各位小伙伴在SAE搭建XSSING平台的时候少走点弯路(同志们 ...