Processes template source files substituting variables and expressions into placeholders in a template source text to produce the desired output using a closure based approach. This engine has equivalent functionality to the SimpleTemplateEngine but creates the template using writable closures making it more scalable for large templates.
Specifically this template engine can handle strings larger than 64k which still causes problems for the other groovy template engines.
The template engine uses JSP style <% %> script and <%= %>
expression syntax or GString style expressions. The variable
'out
' is bound to the writer that the template is being written
to.
def binding = [ firstname : "Grace", lastname : "Hopper", accepted : true, title : 'Groovy for COBOL programmers' ] def text = '''\ DearThis example uses a mix of the JSP style and GString style placeholders but you can typically use just one style if you wish. Running this example will produce this output:<%
out.print firstname%>
${lastname}, We<%
if (accepted) out.print 'are pleased' else out.print 'regret'%>
\ to inform you that your paper entitled '$title' was ${ accepted ? 'accepted' : 'rejected' }. The conference committee. ''' def template = new groovy.text.StreamingTemplateEngine().createTemplate(text) print template.make(binding)
Dear Grace Hopper, We are pleased to inform you that your paper entitled 'Groovy for COBOL programmers' was accepted. The conference committee.
web.xml
file (plus a corresponding servlet-mapping element):
<servlet> <servlet-name>StreamingTemplate</servlet-name> <servlet-class>groovy.servlet.TemplateServlet</servlet-class> <init-param> <param-name>template.engine</param-name> <param-value>groovy.text.StreamingTemplateEngine</param-value> </init-param> </servlet>In this case, your template source file should be HTML with the appropriate embedded placeholders.
The template engine makes an effort to throw descriptive exceptions with context lines, ie:
groovy.text.TemplateExecutionException: Template parse error at line 4: 3: Weand sanitize the exceptions to make things readable.<%
if (accepted) out.print 'are pleased' else out.print 'regret'%>
to inform you that your paper entitled-->
4: '$txitle' was ${ accepted ? 'accepted' : 'rejected' }. 5: at test.run(test.groovy:18) Caused by: groovy.lang.MissingPropertyException: No such property: txitle for class: groovy.tmp.templates.StreamingTemplateScript1 ... 1 more
When the exceptions are not enough, it might sometimes be useful to view the actual script source generated by the template engine. This would conceptually be equivalent to viewing the .java file generated for a jsp page. The source is not currently very readable and until we get a built in groovy code pretty printer, we will probably continue to opt for compactness rather than readability.
With that being said, viewing the source might still have some value. For this reason the script source is accessible via the template.scriptSource property, i.e.:
println template.scriptSourceIn the above example.
Constructor and description |
---|
StreamingTemplateEngine
() Create a streaming template engine instance using the default class loader |
StreamingTemplateEngine
(ClassLoader parentLoader) Create a streaming template engine instance using a custom class loader |
Type Params | Return Type | Name and description |
---|---|---|
|
Template |
createTemplate(Reader reader) Creates a template instance using the template source from the provided Reader. |
Methods inherited from class | Name |
---|---|
class TemplateEngine |
createTemplate, createTemplate, createTemplate, createTemplate |
Create a streaming template engine instance using the default class loader
Create a streaming template engine instance using a custom class loader
The custom loader is used when parsing the template code
parentLoader
- The class loader to use when parsing the template code. Creates a template instance using the template source from the provided Reader.
The template can be applied repeatedly on different bindings to produce custom output.
Technical detail
Under the hood the returned template is represented as a four argument
closure where the three first arguments are curried in
while generating the template.
In essence we start with a closure on the form:
{ parentClass, stringSectionList, binding, out ->
//code generated by parsing the template data
}
We then curry in the parentClass and stringSectionList arguments so that the StreamingTemplate
instance returned from 'createTemplate' internally contains a template closure on the form:
{ binding, out ->
//code generated by parsing the template data
}
Calling template.make(binding)
, curries in the 'binding' argument:
public Writable make(final Map map) { final Closure template = this.template.curry(new Object[]{map}); return (Writable) template; }This only leaves the 'out' argument unbound. The only method on the writable interface is writeTo(Writer out) so groovy rules about casting a closure to a one-method-interface apply and the above works. I.e. we return the now one argument closure as the Writable which can be serialized to System.out, a file, etc according to the Writable interface contract.
Copyright © 2003-2020 The Apache Software Foundation. All rights reserved.