Zend Framework 2:在控制器上做多少单元测试


Zend Framework 2: How much unit testing to do on controllers?

我刚刚开始研究单元测试 Zend 2 应用程序,所以如果这是一个愚蠢的问题,请原谅我。

假设我有一个控制器,它用数据数组更新模型,并返回一些 JSON:

public function testAction($data){
  $model = new 'my'namespace'model();
  $model->updateFromArray($data);
  return new JsonModel(array(
        'success' => true,
  ));
}

在这种情况下,几乎所有的实际工作都是在模型内部的updateFromArray()方法中完成的。控制器所做的只是调用此方法。在为这个控制器编写单元测试时,我基本上有几个测试来测试"testAction"可以通过特定的URL访问,并且该操作返回JsonModel的实例。

现在我知道我还需要对模型的"updateFromArray"方法进行大量单元测试,因为它包含各种验证规则等。

我的问题是,我是否将这些测试应用于控制器(即编写一个到达"testAction"的测试并向其发送许多不同的数据数组),还是通过直接向 updateFromArray 方法提交许多不同的数组来直接测试模型?还是两者兼而有之?

我希望这是有道理的!

更新

澄清一下,我应该像这样测试吗:

//send lots of different requests to the controller and see if it works
class myControllerTest extends 'PHPUnit_Framework_TestCase{
  //some code...//  
  public function testWhatHappensWhenSubmitX(){
    $this->request->getPost()->set('somevariable','x');
    $result   = $this->controller->dispatch($this->request);
    $response = $this->controller->getResponse();
    $this->assertSomething();   
  }
  public function testWhatHappensWhenSubmitY(){
    $this->request->getPost()->set('somevariable','y');
    $result   = $this->controller->dispatch($this->request);
    $response = $this->controller->getResponse();     
    $this->assertSomething();     
  }
  public function testWhatHappensWhenSubmitZ(){
    $this->request->getPost()->set('somevariable','z');
    $result   = $this->controller->dispatch($this->request);
    $response = $this->controller->getResponse(); 
    $this->assertSomething();         
  }
}

或者这个:

//have lots of tests against the model
class myModelTest extends 'PHPUnit_Framework_TestCase{
  public function setUp(){
    $this->model = new 'my'namespace'model()
  }
  public function testWhatHappensWhenSubmitX(){
    $this->$model->doSomething('x');
    $this->assertSomething();   
  }
  public function testWhatHappensWhenSubmitY(){
    $this->$model->doSomething('y');
    $this->assertSomething();     
  }
  public function testWhatHappensWhenSubmitZ(){
    $this->$model->doSomething('z');
    $this->assertSomething();         
  }
}

或者,我应该两者兼而有之吗?

我们两者都做,但程度不同。

您的第一种风格,我们只用于少数几个系统关键功能,作为一种"冒烟测试",以确保所有内容一起运行。重要的是要注意,这些不被视为真正的单元测试,而是更多的"功能"测试,因为结果包括控制器与其使用/交互的任何其他类的交互。

第二种风格是我们测试控制器时最常见的风格。

你应该看看模拟框架,"Phake"http://phake.digitalsandwich.com/docs/html/

这将帮助您隔离控制器的功能并正确测试它。

控制器测试应仅测试控制器本身,而不测试其他任何内容。模型中的方法将由仅测试该模型的另一个类进行测试。

您还应该阅读有关模拟对象的信息,它将很有用。