Grails:如何使用注入的服务对命令对象进行单元测试

时间:2009-11-09 21:27:07

标签: unit-testing grails junit service command-objects

我正在尝试测试控制器 它有一个带有数据绑定的Command对象。

命令对象注入了一个服务。

但是当我尝试测试命令对象注入的服务方法时 永远不会被发现,因为它永远不会被“注入”

有没有办法在命令对象中模拟服务?

测试方法

void testLoginPasswordInvalid() {
    mockRequest.method = 'POST'
    mockDomain(User, [new User(login:"freddy", password:"realpassword")])
    mockLogging(UserService) // userService mocked
    MockUtils.prepareForConstraintsTests(LoginCommand)

    def userService = new UserService()
    def user = userService.getUser("freddy")//Gets called and returns the mockDomain
    assert userService.getUser("freddy")//Passes

    def cmd = new LoginCommand(login:"freddy", password:"letmein")
    cmd.validate() // Fails (userService is nevr injected)
    controller.login(cmd)
    assertTrue cmd.hasErrors()
    assertEquals "user.password.invalid", cmd.errors.password
    assertEquals "/store/index", renderArgs.view
}

找不到userService的getUser()方法

Cannot invoke method getUser() on null object
java.lang.NullPointerException: Cannot invoke method getUser() on null object

代码

调用控制器的登录方法

def login = { LoginCommand cmd ->
  if(request.method == 'POST') {
     if(!cmd.hasErrors()){
       session.user = cmd.getUser()
       redirect(controller:'store')
     }
     else{
       render(view:'/store/index', model:[loginCmd:cmd])
     }
  }else{

    render(view:'/store/index')
  }
}

Command Object注入了一个“userService”。

验证器调用此userService来查找用户

 class LoginCommand {

    def userService

    String login
    String password

    static constraints = {
      login blank:false, validator:{ val, cmd ->
          if(!cmd.userService.getUser()){
             return "user.not.found"
          }
      }
 }

userService.getUser()看起来像这样。

class UserService {

    boolean transactional = true

    User getUser(String login) {
        return User.findByLogin(login)

    }
}

2 个答案:

答案 0 :(得分:11)

使用Spring autowire-by-name完成服务注入。 (为autowire grep Grails源代码树找到一个很好的代码片段,你可以用它来为你的集成测试中的控制器自动装配。)这只在集成测试中起作用,其中有一个Spring应用程序上下文有可以注射的豆子。

在单元测试中,你必须自己做,因为你的东西周围没有Spring-land。这可能是一种痛苦,但会带来一些好处:

1)注入模拟版本的服务很容易 - 例如,使用Expando - 以便更密切地指定控制器协作服务的行为,并允许您仅测试控制器逻辑比控制器和服务在一起。 (你当然可以在单元测试中做后者,但你可以选择如何连接它。)

2)它强迫你明确控制器的依赖关系 - 如果你依赖它,你的测试就会显示它。这使它们成为控制器行为的更好规范。

3)您只能模拟控制器所依赖的外部协作者。这有助于您的测试不那么脆弱 - 当事情发生变化时,不太可能需要改变。

简短回答:您的测试方法需要cmd.userService = userService行。

答案 1 :(得分:8)

约翰所说的是标记。一个例子可能是:

def mockUsers = [new User(login:"freddy", password:"realpassword")]
mockDomain(User, mockUsers)

def userService = [getUser:{String login -> mockUsers[0]}] as UserService

def cmd = new LoginCommand (/*arguments*/)
cmd.userService = userService

您可以在http://groovy.codehaus.org/Groovy+Mocks

查找模拟对象的其他方法