001    /**
002     * Copyright 2005-2011 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.krad.workflow;
017    
018    import org.apache.commons.logging.Log;
019    import org.apache.commons.logging.LogFactory;
020    import org.junit.Ignore;
021    import org.junit.Test;
022    import org.kuali.rice.core.api.impex.xml.XmlConstants;
023    import org.kuali.rice.core.api.resourceloader.GlobalResourceLoader;
024    import org.kuali.rice.core.api.reflect.ObjectDefinition;
025    import org.kuali.rice.kew.api.KewApiConstants;
026    import org.kuali.rice.kew.api.extension.ExtensionDefinition;
027    import org.kuali.rice.kew.rule.bo.RuleAttribute;
028    import org.kuali.rice.kew.rule.xmlrouting.XPathHelper;
029    import org.kuali.rice.krad.datadictionary.AttributeDefinition;
030    import org.kuali.rice.krad.datadictionary.BusinessObjectEntry;
031    import org.kuali.rice.krad.datadictionary.DocumentEntry;
032    import org.kuali.rice.krad.service.DataDictionaryService;
033    import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
034    import org.kuali.rice.krad.workflow.attribute.KualiXmlAttribute;
035    import org.kuali.rice.krad.workflow.attribute.KualiXmlAttributeHelper;
036    import org.kuali.rice.krad.workflow.attribute.KualiXmlRuleAttributeImpl;
037    import org.kuali.rice.krad.workflow.attribute.KualiXmlSearchableAttributeImpl;
038    import org.kuali.test.KRADTestCase;
039    import org.w3c.dom.NamedNodeMap;
040    import org.w3c.dom.Node;
041    import org.w3c.dom.NodeList;
042    import org.xml.sax.InputSource;
043    
044    import javax.xml.transform.Result;
045    import javax.xml.transform.Source;
046    import javax.xml.transform.TransformerException;
047    import javax.xml.transform.TransformerFactory;
048    import javax.xml.transform.dom.DOMSource;
049    import javax.xml.transform.stream.StreamResult;
050    import javax.xml.xpath.XPath;
051    import javax.xml.xpath.XPathConstants;
052    import java.io.StringReader;
053    import java.io.StringWriter;
054    import java.util.HashSet;
055    import java.util.Set;
056    
057    import static org.junit.Assert.*;
058    
059    
060    /**
061     * KualiXMLAttributeImplTest tests the {@link KualiXmlAttributeHelper} operations of getting data from the data dictionary for workflow
062     * attributes
063     * 
064     * @author Kuali Rice Team (rice.collab@kuali.org)
065     */
066    @Ignore
067    public class KualiXMLAttributeImplTest extends KRADTestCase {
068        private static Log LOG = LogFactory.getLog(KualiXMLAttributeImplTest.class);
069    
070        private static final String RULE_ATTRIBUTE_CONFIG_NODE_NAME = XmlConstants.ROUTING_CONFIG;
071        private static final String SEARCH_ATTRIBUTE_CONFIG_NODE_NAME = XmlConstants.SEARCHING_CONFIG;
072    
073        XPath myXPath = XPathHelper.newXPath();
074        String ruleAttributeXml = "";
075        String searchAttributeXml = "";
076    //    private boolean testFailed = false;
077    
078        @Override
079        public void setUp() throws Exception {
080            super.setUp();
081            // read in the file and make it a string
082    
083    //        InputStream is = new FileInputStream(getBaseDir() + "/src/test/resources/org/kuali/rice/kew/batch/data/RuleAttributeContent.xml");
084    //        is.
085    //        if ((StringUtils.isNotBlank(ruleAttributeXml)) && (StringUtils.isNotBlank(searchAttributeXml))) {
086    //            return;
087    //        }
088    //
089    //        DataSource mySource = KEWServiceLocator.getDataSource();
090    //        ruleAttributeXml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<data >\n<ruleAttributes>\n";
091    //        searchAttributeXml = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<data >\n<ruleAttributes>\n";
092    //        Connection dbCon = null;
093    //        try {
094    //
095    //            dbCon = mySource.getConnection();
096    //            Statement dbAsk = dbCon.createStatement();
097    //            ResultSet dbAnswer = dbAsk.executeQuery("select * from EN_RULE_ATTRIB_T");
098    //            // ResultSet dbAnswer = dbAsk.executeQuery("select * from EN_RULE_ATTRIB_T where RULE_ATTRIB_NM =
099    //            // 'SystemParameterRoutingAttribute'");
100    //
101    //            while (dbAnswer.next()) {
102    //                String className = dbAnswer.getString("RULE_ATTRIB_CLS_NM");
103    //                if (StringUtils.isNotBlank(className)) {
104    //                    try {
105    //                        if (KualiXmlAttribute.class.isAssignableFrom(Class.forName(className))) {
106    //                            LOG.debug("Adding attribute to test with class name " + className);
107    //                            String attributeType = dbAnswer.getString("RULE_ATTRIB_TYP");
108    //                            if (KewApiConstants.RULE_XML_ATTRIBUTE_TYPE.equals(attributeType)) {
109    //                                ruleAttributeXml = ruleAttributeXml + "<ruleAttribute>\n\t<name>";
110    //                                ruleAttributeXml = ruleAttributeXml + dbAnswer.getString("RULE_ATTRIB_NM");
111    //                                ruleAttributeXml = ruleAttributeXml + "</name>\n\t<className>";
112    //                                ruleAttributeXml = ruleAttributeXml + className;
113    //                                ruleAttributeXml = ruleAttributeXml + "</className>\n\t<label>";
114    //                                ruleAttributeXml = ruleAttributeXml + dbAnswer.getString("RULE_ATTRIB_LBL_TXT");
115    //                                ruleAttributeXml = ruleAttributeXml + "</label>\n\t<description>";
116    //                                ruleAttributeXml = ruleAttributeXml + dbAnswer.getString("RULE_ATTRIB_DESC");
117    //                                ruleAttributeXml = ruleAttributeXml + "</description>\n\t<type>";
118    //                                ruleAttributeXml = ruleAttributeXml + attributeType;
119    //                                ruleAttributeXml = ruleAttributeXml + "</type>\n\t" + dbAnswer.getString("RULE_ATTRIB_XML_RTE_TXT") + "\n</ruleAttribute>\n";
120    //
121    //                            }
122    //                            else if (KewApiConstants.SEARCHABLE_XML_ATTRIBUTE_TYPE.equals(attributeType)) {
123    //                                searchAttributeXml = searchAttributeXml + "<ruleAttribute>\n\t<name>";
124    //                                searchAttributeXml = searchAttributeXml + dbAnswer.getString("RULE_ATTRIB_NM");
125    //                                searchAttributeXml = searchAttributeXml + "</name>\n\t<className>";
126    //                                searchAttributeXml = searchAttributeXml + className;
127    //                                searchAttributeXml = searchAttributeXml + "</className>\n\t<label>";
128    //                                searchAttributeXml = searchAttributeXml + dbAnswer.getString("RULE_ATTRIB_LBL_TXT");
129    //                                searchAttributeXml = searchAttributeXml + "</label>\n\t<description>";
130    //                                searchAttributeXml = searchAttributeXml + dbAnswer.getString("RULE_ATTRIB_DESC");
131    //                                searchAttributeXml = searchAttributeXml + "</description>\n\t<type>";
132    //                                searchAttributeXml = searchAttributeXml + attributeType;
133    //                                searchAttributeXml = searchAttributeXml + "</type>\n\t" + dbAnswer.getString("RULE_ATTRIB_XML_RTE_TXT") + "\n</ruleAttribute>\n";
134    //
135    //                            }
136    //                        }
137    //                        else {
138    //                            LOG.debug("Skipping attribute with class name " + className);
139    //                        }
140    //                    }
141    //                    catch (ClassNotFoundException cnfe) {
142    //                        LOG.debug("Could not find class for name '" + className + "'");
143    //                    }
144    //                }
145    //            }
146    //            ruleAttributeXml = ruleAttributeXml + "</ruleAttributes>\n</data>\n";
147    //            searchAttributeXml = searchAttributeXml + "</ruleAttributes>\n</data>\n";
148    //
149    //            ruleAttributeXml = ruleAttributeXml.replaceAll(" & ", " &amp; ");
150    //            searchAttributeXml = searchAttributeXml.replaceAll(" & ", " &amp; ");
151    //
152    //            loadDataDictionaryEntries();
153    //        }
154    //        catch (Exception e) {
155    //            e.printStackTrace();
156    //        }
157    //        finally {
158    //            try {
159    //                dbCon.close();
160    //            }
161    //            catch (SQLException sqle2) {
162    //                sqle2.printStackTrace();
163    //            }
164    //        }
165        }
166    
167        /**
168         * goes through all of the ruleAttributes in the inputSource and tries to get a label out of the data dictionary
169         */
170        @Test public void testConfirmLabels() {
171            boolean failed = false;
172            // test rule xml attributes
173            failed |= confirmLabels(KualiXmlAttributeHelper.notFound, ruleAttributeXml, RULE_ATTRIBUTE_CONFIG_NODE_NAME);
174    
175            // test search xml attributes
176            failed |= confirmLabels(KualiXmlAttributeHelper.notFound, searchAttributeXml, SEARCH_ATTRIBUTE_CONFIG_NODE_NAME);
177            
178            assertFalse("At least one label was incorrect", failed);
179        }
180    
181        /**
182         * accepts a Node, and if all goes well, returns the exact same Node with expected changes
183         *
184         * <p>The node should have the name and title attributes added to
185         * the fieldDef element. This exercises the getConfigXML method on the class under test.</p>
186         * 
187         * @param xmlNode - an input node
188         * @return the input node with attributes added
189         * @throws TransformerException
190         */
191        private Node configureRuleAttribute(Node xmlNode, KualiXmlAttribute myAttribute) throws TransformerException {
192            ExtensionDefinition.Builder extensionDefinition = ExtensionDefinition.Builder.create("fakeName", "fakeType", "fakeResourceDescriptor");
193    
194            StringWriter xmlBuffer = new StringWriter();
195            Source source = new DOMSource(xmlNode);
196            Result result = new StreamResult(xmlBuffer);
197            TransformerFactory.newInstance().newTransformer().transform(source, result);
198    
199            extensionDefinition.getConfiguration().put(KewApiConstants.ATTRIBUTE_XML_CONFIG_DATA, new String(xmlBuffer.getBuffer()));
200    
201            if (LOG.isDebugEnabled()) {
202                LOG.debug("This is the XML that was added to the attribute");
203                LOG.debug(new String(xmlBuffer.getBuffer()));
204                StringWriter xmlBuffer2 = new StringWriter();
205                Source source2 = new DOMSource(xmlNode);
206                Result result2 = new StreamResult(xmlBuffer2);
207                TransformerFactory.newInstance().newTransformer().transform(source2, result2);
208                LOG.debug("This is the XML that was returned from the ruleAttribute");
209                LOG.debug(new String(xmlBuffer2.getBuffer()));
210            }
211            return myAttribute.getConfigXML(extensionDefinition.build());
212        }
213    
214        /**
215         * compares the label from the test to the expected, or not expected, value for all of the rule attributes in the file
216         *
217         * <p>The inputSource file should be as close to the production version as possible, as described by the class comments. It
218         * accepts the string to test against as a parameter.</p>
219         * 
220         * @param testString
221         * @param attributeXml
222         * @param configNodeName
223         */
224        private boolean confirmLabels(String testString, String attributeXml, String configNodeName) {
225            boolean testFailed = false;
226            String theTitle = "";
227            String theName = "";
228            String attributeName = "";
229            try {
230                NodeList tempList = (NodeList) myXPath.evaluate("//ruleAttribute", new InputSource(new StringReader(attributeXml)), XPathConstants.NODESET);
231                for (int i = 0; i < tempList.getLength(); i++) { // loop over ruleattributes
232                    Node originalNode = tempList.item(i);
233                    Set ruleAttributeFieldDefNames = new HashSet();
234                    Set ruleAttributeFieldDefTitles = new HashSet();
235                    attributeName = (String) myXPath.evaluate(WorkflowUtils.XSTREAM_MATCH_RELATIVE_PREFIX + "name", originalNode, XPathConstants.STRING);
236                    Node classNameNode = (Node) myXPath.evaluate(WorkflowUtils.XSTREAM_MATCH_RELATIVE_PREFIX + "className", originalNode, XPathConstants.NODE);
237                    if ((classNameNode != null) && (classNameNode.getFirstChild() != null)) {
238                        if (LOG.isInfoEnabled()) {
239                            LOG.info("Checking attribute with name '" + attributeName + "'");
240                        }
241                        KualiXmlAttribute myAttribute = (KualiXmlAttribute) GlobalResourceLoader.getObject(new ObjectDefinition(classNameNode.getFirstChild().getNodeValue()));
242                        Node xmlNode = configureRuleAttribute(originalNode, myAttribute);
243                        NamedNodeMap fieldDefAttributes = null;
244                        String potentialFailMessage = "";
245    
246                        try {
247                            NodeList xmlNodeList = (NodeList) myXPath.evaluate("//fieldDef", xmlNode, XPathConstants.NODESET);
248    
249                            for (int j = 0; j < xmlNodeList.getLength(); j++) {
250                                Node fieldDefXmlNode = xmlNodeList.item(j);
251                                fieldDefAttributes = fieldDefXmlNode.getAttributes();
252    
253                                theTitle = fieldDefAttributes.getNamedItem("title").getNodeValue();// Making sure they are clean
254                                theName = fieldDefAttributes.getNamedItem("name").getNodeValue();
255                                if (LOG.isDebugEnabled()) {
256                                    LOG.debug(attributeName);
257                                    LOG.debug("name=" + theName + "   title=" + theTitle);
258                                }
259                                if (ruleAttributeFieldDefNames.contains(theName)) {
260                                    // names of fieldDefs inside a single attribute must be unique
261                                    potentialFailMessage = "Each fieldDef name on a single attribute must be unique and the fieldDef name '" + theName + "' already exists on the attribute '" + attributeName + "'";
262                                    fail(potentialFailMessage);
263                                }
264                                else {
265                                    ruleAttributeFieldDefNames.add(theName);
266                                }
267                                if (testString.equals(KualiXmlAttributeHelper.notFound)) {
268                                    potentialFailMessage = "Each fieldDef title should be a valid value and currently the title for attribute '" + attributeName + "' is '" + theTitle + "'";
269                                    assertFalse(potentialFailMessage, theTitle.equals(testString));
270                                    if (ruleAttributeFieldDefTitles.contains(theTitle)) {
271                                        /*
272                                         * Titles of fieldDefs inside a single attribute should be unique in the normal case. Having two
273                                         * fields with the same label would certainly confuse the user. However, due to the way the
274                                         * confirmSource test works, all the titles/labels must be the same. So only run this check when
275                                         * not in the confirmSource test.
276                                         */
277                                        potentialFailMessage = "Each fieldDef title on a single attribute must be unique and the fieldDef title '" + theTitle + "' already exists on the attribute '" + attributeName + "'";
278                                        fail(potentialFailMessage);
279                                    }
280                                    else {
281                                        ruleAttributeFieldDefTitles.add(theTitle);
282                                    }
283                                }
284                                else {
285                                    potentialFailMessage = "For attribute '" + attributeName + "' the title should have been '" + testString + "' but was actually '" + theTitle + "'";
286                                    assertEquals(potentialFailMessage, testString, theTitle);
287                                }
288                            }
289                        }
290                        catch (AssertionError afe) {
291                            LOG.warn("Assertion Failed for attribute '" + attributeName + "' with error " + potentialFailMessage, afe);
292                            testFailed = true;
293                        }
294                        finally {
295                            attributeName = "";
296                        }
297                    }
298                    else {
299                        throw new RuntimeException("Could not find class for attribute named '" + attributeName + "'");
300                    }
301                }
302            }
303            catch (Exception e) {
304                LOG.error("General Exception thrown for attribute '" + attributeName + "'", e);
305                testFailed = true;
306            }
307            return testFailed;
308        }
309    
310        /**
311         * confirms that the labels are coming from the data dictionary
312         *
313         * <p>This is done by modifying all the dictionary values
314         * programatically to a nonsense value. It then rebuilds the Hash Table and runs confirmLabels() to make sure the labels have
315         * changed.</p>
316         */
317        @Test public void testLabelSource() {
318            DataDictionaryService myDDService = KRADServiceLocatorWeb.getDataDictionaryService();
319            XPath xpath = XPathHelper.newXPath();
320            String nonsenseString = "BananaRama";
321            for (Object tempEntity : myDDService.getDataDictionary().getBusinessObjectEntries().values()) {
322    
323                for ( AttributeDefinition attribute : ((BusinessObjectEntry) tempEntity).getAttributes() ) {
324                    attribute.setLabel(nonsenseString);
325                    attribute.setShortLabel(nonsenseString);
326                }
327    
328            }
329            for (Object tempEntity : myDDService.getDataDictionary().getDocumentEntries().values()) {
330    
331                for ( AttributeDefinition attribute : ((DocumentEntry) tempEntity).getAttributes() ) {
332                    attribute.setLabel(nonsenseString);
333                    attribute.setShortLabel(nonsenseString);
334                }
335    
336            }
337            // KualiXmlAttributeHelper.buildDictionaryHash();
338    
339            boolean failed = false;
340            assertFalse("At least one label was incorrect", failed);
341            // test rule xml attributes
342            failed |= confirmLabels(nonsenseString, ruleAttributeXml, RULE_ATTRIBUTE_CONFIG_NODE_NAME);
343    
344            // test search xml attributes
345            failed |= confirmLabels(nonsenseString, searchAttributeXml, SEARCH_ATTRIBUTE_CONFIG_NODE_NAME);
346    
347            assertFalse("At least one label was incorrect", failed);
348        }
349    
350        /*
351        unused method
352    
353        private void loadDataDictionaryEntries() throws Exception {
354            KualiXmlRuleAttributeImpl myAttribute = new KualiXmlRuleAttributeImpl();
355            NamedNodeMap fieldDefAttributes = null;
356            NodeList tempList = (NodeList) myXPath.evaluate("//ruleAttribute", new InputSource(new StringReader(ruleAttributeXml)), XPathConstants.NODESET);
357            for (int i = 0; i < tempList.getLength(); i++) {
358                Node xmlNode = configureRuleAttribute(tempList.item(i), myAttribute);
359            }
360            KualiXmlSearchableAttributeImpl mySearchAttribute = new KualiXmlSearchableAttributeImpl();
361            fieldDefAttributes = null;
362            tempList = (NodeList) myXPath.evaluate("//ruleAttribute", new InputSource(new StringReader(searchAttributeXml)), XPathConstants.NODESET);
363            for (int i = 0; i < tempList.getLength(); i++) {
364                Node xmlNode = configureRuleAttribute(tempList.item(i), mySearchAttribute);
365            }
366        }*/
367    }