我遇到了我认为 Vim 保存文件和 Karma 重新运行我的 Jasmine 单元测试之间的竞争条件。下面是四个测试运行的序列,演示了症状(我截断了错误日志中的极长路径):
$ karma start karma.conf.js --auto-watch
[... snip a lot of coding and test running ...]
PhantomJS 1.6 (Linux) LOG: 'Running tests at 2013-08-14T08:19:57.252Z'
PhantomJS 1.6 (Linux): Executed 4 of 4 SUCCESS (0.307 secs / 0.013 secs)
PhantomJS 1.6 (Linux) LOG: 'Running tests at 2013-08-14T08:20:09.866Z'
PhantomJS 1.6 (Linux): Executed 4 of 4 SUCCESS (0.288 secs / 0.012 secs)
PhantomJS 1.6 (Linux) LOG: 'Running tests at 2013-08-14T08:20:14.366Z'
PhantomJS 1.6 (Linux) controller should have a breadcrumb after $routeChangeSuccess FAILED
Error: No module: myapp.question
at /home/rmunn/.../angular.js:1124
at ensure (/home/rmunn/.../angular.js:1065)
at module (/home/rmunn/.../angular.js:1296)
at /home/rmunn/.../angular.js:2806
TypeError: 'undefined' is not an object (evaluating 'rootScope.$broadcast')
at /home/rmunn/.../unit/controllersSpec.js:35
PhantomJS 1.6 (Linux): Executed 4 of 4 (1 FAILED) (0.312 secs / 0.014 secs)
PhantomJS 1.6 (Linux) LOG: 'Running tests at 2013-08-14T08:20:28.588Z'
PhantomJS 1.6 (Linux): Executed 4 of 4 SUCCESS (0.287 secs / 0.013 secs)
我为触发这四次运行所做的唯一更改是在文件中添加和删除空格question.js
;没有实质性的代码更改,但运行 #3 失败,而运行 1、2 和 4 成功。
我的 Vim 配置设置为保留备份文件,默认情况下,Vim 通过重命名原始文件并写入新文件来保留备份文件。(在某些情况下除外,此处不适用)。我认为正在发生的事情是我正在触发一个竞争条件:Vim 重命名question.js
(或我刚刚编辑的任何文件),Karma 检测到该更改并开始运行测试,并且因为我现在的测试很少,所以测试都运行在 Linux 进程调度程序将控制权交还给 Vim 之前。一旦 Vim 再次被调度,它就会写入 的新内容,但是到那时对于依赖于现有且非空question.js
的 Jasmine 测试来说已经太晚了。question.js
我可以通过配置 Vim 不保留备份文件来解决这个问题,但我真的不想冒这个风险。毕竟,我使用备份文件是有原因的。那么有没有办法告诉 Karma “当你检测到文件更改时,暂停 N 毫秒,然后运行单元测试”?虽然这将是一个蛮力解决方案,但它会解决这个特定的竞争条件。或者,如果有人有其他聪明的想法可以提供,我也很想听听。