Grails Goodness: Unit Testing Render Templates from Controller
In a previous blog post we learned how we can unit test a template or view independently. But what if we want to unit test a controller that uses the render()
method and a template with the template
key instead of a view? Normally the view and model are stored in the modelAndView
property of the response. We can even use shortcuts in our test code like view
and model
to check the result. But a render()
method invocation with a template
key will simply execute the template (also in test code) and the result is put in the response. With the text
property of the response we can check the result.
In the following sample controller we use the header
template and pass a username
model property to render output.
%{-- File: /grails-app/views/sample/\_header.gsp --}%
Hi, ${username}
===============
Welcome
=======
package com.mrhaki.grails.web
class SampleController {
def index() {
render template: 'header', model: \[username: params.username\]
}
}
With this Spock specification we test the index()
action:
package com.mrhaki.grails.web
import grails.test.mixin.TestFor
import spock.lang.Specification
@TestFor(SampleController)
class SampleControllerSpec extends Specification {
def "index action renders template with given username"() {
given:
params.username = username
when:
controller.index()
then:
response.text.trim() == expectedOutput
where:
username || expectedOutput
'mrhaki' || '
Hi, mrhaki
==========
'
null || '
Welcome
=======
'
}
}
Suppose we don't want to test the output of the actual template, but we only want to check in our test code that the correct template name is used and the model is correct. We can use the groovyPages
or views
properties in our test code to assign mock implementation for templates. The groovyPages
or views
are added by the ControllerUnitTestMixin
class, which is done automatically if we use the @TestFor()
annotation. The properties are maps where the keys are template locations and the values are strings with mock implementations for the template. For example the template location for our header
template is /sample/_header.gsp
. We can assign a mock String implementation with the following statement: views['/sample/_header.gsp'] = 'mock implementation'
We can rewrite the Spock specification and now use mock implementations for the header
template. We can even use the model in our mock implementation, so we can check if our model is send correctly to the template.
package com.mrhaki.grails.web
import grails.test.mixin.TestFor
import spock.lang.Specification
@TestFor(SampleController)
class SampleControllerSpec extends Specification {
def "index action renders mock template with given username"() {
given:
// Mock implementation with escaped $ (\\$), because otherwise
// the String is interpreted by Groovy as GString.
groovyPages\['/sample/\_header.gsp'\] = "username=\\${username ?: 'empty'}"
// Or we can use views property:
//views\['/sample/\_header.gsp'\] = "username=\\${username ?: 'empty'}"
and:
params.username = username
when:
controller.index()
then:
response.text.trim() == expectedOutput
where:
username || expectedOutput
'mrhaki' || 'username=mrhaki'
null || 'username=empty'
}
}
Code written with Grails 2.2.4