5

使用 Grails 2.1.0

似乎从控制器执行此操作:

render(view: "someView", model: [modelEntry: "hello"]) 

允许我在该控制器的单元测试中执行此操作:

controller.method() 
assert model.modelEntry == "hello" 

但是,如果我更改控制器来执行此操作:

render(template: "someTemplate", model: [modelEntry: "hello"]) 

现在测试中的模型实例是一个空数组。我已经对此进行了大量搜索,大多数解决方案似乎都是针对 Grails 1 的,通常涉及modelAndView对象(在我的测试中不存在)或renderArgs(同上)。

我发现的唯一解决方案是手动覆盖测试中的视图,如下所示:

views['_someTemplate.gsp'] = '${modelEntry}'

然后对字符串进行断言。但我不喜欢这个解决方案,因为它:

  1. 要求测试知道模板的文件名
  2. 使测试没有好的 toString() 方法的模型条目变得困难
  3. 使得对相关模型条目做出多个断言变得困难。

当控制器呈现模板时,有什么方法可以更直接地从测试用例中获取模型中的条目?

4

1 回答 1

10

在 render 方法 () 的代码中挖掘一点,org.codehaus.groovy.grails.web.metaclass.RenderDynamicMethod我可以看到modelAndView只有在渲染view.

渲染模板确实会返回一个 null modelAndView。

要在这种情况下检查模型,我认为您可以使用 Groovy metaClass。思路是截取原方法,存储值,然后调用他。

基于这个问题,我构建了这个(未经测试,可能需要调整):

@TestFor(MyController)
class MyControllerTests

  def templateModel

  @Test
  void inspectTemplateModel() {
    def originalMethod = MyController.metaClass.getMetaMethod('render', [Map] as Class[])
    controller.metaClass.render = { Map args ->
      templateModel = args.model
      originalMethod.invoke(delegate, args)
    }

    controller.method()
    assert templateModel.modelEntry == 'foo'

}
于 2013-02-28T20:15:51.730 回答