001    /**
002     * Copyright 2005-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.rice.kew.dto;
017    
018    import com.google.common.base.Functions;
019    import com.google.common.base.Joiner;
020    import com.google.common.collect.Iterables;
021    import org.apache.commons.lang.StringUtils;
022    import org.apache.log4j.Logger;
023    import org.kuali.rice.core.api.exception.RiceRuntimeException;
024    import org.kuali.rice.core.api.reflect.DataDefinition;
025    import org.kuali.rice.core.api.reflect.ObjectDefinition;
026    import org.kuali.rice.core.api.reflect.PropertyDefinition;
027    import org.kuali.rice.core.api.resourceloader.GlobalResourceLoader;
028    import org.kuali.rice.core.api.uif.RemotableAttributeErrorContract;
029    import org.kuali.rice.core.api.util.xml.XmlHelper;
030    import org.kuali.rice.core.api.util.xml.XmlJotter;
031    import org.kuali.rice.kew.actionrequest.ActionRequestValue;
032    import org.kuali.rice.kew.actiontaken.ActionTakenValue;
033    import org.kuali.rice.kew.api.KewApiServiceLocator;
034    import org.kuali.rice.kew.api.WorkflowRuntimeException;
035    import org.kuali.rice.kew.api.action.ActionRequest;
036    import org.kuali.rice.kew.api.action.ActionTaken;
037    import org.kuali.rice.kew.api.document.DocumentContentUpdate;
038    import org.kuali.rice.kew.api.document.DocumentDetail;
039    import org.kuali.rice.kew.api.document.InvalidDocumentContentException;
040    import org.kuali.rice.kew.api.document.attribute.WorkflowAttributeDefinition;
041    import org.kuali.rice.kew.api.extension.ExtensionDefinition;
042    import org.kuali.rice.kew.api.extension.ExtensionUtils;
043    import org.kuali.rice.kew.definition.AttributeDefinition;
044    import org.kuali.rice.kew.engine.node.RouteNodeInstance;
045    import org.kuali.rice.kew.framework.document.attribute.SearchableAttribute;
046    import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
047    import org.kuali.rice.kew.routeheader.StandardDocumentContent;
048    import org.kuali.rice.kew.rule.WorkflowRuleAttribute;
049    import org.kuali.rice.kew.rule.WorkflowAttributeXmlValidator;
050    import org.kuali.rice.kew.rule.XmlConfiguredAttribute;
051    import org.kuali.rice.kew.rule.bo.RuleAttribute;
052    import org.kuali.rice.kew.rule.xmlrouting.GenericXMLRuleAttribute;
053    import org.kuali.rice.kew.service.KEWServiceLocator;
054    import org.kuali.rice.kew.api.KewApiConstants;
055    import org.w3c.dom.Document;
056    import org.w3c.dom.Element;
057    import org.w3c.dom.NodeList;
058    import org.xml.sax.SAXException;
059    
060    import javax.xml.namespace.QName;
061    import javax.xml.parsers.DocumentBuilder;
062    import javax.xml.parsers.DocumentBuilderFactory;
063    import javax.xml.parsers.ParserConfigurationException;
064    import javax.xml.transform.TransformerException;
065    import java.io.IOException;
066    import java.util.ArrayList;
067    import java.util.HashMap;
068    import java.util.Iterator;
069    import java.util.List;
070    import java.util.Map;
071    
072    /**
073     * Translates Workflow server side beans into client side VO beans.
074     *
075     * @author Kuali Rice Team (rice.collab@kuali.org)
076     */
077    public class DTOConverter {
078        private static final Logger LOG = Logger.getLogger(DTOConverter.class);
079    
080        public static String buildUpdatedDocumentContent(String existingDocContent,
081                DocumentContentUpdate documentContentUpdate, String documentTypeName) {
082            if (existingDocContent == null) {
083                existingDocContent = KewApiConstants.DEFAULT_DOCUMENT_CONTENT;
084            }
085            String documentContent = KewApiConstants.DEFAULT_DOCUMENT_CONTENT;
086            StandardDocumentContent standardDocContent = new StandardDocumentContent(existingDocContent);
087            try {
088                DocumentBuilder builder = DocumentBuilderFactory.newInstance().newDocumentBuilder();
089                Document document = builder.newDocument();
090                Element root = document.createElement(KewApiConstants.DOCUMENT_CONTENT_ELEMENT);
091                document.appendChild(root);
092                Element applicationContentElement = standardDocContent.getApplicationContent();
093                if (documentContentUpdate.getApplicationContent() != null) {
094                    // application content has changed
095                    if (!StringUtils.isEmpty(documentContentUpdate.getApplicationContent())) {
096                        applicationContentElement = document.createElement(KewApiConstants.APPLICATION_CONTENT_ELEMENT);
097                        XmlHelper.appendXml(applicationContentElement, documentContentUpdate.getApplicationContent());
098                    } else {
099                        // they've cleared the application content
100                        applicationContentElement = null;
101                    }
102                }
103                Element attributeContentElement = createDocumentContentSection(document,
104                        standardDocContent.getAttributeContent(), documentContentUpdate.getAttributeDefinitions(),
105                        documentContentUpdate.getAttributeContent(), KewApiConstants.ATTRIBUTE_CONTENT_ELEMENT,
106                        documentTypeName);
107                Element searchableContentElement = createDocumentContentSection(document,
108                        standardDocContent.getSearchableContent(), documentContentUpdate.getSearchableDefinitions(),
109                        documentContentUpdate.getSearchableContent(), KewApiConstants.SEARCHABLE_CONTENT_ELEMENT,
110                        documentTypeName);
111                if (applicationContentElement != null) {
112                    root.appendChild(applicationContentElement);
113                }
114                if (attributeContentElement != null) {
115                    root.appendChild(attributeContentElement);
116                }
117                if (searchableContentElement != null) {
118                    root.appendChild(searchableContentElement);
119                }
120                documentContent = XmlJotter.jotNode(document);
121            } catch (ParserConfigurationException e) {
122                throw new RiceRuntimeException("Failed to initialize XML parser.", e);
123            } catch (SAXException e) {
124                throw new InvalidDocumentContentException("Failed to parse XML.", e);
125            } catch (IOException e) {
126                throw new InvalidDocumentContentException("Failed to parse XML.", e);
127            } catch (TransformerException e) {
128                throw new InvalidDocumentContentException("Failed to parse XML.", e);
129            }
130            return documentContent;
131        }
132    
133        private static Element createDocumentContentSection(Document document, Element existingAttributeElement,
134                List<WorkflowAttributeDefinition> definitions, String content, String elementName,
135                String documentTypeName) throws TransformerException, SAXException, IOException, ParserConfigurationException {
136            Element contentSectionElement = existingAttributeElement;
137            // if they've updated the content, we're going to re-build the content section element from scratch
138            if (content != null) {
139                if (!org.apache.commons.lang.StringUtils.isEmpty(content)) {
140                    contentSectionElement = document.createElement(elementName);
141                    // if they didn't merely clear the content, let's build the content section element by combining the children
142                    // of the incoming XML content
143                    Element incomingAttributeElement = XmlHelper.readXml(content).getDocumentElement();
144                    NodeList children = incomingAttributeElement.getChildNodes();
145                    for (int index = 0; index < children.getLength(); index++) {
146                        contentSectionElement.appendChild(document.importNode(children.item(index), true));
147                    }
148                } else {
149                    contentSectionElement = null;
150                }
151            }
152            // if they have new definitions we're going to append those to the existing content section
153            if (definitions != null && !definitions.isEmpty()) {
154                String errorMessage = "";
155                boolean inError = false;
156                if (contentSectionElement == null) {
157                    contentSectionElement = document.createElement(elementName);
158                }
159                for (WorkflowAttributeDefinition definitionVO : definitions) {
160                    AttributeDefinition definition = convertWorkflowAttributeDefinition(definitionVO);
161                    ExtensionDefinition extensionDefinition = definition.getExtensionDefinition();
162    
163                    Object attribute = null;
164                    attribute = GlobalResourceLoader.getObject(definition.getObjectDefinition());
165                    if (attribute == null) {
166                        attribute = GlobalResourceLoader.getService(QName.valueOf(
167                                definition.getExtensionDefinition().getResourceDescriptor()));
168                    }
169    
170                    if (attribute instanceof XmlConfiguredAttribute) {
171                        ((XmlConfiguredAttribute)attribute).setExtensionDefinition(definition.getExtensionDefinition());
172                    }
173                    boolean propertiesAsMap = false;
174                    if (KewApiConstants.RULE_XML_ATTRIBUTE_TYPE.equals(extensionDefinition.getType())) {
175                        propertiesAsMap = true;
176                    }
177                    if (propertiesAsMap) {
178                        for (org.kuali.rice.kew.api.document.PropertyDefinition propertyDefinitionVO : definitionVO
179                                .getPropertyDefinitions()) {
180                            if (attribute instanceof GenericXMLRuleAttribute) {
181                                ((GenericXMLRuleAttribute) attribute).getParamMap().put(propertyDefinitionVO.getName(),
182                                        propertyDefinitionVO.getValue());
183                            }
184                        }
185                    }
186    
187                    // validate inputs from client application if the attribute is capable
188                    if (attribute instanceof WorkflowAttributeXmlValidator) {
189                        List<? extends RemotableAttributeErrorContract> errors =
190                                ((WorkflowAttributeXmlValidator) attribute).validateClientRoutingData();
191                        if (!errors.isEmpty()) {
192                            inError = true;
193                            errorMessage += "Error validating attribute " + definitionVO.getAttributeName() + " ";
194                            errorMessage += Joiner.on("; ").join(Iterables.transform(errors, Functions.toStringFunction()));
195                        }
196                    }
197                    // dont add to xml if attribute is in error
198                    if (!inError) {
199                        if (attribute instanceof WorkflowRuleAttribute) {
200                            String attributeDocContent = ((WorkflowRuleAttribute) attribute).getDocContent();
201                            if (!StringUtils.isEmpty(attributeDocContent)) {
202                                XmlHelper.appendXml(contentSectionElement, attributeDocContent);
203                            }
204                        } else if (attribute instanceof SearchableAttribute) {
205                            SearchableAttribute searchableAttribute = (SearchableAttribute) attribute;
206                            String searchableAttributeContent = searchableAttribute.generateSearchContent(extensionDefinition, documentTypeName,
207                                    definitionVO);
208                            if (!StringUtils.isBlank(searchableAttributeContent)) {
209                                XmlHelper.appendXml(contentSectionElement, searchableAttributeContent);
210                            }
211                        }
212                    }
213                }
214                if (inError) {
215                    throw new WorkflowRuntimeException(errorMessage);
216                }
217    
218            }
219            if (contentSectionElement != null) {
220                // always be sure and import the element into the new document, if it originated from the existing doc content
221                // and
222                // appended to it, it will need to be imported
223                contentSectionElement = (Element) document.importNode(contentSectionElement, true);
224            }
225            return contentSectionElement;
226        }
227    
228        /**
229         * New for Rice 2.0
230         */
231        public static AttributeDefinition convertWorkflowAttributeDefinition(WorkflowAttributeDefinition definition) {
232            if (definition == null) {
233                return null;
234            }
235            //KULRICE-7643
236            ExtensionDefinition extensionDefinition = null;
237            List<RuleAttribute> ruleAttribute = KEWServiceLocator.getRuleAttributeService().findByClassName(definition.getAttributeName());
238            if (ruleAttribute == null || ruleAttribute.isEmpty()) {
239                extensionDefinition = KewApiServiceLocator.getExtensionRepositoryService().getExtensionByName(definition.getAttributeName());
240            }else{
241                //TODO: Should we do something more intelligent here? Rice 1.x returned only a single entry but we can now have a list
242                RuleAttribute tmpAttr = ruleAttribute.get(0);
243                extensionDefinition = RuleAttribute.to(tmpAttr);
244                if(ruleAttribute.size() > 1){
245                    LOG.warn("AttributeDefinition lookup (findByClassName) returned multiple attribute for the same class name. This should not happen, investigation recommended for classname: " 
246                + definition.getAttributeName() + " which has " + ruleAttribute.size() + " entries.");
247                }
248            }
249            
250            if (extensionDefinition == null) {
251                throw new WorkflowRuntimeException("Extension " + definition.getAttributeName() + " not found");
252            }
253            /*RuleAttribute ruleAttribute = KEWServiceLocator.getRuleAttributeService().findByName(definition.getAttributeName());
254            if (ruleAttribute == null) {
255                throw new WorkflowRuntimeException("Attribute " + definition.getAttributeName() + " not found");
256            }*/
257    
258            ObjectDefinition objectDefinition = new ObjectDefinition(extensionDefinition.getResourceDescriptor());
259            if (definition.getParameters() != null) {
260                for (String parameter : definition.getParameters()) {
261                    objectDefinition.addConstructorParameter(new DataDefinition(parameter, String.class));
262                }
263            }
264            boolean propertiesAsMap = KewApiConstants.RULE_XML_ATTRIBUTE_TYPE.equals(extensionDefinition.getType()) || KewApiConstants
265                    .SEARCHABLE_XML_ATTRIBUTE_TYPE.equals(extensionDefinition.getType());
266            if (!propertiesAsMap && definition.getPropertyDefinitions() != null) {
267                for (org.kuali.rice.kew.api.document.PropertyDefinition propertyDefinition : definition
268                        .getPropertyDefinitions()) {
269                    objectDefinition.addProperty(new PropertyDefinition(propertyDefinition.getName(), new DataDefinition(
270                            propertyDefinition.getValue(), String.class)));
271                }
272            }
273    
274            return new AttributeDefinition(extensionDefinition, objectDefinition);
275        }
276    
277        /**
278         * Interface for a simple service providing RouteNodeInstanceS based on their IDs
279         */
280        public static interface RouteNodeInstanceLoader {
281            RouteNodeInstance load(String routeNodeInstanceID);
282        }
283    
284    
285        public static DocumentDetail convertDocumentDetailNew(DocumentRouteHeaderValue routeHeader) {
286            if (routeHeader == null) {
287                return null;
288            }
289            org.kuali.rice.kew.api.document.Document document = DocumentRouteHeaderValue.to(routeHeader);
290            DocumentDetail.Builder detail = DocumentDetail.Builder.create(document);
291            Map<String, RouteNodeInstance> nodeInstances = new HashMap<String, RouteNodeInstance>();
292            List<ActionRequest> actionRequestVOs = new ArrayList<ActionRequest>();
293            List<ActionRequestValue> rootActionRequests = KEWServiceLocator.getActionRequestService().getRootRequests(
294                    routeHeader.getActionRequests());
295            for (Iterator<ActionRequestValue> iterator = rootActionRequests.iterator(); iterator.hasNext(); ) {
296                ActionRequestValue actionRequest = iterator.next();
297                actionRequestVOs.add(ActionRequestValue.to(actionRequest));
298                RouteNodeInstance nodeInstance = actionRequest.getNodeInstance();
299                if (nodeInstance == null) {
300                    continue;
301                }
302                if (nodeInstance.getRouteNodeInstanceId() == null) {
303                    throw new IllegalStateException(
304                            "Error creating document detail structure because of NULL node instance id.");
305                }
306                nodeInstances.put(nodeInstance.getRouteNodeInstanceId(), nodeInstance);
307            }
308            detail.setActionRequests(actionRequestVOs);
309            List<org.kuali.rice.kew.api.document.node.RouteNodeInstance> nodeInstanceVOs =
310                    new ArrayList<org.kuali.rice.kew.api.document.node.RouteNodeInstance>();
311            for (Iterator<RouteNodeInstance> iterator = nodeInstances.values().iterator(); iterator.hasNext(); ) {
312                RouteNodeInstance nodeInstance = iterator.next();
313                nodeInstanceVOs.add(RouteNodeInstance.to(nodeInstance));
314            }
315            detail.setRouteNodeInstances(nodeInstanceVOs);
316            List<ActionTaken> actionTakenVOs = new ArrayList<ActionTaken>();
317            for (Object element : routeHeader.getActionsTaken()) {
318                ActionTakenValue actionTaken = (ActionTakenValue) element;
319                actionTakenVOs.add(ActionTakenValue.to(actionTaken));
320            }
321            detail.setActionsTaken(actionTakenVOs);
322            return detail.build();
323        }
324    
325    }