gg.jte.TemplateEngine Maven / Gradle / Ivy
package gg.jte;
import gg.jte.html.HtmlPolicy;
import gg.jte.html.HtmlTemplateOutput;
import gg.jte.html.OwaspHtmlTemplateOutput;
import gg.jte.html.HtmlInterceptor;
import gg.jte.runtime.*;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
/**
* jte is a simple, yet powerful template engine for Java.
* All jte templates are compiled to Java class files, meaning jte adds essentially zero overhead to your application.
* jte is designed to introduce as few new keywords as possible and builds upon existing Java features,
* so that it is very easy to reason about what a template does.
*
* Read more at the official documentation
*/
public final class TemplateEngine {
private final TemplateLoader templateLoader;
private final TemplateMode templateMode;
private final ConcurrentHashMap templateCache;
private final TemplateConfig config;
private final ContentType contentType;
private final Path classDirectory;
private final ClassLoader parentClassLoader;
private HtmlInterceptor htmlInterceptor;
/**
* Creates a new template engine.
* All templates are compiled to Java class files on demand.
* A JDK is required.
* Every template has its own class loader.
* This is recommended when running templates on your developer machine.
*
* @param codeResolver to lookup jte templates
* @param contentType the content type of all templates this engine manages
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine create(CodeResolver codeResolver, ContentType contentType) {
return create(codeResolver, Paths.get("jte-classes"), contentType);
}
/**
* Creates a new template engine.
* All templates are compiled to Java class files on demand.
* A JDK is required.
* Every template has its own class loader.
* This is recommended when running templates on your developer machine.
*
* @param codeResolver to lookup jte templates
* @param classDirectory where template class files are compiled to
* @param contentType the content type of all templates this engine manages
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine create(CodeResolver codeResolver, Path classDirectory, ContentType contentType) {
return create(codeResolver, classDirectory, contentType, null);
}
/**
* Creates a new template engine.
* All templates are compiled to Java class files on demand.
* A JDK is required.
* Every template has its own class loader.
* This is recommended when running templates on your developer machine.
*
* @param codeResolver to lookup jte templates
* @param classDirectory where template class files are compiled to
* @param contentType the content type of all templates this engine manages
* @param parentClassLoader the parent classloader to use, or null to use the application class loader as parent
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine create(CodeResolver codeResolver, Path classDirectory, ContentType contentType, ClassLoader parentClassLoader) {
return create(codeResolver, classDirectory, contentType, parentClassLoader, Constants.PACKAGE_NAME_ON_DEMAND);
}
/**
* Creates a new template engine.
* All templates are compiled to Java class files on demand.
* A JDK is required.
* Every template has its own class loader.
* This is recommended when running templates on your developer machine.
*
* @param codeResolver to lookup jte templates
* @param classDirectory where template class files are compiled to
* @param contentType the content type of all templates this engine manages
* @param parentClassLoader the parent classloader to use, or null to use the application class loader as parent
* @param packageName the package name, where template classes are generated to
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine create(CodeResolver codeResolver, Path classDirectory, ContentType contentType, ClassLoader parentClassLoader, String packageName) {
return new TemplateEngine(codeResolver, classDirectory, contentType, TemplateMode.OnDemand, parentClassLoader, packageName);
}
/**
* Creates a new template engine.
* All templates must have been precompiled to Java class files already.
* The template engine will load them from the specified classDirectory.
* No JDK is required.
* All templates share one class loader with each other.
* This is recommended when running templates in production.
* How to precompile templates.
*
* @param classDirectory where template class files are located
* @param contentType the content type of all templates this engine manages
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine createPrecompiled(Path classDirectory, ContentType contentType) {
return createPrecompiled(classDirectory, contentType, null);
}
/**
* Creates a new template engine.
* All templates must have been precompiled to Java class files already.
* The template engine will load them via the application class loader.
* This means all template classes must be bundled in you application JAR file.
* No JDK is required.
* This is recommended when running templates in production.
* How to precompile templates.
*
* @param contentType the content type of all templates this engine manages
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine createPrecompiled(ContentType contentType) {
return createPrecompiled(null, contentType);
}
/**
* Creates a new template engine.
* All templates must have been precompiled to Java class files already.
* The template engine will load them from the specified classDirectory.
* No JDK is required.
* All templates share one class loader with each other.
* This is recommended when running templates in production.
* How to precompile templates.
*
* @param classDirectory where template class files are located
* @param contentType the content type of all templates this engine manages
* @param parentClassLoader the parent classloader to use, or null to use the application class loader as parent (only has an effect if classDirectory is not null)
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine createPrecompiled(Path classDirectory, ContentType contentType, ClassLoader parentClassLoader) {
return createPrecompiled(classDirectory, contentType, parentClassLoader, Constants.PACKAGE_NAME_PRECOMPILED);
}
/**
* Creates a new template engine.
* All templates must have been precompiled to Java class files already.
* The template engine will load them from the specified classDirectory.
* No JDK is required.
* All templates share one class loader with each other.
* This is recommended when running templates in production.
* How to precompile templates.
*
* @param classDirectory where template class files are located
* @param contentType the content type of all templates this engine manages
* @param parentClassLoader the parent classloader to use, or null to use the application class loader as parent (only has an effect if classDirectory is not null)
* @param packageName the package name, where template classes are generated to
* @return a fresh TemplateEngine instance
*/
public static TemplateEngine createPrecompiled(Path classDirectory, ContentType contentType, ClassLoader parentClassLoader, String packageName) {
return new TemplateEngine(null, classDirectory, contentType, TemplateMode.Precompiled, parentClassLoader, packageName);
}
private TemplateEngine(CodeResolver codeResolver, Path classDirectory, ContentType contentType, TemplateMode templateMode, ClassLoader parentClassLoader, String packageName) {
if (contentType == null) {
throw new NullPointerException("Content type must be specified.");
}
this.config = new TemplateConfig(contentType, packageName);
this.templateLoader = createTemplateLoader(config, codeResolver, classDirectory, templateMode, parentClassLoader);
this.templateMode = templateMode;
this.templateCache = new ConcurrentHashMap<>();
this.contentType = contentType;
this.classDirectory = classDirectory;
this.parentClassLoader = parentClassLoader;
if (templateMode == TemplateMode.OnDemand) {
cleanAll();
}
}
private static TemplateLoader createTemplateLoader(TemplateConfig config, CodeResolver codeResolver, Path classDirectory, TemplateMode templateMode, ClassLoader parentClassLoader) {
if (templateMode == TemplateMode.Precompiled) {
return new RuntimeTemplateLoader(classDirectory, parentClassLoader, config.packageName);
} else {
try {
Class> compilerClass = Class.forName("gg.jte.compiler.TemplateCompiler");
return (TemplateLoader)compilerClass.getConstructor(TemplateConfig.class, CodeResolver.class, Path.class, ClassLoader.class).newInstance(config, codeResolver, classDirectory, parentClassLoader);
} catch (Exception e) {
throw new TemplateException("TemplateCompiler could not be located. Maybe jte isn't on your classpath?", e);
}
}
}
/**
* Renders the template with the given name.
* It is preferred to use this method, if all your templates have exactly one parameter.
* @param name the template name relative to the specified root directory, for instance "pages/welcome.jte".
* @param param the param passed to the template.
* @param output any implementation of {@link TemplateOutput}, where the template will be written to.
* @throws TemplateException in case the template failed to render, containing information where the error happened.
*/
public void render(String name, Object param, TemplateOutput output) throws TemplateException {
Template template = resolveTemplate(name);
try {
template.render(checkOutput(output), htmlInterceptor, param);
} catch (Throwable e) {
throw handleRenderException(name, template, e);
}
}
/**
* Renders the template with the given name.
* Parameters in the params map are mapped to the corresponding parameters in the template.
* Template parameters with a default value don't have to be provided in the map.
* @param name the template name relative to the specified root directory, for instance "pages/welcome.jte".
* @param params the parameters passed to the template as key value pairs.
* @param output any implementation of {@link TemplateOutput}, where the template will be written to.
* @throws TemplateException in case the template failed to render, containing information where the error happened.
*/
public void render(String name, Map params, TemplateOutput output) throws TemplateException {
Template template = resolveTemplate(name);
try {
template.renderMap(checkOutput(output), htmlInterceptor, params);
} catch (Throwable e) {
throw handleRenderException(name, template, e);
}
}
private TemplateOutput checkOutput(TemplateOutput templateOutput) {
if (contentType == ContentType.Html && !(templateOutput instanceof HtmlTemplateOutput)) {
return new OwaspHtmlTemplateOutput(templateOutput);
}
return templateOutput;
}
private TemplateException handleRenderException(String name, Template template, Throwable e) {
if (e instanceof TemplateException exception) {
return exception;
}
ClassLoader classLoader = template.getClassLoader();
StackTraceElement[] stackTrace = e.getStackTrace();
DebugInfo debugInfo = templateLoader.resolveDebugInfo(classLoader, stackTrace);
String message = "Failed to render " + name;
if (debugInfo != null) {
message += ", error at " + debugInfo.name + ":" + debugInfo.line;
templateLoader.rewriteStackTrace(e, classLoader, stackTrace);
}
return new TemplateException(message, e);
}
public boolean hasTemplate(String name) {
try {
resolveTemplate(name);
return true;
} catch (TemplateNotFoundException e) {
return false;
}
}
public List getTemplatesUsing(String name) {
return templateLoader.getTemplatesUsing(name);
}
/**
* Obtain parameter information about a specific template.
* @param name the template name relative to the specified root directory, for instance "my/example.jte".
* @return a map containing all template parameters names and their classes
* @throws TemplateException in case parameter information is not available (jte classes must be compiled with -parameters compiler flag.)
*/
public Map> getParamInfo(String name) throws TemplateException {
return resolveTemplate(name).getParamInfo();
}
/**
* Prepares the template with the given name for rendering
* @param name the template name relative to the specified root directory, for instance "pages/welcome.jte".
*/
public void prepareForRendering(String name) {
resolveTemplate(name);
}
/**
* Cleans the directory containing the compiled template classes.
*/
public void cleanAll() {
templateLoader.cleanAll();
}
/**
* Generates all template classes in the sources directory, to the compiled template classes directory.
* This only generates .java files, not .class files.
* @return list of .java template files that were generated
*/
public List generateAll() {
return templateLoader.generateAll();
}
/**
* Compiles all templates located in the sources directory, to the compiled template classes directory.
* @return list of .java template files that were compiled
*/
public List precompileAll() {
return templateLoader.precompileAll();
}
/**
* Compiles all templates located in the sources directory, to the compiled template classes directory.
* @param classPath additional class path arguments for the Java compiler. See also {@link TemplateEngine#setClassPath(List)}
* @return list of .java template files that were compiled
*/
public List precompileAll(List classPath) {
setClassPath(classPath);
return templateLoader.precompileAll();
}
/**
* Clears all cached templates.
* Future invocations of render methods will result in reloaded template classes.
*/
public void clearCache() {
templateCache.clear();
}
private Template resolveTemplate(String name) {
if (templateMode == TemplateMode.OnDemand) {
return resolveTemplateOnDemand(name);
} else {
return templateCache.computeIfAbsent(name, templateLoader::load);
}
}
private Template resolveTemplateOnDemand(String name) {
RuntimeException[] exception = {null};
// We use the guarantees of ConcurrentHashMap#compute to avoid synchronized with double hasChanged() check
Template result = templateCache.compute(name, (templateName, template) -> {
if (template != null && !templateLoader.hasChanged(templateName)) {
return template;
}
try {
if (template == null) {
return templateLoader.load(templateName);
} else {
return templateLoader.hotReload(templateName);
}
} catch (RuntimeException e) {
exception[0] = e; // Store exception to throw later
return null; // null removes the template from the template cache
}
});
if (exception[0] != null) {
throw exception[0];
}
return result;
}
/**
* Useful, if this engine is in precompiled mode (probably production) but you still want to be able to apply a hotfix without deployment.
* This template engine will be entirely unaffected by this call. Instead, a fresh template engine will be created.
* When this call succeeds, you can safely switch the template engine reference to the new instance. The old instance including all
* old templates should then be subject to garbage collection.
*
* This only works if you're running on the JDK and if templates have their own classloader.
*
* @param precompiler a template engine that is configured exactly as you usually would precompile your templates.
* @return a fresh template engine with a warmed up cache.
* @throws TemplateException in case there was a compilation error, in this case you should keep the current engine running!
*/
public TemplateEngine reloadPrecompiled(TemplateEngine precompiler) throws TemplateException {
precompiler.precompileAll();
return reloadPrecompiled(precompiler.classDirectory);
}
/**
* Useful, if this engine is in precompiled mode (probably production) but you still want to be able to apply a hotfix without deployment.
* This template engine will be entirely unaffected by this call. Instead, a fresh template engine will be created.
* When this call succeeds, you can safely switch the template engine reference to the new instance. The old instance including all
* old templates should then be subject to garbage collection.
*
* Basically you could recompile all templates on your build server, upload them to your production server
* and call this method afterwards.
*
* This only works if templates have their own classloader.
*
* @param classDirectory the class directory to load the new templates from.
* @return a fresh template engine with a warmed up cache.
* @throws TemplateException in case there was an error during class loading, in this case you should keep the current engine running!
*/
public TemplateEngine reloadPrecompiled(Path classDirectory) throws TemplateException {
TemplateEngine engine = createPrecompiled(classDirectory, contentType, parentClassLoader, config.packageName);
engine.setHtmlInterceptor(htmlInterceptor);
templateCache.keySet().forEach(engine::prepareForRendering);
return engine;
}
/**
* Sets additional compiler arguments for jte templates.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param compileArgs for instance templateEngine.setCompileArgs("--enable-preview", "--release", "" + Runtime.version().feature());
*/
public void setCompileArgs(String ... compileArgs) {
config.compileArgs = compileArgs;
}
/**
* Sets additional compiler arguments for kte templates.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* Currently, only -jvm-target is supported.
* @param compileArgs for instance templateEngine.setCompileArgs("-jvm-target", "17");
*/
public void setKotlinCompileArgs(String ... compileArgs) {
config.kotlinCompileArgs = compileArgs;
}
/**
* Trims control structures, resulting in prettier output.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param value true, to enable
*/
public void setTrimControlStructures(boolean value) {
config.trimControlStructures = value;
}
/**
* Policy that checks the parsed HTML at compile time.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param htmlPolicy the policy
* @throws NullPointerException if policy is null
*/
public void setHtmlPolicy(HtmlPolicy htmlPolicy) {
if (htmlPolicy == null) {
throw new NullPointerException("htmlPolicy must not be null");
}
config.htmlPolicy = htmlPolicy;
}
/**
* Intercepts the given html tags during template compilation
* and calls the configured htmlInterceptor during template rendering.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param htmlTags tags to be intercepted, for instance setHtmlTags("form", "input");
*/
public void setHtmlTags(String ... htmlTags) {
config.htmlTags = htmlTags;
}
/**
* Interceptor that is called during template rendering when one of the
* configured htmlTags is rendered.
* This allows to integrate existing frameworks into jte.
* @param htmlInterceptor the interceptor
*/
public void setHtmlInterceptor(HtmlInterceptor htmlInterceptor) {
this.htmlInterceptor = htmlInterceptor;
}
/**
* By default, jte omits all HTML/CSS/JS comments, when compiling with {@link ContentType#Html}.
* If you don't want this behavior, you can disable it here.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param htmlCommentsPreserved true, to preserve HTML comments in templates
*/
public void setHtmlCommentsPreserved(boolean htmlCommentsPreserved) {
config.htmlCommentsPreserved = htmlCommentsPreserved;
}
/**
* Setting, that UTF-8 encodes all static template parts at compile time.
* Only makes sense if you use a binary output, like {@link gg.jte.output.Utf8ByteOutput}.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param binaryStaticContent true, to pre-generate UTF-8 encoded byte arrays for all static template parts
*/
public void setBinaryStaticContent(boolean binaryStaticContent) {
config.binaryStaticContent = binaryStaticContent;
}
/**
* The class path used for compiling templates.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param classPath list of elements on the class path
*/
public void setClassPath(List classPath) {
config.classPath = classPath;
}
/**
* Directory in which to generate non-java files (resources). Typically, set by plugin rather than end user.
* Optional - if null, resources will not be generated
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param targetResourceDirectory directory to generate resources in
*/
public void setTargetResourceDirectory(Path targetResourceDirectory) {
config.resourceDirectory = targetResourceDirectory;
}
/**
* "group/artifact" of the project using jte. Typically, set by plugin rather than end user.
* If null, the compiler will make one up.
*
* This is a template compilation setting and has no effect when loading precompiled templates.
* @param projectNamespace "groupId/artifactId"
*/
public void setProjectNamespace(String projectNamespace) {
config.projectNamespace = projectNamespace;
}
/**
* Optional - Extensions this template engine should load. Currently, the following extensions exist:
*
*
* - gg.jte.models.generator.ModelExtension
* - gg.jte.nativeimage.NativeResourcesExtension
*
*
* Sample usage:
*
*
* templateEngine.setExtensions(Map.of("gg.jte.models.generator.ModelExtension", Map.of()));
*
*/
public void setExtensions(Map> extensionSettings) {
config.extensionClasses.putAll(extensionSettings);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy