mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-07-24 17:32:48 +00:00
Refactor of template and script services to allow easy addition of further template and script processors.
Hightlights of check-in include: - Introduction of script processor - Neutralisation of script and template models - The notion of a processor extension introduced - Extensions applied to processor implementation rather than the services - Auto selection of processor based on file extension of template or script git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@5519 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
@@ -3,7 +3,25 @@
|
|||||||
|
|
||||||
<beans>
|
<beans>
|
||||||
|
|
||||||
<bean id="scriptService" class="org.alfresco.repo.jscript.RhinoScriptService">
|
<bean id="scriptService" class="org.alfresco.repo.processor.ScriptServiceImpl">
|
||||||
|
<property name="defaultScriptProcessor">
|
||||||
|
<value>javascript</value>
|
||||||
|
</property>
|
||||||
|
<property name="nodeService">
|
||||||
|
<ref bean="NodeService"/>
|
||||||
|
</property>
|
||||||
|
</bean>
|
||||||
|
|
||||||
|
<bean id="javaScriptProcessor" class="org.alfresco.repo.jscript.RhinoScriptProcessor" init-method="register">
|
||||||
|
<property name="name">
|
||||||
|
<value>javascript</value>
|
||||||
|
</property>
|
||||||
|
<property name="extension">
|
||||||
|
<value>js</value>
|
||||||
|
</property>
|
||||||
|
<property name="scriptService">
|
||||||
|
<ref bean="scriptService"/>
|
||||||
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
<ref bean="ServiceRegistry"/>
|
<ref bean="ServiceRegistry"/>
|
||||||
</property>
|
</property>
|
||||||
@@ -20,20 +38,20 @@
|
|||||||
|
|
||||||
<!-- base config implementation that script extension beans extend from - for auto registration
|
<!-- base config implementation that script extension beans extend from - for auto registration
|
||||||
as a global script with the ScriptService -->
|
as a global script with the ScriptService -->
|
||||||
<bean id="baseScriptImplementation" abstract="true" init-method="register">
|
<bean id="baseJavaScriptExtension" abstract="true" init-method="register">
|
||||||
<property name="scriptService">
|
<property name="processor">
|
||||||
<ref bean="scriptService"/>
|
<ref bean="javaScriptProcessor"/>
|
||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="loggerScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.ScriptLogger">
|
<bean id="loggerScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.ScriptLogger">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>logger</value>
|
<value>logger</value>
|
||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="utilsScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.ScriptUtils">
|
<bean id="utilsScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.ScriptUtils">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>utils</value>
|
<value>utils</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -41,8 +59,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="actionsScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.Actions">
|
<bean id="actionsScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.Actions">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>actions</value>
|
<value>actions</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -50,8 +68,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="searchScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.Search">
|
<bean id="searchScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.Search">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>search</value>
|
<value>search</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -62,8 +80,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="classificationScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.Classification">
|
<bean id="classificationScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.Classification">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>classification</value>
|
<value>classification</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -74,8 +92,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="peopleScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.People">
|
<bean id="peopleScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.People">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>people</value>
|
<value>people</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -89,8 +107,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="sessionScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.Session">
|
<bean id="sessionScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.Session">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>session</value>
|
<value>session</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -104,8 +122,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="avmScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.AVM">
|
<bean id="avmScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.AVM">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>avm</value>
|
<value>avm</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
@@ -113,8 +131,8 @@
|
|||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<bean id="crossCopyScript" parent="baseScriptImplementation" class="org.alfresco.repo.jscript.CrossRepositoryCopy">
|
<bean id="crossCopyScript" parent="baseJavaScriptExtension" class="org.alfresco.repo.jscript.CrossRepositoryCopy">
|
||||||
<property name="scriptName">
|
<property name="extensionName">
|
||||||
<value>crossRepoCopy</value>
|
<value>crossRepoCopy</value>
|
||||||
</property>
|
</property>
|
||||||
<property name="serviceRegistry">
|
<property name="serviceRegistry">
|
||||||
|
@@ -3,44 +3,39 @@
|
|||||||
|
|
||||||
<beans>
|
<beans>
|
||||||
|
|
||||||
<bean id="templateService" class="org.alfresco.repo.template.TemplateServiceImpl">
|
<bean id="templateService" class="org.alfresco.repo.processor.TemplateServiceImpl">
|
||||||
<!-- A Map of named template engines to class implementations/Spring bean IDs -->
|
|
||||||
<!-- The key of each property is the name of the engine - this is the name that is passed in -->
|
|
||||||
<!-- to the TemplateService by the caller. The value is either a fully qualified class name for -->
|
|
||||||
<!-- the object to create, or the Spring bean ID if the object requires Spring service injection. -->
|
|
||||||
<property name="templateEngines">
|
|
||||||
<map>
|
|
||||||
<entry key="freemarker">
|
|
||||||
<value>freeMarkerProcessor</value>
|
|
||||||
</entry>
|
|
||||||
</map>
|
|
||||||
</property>
|
|
||||||
<property name="defaultTemplateEngine">
|
<property name="defaultTemplateEngine">
|
||||||
<value>freemarker</value>
|
<value>freemarker</value>
|
||||||
</property>
|
</property>
|
||||||
</bean>
|
<property name="nodeService">
|
||||||
|
|
||||||
<!-- This engine requires Spring config setup to use Repository services -->
|
|
||||||
<!-- The beans are not thread safe and therefore we create one per request -->
|
|
||||||
<bean id="freeMarkerProcessor" class="org.alfresco.repo.template.FreeMarkerProcessor" singleton="false">
|
|
||||||
<property name="nodeService">
|
|
||||||
<ref bean="NodeService"/>
|
<ref bean="NodeService"/>
|
||||||
</property>
|
</property>
|
||||||
<property name="contentService">
|
</bean>
|
||||||
<ref bean="ContentService"/>
|
|
||||||
|
<bean id="baseTemplateProcessor" abstract="true" init-method="register">
|
||||||
|
<property name="templateService">
|
||||||
|
<ref bean="templateService"/>
|
||||||
</property>
|
</property>
|
||||||
<!--
|
<property name="serviceRegistry">
|
||||||
<property name="defaultEncoding">
|
<ref bean="ServiceRegistry"/>
|
||||||
<value>UTF-8</value>
|
|
||||||
</property>
|
</property>
|
||||||
-->
|
</bean>
|
||||||
|
|
||||||
|
<!-- The beans are not thread safe and therefore we create one per request -->
|
||||||
|
<bean id="freeMarkerProcessor" parent="baseTemplateProcessor" class="org.alfresco.repo.template.FreeMarkerProcessor">
|
||||||
|
<property name="name">
|
||||||
|
<value>freemarker</value>
|
||||||
|
</property>
|
||||||
|
<property name="extension">
|
||||||
|
<value>ftl</value>
|
||||||
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
<!-- base config implementation that template extension beans extend from - for auto registration
|
<!-- base config implementation that template extension beans extend from - for auto registration
|
||||||
as a global template helper with the TemplateService -->
|
as a global template helper with the TemplateService -->
|
||||||
<bean id="baseTemplateImplementation" abstract="true" init-method="register">
|
<bean id="baseTemplateImplementation" abstract="true" init-method="register">
|
||||||
<property name="templateService">
|
<property name="processor">
|
||||||
<ref bean="templateService"/>
|
<ref bean="freeMarkerProcessor"/>
|
||||||
</property>
|
</property>
|
||||||
</bean>
|
</bean>
|
||||||
|
|
||||||
|
@@ -29,7 +29,6 @@ import java.util.Map;
|
|||||||
|
|
||||||
import org.alfresco.model.ContentModel;
|
import org.alfresco.model.ContentModel;
|
||||||
import org.alfresco.repo.action.ParameterDefinitionImpl;
|
import org.alfresco.repo.action.ParameterDefinitionImpl;
|
||||||
import org.alfresco.repo.jscript.RhinoScriptService;
|
|
||||||
import org.alfresco.repo.jscript.ScriptAction;
|
import org.alfresco.repo.jscript.ScriptAction;
|
||||||
import org.alfresco.service.ServiceRegistry;
|
import org.alfresco.service.ServiceRegistry;
|
||||||
import org.alfresco.service.cmr.action.Action;
|
import org.alfresco.service.cmr.action.Action;
|
||||||
@@ -129,8 +128,7 @@ public class ScriptActionExecuter extends ActionExecuterAbstractBase
|
|||||||
|
|
||||||
// the default scripting model provides access to well known objects and searching
|
// the default scripting model provides access to well known objects and searching
|
||||||
// facilities - it also provides basic create/update/delete/copy/move services
|
// facilities - it also provides basic create/update/delete/copy/move services
|
||||||
Map<String, Object> model = RhinoScriptService.buildDefaultModel(
|
Map<String, Object> model = this.serviceRegistry.getScriptService().buildDefaultModel(
|
||||||
this.serviceRegistry,
|
|
||||||
personRef,
|
personRef,
|
||||||
getCompanyHome(),
|
getCompanyHome(),
|
||||||
homeSpaceRef,
|
homeSpaceRef,
|
||||||
|
@@ -24,6 +24,8 @@
|
|||||||
*/
|
*/
|
||||||
package org.alfresco.repo.action.scheduled;
|
package org.alfresco.repo.action.scheduled;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import org.alfresco.service.cmr.repository.NodeRef;
|
import org.alfresco.service.cmr.repository.NodeRef;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -46,7 +48,7 @@ public interface TemplateActionModelFactory
|
|||||||
*
|
*
|
||||||
* @return - the model for the template engine.
|
* @return - the model for the template engine.
|
||||||
*/
|
*/
|
||||||
public Object getModel();
|
public Map<String, Object> getModel();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build a model with a default node context.
|
* Build a model with a default node context.
|
||||||
@@ -54,5 +56,5 @@ public interface TemplateActionModelFactory
|
|||||||
* @param nodeRef
|
* @param nodeRef
|
||||||
* @return - the model (with nodeRef as its context).
|
* @return - the model (with nodeRef as its context).
|
||||||
*/
|
*/
|
||||||
public Object getModel(NodeRef nodeRef);
|
public Map<String, Object> getModel(NodeRef nodeRef);
|
||||||
}
|
}
|
||||||
|
@@ -34,7 +34,7 @@ import org.alfresco.service.cmr.avm.AVMNodeDescriptor;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public final class AVM extends BaseScopableScriptImplementation
|
public final class AVM extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
/** Repository Service Registry */
|
/** Repository Service Registry */
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
@@ -36,7 +36,7 @@ import org.alfresco.service.cmr.action.ActionService;
|
|||||||
*
|
*
|
||||||
* @author davidc
|
* @author davidc
|
||||||
*/
|
*/
|
||||||
public final class Actions extends BaseScopableScriptImplementation
|
public final class Actions extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
/** Repository Service Registry */
|
/** Repository Service Registry */
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
@@ -16,6 +16,7 @@
|
|||||||
*/
|
*/
|
||||||
package org.alfresco.repo.jscript;
|
package org.alfresco.repo.jscript;
|
||||||
|
|
||||||
|
import org.alfresco.repo.processor.BaseProcessorExtension;
|
||||||
import org.mozilla.javascript.Scriptable;
|
import org.mozilla.javascript.Scriptable;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -25,7 +26,7 @@ import org.mozilla.javascript.Scriptable;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public class BaseScopableScriptImplementation extends BaseScriptImplementation implements Scopeable
|
public class BaseScopableProcessorExtension extends BaseProcessorExtension implements Scopeable
|
||||||
{
|
{
|
||||||
private static ThreadLocal<Scriptable> scope = new ThreadLocal<Scriptable>();
|
private static ThreadLocal<Scriptable> scope = new ThreadLocal<Scriptable>();
|
||||||
|
|
||||||
@@ -36,7 +37,7 @@ public class BaseScopableScriptImplementation extends BaseScriptImplementation i
|
|||||||
*/
|
*/
|
||||||
public void setScope(Scriptable scope)
|
public void setScope(Scriptable scope)
|
||||||
{
|
{
|
||||||
this.scope.set(scope);
|
BaseScopableProcessorExtension.scope.set(scope);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -44,6 +45,6 @@ public class BaseScopableScriptImplementation extends BaseScriptImplementation i
|
|||||||
*/
|
*/
|
||||||
public Scriptable getScope()
|
public Scriptable getScope()
|
||||||
{
|
{
|
||||||
return this.scope.get();
|
return BaseScopableProcessorExtension.scope.get();
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -37,7 +37,7 @@ import org.alfresco.service.namespace.QName;
|
|||||||
*
|
*
|
||||||
* @author Andy Hind
|
* @author Andy Hind
|
||||||
*/
|
*/
|
||||||
public final class Classification extends BaseScopableScriptImplementation
|
public final class Classification extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
|
||||||
|
@@ -37,7 +37,7 @@ import org.mozilla.javascript.Scriptable;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public final class CrossRepositoryCopy extends BaseScopableScriptImplementation
|
public final class CrossRepositoryCopy extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
public final static String BEAN_NAME = "crossCopyScript";
|
public final static String BEAN_NAME = "crossCopyScript";
|
||||||
|
|
||||||
|
@@ -39,8 +39,6 @@ import org.alfresco.model.ContentModel;
|
|||||||
import org.alfresco.repo.action.executer.TransformActionExecuter;
|
import org.alfresco.repo.action.executer.TransformActionExecuter;
|
||||||
import org.alfresco.repo.content.transform.magick.ImageMagickContentTransformer;
|
import org.alfresco.repo.content.transform.magick.ImageMagickContentTransformer;
|
||||||
import org.alfresco.repo.search.QueryParameterDefImpl;
|
import org.alfresco.repo.search.QueryParameterDefImpl;
|
||||||
import org.alfresco.repo.template.FreeMarkerProcessor;
|
|
||||||
import org.alfresco.repo.template.TemplateNode;
|
|
||||||
import org.alfresco.repo.version.VersionModel;
|
import org.alfresco.repo.version.VersionModel;
|
||||||
import org.alfresco.service.ServiceRegistry;
|
import org.alfresco.service.ServiceRegistry;
|
||||||
import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
|
import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
|
||||||
@@ -1739,8 +1737,7 @@ public class Node implements Serializable, Scopeable
|
|||||||
private String processTemplate(String template, NodeRef templateRef, ScriptableObject args)
|
private String processTemplate(String template, NodeRef templateRef, ScriptableObject args)
|
||||||
{
|
{
|
||||||
// build default model for the template processing
|
// build default model for the template processing
|
||||||
Map<String, Object> model = FreeMarkerProcessor.buildDefaultModel(
|
Map<String, Object> model = this.services.getTemplateService().buildDefaultModel(
|
||||||
services,
|
|
||||||
((Node)((Wrapper)scope.get("person", scope)).unwrap()).getNodeRef(),
|
((Node)((Wrapper)scope.get("person", scope)).unwrap()).getNodeRef(),
|
||||||
((Node)((Wrapper)scope.get("companyhome", scope)).unwrap()).getNodeRef(),
|
((Node)((Wrapper)scope.get("companyhome", scope)).unwrap()).getNodeRef(),
|
||||||
((Node)((Wrapper)scope.get("userhome", scope)).unwrap()).getNodeRef(),
|
((Node)((Wrapper)scope.get("userhome", scope)).unwrap()).getNodeRef(),
|
||||||
@@ -1750,12 +1747,12 @@ public class Node implements Serializable, Scopeable
|
|||||||
// add the current node as either the document/space as appropriate
|
// add the current node as either the document/space as appropriate
|
||||||
if (this.getIsDocument())
|
if (this.getIsDocument())
|
||||||
{
|
{
|
||||||
model.put("document", new TemplateNode(this.nodeRef, this.services, null));
|
model.put("document", this.nodeRef);
|
||||||
model.put("space", new TemplateNode(getPrimaryParentAssoc().getParentRef(), this.services, null));
|
model.put("space", getPrimaryParentAssoc().getParentRef());
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
model.put("space", new TemplateNode(this.nodeRef, this.services, null));
|
model.put("space", this.nodeRef);
|
||||||
}
|
}
|
||||||
|
|
||||||
// add the supplied args to the 'args' root object
|
// add the supplied args to the 'args' root object
|
||||||
|
@@ -40,7 +40,7 @@ import org.alfresco.util.ParameterCheck;
|
|||||||
*
|
*
|
||||||
* @author davidc
|
* @author davidc
|
||||||
*/
|
*/
|
||||||
public final class People extends BaseScopableScriptImplementation
|
public final class People extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
/** Repository Service Registry */
|
/** Repository Service Registry */
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
@@ -36,19 +36,18 @@ import java.util.StringTokenizer;
|
|||||||
|
|
||||||
import org.alfresco.error.AlfrescoRuntimeException;
|
import org.alfresco.error.AlfrescoRuntimeException;
|
||||||
import org.alfresco.model.ContentModel;
|
import org.alfresco.model.ContentModel;
|
||||||
import org.alfresco.service.ServiceRegistry;
|
import org.alfresco.repo.processor.BaseProcessor;
|
||||||
import org.alfresco.service.cmr.model.FileInfo;
|
import org.alfresco.service.cmr.model.FileInfo;
|
||||||
import org.alfresco.service.cmr.model.FileNotFoundException;
|
import org.alfresco.service.cmr.model.FileNotFoundException;
|
||||||
import org.alfresco.service.cmr.repository.ContentIOException;
|
import org.alfresco.service.cmr.repository.ContentIOException;
|
||||||
import org.alfresco.service.cmr.repository.ContentReader;
|
import org.alfresco.service.cmr.repository.ContentReader;
|
||||||
import org.alfresco.service.cmr.repository.NodeRef;
|
import org.alfresco.service.cmr.repository.NodeRef;
|
||||||
|
import org.alfresco.service.cmr.repository.ProcessorExtension;
|
||||||
import org.alfresco.service.cmr.repository.ScriptException;
|
import org.alfresco.service.cmr.repository.ScriptException;
|
||||||
import org.alfresco.service.cmr.repository.ScriptImplementation;
|
|
||||||
import org.alfresco.service.cmr.repository.ScriptLocation;
|
import org.alfresco.service.cmr.repository.ScriptLocation;
|
||||||
import org.alfresco.service.cmr.repository.ScriptService;
|
import org.alfresco.service.cmr.repository.ScriptProcessor;
|
||||||
import org.alfresco.service.cmr.repository.StoreRef;
|
import org.alfresco.service.cmr.repository.StoreRef;
|
||||||
import org.alfresco.service.namespace.QName;
|
import org.alfresco.service.namespace.QName;
|
||||||
import org.alfresco.util.ParameterCheck;
|
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
import org.mozilla.javascript.Context;
|
import org.mozilla.javascript.Context;
|
||||||
import org.mozilla.javascript.NativeArray;
|
import org.mozilla.javascript.NativeArray;
|
||||||
@@ -58,41 +57,25 @@ import org.mozilla.javascript.Wrapper;
|
|||||||
import org.springframework.util.FileCopyUtils;
|
import org.springframework.util.FileCopyUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Implementation of the ScriptService using the Rhino JavaScript engine.
|
* Implementation of the ScriptEninge using the Rhino JavaScript engine.
|
||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public class RhinoScriptService implements ScriptService
|
public class RhinoScriptProcessor extends BaseProcessor implements ScriptProcessor
|
||||||
{
|
{
|
||||||
private static final Logger logger = Logger.getLogger(RhinoScriptService.class);
|
private static final Logger logger = Logger.getLogger(RhinoScriptProcessor.class);
|
||||||
|
|
||||||
private static final String IMPORT_PREFIX = "<import";
|
private static final String IMPORT_PREFIX = "<import";
|
||||||
private static final String IMPORT_RESOURCE = "resource=\"";
|
private static final String IMPORT_RESOURCE = "resource=\"";
|
||||||
private static final String PATH_CLASSPATH = "classpath:";
|
private static final String PATH_CLASSPATH = "classpath:";
|
||||||
private static final String SCRIPT_ROOT = "_root";
|
private static final String SCRIPT_ROOT = "_root";
|
||||||
|
|
||||||
/** Repository Service Registry */
|
|
||||||
private ServiceRegistry services;
|
|
||||||
|
|
||||||
/** Store into which to resolve cm:name based script paths */
|
/** Store into which to resolve cm:name based script paths */
|
||||||
private StoreRef storeRef;
|
private StoreRef storeRef;
|
||||||
|
|
||||||
/** Store root path to resolve cm:name based scripts path from */
|
/** Store root path to resolve cm:name based scripts path from */
|
||||||
private String storePath;
|
private String storePath;
|
||||||
|
|
||||||
/** List of global scripts */
|
|
||||||
private List<ScriptImplementation> globalScripts = new ArrayList<ScriptImplementation>();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the Service Registry
|
|
||||||
*
|
|
||||||
* @param service registry
|
|
||||||
*/
|
|
||||||
public void setServiceRegistry(ServiceRegistry services)
|
|
||||||
{
|
|
||||||
this.services = services;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the default store reference
|
* Set the default store reference
|
||||||
*
|
*
|
||||||
@@ -112,102 +95,10 @@ public class RhinoScriptService implements ScriptService
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see org.alfresco.service.cmr.repository.ScriptService#registerScript(java.lang.Object)
|
* @see org.alfresco.service.cmr.repository.ScriptProcessor#execute(org.alfresco.service.cmr.repository.ScriptLocation, java.util.Map)
|
||||||
*/
|
*/
|
||||||
public void registerScript(ScriptImplementation script)
|
public Object execute(ScriptLocation location, Map<String, Object> model)
|
||||||
{
|
{
|
||||||
this.globalScripts.add(script);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(java.lang.String, java.util.Map)
|
|
||||||
*/
|
|
||||||
public Object executeScript(String scriptClasspath, Map<String, Object> model)
|
|
||||||
throws ScriptException
|
|
||||||
{
|
|
||||||
if (scriptClasspath == null)
|
|
||||||
{
|
|
||||||
throw new IllegalArgumentException("Script ClassPath is mandatory.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (logger.isDebugEnabled())
|
|
||||||
{
|
|
||||||
logger.debug("Executing script: " + scriptClasspath);
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
InputStream stream = getClass().getClassLoader().getResourceAsStream(scriptClasspath);
|
|
||||||
if (stream == null)
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Unable to load classpath resource: " + scriptClasspath);
|
|
||||||
}
|
|
||||||
ByteArrayOutputStream os = new ByteArrayOutputStream();
|
|
||||||
FileCopyUtils.copy(stream, os); // both streams are closed
|
|
||||||
byte[] bytes = os.toByteArray();
|
|
||||||
|
|
||||||
return executeScriptImpl(resolveScriptImports(new String(bytes)), model);
|
|
||||||
}
|
|
||||||
catch (Throwable err)
|
|
||||||
{
|
|
||||||
throw new ScriptException("Failed to execute script '" + scriptClasspath + "': " + err.getMessage(), err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.util.Map)
|
|
||||||
*/
|
|
||||||
public Object executeScript(NodeRef scriptRef, QName contentProp, Map<String, Object> model)
|
|
||||||
throws ScriptException
|
|
||||||
{
|
|
||||||
if (scriptRef == null)
|
|
||||||
{
|
|
||||||
throw new IllegalArgumentException("Script NodeRef is mandatory.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (logger.isDebugEnabled())
|
|
||||||
{
|
|
||||||
logger.debug("Executing script: " + scriptRef.toString());
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (this.services.getNodeService().exists(scriptRef) == false)
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Script Node does not exist: " + scriptRef);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (contentProp == null)
|
|
||||||
{
|
|
||||||
contentProp = ContentModel.PROP_CONTENT;
|
|
||||||
}
|
|
||||||
ContentReader cr = this.services.getContentService().getReader(scriptRef, contentProp);
|
|
||||||
if (cr == null || cr.exists() == false)
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Script Node content not found: " + scriptRef);
|
|
||||||
}
|
|
||||||
|
|
||||||
return executeScriptImpl(resolveScriptImports(cr.getContentString()), model);
|
|
||||||
}
|
|
||||||
catch (Throwable err)
|
|
||||||
{
|
|
||||||
throw new ScriptException("Failed to execute script '" + scriptRef.toString() + "': " + err.getMessage(), err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(org.alfresco.service.cmr.repository.ScriptLocation, java.util.Map)
|
|
||||||
*/
|
|
||||||
public Object executeScript(ScriptLocation location, Map<String, Object> model)
|
|
||||||
throws ScriptException
|
|
||||||
{
|
|
||||||
ParameterCheck.mandatory("Location", location);
|
|
||||||
|
|
||||||
if (logger.isDebugEnabled())
|
|
||||||
{
|
|
||||||
logger.debug("Executing script: " + location.toString());
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
ByteArrayOutputStream os = new ByteArrayOutputStream();
|
ByteArrayOutputStream os = new ByteArrayOutputStream();
|
||||||
@@ -223,21 +114,64 @@ public class RhinoScriptService implements ScriptService
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see org.alfresco.service.cmr.repository.ScriptService#executeScriptString(java.lang.String, java.util.Map)
|
* @see org.alfresco.service.cmr.repository.ScriptProcessor#execute(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.util.Map)
|
||||||
*/
|
*/
|
||||||
public Object executeScriptString(String script, Map<String, Object> model)
|
public Object execute(NodeRef nodeRef, QName contentProp, Map<String, Object> model)
|
||||||
throws ScriptException
|
|
||||||
{
|
{
|
||||||
if (script == null || script.length() == 0)
|
try
|
||||||
{
|
{
|
||||||
throw new IllegalArgumentException("Script argument is mandatory.");
|
if (this.services.getNodeService().exists(nodeRef) == false)
|
||||||
}
|
{
|
||||||
|
throw new AlfrescoRuntimeException("Script Node does not exist: " + nodeRef);
|
||||||
|
}
|
||||||
|
|
||||||
if (logger.isDebugEnabled())
|
if (contentProp == null)
|
||||||
|
{
|
||||||
|
contentProp = ContentModel.PROP_CONTENT;
|
||||||
|
}
|
||||||
|
ContentReader cr = this.services.getContentService().getReader(nodeRef, contentProp);
|
||||||
|
if (cr == null || cr.exists() == false)
|
||||||
|
{
|
||||||
|
throw new AlfrescoRuntimeException("Script Node content not found: " + nodeRef);
|
||||||
|
}
|
||||||
|
|
||||||
|
return executeScriptImpl(resolveScriptImports(cr.getContentString()), model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
{
|
{
|
||||||
logger.debug("Executing script:\n" + script);
|
throw new ScriptException("Failed to execute script '" + nodeRef.toString() + "': " + err.getMessage(), err);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptProcessor#execute(java.lang.String, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object execute(String location, Map<String, Object> model)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
InputStream stream = getClass().getClassLoader().getResourceAsStream(location);
|
||||||
|
if (stream == null)
|
||||||
|
{
|
||||||
|
throw new AlfrescoRuntimeException("Unable to load classpath resource: " + location);
|
||||||
|
}
|
||||||
|
ByteArrayOutputStream os = new ByteArrayOutputStream();
|
||||||
|
FileCopyUtils.copy(stream, os); // both streams are closed
|
||||||
|
byte[] bytes = os.toByteArray();
|
||||||
|
|
||||||
|
return executeScriptImpl(resolveScriptImports(new String(bytes)), model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + location + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptProcessor#executeString(java.lang.String, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeString(String script, Map<String, Object> model)
|
||||||
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return executeScriptImpl(resolveScriptImports(script), model);
|
return executeScriptImpl(resolveScriptImports(script), model);
|
||||||
@@ -248,6 +182,7 @@ public class RhinoScriptService implements ScriptService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Resolve the imports in the specified script. Include directives are of the following form:
|
* Resolve the imports in the specified script. Include directives are of the following form:
|
||||||
* <pre>
|
* <pre>
|
||||||
@@ -453,7 +388,7 @@ public class RhinoScriptService implements ScriptService
|
|||||||
if (resource.startsWith("/"))
|
if (resource.startsWith("/"))
|
||||||
{
|
{
|
||||||
// resolve from default SpacesStore as cm:name based path
|
// resolve from default SpacesStore as cm:name based path
|
||||||
// we have to assume "/Company Home" as the root for now
|
// TODO: remove this once FFS correctly allows name path resolving from store root!
|
||||||
NodeRef rootNodeRef = this.services.getNodeService().getRootNode(this.storeRef);
|
NodeRef rootNodeRef = this.services.getNodeService().getRootNode(this.storeRef);
|
||||||
List<NodeRef> nodes = this.services.getSearchService().selectNodes(
|
List<NodeRef> nodes = this.services.getSearchService().selectNodes(
|
||||||
rootNodeRef, this.storePath, null, this.services.getNamespaceService(), false);
|
rootNodeRef, this.storePath, null, this.services.getNamespaceService(), false);
|
||||||
@@ -516,7 +451,7 @@ public class RhinoScriptService implements ScriptService
|
|||||||
*
|
*
|
||||||
* @throws AlfrescoRuntimeException
|
* @throws AlfrescoRuntimeException
|
||||||
*/
|
*/
|
||||||
private Object executeScriptImpl(String script, Map<String, Object> model)
|
private Object executeScriptImpl(String script, Map<String, Object> origModel)
|
||||||
throws AlfrescoRuntimeException
|
throws AlfrescoRuntimeException
|
||||||
{
|
{
|
||||||
long startTime = 0;
|
long startTime = 0;
|
||||||
@@ -525,6 +460,9 @@ public class RhinoScriptService implements ScriptService
|
|||||||
startTime = System.currentTimeMillis();
|
startTime = System.currentTimeMillis();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Convert the model
|
||||||
|
Map<String, Object> model = convertToRhinoModel(origModel);
|
||||||
|
|
||||||
// check that rhino script engine is available
|
// check that rhino script engine is available
|
||||||
Context cx = Context.enter();
|
Context cx = Context.enter();
|
||||||
try
|
try
|
||||||
@@ -541,9 +479,9 @@ public class RhinoScriptService implements ScriptService
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add the global scripts
|
// add the global scripts
|
||||||
for (ScriptImplementation ex : this.globalScripts)
|
for (ProcessorExtension ex : this.processExtensions.values())
|
||||||
{
|
{
|
||||||
model.put(ex.getScriptName(), ex);
|
model.put(ex.getExtensionName(), ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
// insert supplied object model into root of the default scope
|
// insert supplied object model into root of the default scope
|
||||||
@@ -596,50 +534,25 @@ public class RhinoScriptService implements ScriptService
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create the default data-model available to scripts as global scope level objects:
|
* Converts the passed model into a Rhino model
|
||||||
* <p>
|
|
||||||
* 'companyhome' - the Company Home node<br>
|
|
||||||
* 'userhome' - the current user home space node<br>
|
|
||||||
* 'person' - the node representing the current user Person<br>
|
|
||||||
* 'script' - the node representing the script itself (may not be available)<br>
|
|
||||||
* 'document' - document context node (may not be available)<br>
|
|
||||||
* 'space' - space context node (may not be available)
|
|
||||||
*
|
*
|
||||||
* @param services ServiceRegistry
|
* @param model the model
|
||||||
* @param person The current user Person Node
|
* @return Map<String, Object> the converted model
|
||||||
* @param companyHome The CompanyHome ref
|
|
||||||
* @param userHome The User home space ref
|
|
||||||
* @param script Optional ref to the script itself
|
|
||||||
* @param document Optional ref to a document Node
|
|
||||||
* @param space Optional ref to a space Node
|
|
||||||
* @param resolver Image resolver to resolve icon images etc.
|
|
||||||
*
|
|
||||||
* @return A Map of global scope scriptable Node objects
|
|
||||||
*/
|
*/
|
||||||
public static Map<String, Object> buildDefaultModel(
|
private Map<String, Object> convertToRhinoModel(Map<String, Object> model)
|
||||||
ServiceRegistry services,
|
|
||||||
NodeRef person, NodeRef companyHome, NodeRef userHome,
|
|
||||||
NodeRef script, NodeRef document, NodeRef space)
|
|
||||||
{
|
{
|
||||||
Map<String, Object> model = new HashMap<String, Object>();
|
Map<String, Object> newModel = new HashMap<String, Object>(model.size());
|
||||||
|
for (Map.Entry<String, Object> entry : model.entrySet())
|
||||||
// add the well known node wrapper objects
|
|
||||||
model.put("companyhome", new Node(companyHome, services));
|
|
||||||
model.put("userhome", new Node(userHome, services));
|
|
||||||
model.put("person", new Node(person, services));
|
|
||||||
if (script != null)
|
|
||||||
{
|
{
|
||||||
model.put("script", new Node(script, services));
|
if (entry.getValue() instanceof NodeRef)
|
||||||
|
{
|
||||||
|
newModel.put(entry.getKey(), new Node((NodeRef)entry.getValue(), this.services));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
newModel.put(entry.getKey(), entry.getValue());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (document != null)
|
return newModel;
|
||||||
{
|
|
||||||
model.put("document", new Node(document, services));
|
|
||||||
}
|
|
||||||
if (space != null)
|
|
||||||
{
|
|
||||||
model.put("space", new Node(space, services));
|
|
||||||
}
|
|
||||||
|
|
||||||
return model;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -260,7 +260,7 @@ public class RhinoScriptTest extends TestCase
|
|||||||
scriptService.executeScript(contentNodeRef, BaseNodeServiceTest.PROP_QNAME_TEST_CONTENT, model);
|
scriptService.executeScript(contentNodeRef, BaseNodeServiceTest.PROP_QNAME_TEST_CONTENT, model);
|
||||||
|
|
||||||
// test executing a script directly as a String
|
// test executing a script directly as a String
|
||||||
scriptService.executeScriptString(TESTSCRIPT1, model);
|
scriptService.executeScriptString("javascript", TESTSCRIPT1, model);
|
||||||
}
|
}
|
||||||
catch (Throwable err)
|
catch (Throwable err)
|
||||||
{
|
{
|
||||||
|
@@ -24,12 +24,13 @@
|
|||||||
*/
|
*/
|
||||||
package org.alfresco.repo.jscript;
|
package org.alfresco.repo.jscript;
|
||||||
|
|
||||||
|
import org.alfresco.repo.processor.BaseProcessorExtension;
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public final class ScriptLogger extends BaseScriptImplementation
|
public final class ScriptLogger extends BaseProcessorExtension
|
||||||
{
|
{
|
||||||
private static final Logger logger = Logger.getLogger(ScriptLogger.class);
|
private static final Logger logger = Logger.getLogger(ScriptLogger.class);
|
||||||
|
|
||||||
|
@@ -32,7 +32,7 @@ import org.alfresco.service.cmr.repository.NodeRef;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public final class ScriptUtils extends BaseScopableScriptImplementation
|
public final class ScriptUtils extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
/** Services */
|
/** Services */
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
@@ -54,7 +54,7 @@ import org.dom4j.io.SAXReader;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public final class Search extends BaseScopableScriptImplementation
|
public final class Search extends BaseScopableProcessorExtension
|
||||||
{
|
{
|
||||||
/** Service registry */
|
/** Service registry */
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
@@ -24,6 +24,7 @@
|
|||||||
*/
|
*/
|
||||||
package org.alfresco.repo.jscript;
|
package org.alfresco.repo.jscript;
|
||||||
|
|
||||||
|
import org.alfresco.repo.processor.BaseProcessorExtension;
|
||||||
import org.alfresco.service.ServiceRegistry;
|
import org.alfresco.service.ServiceRegistry;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -33,7 +34,7 @@ import org.alfresco.service.ServiceRegistry;
|
|||||||
*
|
*
|
||||||
* @author Andy Hind
|
* @author Andy Hind
|
||||||
*/
|
*/
|
||||||
public class Session extends BaseScriptImplementation
|
public class Session extends BaseProcessorExtension
|
||||||
{
|
{
|
||||||
/** Service registry */
|
/** Service registry */
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
@@ -6,3 +6,4 @@
|
|||||||
/web/css=/css
|
/web/css=/css
|
||||||
/web/images=/images
|
/web/images=/images
|
||||||
/web/scripts=/scripts
|
/web/scripts=/scripts
|
||||||
|
/web/php=/php
|
158
source/java/org/alfresco/repo/processor/BaseProcessor.java
Normal file
158
source/java/org/alfresco/repo/processor/BaseProcessor.java
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
* As a special exception to the terms and conditions of version 2.0 of
|
||||||
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
||||||
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
||||||
|
* FLOSS exception. You should have recieved a copy of the text describing
|
||||||
|
* the FLOSS exception, and it is also available here:
|
||||||
|
* http://www.alfresco.com/legal/licensing
|
||||||
|
*/
|
||||||
|
package org.alfresco.repo.processor;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.alfresco.service.ServiceRegistry;
|
||||||
|
import org.alfresco.service.cmr.repository.Processor;
|
||||||
|
import org.alfresco.service.cmr.repository.ProcessorExtension;
|
||||||
|
import org.alfresco.service.cmr.repository.ScriptProcessor;
|
||||||
|
import org.alfresco.service.cmr.repository.ScriptService;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateProcessor;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Base class of a processor, encapsulates the implementation reguarding the registration of the processor
|
||||||
|
* with the relevant services and the handling of processor extensions.
|
||||||
|
*
|
||||||
|
* @author Roy Wetherall
|
||||||
|
*/
|
||||||
|
public abstract class BaseProcessor implements Processor
|
||||||
|
{
|
||||||
|
/** The name of the processor */
|
||||||
|
protected String name;
|
||||||
|
|
||||||
|
/** The file extension that this processor understands */
|
||||||
|
protected String extension;
|
||||||
|
|
||||||
|
/** The script service */
|
||||||
|
protected ScriptService scriptService;
|
||||||
|
|
||||||
|
/** The template service */
|
||||||
|
protected TemplateService templateService;
|
||||||
|
|
||||||
|
/** The service registry */
|
||||||
|
protected ServiceRegistry services;
|
||||||
|
|
||||||
|
/** A map containing all the processor extenstions */
|
||||||
|
protected Map<String, ProcessorExtension> processExtensions = new HashMap<String, ProcessorExtension>(10);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers this processor with the relevant services
|
||||||
|
*/
|
||||||
|
public void register()
|
||||||
|
{
|
||||||
|
if (this instanceof ScriptProcessor)
|
||||||
|
{
|
||||||
|
scriptService.registerScriptProcessor((ScriptProcessor)this);
|
||||||
|
}
|
||||||
|
if (this instanceof TemplateProcessor)
|
||||||
|
{
|
||||||
|
templateService.registerTemplateProcessor((TemplateProcessor)this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the script service
|
||||||
|
*
|
||||||
|
* @param scriptService the script service
|
||||||
|
*/
|
||||||
|
public void setScriptService(ScriptService scriptService)
|
||||||
|
{
|
||||||
|
this.scriptService = scriptService;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the template service
|
||||||
|
*
|
||||||
|
* @param templateService the template service
|
||||||
|
*/
|
||||||
|
public void setTemplateService(TemplateService templateService)
|
||||||
|
{
|
||||||
|
this.templateService = templateService;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the service registry
|
||||||
|
*
|
||||||
|
* @param serviceRegistry the service registry
|
||||||
|
*/
|
||||||
|
public void setServiceRegistry(ServiceRegistry serviceRegistry)
|
||||||
|
{
|
||||||
|
this.services = serviceRegistry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the name of the processor
|
||||||
|
*
|
||||||
|
* @return String the name of the processor
|
||||||
|
*/
|
||||||
|
public String getName()
|
||||||
|
{
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the name of the processor
|
||||||
|
*
|
||||||
|
* @param name the name of the processor
|
||||||
|
*/
|
||||||
|
public void setName(String name)
|
||||||
|
{
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the extension that the processor understands
|
||||||
|
*
|
||||||
|
* @return String the extension
|
||||||
|
*/
|
||||||
|
public String getExtension()
|
||||||
|
{
|
||||||
|
return extension;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the extenstion that the processor understands
|
||||||
|
*
|
||||||
|
* @param extension the extension
|
||||||
|
*/
|
||||||
|
public void setExtension(String extension)
|
||||||
|
{
|
||||||
|
this.extension = extension;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a processor extension with the processor
|
||||||
|
*
|
||||||
|
* @param processorExtension the processor extension
|
||||||
|
*/
|
||||||
|
public void registerProcessorExtension(ProcessorExtension processorExtension)
|
||||||
|
{
|
||||||
|
this.processExtensions.put(processorExtension.getExtensionName(), processorExtension);
|
||||||
|
}
|
||||||
|
}
|
@@ -22,32 +22,32 @@
|
|||||||
* the FLOSS exception, and it is also available here:
|
* the FLOSS exception, and it is also available here:
|
||||||
* http://www.alfresco.com/legal/licensing"
|
* http://www.alfresco.com/legal/licensing"
|
||||||
*/
|
*/
|
||||||
package org.alfresco.repo.jscript;
|
package org.alfresco.repo.processor;
|
||||||
|
|
||||||
import org.alfresco.service.cmr.repository.ScriptImplementation;
|
import org.alfresco.service.cmr.repository.Processor;
|
||||||
import org.alfresco.service.cmr.repository.ScriptService;
|
import org.alfresco.service.cmr.repository.ProcessorExtension;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract base class for a script implementation
|
* Abstract base class for a processor extension
|
||||||
*
|
*
|
||||||
* @author Roy Wetherall
|
* @author Roy Wetherall
|
||||||
*/
|
*/
|
||||||
public abstract class BaseScriptImplementation implements ScriptImplementation
|
public abstract class BaseProcessorExtension implements ProcessorExtension
|
||||||
{
|
{
|
||||||
/** The script service */
|
/** The processor */
|
||||||
private ScriptService scriptService;
|
private Processor processor;
|
||||||
|
|
||||||
/** The name of the script */
|
/** The name of the extension */
|
||||||
private String scriptName;
|
private String extensionName;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the script service
|
* Sets the processor
|
||||||
*
|
*
|
||||||
* @param scriptService the script service
|
* @param scriptProcessor the processor
|
||||||
*/
|
*/
|
||||||
public void setScriptService(ScriptService scriptService)
|
public void setProcessor(Processor processor)
|
||||||
{
|
{
|
||||||
this.scriptService = scriptService;
|
this.processor = processor;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -55,24 +55,24 @@ public abstract class BaseScriptImplementation implements ScriptImplementation
|
|||||||
*/
|
*/
|
||||||
public void register()
|
public void register()
|
||||||
{
|
{
|
||||||
this.scriptService.registerScript(this);
|
this.processor.registerProcessorExtension(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the script name
|
* Sets the extension name
|
||||||
*
|
*
|
||||||
* @param scriptName the script name
|
* @param extensionName the extension name
|
||||||
*/
|
*/
|
||||||
public void setScriptName(String scriptName)
|
public void setExtensionName(String extension)
|
||||||
{
|
{
|
||||||
this.scriptName = scriptName;
|
this.extensionName = extension;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see org.alfresco.service.cmr.repository.ScriptImplementation#getScriptName()
|
* @see org.alfresco.service.cmr.repository.ProcessorExtension#getExtensionName()
|
||||||
*/
|
*/
|
||||||
public String getScriptName()
|
public String getExtensionName()
|
||||||
{
|
{
|
||||||
return this.scriptName;
|
return this.extensionName;
|
||||||
}
|
}
|
||||||
}
|
}
|
403
source/java/org/alfresco/repo/processor/ScriptServiceImpl.java
Normal file
403
source/java/org/alfresco/repo/processor/ScriptServiceImpl.java
Normal file
@@ -0,0 +1,403 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
* As a special exception to the terms and conditions of version 2.0 of
|
||||||
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
||||||
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
||||||
|
* FLOSS exception. You should have recieved a copy of the text describing
|
||||||
|
* the FLOSS exception, and it is also available here:
|
||||||
|
* http://www.alfresco.com/legal/licensing"
|
||||||
|
*/
|
||||||
|
package org.alfresco.repo.processor;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.alfresco.model.ContentModel;
|
||||||
|
import org.alfresco.service.cmr.repository.NodeRef;
|
||||||
|
import org.alfresco.service.cmr.repository.NodeService;
|
||||||
|
import org.alfresco.service.cmr.repository.ScriptException;
|
||||||
|
import org.alfresco.service.cmr.repository.ScriptLocation;
|
||||||
|
import org.alfresco.service.cmr.repository.ScriptProcessor;
|
||||||
|
import org.alfresco.service.cmr.repository.ScriptService;
|
||||||
|
import org.alfresco.service.cmr.repository.StoreRef;
|
||||||
|
import org.alfresco.service.namespace.QName;
|
||||||
|
import org.alfresco.util.ParameterCheck;
|
||||||
|
import org.apache.log4j.Logger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Script service implementation
|
||||||
|
*
|
||||||
|
* @author Kevin Roast
|
||||||
|
* @author Roy Wetherall
|
||||||
|
*/
|
||||||
|
public class ScriptServiceImpl implements ScriptService
|
||||||
|
{
|
||||||
|
/** Logger */
|
||||||
|
private static final Logger logger = Logger.getLogger(ScriptServiceImpl.class);
|
||||||
|
|
||||||
|
/** The name of the default script processor */
|
||||||
|
private String defaultScriptProcessor;
|
||||||
|
|
||||||
|
/** Maps containing the script processors */
|
||||||
|
private Map<String, ScriptProcessor> scriptProcessors = new HashMap<String, ScriptProcessor>(5);
|
||||||
|
private Map<String, String> scriptProcessorNamesByExtension = new HashMap<String, String>(5);
|
||||||
|
|
||||||
|
/** The node service */
|
||||||
|
private NodeService nodeService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the name of the default script processor
|
||||||
|
*
|
||||||
|
* @param defaultScriptProcessor the name of the default script processor
|
||||||
|
*/
|
||||||
|
public void setDefaultScriptProcessor(String defaultScriptProcessor)
|
||||||
|
{
|
||||||
|
this.defaultScriptProcessor = defaultScriptProcessor;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the node service
|
||||||
|
*
|
||||||
|
* @param nodeService the node service
|
||||||
|
*/
|
||||||
|
public void setNodeService(NodeService nodeService)
|
||||||
|
{
|
||||||
|
this.nodeService = nodeService;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a script processor
|
||||||
|
*
|
||||||
|
* @param scriptProcessor the script processor to register with the script service
|
||||||
|
*/
|
||||||
|
public void registerScriptProcessor(ScriptProcessor scriptProcessor)
|
||||||
|
{
|
||||||
|
this.scriptProcessors.put(scriptProcessor.getName(), scriptProcessor);
|
||||||
|
this.scriptProcessorNamesByExtension.put(scriptProcessor.getExtension(), scriptProcessor.getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(java.lang.String, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScript(String scriptClasspath, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
if (scriptClasspath == null)
|
||||||
|
{
|
||||||
|
throw new IllegalArgumentException("Script ClassPath is mandatory.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script: " + scriptClasspath);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = getScriptProcessor(scriptClasspath);
|
||||||
|
return scriptProcessor.execute(scriptClasspath, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + scriptClasspath + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(java.lang.String, java.lang.String, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScript(String engine, String scriptClasspath, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
if (scriptClasspath == null)
|
||||||
|
{
|
||||||
|
throw new IllegalArgumentException("Script ClassPath is mandatory.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script: " + scriptClasspath);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = lookupScriptProcessor(engine);
|
||||||
|
return scriptProcessor.execute(scriptClasspath, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + scriptClasspath + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScript(NodeRef scriptRef, QName contentProp, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
if (scriptRef == null)
|
||||||
|
{
|
||||||
|
throw new IllegalArgumentException("Script NodeRef is mandatory.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script: " + scriptRef.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = getScriptProcessor(scriptRef);
|
||||||
|
return scriptProcessor.execute(scriptRef, contentProp, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + scriptRef.toString() + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(java.lang.String, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScript(String engine, NodeRef scriptRef, QName contentProp, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
if (scriptRef == null)
|
||||||
|
{
|
||||||
|
throw new IllegalArgumentException("Script NodeRef is mandatory.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script: " + scriptRef.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = lookupScriptProcessor(engine);
|
||||||
|
return scriptProcessor.execute(scriptRef, contentProp, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + scriptRef.toString() + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(org.alfresco.service.cmr.repository.ScriptLocation, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScript(ScriptLocation location, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
ParameterCheck.mandatory("Location", location);
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script: " + location.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = getScriptProcessor(location.toString());
|
||||||
|
return scriptProcessor.execute(location, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + location.toString() + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScript(java.lang.String, org.alfresco.service.cmr.repository.ScriptLocation, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScript(String engine, ScriptLocation location, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
ParameterCheck.mandatory("Location", location);
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script: " + location.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = lookupScriptProcessor(engine);
|
||||||
|
return scriptProcessor.execute(location, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute script '" + location.toString() + "': " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScriptString(java.lang.String, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScriptString(String script, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
return executeScriptString(this.defaultScriptProcessor, script, model);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#executeScriptString(java.lang.String, java.util.Map)
|
||||||
|
*/
|
||||||
|
public Object executeScriptString(String engine, String script, Map<String, Object> model)
|
||||||
|
throws ScriptException
|
||||||
|
{
|
||||||
|
if (script == null || script.length() == 0)
|
||||||
|
{
|
||||||
|
throw new IllegalArgumentException("Script argument is mandatory.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled())
|
||||||
|
{
|
||||||
|
logger.debug("Executing script:\n" + script);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = lookupScriptProcessor(engine);
|
||||||
|
return scriptProcessor.executeString(script, model);
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new ScriptException("Failed to execute supplied script: " + err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper method to lookup the script proecessor based on a name
|
||||||
|
*
|
||||||
|
* @param name the name of the script processor
|
||||||
|
* @return ScriptProcessor the script processor, default processor if no match found
|
||||||
|
*/
|
||||||
|
protected ScriptProcessor lookupScriptProcessor(String name)
|
||||||
|
{
|
||||||
|
ScriptProcessor scriptProcessor = this.scriptProcessors.get(name);
|
||||||
|
if (scriptProcessor == null)
|
||||||
|
{
|
||||||
|
scriptProcessor = this.scriptProcessors.get(this.defaultScriptProcessor);
|
||||||
|
}
|
||||||
|
return scriptProcessor;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets a scipt processor based on the node reference of a script
|
||||||
|
*
|
||||||
|
* @param scriptNode the node reference of the script
|
||||||
|
* @return ScriptProcessor the script processor
|
||||||
|
*/
|
||||||
|
protected ScriptProcessor getScriptProcessor(NodeRef scriptNode)
|
||||||
|
{
|
||||||
|
String scriptName = (String)this.nodeService.getProperty(scriptNode, ContentModel.PROP_NAME);
|
||||||
|
return getScriptProcessorImpl(scriptName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets a script processor based on the script location string
|
||||||
|
*
|
||||||
|
* @param scriptLocation the script location
|
||||||
|
* @return ScriptProcessor the script processor
|
||||||
|
*/
|
||||||
|
protected ScriptProcessor getScriptProcessor(String scriptLocation)
|
||||||
|
{
|
||||||
|
if (scriptLocation.indexOf(StoreRef.URI_FILLER) != -1)
|
||||||
|
{
|
||||||
|
// Try and create the nodeRef
|
||||||
|
NodeRef nodeRef = new NodeRef(scriptLocation);
|
||||||
|
scriptLocation = (String)this.nodeService.getProperty(nodeRef, ContentModel.PROP_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
return getScriptProcessorImpl(scriptLocation);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets a script processor based on the scripts file name
|
||||||
|
*
|
||||||
|
* @param scriptFileName the scripts file name
|
||||||
|
* @return ScriptProcessor the matching script processor
|
||||||
|
*/
|
||||||
|
protected ScriptProcessor getScriptProcessorImpl(String scriptFileName)
|
||||||
|
{
|
||||||
|
String engine = null;
|
||||||
|
|
||||||
|
if (scriptFileName != null)
|
||||||
|
{
|
||||||
|
String extension = getFileExtension(scriptFileName);
|
||||||
|
if (extension != null)
|
||||||
|
{
|
||||||
|
engine = this.scriptProcessorNamesByExtension.get(extension);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return lookupScriptProcessor(engine);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the file extension of a file
|
||||||
|
*
|
||||||
|
* @param fileName the file name
|
||||||
|
* @return the file extension
|
||||||
|
*/
|
||||||
|
private String getFileExtension(String fileName)
|
||||||
|
{
|
||||||
|
String extension = null;
|
||||||
|
int index = fileName.lastIndexOf('.');
|
||||||
|
if (index > -1 && (index < fileName.length() - 1))
|
||||||
|
{
|
||||||
|
extension = fileName.substring(index + 1);
|
||||||
|
}
|
||||||
|
return extension;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.ScriptService#buildDefaultModel(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef)
|
||||||
|
*/
|
||||||
|
public Map<String, Object> buildDefaultModel(
|
||||||
|
NodeRef person,
|
||||||
|
NodeRef companyHome,
|
||||||
|
NodeRef userHome,
|
||||||
|
NodeRef script,
|
||||||
|
NodeRef document,
|
||||||
|
NodeRef space)
|
||||||
|
{
|
||||||
|
Map<String, Object> model = new HashMap<String, Object>();
|
||||||
|
|
||||||
|
// add the well known node wrapper objects
|
||||||
|
model.put("companyhome", companyHome);
|
||||||
|
model.put("userhome", userHome);
|
||||||
|
model.put("person", person);
|
||||||
|
if (script != null)
|
||||||
|
{
|
||||||
|
model.put("script", script);
|
||||||
|
}
|
||||||
|
if (document != null)
|
||||||
|
{
|
||||||
|
model.put("document", document);
|
||||||
|
}
|
||||||
|
if (space != null)
|
||||||
|
{
|
||||||
|
model.put("space", space);
|
||||||
|
}
|
||||||
|
|
||||||
|
return model;
|
||||||
|
}
|
||||||
|
}
|
262
source/java/org/alfresco/repo/processor/TemplateServiceImpl.java
Normal file
262
source/java/org/alfresco/repo/processor/TemplateServiceImpl.java
Normal file
@@ -0,0 +1,262 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
* As a special exception to the terms and conditions of version 2.0 of
|
||||||
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
||||||
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
||||||
|
* FLOSS exception. You should have recieved a copy of the text describing
|
||||||
|
* the FLOSS exception, and it is also available here:
|
||||||
|
* http://www.alfresco.com/legal/licensing"
|
||||||
|
*/
|
||||||
|
package org.alfresco.repo.processor;
|
||||||
|
|
||||||
|
import java.io.StringWriter;
|
||||||
|
import java.io.Writer;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.alfresco.error.AlfrescoRuntimeException;
|
||||||
|
import org.alfresco.model.ContentModel;
|
||||||
|
import org.alfresco.service.cmr.repository.NodeRef;
|
||||||
|
import org.alfresco.service.cmr.repository.NodeService;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateException;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateProcessor;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implementation of the TemplateService using Spring configured script engines.
|
||||||
|
*
|
||||||
|
* @author Kevin Roast
|
||||||
|
*/
|
||||||
|
public class TemplateServiceImpl implements TemplateService
|
||||||
|
{
|
||||||
|
/** Default Template processor engine to use */
|
||||||
|
private String defaultTemplateEngine;
|
||||||
|
|
||||||
|
/** List of available template processors */
|
||||||
|
private Map<String, TemplateProcessor> processors = new HashMap<String, TemplateProcessor>(5);
|
||||||
|
private Map<String, String> processorNamesByExtension = new HashMap<String, String>(5);
|
||||||
|
|
||||||
|
/** The node service */
|
||||||
|
private NodeService nodeService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param defaultTemplateEngine The default Template Engine name to set.
|
||||||
|
*/
|
||||||
|
public void setDefaultTemplateEngine(String defaultTemplateEngine)
|
||||||
|
{
|
||||||
|
this.defaultTemplateEngine = defaultTemplateEngine;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the node service
|
||||||
|
*
|
||||||
|
* @param nodeService the node service
|
||||||
|
*/
|
||||||
|
public void setNodeService(NodeService nodeService)
|
||||||
|
{
|
||||||
|
this.nodeService = nodeService;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#getTemplateProcessor(java.lang.String)
|
||||||
|
*/
|
||||||
|
public TemplateProcessor getTemplateProcessor(String engine)
|
||||||
|
{
|
||||||
|
return this.processors.get(engine);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#registerTemplateProcessor(org.alfresco.service.cmr.repository.TemplateProcessor)
|
||||||
|
*/
|
||||||
|
public void registerTemplateProcessor(TemplateProcessor templateProcessor)
|
||||||
|
{
|
||||||
|
this.processors.put(templateProcessor.getName(), templateProcessor);
|
||||||
|
this.processorNamesByExtension.put(templateProcessor.getExtension(), templateProcessor.getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplate(java.lang.String, java.lang.Object)
|
||||||
|
*/
|
||||||
|
public String processTemplate(String template, Object model) throws TemplateException
|
||||||
|
{
|
||||||
|
return processTemplate(getTemplateProcessorName(template), template, model);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplate(java.lang.String, java.lang.Object, java.io.Writer)
|
||||||
|
*/
|
||||||
|
public void processTemplate(String template, Object model, Writer out) throws TemplateException
|
||||||
|
{
|
||||||
|
processTemplate(getTemplateProcessorName(template), template, model, out);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the template processor name from the template string
|
||||||
|
*
|
||||||
|
* @param template the template string location
|
||||||
|
* @return the template processor string
|
||||||
|
*/
|
||||||
|
private String getTemplateProcessorName(String template)
|
||||||
|
{
|
||||||
|
String engine = null;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Try and create the nodeRef
|
||||||
|
NodeRef templateNodeRef = new NodeRef(template);
|
||||||
|
String templateName = (String)this.nodeService.getProperty(templateNodeRef, ContentModel.PROP_NAME);
|
||||||
|
String extension = getFileExtension(templateName);
|
||||||
|
if (extension != null)
|
||||||
|
{
|
||||||
|
engine = this.processorNamesByExtension.get(extension);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (AlfrescoRuntimeException exception)
|
||||||
|
{
|
||||||
|
// Presume that the provided template is a classpath
|
||||||
|
String extension = getFileExtension(template);
|
||||||
|
if (extension != null)
|
||||||
|
{
|
||||||
|
engine = this.processorNamesByExtension.get(extension);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the default engine if none found
|
||||||
|
if (engine == null)
|
||||||
|
{
|
||||||
|
engine = this.defaultTemplateEngine;
|
||||||
|
}
|
||||||
|
|
||||||
|
return engine;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the file extension of a file
|
||||||
|
*
|
||||||
|
* @param fileName the file name
|
||||||
|
* @return the file extension
|
||||||
|
*/
|
||||||
|
private String getFileExtension(String fileName)
|
||||||
|
{
|
||||||
|
String extension = null;
|
||||||
|
int index = fileName.lastIndexOf('.');
|
||||||
|
if (index > -1 && (index < fileName.length() - 1))
|
||||||
|
{
|
||||||
|
extension = fileName.substring(index + 1);
|
||||||
|
}
|
||||||
|
return extension;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplate(java.lang.String, java.lang.String, java.lang.Object, java.io.Writer)
|
||||||
|
*/
|
||||||
|
public void processTemplate(String engine, String template, Object model, Writer out)
|
||||||
|
throws TemplateException
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// execute template processor
|
||||||
|
TemplateProcessor processor = getTemplateProcessor(engine);
|
||||||
|
processor.process(template, model, out);
|
||||||
|
}
|
||||||
|
catch (TemplateException terr)
|
||||||
|
{
|
||||||
|
throw terr;
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new TemplateException(err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplate(java.lang.String, java.lang.String, java.lang.Object)
|
||||||
|
*/
|
||||||
|
public String processTemplate(String engine, String template, Object model)
|
||||||
|
throws TemplateException
|
||||||
|
{
|
||||||
|
Writer out = new StringWriter(1024);
|
||||||
|
processTemplate(engine, template, model, out);
|
||||||
|
return out.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplateString(java.lang.String, java.lang.String, java.lang.Object, java.io.Writer)
|
||||||
|
*/
|
||||||
|
public void processTemplateString(String engine, String template, Object model, Writer out)
|
||||||
|
throws TemplateException
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// execute template processor
|
||||||
|
TemplateProcessor processor = getTemplateProcessor(engine);
|
||||||
|
processor.processString(template, model, out);
|
||||||
|
}
|
||||||
|
catch (TemplateException terr)
|
||||||
|
{
|
||||||
|
throw terr;
|
||||||
|
}
|
||||||
|
catch (Throwable err)
|
||||||
|
{
|
||||||
|
throw new TemplateException(err.getMessage(), err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplateString(java.lang.String, java.lang.String, java.lang.Object)
|
||||||
|
*/
|
||||||
|
public String processTemplateString(String engine, String template, Object model)
|
||||||
|
throws TemplateException
|
||||||
|
{
|
||||||
|
Writer out = new StringWriter(1024);
|
||||||
|
processTemplateString(engine, template, model, out);
|
||||||
|
return out.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see org.alfresco.service.cmr.repository.TemplateService#buildDefaultModel(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.TemplateImageResolver)
|
||||||
|
*/
|
||||||
|
public Map<String, Object> buildDefaultModel(NodeRef person, NodeRef companyHome, NodeRef userHome, NodeRef template, TemplateImageResolver imageResolver)
|
||||||
|
{
|
||||||
|
Map<String, Object> model = new HashMap<String, Object>(16, 1.0f);
|
||||||
|
|
||||||
|
// Place the image resolver into the model
|
||||||
|
if (imageResolver != null)
|
||||||
|
{
|
||||||
|
model.put(KEY_IMAGE_RESOLVER, imageResolver);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Put the common node reference into the model
|
||||||
|
model.put(KEY_COMPANY_HOME, companyHome);
|
||||||
|
model.put(KEY_USER_HOME, userHome);
|
||||||
|
model.put(KEY_PERSON, person);
|
||||||
|
|
||||||
|
// add the template itself as "template" if it comes from content on a node
|
||||||
|
if (template != null)
|
||||||
|
{
|
||||||
|
model.put(KEY_TEMPLATE, template);
|
||||||
|
}
|
||||||
|
|
||||||
|
// current date/time is useful to have and isn't supplied by FreeMarker by default
|
||||||
|
model.put(KEY_DATE, new Date());
|
||||||
|
|
||||||
|
return model;
|
||||||
|
}
|
||||||
|
}
|
@@ -34,7 +34,7 @@ import org.alfresco.service.cmr.avm.AVMStoreDescriptor;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public class AVM extends BaseTemplateExtensionImplementation
|
public class AVM extends BaseTemplateProcessorExtension
|
||||||
{
|
{
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
|
||||||
|
@@ -24,37 +24,22 @@
|
|||||||
*/
|
*/
|
||||||
package org.alfresco.repo.template;
|
package org.alfresco.repo.template;
|
||||||
|
|
||||||
import org.alfresco.service.cmr.repository.TemplateExtensionImplementation;
|
import org.alfresco.repo.processor.BaseProcessorExtension;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateProcessorExtension;
|
||||||
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
||||||
import org.alfresco.service.cmr.repository.TemplateService;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract base class for a template extension implementation
|
* Abstract base class for a template extension implementation
|
||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public abstract class BaseTemplateExtensionImplementation implements TemplateExtensionImplementation
|
public abstract class BaseTemplateProcessorExtension extends BaseProcessorExtension implements TemplateProcessorExtension
|
||||||
{
|
{
|
||||||
/** The template service instance */
|
|
||||||
private TemplateService templateService;
|
|
||||||
|
|
||||||
/** The name of the template extension */
|
|
||||||
private String extensionName;
|
|
||||||
|
|
||||||
/** The TemplateImageResolver for the current template execution thread */
|
/** The TemplateImageResolver for the current template execution thread */
|
||||||
private ThreadLocal<TemplateImageResolver> resolver = new ThreadLocal<TemplateImageResolver>();
|
private ThreadLocal<TemplateImageResolver> resolver = new ThreadLocal<TemplateImageResolver>();
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param templateService The TemplateService to set.
|
* @see org.alfresco.service.cmr.repository.TemplateProcessorExtension#setTemplateImageResolver(org.alfresco.service.cmr.repository.TemplateImageResolver)
|
||||||
*/
|
|
||||||
public void setTemplateService(TemplateService templateService)
|
|
||||||
{
|
|
||||||
this.templateService = templateService;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateExtensionImplementation#setTemplateImageResolver(org.alfresco.service.cmr.repository.TemplateImageResolver)
|
|
||||||
*/
|
*/
|
||||||
public void setTemplateImageResolver(TemplateImageResolver resolver)
|
public void setTemplateImageResolver(TemplateImageResolver resolver)
|
||||||
{
|
{
|
||||||
@@ -62,36 +47,10 @@ public abstract class BaseTemplateExtensionImplementation implements TemplateExt
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateExtensionImplementation#getTemplateImageResolver()
|
* @see org.alfresco.service.cmr.repository.TemplateProcessorExtension#getTemplateImageResolver()
|
||||||
*/
|
*/
|
||||||
public TemplateImageResolver getTemplateImageResolver()
|
public TemplateImageResolver getTemplateImageResolver()
|
||||||
{
|
{
|
||||||
return this.resolver.get();
|
return this.resolver.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Registers this template extension with the Template Service
|
|
||||||
*/
|
|
||||||
public void register()
|
|
||||||
{
|
|
||||||
this.templateService.registerExtension(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the name of the template extension
|
|
||||||
*
|
|
||||||
* @return the name of the template extension
|
|
||||||
*/
|
|
||||||
public String getExtensionName()
|
|
||||||
{
|
|
||||||
return extensionName;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param extensionName The template extension name.
|
|
||||||
*/
|
|
||||||
public void setExtensionName(String extensionName)
|
|
||||||
{
|
|
||||||
this.extensionName = extensionName;
|
|
||||||
}
|
|
||||||
}
|
}
|
@@ -40,7 +40,7 @@ import org.alfresco.service.namespace.QName;
|
|||||||
*
|
*
|
||||||
* @author Andy Hind
|
* @author Andy Hind
|
||||||
*/
|
*/
|
||||||
public class Classification extends BaseTemplateExtensionImplementation
|
public class Classification extends BaseTemplateProcessorExtension
|
||||||
{
|
{
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
private StoreRef storeRef;
|
private StoreRef storeRef;
|
||||||
|
@@ -43,7 +43,7 @@ import freemarker.template.TemplateNumberModel;
|
|||||||
* dateCompare(dateA, dateB) - 1 if dateA if greater than dateB
|
* dateCompare(dateA, dateB) - 1 if dateA if greater than dateB
|
||||||
* dateCompare(dateA, dateB, millis) - 1 if dateA is greater than dateB by at least millis, else 0
|
* dateCompare(dateA, dateB, millis) - 1 if dateA is greater than dateB by at least millis, else 0
|
||||||
*/
|
*/
|
||||||
public class DateCompareMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class DateCompareMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -35,7 +35,7 @@ import freemarker.template.TemplateNumberModel;
|
|||||||
/**
|
/**
|
||||||
* @author Roy Wetherall
|
* @author Roy Wetherall
|
||||||
*/
|
*/
|
||||||
public class DateIncrementMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class DateIncrementMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -26,18 +26,17 @@ package org.alfresco.repo.template;
|
|||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.Writer;
|
import java.io.Writer;
|
||||||
import java.util.Date;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.alfresco.service.ServiceRegistry;
|
import org.alfresco.repo.processor.BaseProcessor;
|
||||||
import org.alfresco.service.cmr.repository.ContentService;
|
|
||||||
import org.alfresco.service.cmr.repository.NodeRef;
|
import org.alfresco.service.cmr.repository.NodeRef;
|
||||||
import org.alfresco.service.cmr.repository.NodeService;
|
import org.alfresco.service.cmr.repository.ProcessorExtension;
|
||||||
import org.alfresco.service.cmr.repository.TemplateException;
|
import org.alfresco.service.cmr.repository.TemplateException;
|
||||||
import org.alfresco.service.cmr.repository.TemplateExtensionImplementation;
|
|
||||||
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
||||||
import org.alfresco.service.cmr.repository.TemplateProcessor;
|
import org.alfresco.service.cmr.repository.TemplateProcessor;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateProcessorExtension;
|
||||||
|
import org.alfresco.service.cmr.repository.TemplateService;
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
|
|
||||||
import freemarker.cache.MruCacheStorage;
|
import freemarker.cache.MruCacheStorage;
|
||||||
@@ -61,7 +60,7 @@ import freemarker.template.TemplateExceptionHandler;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public class FreeMarkerProcessor implements TemplateProcessor
|
public class FreeMarkerProcessor extends BaseProcessor implements TemplateProcessor
|
||||||
{
|
{
|
||||||
private final static String MSG_ERROR_NO_TEMPLATE = "error_no_template";
|
private final static String MSG_ERROR_NO_TEMPLATE = "error_no_template";
|
||||||
private final static String MSG_ERROR_TEMPLATE_FAIL = "error_template_fail";
|
private final static String MSG_ERROR_TEMPLATE_FAIL = "error_template_fail";
|
||||||
@@ -72,35 +71,9 @@ public class FreeMarkerProcessor implements TemplateProcessor
|
|||||||
/** Pseudo path to String based template */
|
/** Pseudo path to String based template */
|
||||||
private static final String PATH = "string://fixed";
|
private static final String PATH = "string://fixed";
|
||||||
|
|
||||||
/** The permission-safe node service */
|
|
||||||
private NodeService nodeService;
|
|
||||||
|
|
||||||
/** The Content Service to use */
|
|
||||||
private ContentService contentService;
|
|
||||||
|
|
||||||
/** Template encoding */
|
/** Template encoding */
|
||||||
private String defaultEncoding;
|
private String defaultEncoding;
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the node service
|
|
||||||
*
|
|
||||||
* @param nodeService The permission-safe node service
|
|
||||||
*/
|
|
||||||
public void setNodeService(NodeService nodeService)
|
|
||||||
{
|
|
||||||
this.nodeService = nodeService;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the content service
|
|
||||||
*
|
|
||||||
* @param contentService The ContentService to use
|
|
||||||
*/
|
|
||||||
public void setContentService(ContentService contentService)
|
|
||||||
{
|
|
||||||
this.contentService = contentService;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the default template encoding
|
* Set the default template encoding
|
||||||
*
|
*
|
||||||
@@ -124,7 +97,7 @@ public class FreeMarkerProcessor implements TemplateProcessor
|
|||||||
config.setCacheStorage(new MruCacheStorage(2, 0));
|
config.setCacheStorage(new MruCacheStorage(2, 0));
|
||||||
|
|
||||||
// use our custom loader to find templates on the ClassPath
|
// use our custom loader to find templates on the ClassPath
|
||||||
config.setTemplateLoader(new ClassPathRepoTemplateLoader(nodeService, contentService));
|
config.setTemplateLoader(new ClassPathRepoTemplateLoader(this.services.getNodeService(), this.services.getContentService()));
|
||||||
|
|
||||||
// use our custom object wrapper that can deal with QNameMap objects directly
|
// use our custom object wrapper that can deal with QNameMap objects directly
|
||||||
config.setObjectWrapper(new QNameAwareObjectWrapper());
|
config.setObjectWrapper(new QNameAwareObjectWrapper());
|
||||||
@@ -213,7 +186,8 @@ public class FreeMarkerProcessor implements TemplateProcessor
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
// perform the template processing against supplied data model
|
// perform the template processing against supplied data model
|
||||||
t.process(model, out);
|
Object freeMarkerModel = convertToFreeMarkerModel(model);
|
||||||
|
t.process(freeMarkerModel, out);
|
||||||
}
|
}
|
||||||
catch (Throwable err)
|
catch (Throwable err)
|
||||||
{
|
{
|
||||||
@@ -270,7 +244,8 @@ public class FreeMarkerProcessor implements TemplateProcessor
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
// perform the template processing against supplied data model
|
// perform the template processing against supplied data model
|
||||||
t.process(model, out);
|
Object freeMarkerModel = convertToFreeMarkerModel(model);
|
||||||
|
t.process(freeMarkerModel, out);
|
||||||
|
|
||||||
if (logger.isDebugEnabled())
|
if (logger.isDebugEnabled())
|
||||||
{
|
{
|
||||||
@@ -294,58 +269,52 @@ public class FreeMarkerProcessor implements TemplateProcessor
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
private Object convertToFreeMarkerModel(Object model)
|
||||||
* Create the default data-model available to templates as global objects.
|
|
||||||
* <p>
|
|
||||||
* 'companyhome' - the Company Home node<br>
|
|
||||||
* 'userhome' - the current user home space node<br>
|
|
||||||
* 'person' - the node representing the current user Person<br>
|
|
||||||
* 'template' - the node representing the template itself (may not be available)
|
|
||||||
* <p>
|
|
||||||
* Also adds various helper util objects and methods.
|
|
||||||
*
|
|
||||||
* @param services ServiceRegistry
|
|
||||||
* @param person The current user Person Node
|
|
||||||
* @param companyHome The CompanyHome ref
|
|
||||||
* @param userHome The User home space ref
|
|
||||||
* @param template Optional ref to the template itself
|
|
||||||
* @param resolver Image resolver to resolve icon images etc.
|
|
||||||
*
|
|
||||||
* @return A Map of Templatable Node objects and util objects.
|
|
||||||
*/
|
|
||||||
public static Map<String, Object> buildDefaultModel(
|
|
||||||
ServiceRegistry services,
|
|
||||||
NodeRef person, NodeRef companyHome, NodeRef userHome, NodeRef template,
|
|
||||||
TemplateImageResolver imageResolver)
|
|
||||||
{
|
{
|
||||||
Map<String, Object> model = new HashMap<String, Object>(16, 1.0f);
|
// If we dont have a map in our hand we just return the passes model
|
||||||
|
if (model instanceof Map)
|
||||||
// supply the Company Home space as "companyhome"
|
|
||||||
model.put("companyhome", new TemplateNode(companyHome, services, imageResolver));
|
|
||||||
|
|
||||||
// supply the users Home Space as "userhome"
|
|
||||||
model.put("userhome", new TemplateNode(userHome, services, imageResolver));
|
|
||||||
|
|
||||||
// supply the current user Node as "person"
|
|
||||||
model.put("person", new TemplateNode(person, services, imageResolver));
|
|
||||||
|
|
||||||
// add the template itself as "template" if it comes from content on a node
|
|
||||||
if (template != null)
|
|
||||||
{
|
{
|
||||||
model.put("template", new TemplateNode(template, services, imageResolver));
|
Map<String, Object> freeMarkerModel = new HashMap<String, Object>(((Map)model).size());
|
||||||
|
|
||||||
|
// Look for the image resolver in the model
|
||||||
|
TemplateImageResolver imageResolver = (TemplateImageResolver)((Map)model).get(TemplateService.KEY_IMAGE_RESOLVER);
|
||||||
|
|
||||||
|
for (Object objKey : ((Map)model).keySet())
|
||||||
|
{
|
||||||
|
String key = (String)objKey;
|
||||||
|
if (key.equals(TemplateService.KEY_IMAGE_RESOLVER) == false)
|
||||||
|
{
|
||||||
|
Object value = ((Map)model).get(key);
|
||||||
|
if (value instanceof NodeRef)
|
||||||
|
{
|
||||||
|
// Concer the node reference to a template node
|
||||||
|
freeMarkerModel.put(key, new TemplateNode((NodeRef)value, this.services, imageResolver));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Just add the objec to the free marker model
|
||||||
|
freeMarkerModel.put(key, ((Map)model).get(key));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// add the template extensions to the model
|
||||||
|
// the extensions include custom root helper objects and custom template method objects
|
||||||
|
for (ProcessorExtension ext : this.processExtensions.values())
|
||||||
|
{
|
||||||
|
if (ext instanceof TemplateProcessorExtension)
|
||||||
|
{
|
||||||
|
((TemplateProcessorExtension)ext).setTemplateImageResolver(imageResolver);
|
||||||
|
}
|
||||||
|
freeMarkerModel.put(ext.getExtensionName(), ext);
|
||||||
|
}
|
||||||
|
|
||||||
|
return freeMarkerModel;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
// current date/time is useful to have and isn't supplied by FreeMarker by default
|
|
||||||
model.put("date", new Date());
|
|
||||||
|
|
||||||
// add the template extensions to the model
|
|
||||||
// the extensions include custom root helper objects and custom template method objects
|
|
||||||
for (TemplateExtensionImplementation ext : services.getTemplateService().getExtensions())
|
|
||||||
{
|
{
|
||||||
ext.setTemplateImageResolver(imageResolver);
|
return model;
|
||||||
model.put(ext.getExtensionName(), ext);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return model;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -42,7 +42,7 @@ import freemarker.template.TemplateScalarModel;
|
|||||||
* <p>
|
* <p>
|
||||||
* Usage: hasAspect(TemplateNode node, String aspect) - 1 on true, 0 on false
|
* Usage: hasAspect(TemplateNode node, String aspect) - 1 on true, 0 on false
|
||||||
*/
|
*/
|
||||||
public class HasAspectMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class HasAspectMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -41,7 +41,7 @@ import freemarker.template.TemplateScalarModel;
|
|||||||
* <p>
|
* <p>
|
||||||
* Usage: hasPermission(TemplateNode node, String permission) - 1 on true, 0 on false
|
* Usage: hasPermission(TemplateNode node, String permission) - 1 on true, 0 on false
|
||||||
*/
|
*/
|
||||||
public class HasPermissionMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class HasPermissionMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -41,7 +41,7 @@ import freemarker.template.TemplateScalarModel;
|
|||||||
* <p>
|
* <p>
|
||||||
* Usage: message(String id)
|
* Usage: message(String id)
|
||||||
*/
|
*/
|
||||||
public class I18NMessageMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class I18NMessageMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -41,7 +41,7 @@ import freemarker.template.TemplateModelException;
|
|||||||
* <p>
|
* <p>
|
||||||
* Usage: xmldate(Date date)
|
* Usage: xmldate(Date date)
|
||||||
*/
|
*/
|
||||||
public class ISO8601DateFormatMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class ISO8601DateFormatMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -32,7 +32,7 @@ import org.alfresco.service.cmr.repository.TemplateImageResolver;
|
|||||||
*
|
*
|
||||||
* @author Andy Hind
|
* @author Andy Hind
|
||||||
*/
|
*/
|
||||||
public class Session extends BaseTemplateExtensionImplementation
|
public class Session extends BaseTemplateProcessorExtension
|
||||||
{
|
{
|
||||||
private ServiceRegistry services;
|
private ServiceRegistry services;
|
||||||
|
|
||||||
|
@@ -1,271 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU General Public License
|
|
||||||
* as published by the Free Software Foundation; either version 2
|
|
||||||
* of the License, or (at your option) any later version.
|
|
||||||
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
||||||
|
|
||||||
* As a special exception to the terms and conditions of version 2.0 of
|
|
||||||
* the GPL, you may redistribute this Program in connection with Free/Libre
|
|
||||||
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
|
||||||
* FLOSS exception. You should have recieved a copy of the text describing
|
|
||||||
* the FLOSS exception, and it is also available here:
|
|
||||||
* http://www.alfresco.com/legal/licensing"
|
|
||||||
*/
|
|
||||||
package org.alfresco.repo.template;
|
|
||||||
|
|
||||||
import java.io.StringWriter;
|
|
||||||
import java.io.Writer;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
import org.alfresco.error.AlfrescoRuntimeException;
|
|
||||||
import org.alfresco.service.cmr.repository.ScriptImplementation;
|
|
||||||
import org.alfresco.service.cmr.repository.TemplateException;
|
|
||||||
import org.alfresco.service.cmr.repository.TemplateExtensionImplementation;
|
|
||||||
import org.alfresco.service.cmr.repository.TemplateProcessor;
|
|
||||||
import org.alfresco.service.cmr.repository.TemplateService;
|
|
||||||
import org.apache.commons.logging.Log;
|
|
||||||
import org.apache.commons.logging.LogFactory;
|
|
||||||
import org.springframework.beans.BeansException;
|
|
||||||
import org.springframework.context.ApplicationContext;
|
|
||||||
import org.springframework.context.ApplicationContextAware;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Implementation of the TemplateService using Spring configured template engines.
|
|
||||||
*
|
|
||||||
* @author Kevin Roast
|
|
||||||
*/
|
|
||||||
public class TemplateServiceImpl implements TemplateService, ApplicationContextAware
|
|
||||||
{
|
|
||||||
private static Log logger = LogFactory.getLog(TemplateService.class);
|
|
||||||
|
|
||||||
/** Spring ApplicationContext for bean lookup by ID */
|
|
||||||
private ApplicationContext applicationContext;
|
|
||||||
|
|
||||||
/** Default Template processor engine to use */
|
|
||||||
private String defaultTemplateEngine;
|
|
||||||
|
|
||||||
/** Available template engine names to impl class names */
|
|
||||||
private Map<String, String> templateEngines;
|
|
||||||
|
|
||||||
/** Threadlocal instance for template processor cache */
|
|
||||||
private static ThreadLocal<Map<String, TemplateProcessor>> processors = new ThreadLocal<Map<String, TemplateProcessor>>();
|
|
||||||
|
|
||||||
/** List of global template extensions */
|
|
||||||
private List<TemplateExtensionImplementation> globalExtensions = new ArrayList<TemplateExtensionImplementation>();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the application context
|
|
||||||
*
|
|
||||||
* @param applicationContext the application context
|
|
||||||
*/
|
|
||||||
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException
|
|
||||||
{
|
|
||||||
this.applicationContext = applicationContext;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#registerExtension(org.alfresco.service.cmr.repository.TemplateExtensionImplementation)
|
|
||||||
*/
|
|
||||||
public void registerExtension(TemplateExtensionImplementation extension)
|
|
||||||
{
|
|
||||||
this.globalExtensions.add(extension);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#getExtensions()
|
|
||||||
*/
|
|
||||||
public List<TemplateExtensionImplementation> getExtensions()
|
|
||||||
{
|
|
||||||
return this.globalExtensions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param defaultTemplateEngine The default Template Engine name to set.
|
|
||||||
*/
|
|
||||||
public void setDefaultTemplateEngine(String defaultTemplateEngine)
|
|
||||||
{
|
|
||||||
this.defaultTemplateEngine = defaultTemplateEngine;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param templateEngines The Map of template engine name to impl class name to set.
|
|
||||||
*/
|
|
||||||
public void setTemplateEngines(Map<String, String> templateEngines)
|
|
||||||
{
|
|
||||||
this.templateEngines = templateEngines;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#getTemplateProcessor(java.lang.String)
|
|
||||||
*/
|
|
||||||
public TemplateProcessor getTemplateProcessor(String engine)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
return getTemplateProcessorImpl(engine);
|
|
||||||
}
|
|
||||||
catch (Throwable err)
|
|
||||||
{
|
|
||||||
if (logger.isDebugEnabled())
|
|
||||||
logger.debug("Unable to load template processor.", err);
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplate(java.lang.String, java.lang.String, java.lang.Object, java.io.Writer)
|
|
||||||
*/
|
|
||||||
public void processTemplate(String engine, String template, Object model, Writer out)
|
|
||||||
throws TemplateException
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// execute template processor
|
|
||||||
TemplateProcessor processor = getTemplateProcessorImpl(engine);
|
|
||||||
processor.process(template, model, out);
|
|
||||||
}
|
|
||||||
catch (TemplateException terr)
|
|
||||||
{
|
|
||||||
throw terr;
|
|
||||||
}
|
|
||||||
catch (Throwable err)
|
|
||||||
{
|
|
||||||
throw new TemplateException(err.getMessage(), err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplate(java.lang.String, java.lang.String, java.lang.Object)
|
|
||||||
*/
|
|
||||||
public String processTemplate(String engine, String template, Object model)
|
|
||||||
throws TemplateException
|
|
||||||
{
|
|
||||||
Writer out = new StringWriter(1024);
|
|
||||||
processTemplate(engine, template, model, out);
|
|
||||||
return out.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplateString(java.lang.String, java.lang.String, java.lang.Object, java.io.Writer)
|
|
||||||
*/
|
|
||||||
public void processTemplateString(String engine, String template, Object model, Writer out)
|
|
||||||
throws TemplateException
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// execute template processor
|
|
||||||
TemplateProcessor processor = getTemplateProcessorImpl(engine);
|
|
||||||
processor.processString(template, model, out);
|
|
||||||
}
|
|
||||||
catch (TemplateException terr)
|
|
||||||
{
|
|
||||||
throw terr;
|
|
||||||
}
|
|
||||||
catch (Throwable err)
|
|
||||||
{
|
|
||||||
throw new TemplateException(err.getMessage(), err);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see org.alfresco.service.cmr.repository.TemplateService#processTemplateString(java.lang.String, java.lang.String, java.lang.Object)
|
|
||||||
*/
|
|
||||||
public String processTemplateString(String engine, String template, Object model)
|
|
||||||
throws TemplateException
|
|
||||||
{
|
|
||||||
Writer out = new StringWriter(1024);
|
|
||||||
processTemplateString(engine, template, model, out);
|
|
||||||
return out.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return the TemplateProcessor implementation for the named template engine
|
|
||||||
*
|
|
||||||
* @param name Template Engine name
|
|
||||||
*
|
|
||||||
* @return TemplateProcessor
|
|
||||||
*/
|
|
||||||
private TemplateProcessor getTemplateProcessorImpl(String name)
|
|
||||||
{
|
|
||||||
// use the ThreadLocal map to find the processors instance
|
|
||||||
// create the cache map for this thread if required
|
|
||||||
Map<String, TemplateProcessor> procMap = processors.get();
|
|
||||||
if (procMap == null)
|
|
||||||
{
|
|
||||||
procMap = new HashMap<String, TemplateProcessor>(2, 1.0f);
|
|
||||||
processors.set(procMap);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (name == null)
|
|
||||||
{
|
|
||||||
name = defaultTemplateEngine;
|
|
||||||
}
|
|
||||||
|
|
||||||
// find the impl for the named processor
|
|
||||||
TemplateProcessor processor = procMap.get(name);
|
|
||||||
if (processor == null)
|
|
||||||
{
|
|
||||||
String className = templateEngines.get(name);
|
|
||||||
if (className == null)
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Unable to find configured ClassName for template engine: " + name);
|
|
||||||
}
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Object obj;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
obj = this.applicationContext.getBean(className);
|
|
||||||
}
|
|
||||||
catch (BeansException err)
|
|
||||||
{
|
|
||||||
// instantiate the processor class directory if not a Spring bean
|
|
||||||
obj = Class.forName(className).newInstance();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (obj instanceof TemplateProcessor)
|
|
||||||
{
|
|
||||||
processor = (TemplateProcessor)obj;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Supplied template processors does not implement TemplateProcessor: " + className);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (ClassNotFoundException err1)
|
|
||||||
{
|
|
||||||
// if the bean is not a classname, then it may be a spring bean Id
|
|
||||||
throw new AlfrescoRuntimeException("Unable to load class for supplied template processors: " + className, err1);
|
|
||||||
}
|
|
||||||
catch (IllegalAccessException err2)
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Unable to load class for supplied template processors: " + className, err2);
|
|
||||||
}
|
|
||||||
catch (InstantiationException err3)
|
|
||||||
{
|
|
||||||
throw new AlfrescoRuntimeException("Unable to instantiate class for supplied template processors: " + className, err3);
|
|
||||||
}
|
|
||||||
|
|
||||||
// cache for later
|
|
||||||
procMap.put(name, processor);
|
|
||||||
}
|
|
||||||
|
|
||||||
return processor;
|
|
||||||
}
|
|
||||||
}
|
|
@@ -41,7 +41,7 @@ import freemarker.template.TemplateScalarModel;
|
|||||||
* <p>
|
* <p>
|
||||||
* Usage: urlencode(String url)
|
* Usage: urlencode(String url)
|
||||||
*/
|
*/
|
||||||
public class UrlEncodeMethod extends BaseTemplateExtensionImplementation implements TemplateMethodModelEx
|
public class UrlEncodeMethod extends BaseTemplateProcessorExtension implements TemplateMethodModelEx
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
* @see freemarker.template.TemplateMethodModel#exec(java.util.List)
|
||||||
|
@@ -0,0 +1,55 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
* As a special exception to the terms and conditions of version 2.0 of
|
||||||
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
||||||
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
||||||
|
* FLOSS exception. You should have recieved a copy of the text describing
|
||||||
|
* the FLOSS exception, and it is also available here:
|
||||||
|
* http://www.alfresco.com/legal/licensing"
|
||||||
|
*/
|
||||||
|
package org.alfresco.service.cmr.repository;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Processor interface.
|
||||||
|
*
|
||||||
|
* @author Roy Wetherall
|
||||||
|
*/
|
||||||
|
public interface Processor
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Get the name of the processor
|
||||||
|
*
|
||||||
|
* @return the name of the processor
|
||||||
|
*/
|
||||||
|
public String getName();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The file extension that the processor is associated with, null if none.
|
||||||
|
*
|
||||||
|
* @return the extension
|
||||||
|
*/
|
||||||
|
public String getExtension();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a processor extension with the processor
|
||||||
|
*
|
||||||
|
* @param processorExtension the process extension
|
||||||
|
*/
|
||||||
|
public void registerProcessorExtension(ProcessorExtension processorExtension);
|
||||||
|
}
|
@@ -29,12 +29,12 @@ package org.alfresco.service.cmr.repository;
|
|||||||
*
|
*
|
||||||
* @author Roy Wetherall
|
* @author Roy Wetherall
|
||||||
*/
|
*/
|
||||||
public interface ScriptImplementation
|
public interface ProcessorExtension
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Returns the name of the script
|
* Returns the name of the extension
|
||||||
*
|
*
|
||||||
* @return the name of the script
|
* @return the name of the extension
|
||||||
*/
|
*/
|
||||||
String getScriptName();
|
String getExtensionName();
|
||||||
}
|
}
|
@@ -31,6 +31,8 @@ import org.alfresco.error.AlfrescoRuntimeException;
|
|||||||
*/
|
*/
|
||||||
public class ScriptException extends AlfrescoRuntimeException
|
public class ScriptException extends AlfrescoRuntimeException
|
||||||
{
|
{
|
||||||
|
private static final long serialVersionUID = 1739480648583299623L;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param msgId
|
* @param msgId
|
||||||
*/
|
*/
|
||||||
|
@@ -0,0 +1,75 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2005-2007 Alfresco Software Limited.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
* As a special exception to the terms and conditions of version 2.0 of
|
||||||
|
* the GPL, you may redistribute this Program in connection with Free/Libre
|
||||||
|
* and Open Source Software ("FLOSS") applications as described in Alfresco's
|
||||||
|
* FLOSS exception. You should have recieved a copy of the text describing
|
||||||
|
* the FLOSS exception, and it is also available here:
|
||||||
|
* http://www.alfresco.com/legal/licensing"
|
||||||
|
*/
|
||||||
|
package org.alfresco.service.cmr.repository;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.alfresco.service.namespace.QName;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Script processor interface
|
||||||
|
*
|
||||||
|
* @author Roy Wetherall
|
||||||
|
*/
|
||||||
|
public interface ScriptProcessor extends Processor
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Execute script
|
||||||
|
*
|
||||||
|
* @param location the location of the script
|
||||||
|
* @param model context model
|
||||||
|
* @return Object the result of the script
|
||||||
|
*/
|
||||||
|
public Object execute(ScriptLocation location, Map<String, Object> model);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute script
|
||||||
|
*
|
||||||
|
* @param nodeRef the script node reference
|
||||||
|
* @param contentProp the content property of the script
|
||||||
|
* @param model the context model
|
||||||
|
* @return Object the result of the script
|
||||||
|
*/
|
||||||
|
public Object execute(NodeRef nodeRef, QName contentProp, Map<String, Object> model);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute script
|
||||||
|
*
|
||||||
|
* @param location the classpath string locating the script
|
||||||
|
* @param model the context model
|
||||||
|
* @return Object the result of the script
|
||||||
|
*/
|
||||||
|
public Object execute(String location, Map<String, Object> model);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute script string
|
||||||
|
*
|
||||||
|
* @param script the script string
|
||||||
|
* @param model the context model
|
||||||
|
* @return Obejct the result of the script
|
||||||
|
*/
|
||||||
|
public Object executeString(String script, Map<String, Object> model);
|
||||||
|
|
||||||
|
}
|
@@ -51,6 +51,8 @@ public interface ScriptService
|
|||||||
/**
|
/**
|
||||||
* Process a script against the supplied data model.
|
* Process a script against the supplied data model.
|
||||||
*
|
*
|
||||||
|
* Uses the most approparite script engine or the default if none found.
|
||||||
|
*
|
||||||
* @param scriptClasspath Script location as qualified classpath name
|
* @param scriptClasspath Script location as qualified classpath name
|
||||||
* @param model Object model to process script against
|
* @param model Object model to process script against
|
||||||
*
|
*
|
||||||
@@ -61,10 +63,28 @@ public interface ScriptService
|
|||||||
@Auditable(parameters = {"scriptClasspath", "model"})
|
@Auditable(parameters = {"scriptClasspath", "model"})
|
||||||
public Object executeScript(String scriptClasspath, Map<String, Object> model)
|
public Object executeScript(String scriptClasspath, Map<String, Object> model)
|
||||||
throws ScriptException;
|
throws ScriptException;
|
||||||
|
/**
|
||||||
|
* Process a script against the supplied data model.
|
||||||
|
*
|
||||||
|
* Use the
|
||||||
|
*
|
||||||
|
* @param engine the script engine to use
|
||||||
|
* @param scriptClasspath Script location as qualified classpath name
|
||||||
|
* @param model Object model to process script against
|
||||||
|
*
|
||||||
|
* @return output of the script (may be null or any valid wrapped JavaScript object)
|
||||||
|
*
|
||||||
|
* @throws ScriptException
|
||||||
|
*/
|
||||||
|
@Auditable(parameters = {"engine", "scriptClasspath", "model"})
|
||||||
|
public Object executeScript(String engine, String scriptClasspath, Map<String, Object> model)
|
||||||
|
throws ScriptException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Process a script against the supplied data model.
|
* Process a script against the supplied data model.
|
||||||
*
|
*
|
||||||
|
* Uses the most approparite script engine or the default if none found.
|
||||||
|
*
|
||||||
* @param scriptRef Script NodeRef location
|
* @param scriptRef Script NodeRef location
|
||||||
* @param contentProp QName of the property on the node that contains the content, null can
|
* @param contentProp QName of the property on the node that contains the content, null can
|
||||||
* be passed to indicate the default property of 'cm:content'
|
* be passed to indicate the default property of 'cm:content'
|
||||||
@@ -81,6 +101,25 @@ public interface ScriptService
|
|||||||
/**
|
/**
|
||||||
* Process a script against the supplied data model.
|
* Process a script against the supplied data model.
|
||||||
*
|
*
|
||||||
|
* @param engine the script engine to use
|
||||||
|
* @param scriptRef Script NodeRef location
|
||||||
|
* @param contentProp QName of the property on the node that contains the content, null can
|
||||||
|
* be passed to indicate the default property of 'cm:content'
|
||||||
|
* @param model Object model to process script against
|
||||||
|
*
|
||||||
|
* @return output of the script (may be null or any valid wrapped JavaScript object)
|
||||||
|
*
|
||||||
|
* @throws ScriptException
|
||||||
|
*/
|
||||||
|
@Auditable(key = Auditable.Key.ARG_1, parameters = {"engine", "scriptRef", "contentProp", "model"})
|
||||||
|
public Object executeScript(String engine, NodeRef scriptRef, QName contentProp, Map<String, Object> model)
|
||||||
|
throws ScriptException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process a script against the supplied data model
|
||||||
|
*
|
||||||
|
* Uses the most approparite script engine or the default if none found.
|
||||||
|
*
|
||||||
* @param scriptLocation object representing the script location
|
* @param scriptLocation object representing the script location
|
||||||
* @param model Object model to process script against
|
* @param model Object model to process script against
|
||||||
*
|
*
|
||||||
@@ -95,6 +134,21 @@ public interface ScriptService
|
|||||||
/**
|
/**
|
||||||
* Process a script against the supplied data model.
|
* Process a script against the supplied data model.
|
||||||
*
|
*
|
||||||
|
* @param engine the script engine to use
|
||||||
|
* @param scriptLocation object representing the script location
|
||||||
|
* @param model Object model to process script against
|
||||||
|
*
|
||||||
|
* @return output of the script (may be null or any other valid wrapped JavaScript object)
|
||||||
|
*
|
||||||
|
* @throws ScriptException
|
||||||
|
*/
|
||||||
|
@Auditable(parameters = {"engine", "scriptLocation", "model"})
|
||||||
|
public Object executeScript(String engine, ScriptLocation scriptLocation, Map<String, Object> model)
|
||||||
|
throws ScriptException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process a script against the supplied data model. Uses the default script engine.
|
||||||
|
*
|
||||||
* @param script Script content as a String.
|
* @param script Script content as a String.
|
||||||
* @param model Object model to process script against
|
* @param model Object model to process script against
|
||||||
*
|
*
|
||||||
@@ -107,10 +161,53 @@ public interface ScriptService
|
|||||||
throws ScriptException;
|
throws ScriptException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Registers a script implementation with the script service
|
* Process a script against the supplied data model.
|
||||||
*
|
*
|
||||||
* @param script the script implementation
|
* @param engine the script engine to use
|
||||||
|
* @param script Script content as a String.
|
||||||
|
* @param model Object model to process script against
|
||||||
|
*
|
||||||
|
* @return output of the script (may be null or any valid wrapped JavaScript object)
|
||||||
|
*
|
||||||
|
* @throws ScriptException
|
||||||
*/
|
*/
|
||||||
@Auditable(parameters = {"script"})
|
@Auditable(parameters = {"engine", "script", "model"})
|
||||||
public void registerScript(ScriptImplementation script);
|
public Object executeScriptString(String engine, String script, Map<String, Object> model)
|
||||||
|
throws ScriptException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Registers a script processor with the script service
|
||||||
|
*
|
||||||
|
* @param scriptProcessor
|
||||||
|
*/
|
||||||
|
@Auditable(parameters = {"scriptProcessot"})
|
||||||
|
public void registerScriptProcessor(ScriptProcessor scriptProcessor);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create the default data-model available to scripts as global scope level objects:
|
||||||
|
* <p>
|
||||||
|
* 'companyhome' - the Company Home node<br>
|
||||||
|
* 'userhome' - the current user home space node<br>
|
||||||
|
* 'person' - the node representing the current user Person<br>
|
||||||
|
* 'script' - the node representing the script itself (may not be available)<br>
|
||||||
|
* 'document' - document context node (may not be available)<br>
|
||||||
|
* 'space' - space context node (may not be available)
|
||||||
|
*
|
||||||
|
* @param person The current user Person Node
|
||||||
|
* @param companyHome The CompanyHome ref
|
||||||
|
* @param userHome The User home space ref
|
||||||
|
* @param script Optional ref to the script itself
|
||||||
|
* @param document Optional ref to a document Node
|
||||||
|
* @param space Optional ref to a space Node
|
||||||
|
* @param resolver Image resolver to resolve icon images etc.
|
||||||
|
*
|
||||||
|
* @return A Map of global scope scriptable Node objects
|
||||||
|
*/
|
||||||
|
public Map<String, Object> buildDefaultModel(
|
||||||
|
NodeRef person,
|
||||||
|
NodeRef companyHome,
|
||||||
|
NodeRef userHome,
|
||||||
|
NodeRef script,
|
||||||
|
NodeRef document,
|
||||||
|
NodeRef space);
|
||||||
}
|
}
|
||||||
|
@@ -31,6 +31,8 @@ import org.alfresco.error.AlfrescoRuntimeException;
|
|||||||
*/
|
*/
|
||||||
public class TemplateException extends AlfrescoRuntimeException
|
public class TemplateException extends AlfrescoRuntimeException
|
||||||
{
|
{
|
||||||
|
private static final long serialVersionUID = 2863142603098852564L;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param msgId
|
* @param msgId
|
||||||
*/
|
*/
|
||||||
|
@@ -33,7 +33,7 @@ import java.io.Writer;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public interface TemplateProcessor
|
public interface TemplateProcessor extends Processor
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Process a template against the supplied data model and write to the out.
|
* Process a template against the supplied data model and write to the out.
|
||||||
|
@@ -29,15 +29,8 @@ package org.alfresco.service.cmr.repository;
|
|||||||
*
|
*
|
||||||
* @author Kevin Roast
|
* @author Kevin Roast
|
||||||
*/
|
*/
|
||||||
public interface TemplateExtensionImplementation
|
public interface TemplateProcessorExtension extends ProcessorExtension
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* Returns the name of the template extension
|
|
||||||
*
|
|
||||||
* @return the name of the template extension
|
|
||||||
*/
|
|
||||||
String getExtensionName();
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the template image resolver for this extension
|
* Set the template image resolver for this extension
|
||||||
*
|
*
|
@@ -25,7 +25,7 @@
|
|||||||
package org.alfresco.service.cmr.repository;
|
package org.alfresco.service.cmr.repository;
|
||||||
|
|
||||||
import java.io.Writer;
|
import java.io.Writer;
|
||||||
import java.util.List;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.alfresco.service.Auditable;
|
import org.alfresco.service.Auditable;
|
||||||
import org.alfresco.service.PublicService;
|
import org.alfresco.service.PublicService;
|
||||||
@@ -47,6 +47,44 @@ import org.alfresco.service.PublicService;
|
|||||||
@PublicService
|
@PublicService
|
||||||
public interface TemplateService
|
public interface TemplateService
|
||||||
{
|
{
|
||||||
|
/** Keys for default model values */
|
||||||
|
public static final String KEY_IMAGE_RESOLVER = "imageresolver";
|
||||||
|
public static final String KEY_COMPANY_HOME = "companyhome";
|
||||||
|
public static final String KEY_USER_HOME = "userhome";
|
||||||
|
public static final String KEY_PERSON = "person";
|
||||||
|
public static final String KEY_TEMPLATE = "template";
|
||||||
|
public static final String KEY_DATE = "date";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process a template against the upplied data model and return the result as
|
||||||
|
* a string.
|
||||||
|
*
|
||||||
|
* The template engine used will be determined by the extension of the template.
|
||||||
|
*
|
||||||
|
* @param template Template (qualified classpath name or noderef)
|
||||||
|
* @param model Object model to process template against
|
||||||
|
*
|
||||||
|
* @return output of the template process as a String
|
||||||
|
* @throws TemplateException
|
||||||
|
*/
|
||||||
|
@Auditable(parameters = {"template", "model"})
|
||||||
|
public String processTemplate(String template, Object model)
|
||||||
|
throws TemplateException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Process a template against the supplied data model and write to the out.
|
||||||
|
*
|
||||||
|
* The template engine used will be determined by the extension of the template.
|
||||||
|
*
|
||||||
|
* @param engine Name of the template engine to use
|
||||||
|
* @param template Template (qualified classpath name or noderef)
|
||||||
|
* @param model Object model to process template against
|
||||||
|
* @param out Writer object to send output too
|
||||||
|
*/
|
||||||
|
@Auditable(parameters = {"template", "model", "out"})
|
||||||
|
public void processTemplate(String template, Object model, Writer out)
|
||||||
|
throws TemplateException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Process a template against the supplied data model and write to the out.
|
* Process a template against the supplied data model and write to the out.
|
||||||
*
|
*
|
||||||
@@ -114,18 +152,28 @@ public interface TemplateService
|
|||||||
public TemplateProcessor getTemplateProcessor(String engine);
|
public TemplateProcessor getTemplateProcessor(String engine);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Registers a template extension implementation with the Template service
|
* Registers a new template processor with the template service
|
||||||
*
|
*
|
||||||
* @param extension the template extension implementation
|
* @param templateProcessor the template processor to register
|
||||||
*/
|
*/
|
||||||
@Auditable(parameters = {"extension"})
|
@Auditable(parameters = {"templateProcessor"})
|
||||||
public void registerExtension(TemplateExtensionImplementation extension);
|
public void registerTemplateProcessor(TemplateProcessor templateProcessor);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a list of the Template Extension objects configured for this service
|
* Helper method to build a default model
|
||||||
*
|
*
|
||||||
* @return list of the Template Extension objects configured for this service
|
* @param person the person node reference
|
||||||
|
* @param companyHome the company home node refereence
|
||||||
|
* @param userHome the user home node reference
|
||||||
|
* @param template the node ref for the template (optional)
|
||||||
|
* @param imageResolver the image resolver (optional)
|
||||||
|
* @return
|
||||||
*/
|
*/
|
||||||
@Auditable(warn = true, recordReturnedObject = false)
|
@Auditable(parameters = {"person", "compantHome", "userHome", "template", "imageResolver"})
|
||||||
public List<TemplateExtensionImplementation> getExtensions();
|
public Map<String, Object> buildDefaultModel(
|
||||||
|
NodeRef person,
|
||||||
|
NodeRef companyHome,
|
||||||
|
NodeRef userHome,
|
||||||
|
NodeRef template,
|
||||||
|
TemplateImageResolver imageResolver);
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user