【问题标题】:How does variable scope work within the Mocha test framework?变量范围在 Mocha 测试框架中是如何工作的?
【发布时间】:2015-06-09 00:08:20
【问题描述】:

我是 javascript、node.js、mocha 等所有东西的相对新手。

在我的代码中,我有一个 Unit 对象,它有一个将 disabled 属性设置为 true 的 disable() 和一个返回 disabled 属性的 isDisabled()。它还有一个方法nextTurnReset(),可以在下一回合开始时重置单位。我已经编写了一个测试套件来测试这种行为。我首先禁用该对象,然后尝试测试它是否被禁用。但是,我的第一个测试中的单元变量(在传递给 Mocha 的 it() 方法的匿名函数中)处于非禁用状态,正如我在节点的调试器中观察到的那样。

describe('#disable()', function() {
    var unit = tests.newUnit();
    unit.disable();
    debugger;
    it('disabled off turn?', function() {
        debugger;
        (unit.isDisabled()).should.be.exactly(true);
    });
    unit.nextTurnReset();
    it('disabled on next turn?', function() {
        (unit.isDisabled()).should.be.exactly(true);
    });
    unit.nextTurnReset();
    it('disabled on 2nd turn?', function() {
        (unit.isDisabled()).should.be.exactly(false);
    });
});

作为记录,前两个测试失败,最后一个成功,表明该单元从未被禁用。

使用节点调试器的repl:在第一个debugger; 语句之后,unit.disabled == true,但在第二个debugger; 语句unit.disabled == false 之后。我希望这两种情况下的价值都是真实的。

知道为什么会这样吗?另外,编写 Mocha 测试以获得预期结果的正确方法是什么?

非常感谢!

【问题讨论】:

    标签: node.js scope mocha.js anonymous-function


    【解决方案1】:

    Mocha 中的变量作用域与任何其他 JavaScript 代码完全相同。您的问题是您没有意识到 Mocha 将执行您的代码的顺序。这就是发生的事情:

    1. 您创建您的单元实例并在其上调用disable

    2. 您注册了您的第一个测试。这就是it 所做的:它为future 执行注册一个测试。测试现在不执行

    3. 您调用 unit.nextTurnReset(); 会重置对象的状态。

    4. 您注册了第二个测试。 它现在不执行

    5. 您再次重置对象。

    6. 您注册了最后一次测试。

    之后,Mocha 会接受您注册的测试并运行它们。 在您的测试运行时,您的对象处于重置状态,而不是禁用。

    在我看来,鉴于您描述的期望行为,您的代码应该是:

    describe('#disable()', function() {
        var unit = tests.newUnit();
    
        beforeEach(function () {
            unit.nextTurnReset();
        });
    
        it('disabled off turn?', function() {
            unit.disable();
            (unit.isDisabled()).should.be.exactly(true);
        });
    
        it('disabled on next turn?', function() {
            (unit.isDisabled()).should.be.exactly(false);
        });
    });
    

    传递给beforeEach 的代码在您注册的每个测试之前运行,因此它会在正确的时间重置对象。

    【讨论】:

      【解决方案2】:

      在示例之间传递状态不是一个很好的做法。如果您需要以随机顺序运行测试会发生什么?或者项目中的某个人决定移动示例?

      对我来说,拥有以下两个示例似乎足以正确测试 Unit#disable

      describe('#disable()', function() {
          it('gets disabled when called on an enabled', function() {
              var unit = tests.newUnit();
              unit.disable();
      
              (unit.isDisabled()).should.be.exactly(true);
          });
      
          it('gets enabled when called on a disabled', function() {
              var unit = tests.newUnit();
              unit.disable();
              unit.disable();
      
              (unit.isDisabled()).should.be.exactly(false);
          });
      });
      

      【讨论】:

      • OP 做错了,但在测试之间将夹具重置为已知状态是一种完全有效的策略。如果整个夹具的创建不繁琐,我更喜欢在每次测试中重新创建它,但有时最好在测试之间重置为已知状态,而不是重新创建昂贵的夹具。在beforeEach 挂钩中调用unit.nextTurnReset() 会将测试夹具重置为 测试之间的已知状态。由于它在beforeEach 挂钩中,因此在每次测试之前对unit.nextTurnReset() 的调用不依赖于哪些测试运行或不运行,或测试顺序。
      猜你喜欢
      • 2017-01-25
      • 1970-01-01
      • 2014-06-17
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多