public class StreamingTemplateEngine extends TemplateEngine
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 = '''\ Dear <% 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)This 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:
Dear Grace Hopper, We are pleased to inform you that your paper entitled 'Groovy for COBOL programmers' was accepted. The conference committee.
TemplateServlet
by placing the following in your
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: We <% 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 moreand sanitize the exceptions to make things readable.
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(java.lang.ClassLoader parentLoader)
Create a streaming template engine instance using a custom class loader
|
Modifier and Type | Method and Description |
---|---|
Template |
createTemplate(java.io.Reader reader)
Creates a template instance using the template source from the provided Reader.
|
createTemplate, createTemplate, createTemplate
public StreamingTemplateEngine()
public StreamingTemplateEngine(java.lang.ClassLoader parentLoader)
The custom loader is used when parsing the template code
parentLoader
- The class loader to use when parsing the template code.public Template createTemplate(java.io.Reader reader) throws CompilationFailedException, java.lang.ClassNotFoundException, java.io.IOException
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 detailcurried
in
while generating the template. { 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; }which 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.
createTemplate
in class TemplateEngine
CompilationFailedException
java.lang.ClassNotFoundException
java.io.IOException
TemplateEngine.createTemplate(java.io.Reader)