2011-04-07 16:15:49 +00:00
|
|
|
/* $This file is distributed under the terms of the license in /doc/license.txt$ */
|
|
|
|
|
|
|
|
package freemarker.ext.dump;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.StringWriter;
|
|
|
|
import java.io.Writer;
|
2011-04-08 16:50:33 +00:00
|
|
|
import java.lang.reflect.Method;
|
2011-04-08 17:23:16 +00:00
|
|
|
import java.util.ArrayList;
|
2011-04-07 16:15:49 +00:00
|
|
|
import java.util.HashMap;
|
2011-04-08 17:23:16 +00:00
|
|
|
import java.util.List;
|
2011-04-07 16:15:49 +00:00
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
import org.apache.commons.logging.Log;
|
|
|
|
import org.apache.commons.logging.LogFactory;
|
|
|
|
|
|
|
|
import freemarker.core.Environment;
|
|
|
|
import freemarker.template.Template;
|
|
|
|
import freemarker.template.TemplateBooleanModel;
|
2011-04-11 15:32:12 +00:00
|
|
|
import freemarker.template.TemplateCollectionModel;
|
2011-04-07 16:15:49 +00:00
|
|
|
import freemarker.template.TemplateDateModel;
|
|
|
|
import freemarker.template.TemplateDirectiveModel;
|
|
|
|
import freemarker.template.TemplateException;
|
|
|
|
import freemarker.template.TemplateHashModel;
|
|
|
|
import freemarker.template.TemplateHashModelEx;
|
|
|
|
import freemarker.template.TemplateMethodModel;
|
|
|
|
import freemarker.template.TemplateModel;
|
|
|
|
import freemarker.template.TemplateModelException;
|
2011-04-11 15:32:12 +00:00
|
|
|
import freemarker.template.TemplateModelIterator;
|
2011-04-07 16:15:49 +00:00
|
|
|
import freemarker.template.TemplateNumberModel;
|
|
|
|
import freemarker.template.TemplateScalarModel;
|
|
|
|
import freemarker.template.TemplateSequenceModel;
|
|
|
|
import freemarker.template.utility.DeepUnwrap;
|
|
|
|
|
|
|
|
/* TODO
|
|
|
|
* - Check error messages generated for TemplateModelException-s. If too generic, need to catch, create specific
|
|
|
|
* error message, and rethrow.
|
|
|
|
*/
|
|
|
|
|
|
|
|
public abstract class BaseDumpDirective implements TemplateDirectiveModel {
|
|
|
|
|
|
|
|
private static final Log log = LogFactory.getLog(BaseDumpDirective.class);
|
|
|
|
|
2011-04-11 22:05:36 +00:00
|
|
|
enum Key {
|
|
|
|
NAME("name"),
|
|
|
|
TYPE("type"),
|
|
|
|
VALUE("value"),
|
|
|
|
DATE_TYPE("dateType");
|
|
|
|
|
|
|
|
private final String key;
|
|
|
|
|
|
|
|
Key(String key) {
|
|
|
|
this.key = key;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String toString() {
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum Type {
|
|
|
|
STRING("String"),
|
|
|
|
NUMBER("Number"),
|
|
|
|
BOOLEAN("Boolean"),
|
|
|
|
DATE("Date"),
|
|
|
|
SEQUENCE("Sequence"),
|
|
|
|
HASH("Hash"),
|
|
|
|
HASH_EX("HashEx"),
|
|
|
|
COLLECTION("Collection"),
|
|
|
|
METHOD("Method"),
|
|
|
|
DIRECTIVE("Directive");
|
|
|
|
|
|
|
|
private final String type;
|
|
|
|
|
|
|
|
Type(String type) {
|
|
|
|
this.type = type;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String toString() {
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum DateType {
|
|
|
|
DATE("Date"),
|
|
|
|
DATETIME("DateTime"),
|
|
|
|
TIME("Time"),
|
|
|
|
UNKNOWN("Unknown");
|
|
|
|
|
|
|
|
private final String type;
|
|
|
|
|
|
|
|
DateType(String type) {
|
|
|
|
this.type = type;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String toString() {
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-07 16:15:49 +00:00
|
|
|
protected Map<String, Object> getTemplateVariableData(String varName, Environment env)
|
|
|
|
throws TemplateModelException {
|
|
|
|
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.NAME.toString(), varName);
|
2011-04-07 16:15:49 +00:00
|
|
|
|
|
|
|
TemplateHashModel dataModel = env.getDataModel();
|
|
|
|
TemplateModel model = dataModel.get(varName);
|
2011-04-11 22:05:36 +00:00
|
|
|
|
|
|
|
// Don't return null if model == null. We still want to send the map to the template.
|
|
|
|
if (model != null) {
|
|
|
|
// TemplateMethodModel and TemplateDirectiveModel objects can only be
|
|
|
|
// included in the data model at the top level.
|
|
|
|
if (model instanceof TemplateMethodModel) {
|
|
|
|
map.putAll( getTemplateModelData( ( TemplateMethodModel)model, varName ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateDirectiveModel) {
|
|
|
|
map.putAll( getTemplateModelData( ( TemplateDirectiveModel)model, varName ) );
|
|
|
|
|
|
|
|
} else {
|
|
|
|
map.putAll(getData(model));
|
|
|
|
}
|
2011-04-08 16:50:33 +00:00
|
|
|
}
|
2011-04-07 16:15:49 +00:00
|
|
|
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getData(TemplateModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
|
|
|
|
|
|
|
// Don't return null if model == null. We still want to send the map to the template.
|
|
|
|
if (model != null) {
|
2011-04-08 16:50:33 +00:00
|
|
|
// Do TemplateHashModel cases first, because models of some Java objects are
|
|
|
|
// StringModels, and so are both TemplateScalarModels and TemplateHashModelExs.
|
|
|
|
if (model instanceof TemplateHashModelEx) {
|
|
|
|
map.putAll( getTemplateModelData( ( TemplateHashModelEx)model ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateHashModel) {
|
2011-04-07 16:15:49 +00:00
|
|
|
map.putAll( getTemplateModelData( ( TemplateHashModel)model ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateScalarModel) {
|
|
|
|
map.putAll( getTemplateModelData( (TemplateScalarModel)model ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateBooleanModel) {
|
|
|
|
map.putAll( getTemplateModelData( (TemplateBooleanModel)model ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateNumberModel) {
|
|
|
|
map.putAll( getTemplateModelData( (TemplateNumberModel)model ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateDateModel) {
|
|
|
|
map.putAll( getTemplateModelData( (TemplateDateModel)model ) );
|
|
|
|
|
|
|
|
} else if (model instanceof TemplateSequenceModel){
|
|
|
|
map.putAll( getTemplateModelData( ( TemplateSequenceModel)model ) );
|
2011-04-08 16:50:33 +00:00
|
|
|
|
2011-04-11 15:32:12 +00:00
|
|
|
} else if (model instanceof TemplateCollectionModel) {
|
|
|
|
map.putAll( getTemplateModelData( ( TemplateCollectionModel)model ) );
|
|
|
|
|
|
|
|
// Nodes and transforms not included here
|
|
|
|
|
2011-04-07 16:15:49 +00:00
|
|
|
} else {
|
|
|
|
map.putAll( getTemplateModelData( (TemplateModel)model ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateScalarModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.STRING);
|
|
|
|
map.put(Key.VALUE.toString(), model.getAsString());
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateBooleanModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.BOOLEAN);
|
|
|
|
map.put(Key.VALUE.toString(), model.getAsBoolean());
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateNumberModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.NUMBER);
|
|
|
|
map.put(Key.VALUE.toString(), model.getAsNumber());
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateDateModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.DATE);
|
2011-04-08 21:23:32 +00:00
|
|
|
int dateType = model.getDateType();
|
|
|
|
|
2011-04-11 22:05:36 +00:00
|
|
|
DateType type;
|
2011-04-08 21:23:32 +00:00
|
|
|
switch (dateType) {
|
|
|
|
case TemplateDateModel.DATE:
|
2011-04-11 22:05:36 +00:00
|
|
|
type = DateType.DATE;
|
2011-04-08 21:23:32 +00:00
|
|
|
break;
|
|
|
|
case TemplateDateModel.DATETIME:
|
2011-04-11 22:05:36 +00:00
|
|
|
type = DateType.DATETIME;
|
2011-04-08 21:23:32 +00:00
|
|
|
break;
|
|
|
|
case TemplateDateModel.TIME:
|
2011-04-11 22:05:36 +00:00
|
|
|
type = DateType.TIME;
|
2011-04-08 21:23:32 +00:00
|
|
|
break;
|
|
|
|
default:
|
2011-04-11 22:05:36 +00:00
|
|
|
type = DateType.UNKNOWN;
|
2011-04-08 21:23:32 +00:00
|
|
|
}
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.DATE_TYPE.toString(), type);
|
2011-04-08 21:23:32 +00:00
|
|
|
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.VALUE.toString(), model.getAsDate());
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateHashModel model) throws TemplateModelException {
|
2011-04-11 15:32:12 +00:00
|
|
|
// The data model is a hash; when else do we get here?
|
|
|
|
log.debug("Dumping model " + model);
|
2011-04-07 16:15:49 +00:00
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.HASH);
|
|
|
|
//map.put(Key.VALUE.toString(), ????);
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateSequenceModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.SEQUENCE);
|
2011-04-08 17:23:16 +00:00
|
|
|
int itemCount = model.size();
|
|
|
|
List<Map<String, Object>> items = new ArrayList<Map<String, Object>>(itemCount);
|
|
|
|
for ( int i = 0; i < itemCount; i++ ) {
|
|
|
|
TemplateModel item = model.get(i);
|
|
|
|
items.add(getData(item));
|
|
|
|
}
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.VALUE.toString(), items);
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateHashModelEx model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.HASH_EX);
|
2011-04-11 15:32:12 +00:00
|
|
|
Map<String, Object> items = new HashMap<String, Object>();
|
|
|
|
// keys() gets only values visible to template
|
|
|
|
TemplateCollectionModel keys = model.keys();
|
|
|
|
TemplateModelIterator iModel = keys.iterator();
|
|
|
|
while (iModel.hasNext()) {
|
|
|
|
String key = iModel.next().toString();
|
|
|
|
TemplateModel value = model.get(key);
|
|
|
|
items.put(key, getData(value));
|
|
|
|
}
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.VALUE.toString(), items);
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
2011-04-11 15:32:12 +00:00
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateCollectionModel model) throws TemplateModelException {
|
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.COLLECTION);
|
2011-04-11 15:32:12 +00:00
|
|
|
List<Map<String, Object>> items = new ArrayList<Map<String, Object>>();
|
|
|
|
TemplateModelIterator iModel = model.iterator();
|
|
|
|
while (iModel.hasNext()) {
|
|
|
|
TemplateModel m = iModel.next();
|
|
|
|
items.add(getData(m));
|
|
|
|
}
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.VALUE.toString(), items);
|
2011-04-11 15:32:12 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
2011-04-08 16:50:33 +00:00
|
|
|
private Map<String, Object> getTemplateModelData(TemplateMethodModel model, String varName) throws TemplateModelException {
|
2011-04-07 16:15:49 +00:00
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.METHOD);
|
2011-04-08 16:50:33 +00:00
|
|
|
map.put("help", getHelp(model, varName));
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
2011-04-08 16:50:33 +00:00
|
|
|
private Map<String, Object> getTemplateModelData(TemplateDirectiveModel model, String varName) throws TemplateModelException {
|
2011-04-07 16:15:49 +00:00
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), Type.DIRECTIVE);
|
2011-04-08 16:50:33 +00:00
|
|
|
map.put("help", getHelp(model, varName));
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
@SuppressWarnings("unchecked")
|
|
|
|
private Map<String, Object> getHelp(TemplateModel model, String varName) {
|
|
|
|
Map<String, Object> map = null;
|
|
|
|
if ( model instanceof TemplateMethodModel || model instanceof TemplateDirectiveModel ) {
|
|
|
|
Class<?> cls = model.getClass();
|
|
|
|
Method[] methods = cls.getMethods();
|
|
|
|
for (Method method : methods) {
|
|
|
|
if ( method.getName().equals("help") ) {
|
|
|
|
try {
|
|
|
|
map = (Map<String, Object>) method.invoke(model, varName);
|
|
|
|
} catch (Exception e) {
|
|
|
|
String modelClass = model instanceof TemplateMethodModel ? "TemplateMethodModel" : "TemplateDirectiveModel";
|
|
|
|
log.error("Error invoking method help() on " + modelClass + " of class " + cls.getName());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Map<String, Object> getTemplateModelData(TemplateModel model) throws TemplateModelException {
|
|
|
|
// One of the above cases should have applied. Track whether this actually occurs.
|
2011-04-08 16:50:33 +00:00
|
|
|
log.debug("Found model with no known type");
|
2011-04-07 16:15:49 +00:00
|
|
|
Map<String, Object> map = new HashMap<String, Object>();
|
|
|
|
Object unwrappedModel = DeepUnwrap.permissiveUnwrap(model);
|
2011-04-11 22:05:36 +00:00
|
|
|
map.put(Key.TYPE.toString(), unwrappedModel.getClass().getName());
|
|
|
|
map.put(Key.VALUE.toString(), unwrappedModel.toString());
|
2011-04-07 16:15:49 +00:00
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void dump(String templateName, Map<String, Object> map, String modelName, Environment env)
|
|
|
|
throws TemplateException, IOException {
|
|
|
|
|
|
|
|
Template template = env.getConfiguration().getTemplate(templateName);
|
|
|
|
StringWriter sw = new StringWriter();
|
|
|
|
template.process(map, sw);
|
|
|
|
Writer out = env.getOut();
|
|
|
|
out.write(sw.toString());
|
|
|
|
|
|
|
|
}
|
2011-04-11 15:32:12 +00:00
|
|
|
|
2011-04-07 16:15:49 +00:00
|
|
|
}
|