com.aspectran.freemarker.ConfigurationFactory Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2025 The Aspectran Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.aspectran.freemarker;
import com.aspectran.core.component.bean.annotation.AvoidAdvice;
import com.aspectran.core.component.bean.aware.ActivityContextAware;
import com.aspectran.core.context.ActivityContext;
import com.aspectran.freemarker.directive.CustomTrimDirective;
import com.aspectran.freemarker.directive.TrimDirective;
import com.aspectran.freemarker.directive.TrimDirectiveGroup;
import com.aspectran.freemarker.directive.Trimmer;
import com.aspectran.utils.PropertiesLoaderUtils;
import com.aspectran.utils.ResourceUtils;
import com.aspectran.utils.annotation.jsr305.NonNull;
import com.aspectran.utils.apon.Parameters;
import com.aspectran.utils.logging.Logger;
import com.aspectran.utils.logging.LoggerFactory;
import freemarker.cache.ClassTemplateLoader;
import freemarker.cache.FileTemplateLoader;
import freemarker.cache.MultiTemplateLoader;
import freemarker.cache.TemplateLoader;
import freemarker.template.Configuration;
import freemarker.template.SimpleHash;
import freemarker.template.TemplateException;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Properties;
/**
* Factory that configures a FreeMarker Configuration.
*
* Created: 2016. 1. 9.
*/
public class ConfigurationFactory implements ActivityContextAware {
private static final Logger logger = LoggerFactory.getLogger(ConfigurationFactory.class);
private static final String DIRECTIVE_NAME_PARAM_NAME = "name";
private ActivityContext context;
private String configLocation;
private Properties freemarkerSettings;
private Map freemarkerVariables;
private String defaultEncoding;
private String[] templateLoaderPaths;
private TemplateLoader[] templateLoaders;
private TrimDirective[] trimDirectives;
@Override
@AvoidAdvice
public void setActivityContext(@NonNull ActivityContext context) {
this.context = context;
}
/**
* Set the location of the Freemarker settings file.
* Alternatively, you can specify all properties locally.
*/
public void setConfigLocation(String configLocation) {
this.configLocation = configLocation;
}
/**
* Set properties that contain well-known FreeMarker keys which will be
* passed to FreeMarker's {@code Configuration.setSettings} method.
* @param settings the settings
* @see freemarker.template.Configuration#setSettings freemarker.template.Configuration#setSettings
*/
public void setFreemarkerSettings(Properties settings) {
this.freemarkerSettings = settings;
}
/**
* Set a Map that contains well-known FreeMarker objects which will be passed
* to FreeMarker's {@code Configuration.setAllSharedVariables()} method.
* @param variables the variables
* @see freemarker.template.Configuration#setAllSharedVariables freemarker.template.Configuration#setAllSharedVariables
*/
public void setFreemarkerVariables(Map variables) {
this.freemarkerVariables = variables;
}
/**
* Set the default encoding for the FreeMarker configuration.
* If not specified, FreeMarker will use the platform file encoding.
* Used for template rendering unless there is an explicit encoding specified
* for the rendering process (for example, on Spring's FreeMarkerView).
* @param defaultEncoding the default encoding
* @see freemarker.template.Configuration#setDefaultEncoding freemarker.template.Configuration#setDefaultEncoding
*/
public void setDefaultEncoding(String defaultEncoding) {
this.defaultEncoding = defaultEncoding;
}
/**
* Set the Freemarker template loader path.
* @param templateLoaderPath the Freemarker template loader path
*/
public void setTemplateLoaderPath(String templateLoaderPath) {
this.templateLoaderPaths = new String[] { templateLoaderPath };
}
/**
* Set multiple Freemarker template loader paths.
* @param templateLoaderPaths the multiple Freemarker template loader paths
*/
public void setTemplateLoaderPath(String... templateLoaderPaths) {
this.templateLoaderPaths = templateLoaderPaths;
}
/**
* Set a List of Freemarker template loader paths.
* @param templateLoaderPathList a List of Freemarker template loader paths
*/
public void setTemplateLoaderPath(@NonNull List templateLoaderPathList) {
this.templateLoaderPaths = templateLoaderPathList.toArray(new String[0]);
}
/**
* Set a TemplateLoader that will be used to search for templates.
* @param templateLoader the template loader
*/
public void setTemplateLoader(TemplateLoader templateLoader) {
this.templateLoaders = new TemplateLoader[] { templateLoader };
}
/**
* Set multiple TemplateLoaders that will be used to search for templates.
* @param templateLoaders the multiple TemplateLoaders
*/
public void setTemplateLoader(TemplateLoader... templateLoaders) {
this.templateLoaders = templateLoaders;
}
/**
* Set a List of TemplateLoaders that will be used to search for templates.
* @param templateLoaderList a List of TemplateLoaders
*/
public void setTemplateLoader(@NonNull List templateLoaderList) {
this.templateLoaders = templateLoaderList.toArray(new TemplateLoader[0]);
}
public void setTrimDirectives(TrimDirective... trimDirectives) {
this.trimDirectives = trimDirectives;
}
public void setTrimDirectives(@NonNull Parameters parameters) {
String[] directiveGroupNames = parameters.getParameterNames();
List list = new ArrayList<>();
for (String groupName : directiveGroupNames) {
List paramsList = parameters.getParametersList(groupName);
for (Parameters p : paramsList) {
if (p != null) {
String directiveName = p.getString(DIRECTIVE_NAME_PARAM_NAME);
String prefix = p.getString(TrimDirective.PREFIX_PARAM_NAME);
String suffix = p.getString(TrimDirective.SUFFIX_PARAM_NAME);
String[] deprefixes = p.getStringArray(TrimDirective.DEPREFIXES_PARAM_NAME);
String[] desuffixes = p.getStringArray(TrimDirective.DESUFFIXES_PARAM_NAME);
boolean caseSensitive = Boolean.parseBoolean(p.getString(TrimDirective.CASE_SENSITIVE_PARAM_NAME));
if (directiveName != null) {
Trimmer trimmer;
if (prefix != null || suffix != null || deprefixes != null || desuffixes != null) {
trimmer = new Trimmer();
trimmer.setPrefix(prefix);
trimmer.setSuffix(suffix);
trimmer.setDeprefixes(deprefixes);
trimmer.setDesuffixes(desuffixes);
trimmer.setCaseSensitive(caseSensitive);
} else {
trimmer = null;
}
TrimDirective ctd = new CustomTrimDirective(groupName, directiveName, trimmer);
list.add(ctd);
if (logger.isDebugEnabled()) {
logger.debug("CustomTrimDirective " + ctd);
}
}
}
}
}
if (!list.isEmpty()) {
trimDirectives = list.toArray(new TrimDirective[0]);
} else {
trimDirectives = null;
}
}
/**
* Prepare the FreeMarker Configuration and return it.
* @return the FreeMarker Configuration object
* @throws IOException if the config file wasn't found
* @throws TemplateException on FreeMarker initialization failure
*/
public Configuration createConfiguration() throws IOException, TemplateException {
Configuration config = newConfiguration();
Properties props = new Properties();
// Load config file if set.
if (this.configLocation != null) {
logger.info("Loading Freemarker settings from [" + this.configLocation + "]");
props.putAll(PropertiesLoaderUtils.loadProperties(this.configLocation));
}
// Merge local properties if specified.
if (this.freemarkerSettings != null) {
props.putAll(this.freemarkerSettings);
}
// FreeMarker will only accept known keys in its setSettings and
// setAllSharedVariables methods.
if (!props.isEmpty()) {
config.setSettings(props);
}
if (this.freemarkerVariables != null && !freemarkerVariables.isEmpty()) {
config.setAllSharedVariables(new SimpleHash(this.freemarkerVariables, config.getObjectWrapper()));
}
if (this.defaultEncoding != null) {
config.setDefaultEncoding(this.defaultEncoding);
}
// determine FreeMarker TemplateLoader
if (templateLoaders == null) {
if (templateLoaderPaths != null && templateLoaderPaths.length > 0) {
List templateLoaderList = new ArrayList<>();
for (String path : templateLoaderPaths) {
templateLoaderList.add(getTemplateLoaderForPath(path));
}
setTemplateLoader(templateLoaderList);
}
}
TemplateLoader templateLoader = getAggregateTemplateLoader(templateLoaders);
if (templateLoader != null) {
config.setTemplateLoader(templateLoader);
}
// determine CustomTrimDirectives
if (trimDirectives != null && trimDirectives.length > 0) {
TrimDirectiveGroup group = new TrimDirectiveGroup(trimDirectives);
for (Map.Entry> directives : group.entrySet()) {
config.setSharedVariable(directives.getKey(), directives.getValue());
}
}
return config;
}
/**
* Return a new Configuration object. Subclasses can override this for custom
* initialization (e.g. specifying a FreeMarker compatibility level which is a
* new feature in FreeMarker 2.3.21), or for using a mock object for testing.
* Called by {@code createConfiguration()}.
* @return the Configuration object
*/
protected Configuration newConfiguration() {
return new Configuration(Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS);
}
/**
* Return a TemplateLoader based on the given TemplateLoader list.
* If more than one TemplateLoader has been registered, a FreeMarker
* MultiTemplateLoader needs to be created.
* @param templateLoaders the final List of TemplateLoader instances
* @return the aggregate TemplateLoader
*/
protected TemplateLoader getAggregateTemplateLoader(TemplateLoader[] templateLoaders) {
int loaderCount = (templateLoaders != null ? templateLoaders.length : 0);
switch (loaderCount) {
case 0:
if (logger.isDebugEnabled()) {
logger.debug("No FreeMarker TemplateLoaders specified; Can be used only inner template source");
}
return null;
case 1:
if (logger.isDebugEnabled()) {
logger.debug("One FreeMarker TemplateLoader registered: " + templateLoaders[0]);
}
return templateLoaders[0];
default:
TemplateLoader loader = new MultiTemplateLoader(templateLoaders);
if (logger.isDebugEnabled()) {
logger.debug("Multiple FreeMarker TemplateLoader registered: " + loader);
}
return loader;
}
}
/**
* Determine a FreeMarker TemplateLoader for the given path.
* @param templateLoaderPath the path to load templates from
* @return an appropriate TemplateLoader
* @throws IOException if an I/O error has occurred
* @see freemarker.cache.FileTemplateLoader
*/
protected TemplateLoader getTemplateLoaderForPath(@NonNull String templateLoaderPath) throws IOException {
if (templateLoaderPath.startsWith(ResourceUtils.CLASSPATH_URL_PREFIX)) {
String basePackagePath = templateLoaderPath.substring(ResourceUtils.CLASSPATH_URL_PREFIX.length());
if (logger.isDebugEnabled()) {
logger.debug("Template loader path [" + templateLoaderPath +
"] resolved to class path [" + basePackagePath + "]");
}
return new ClassTemplateLoader(context.getAvailableActivity().getClassLoader(), basePackagePath);
} else if (templateLoaderPath.startsWith(ResourceUtils.FILE_URL_PREFIX)) {
File file = new File(templateLoaderPath.substring(ResourceUtils.FILE_URL_PREFIX.length()));
if (logger.isDebugEnabled()) {
logger.debug("Template loader path [" + templateLoaderPath +
"] resolved to file path [" + file.getAbsolutePath() + "]");
}
return new FileTemplateLoader(file);
} else {
File file = new File(context.getApplicationAdapter().getBasePath(), templateLoaderPath);
if (logger.isDebugEnabled()) {
logger.debug("Template loader path [" + templateLoaderPath +
"] resolved to file path [" + file.getAbsolutePath() + "]");
}
return new FileTemplateLoader(file);
}
}
}