10.4 PHPUnit Doubles / Mocks / Stubs

我们写单体测试的目的是要测试个别组件,例如一个服务。通常,服务需要系统其他部分的功能。这种情况下,需要制造依赖的一个模拟(Mock)版本,以便于我们把焦点放在测试服务本身。

创建一个模拟

当你创建一个模拟时,其实你正在创建那个类的一个副本,只是它的所有方法被替换返回 NULL。

例如,如果我们需要一个查询对象,可以使用 getMockBuilder() 和  getMock() 方法取得一个。

你能看到我们在本例中传递了类名,之后调用了 getMock()。

$query = $this->getMockBuilder('\Drupal\Core\Entity\Query\QueryInterface')
                       ->getMock();

 

禁用原始构造器

这种模式适用于大多数类。一些类有一个 __construct() 函数,需要被移除。我们知道的最简点方式是用基本模式运行你的测试,如果收到一个错误,就添加 disableOriginalConstructor() 方法移除它。

$query_factory = $this->getMockBuilder('\Drupal\Core\Entity\Query\QueryFactory')
                                    ->disableOriginalConstructor()
                                    ->getMock();

 

抽象类

某些情况下,你将使用一个抽象类。再试下常规模式,如果出错,可以使用下面的方法:

$filter_plugin = $this->getMockBuilder('\Drupal\filter\Plugin\FilterBase')
                                ->getMockForAbstractClass();

 

模拟方法

当你需要从模拟返回测试值时,你可以为任意类方法创建置换。

PHPUnit 里,有很多可选的方式写东西。第一种实现是第二种的速写。两种写法都是告诉系统: 如果调用查询对象的 execute 方法,会返回数组 [1 => 1] 。大多数情况下,把 $this->any() 作为条件传递足够了。对于较复杂的模拟,你可以发送另一些值。

$query = $this->getMockBuilder('\Drupal\Core\Entity\Query\QueryInterface')
                       ->getMock();

$query->expects($this->any())->method('execute')->willReturn([1 => 1]);

// or
$query->expects($this->any())->method('execute')->will($this->returnValue([1 => 1]));

 

返回映射(maps)的模拟方法

前一个例子,我们从模拟方法返回了一个值。我们也可以根据输入返回不同的值。

这个例子返回一个映射数组。数组中的第一项是传递给 get 方法的参数,最后一项是返回值。我们告诉模拟对象当 $fieldDefinition->get(‘field_name’) 被调用时返回字符串 test_field 。类似的,当 $fieldDefinition->get(‘field_type’) 被调用时,返回 text 。如果方法有两个输入,你返回的映射应该是这样:  ['argument_one', 'argument_two', 'return_value']

$fieldDefinition = $this->getMockBuilder('\Drupal\Core\Field\BaseFieldDefinition')
                                    ->getMock();

$fieldDefinition->expects($this->any())->method('get')
  ->willReturnMap([
    ['field_name', 'test_field'],
    ['field_type', 'text'],
    ['fieldStorage', $fieldStorage],
  ]);

 

使用回调的模拟方法

如果你需要做复杂处理,或者访问可能变化的类变量,你可以提供一个函数作为返回值。新版本 PHP 支持匿名函数,因此你可以把函数声明作为参数。

$this->field = $this->getMockBuilder('\Drupal\Core\Field\FieldItemList')
                              ->disableOriginalConstructor()
                              ->getMock();

$this->field->expects($this->any())->method('getValue')
  // We use a function here so that the value is calculated at runtime.
  ->will($this->returnCallback(
    function() {
      return $this->testValue;
    }
  ));

 

评论 (写第一个评论)