1 package org
.argeo
.slc
.core
.execution
.xml
;
3 import java
.util
.ArrayList
;
6 import org
.argeo
.api
.cms
.CmsLog
;
7 import org
.argeo
.slc
.SlcException
;
8 import org
.argeo
.slc
.core
.execution
.DefaultExecutionFlow
;
9 import org
.argeo
.slc
.execution
.ExecutionFlow
;
10 import org
.springframework
.beans
.factory
.BeanDefinitionStoreException
;
11 import org
.springframework
.beans
.factory
.config
.RuntimeBeanReference
;
12 import org
.springframework
.beans
.factory
.support
.AbstractBeanDefinition
;
13 import org
.springframework
.beans
.factory
.support
.BeanDefinitionBuilder
;
14 import org
.springframework
.beans
.factory
.support
.ManagedList
;
15 import org
.springframework
.beans
.factory
.support
.ManagedMap
;
16 import org
.springframework
.beans
.factory
.xml
.AbstractSingleBeanDefinitionParser
;
17 import org
.springframework
.beans
.factory
.xml
.ParserContext
;
18 import org
.springframework
.util
.StringUtils
;
19 import org
.springframework
.util
.xml
.DomUtils
;
20 import org
.w3c
.dom
.Element
;
21 import org
.w3c
.dom
.Node
;
22 import org
.w3c
.dom
.NodeList
;
24 /** Interprets the <flow:flow> tag */
25 public class FlowBeanDefinitionParser
extends
26 AbstractSingleBeanDefinitionParser
{
27 private CmsLog log
= CmsLog
.getLog(FlowBeanDefinitionParser
.class);
29 /** Whether the user has already be warned on path attribute usage. */
30 private Boolean warnedAboutPathAttribute
= false;
33 protected void doParse(Element element
, ParserContext parserContext
,
34 BeanDefinitionBuilder builder
) {
35 String path
= element
.getAttribute("path");
36 if (StringUtils
.hasText(path
)) {
37 builder
.addPropertyValue("path", path
);
39 // warns user only once
40 if (!warnedAboutPathAttribute
)
41 log
.warn("The path=\"\" attribute is deprecated"
42 + " and will be removed in a later release."
43 + " Use <flow:flow name=\"/my/path/flowName\">.");
44 warnedAboutPathAttribute
= true;
47 String spec
= element
.getAttribute("spec");
48 if (StringUtils
.hasText(spec
))
49 builder
.getBeanDefinition().getConstructorArgumentValues()
50 .addGenericArgumentValue(new RuntimeBeanReference(spec
));
52 String abstrac
= element
.getAttribute("abstract");
53 if (StringUtils
.hasText(abstrac
))
54 builder
.setAbstract(Boolean
.parseBoolean(abstrac
));
56 String parent
= element
.getAttribute("parent");
57 if (StringUtils
.hasText(parent
))
58 builder
.setParentName(parent
);
60 builder
.getBeanDefinition().setDescription(
61 DomUtils
.getChildElementValueByTagName(element
, "description"));
63 List
<Element
> argsElems
= new ArrayList
<Element
>();
64 List
<Element
> execElems
= new ArrayList
<Element
>();
65 List
<Element
> specElems
= new ArrayList
<Element
>();
66 NodeList nodeList
= element
.getChildNodes();
67 for (int i
= 0; i
< nodeList
.getLength(); i
++) {
68 Node node
= nodeList
.item(i
);
69 if (node
instanceof Element
) {
70 if (DomUtils
.nodeNameEquals(node
, "arg"))
71 argsElems
.add((Element
) node
);
72 else if (DomUtils
.nodeNameEquals(node
, "spec"))
73 specElems
.add((Element
) node
);
74 else if (!DomUtils
.nodeNameEquals(node
, "description"))
75 execElems
.add((Element
) node
);
80 if (argsElems
.size() != 0) {
81 ManagedMap
<String
, Object
> args
= new ManagedMap
<String
, Object
>(
83 for (Element argElem
: argsElems
) {
84 Object value
= NamespaceUtils
.parseValue(argElem
,
85 parserContext
, builder
.getBeanDefinition(), null);
87 args
.put(argElem
.getAttribute("name"), value
);
89 throw new SlcException("No value defined.");
91 builder
.getBeanDefinition().getConstructorArgumentValues()
92 .addGenericArgumentValue(args
);
96 if (StringUtils
.hasText(spec
) && specElems
.size() != 0)
97 throw new SlcException("A flow cannot have multiple specs");
98 if (specElems
.size() == 1) {
99 Object specObj
= NamespaceUtils
.parseBeanOrReference(
100 specElems
.get(0), parserContext
,
101 builder
.getBeanDefinition());
102 builder
.getBeanDefinition().getConstructorArgumentValues()
103 .addGenericArgumentValue(specObj
);
104 } else if (specElems
.size() > 1)
105 throw new SlcException("A flow cannot have multiple specs");
108 if (execElems
.size() != 0) {
109 ManagedList
<Object
> executables
= new ManagedList
<Object
>(
111 for (Element child
: execElems
) {
112 // child validity check is performed in xsd
113 executables
.add(NamespaceUtils
.parseBeanOrReference(child
,
114 parserContext
, builder
.getBeanDefinition()));
116 if (executables
.size() > 0)
117 builder
.addPropertyValue("executables", executables
);
121 @SuppressWarnings("unchecked")
123 protected Class
<?
extends ExecutionFlow
> getBeanClass(Element element
) {
124 String clss
= element
.getAttribute("class");
125 if (StringUtils
.hasText(clss
))
126 // TODO: check that it actually works
128 return (Class
<?
extends ExecutionFlow
>) getClass()
129 .getClassLoader().loadClass(clss
);
130 } catch (ClassNotFoundException e
) {
132 return (Class
<?
extends ExecutionFlow
>) Thread
133 .currentThread().getContextClassLoader()
135 } catch (ClassNotFoundException e1
) {
136 throw new SlcException("Cannot load class " + clss
, e
);
140 return DefaultExecutionFlow
.class;
143 // parse nested bean definition
144 // private Object parseBeanReference(Element element,
145 // ParserContext parserContext, BeanDefinitionBuilder builder) {
146 // return parserContext.getDelegate().parsePropertySubElement(element,
147 // builder.getBeanDefinition());
151 protected String
resolveId(Element element
,
152 AbstractBeanDefinition definition
, ParserContext parserContext
)
153 throws BeanDefinitionStoreException
{
154 String name
= element
.getAttribute("name");
155 if (StringUtils
.hasText(name
)) {
158 return super.resolveId(element
, definition
, parserContext
);
162 protected boolean shouldGenerateIdAsFallback() {