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