Coverage Report - org.kuali.rice.kew.xml.RuleTemplateXmlParser
 
Classes in this File Line Coverage Branch Coverage Complexity
RuleTemplateXmlParser
0%
0/190
0%
0/84
5.833
 
 1  
 /*
 2  
  * Copyright 2005-2007 The Kuali Foundation
 3  
  *
 4  
  *
 5  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 6  
  * you may not use this file except in compliance with the License.
 7  
  * You may obtain a copy of the License at
 8  
  *
 9  
  * http://www.opensource.org/licenses/ecl2.php
 10  
  *
 11  
  * Unless required by applicable law or agreed to in writing, software
 12  
  * distributed under the License is distributed on an "AS IS" BASIS,
 13  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 14  
  * See the License for the specific language governing permissions and
 15  
  * limitations under the License.
 16  
  */
 17  
 package org.kuali.rice.kew.xml;
 18  
 
 19  
 import java.io.IOException;
 20  
 import java.io.InputStream;
 21  
 import java.sql.Timestamp;
 22  
 import java.text.ParseException;
 23  
 import java.util.ArrayList;
 24  
 import java.util.Iterator;
 25  
 import java.util.List;
 26  
 import java.util.Vector;
 27  
 
 28  
 import javax.xml.parsers.ParserConfigurationException;
 29  
 
 30  
 import org.apache.commons.lang.BooleanUtils;
 31  
 import org.apache.commons.lang.StringUtils;
 32  
 import org.jdom.Attribute;
 33  
 import org.jdom.Document;
 34  
 import org.jdom.Element;
 35  
 import org.jdom.JDOMException;
 36  
 import org.kuali.rice.core.util.RiceConstants;
 37  
 import org.kuali.rice.kew.exception.InvalidXmlException;
 38  
 import org.kuali.rice.kew.rule.RuleBaseValues;
 39  
 import org.kuali.rice.kew.rule.RuleDelegation;
 40  
 import org.kuali.rice.kew.rule.RuleTemplateOption;
 41  
 import org.kuali.rice.kew.rule.bo.RuleAttribute;
 42  
 import org.kuali.rice.kew.rule.bo.RuleTemplate;
 43  
 import org.kuali.rice.kew.rule.bo.RuleTemplateAttribute;
 44  
 import org.kuali.rice.kew.service.KEWServiceLocator;
 45  
 import org.kuali.rice.kew.util.KEWConstants;
 46  
 import org.kuali.rice.kew.util.Utilities;
 47  
 import org.kuali.rice.kew.util.XmlHelper;
 48  
 import org.xml.sax.SAXException;
 49  
 
 50  
 
 51  
 /**
 52  
  * Parses {@link RuleTemplate}s from XML.
 53  
  *
 54  
  * @see RuleTemplate
 55  
  *
 56  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 57  
  */
 58  0
 public class RuleTemplateXmlParser implements XmlConstants {
 59  
 
 60  0
     private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(RuleTemplateXmlParser.class);
 61  
 
 62  
     /**
 63  
      * By default make attributes defined without a <required> element
 64  
      */
 65  
     private static final boolean DEFAULT_ATTRIBUTE_REQUIRED = true;
 66  
     private static final boolean DEFAULT_ATTRIBUTE_ACTIVE = true;
 67  
 
 68  
     /**
 69  
      * A dummy document type used in the default rule
 70  
      */
 71  
     private static final String DUMMY_DOCUMENT_TYPE = "dummyDocumentType";
 72  
 
 73  
     /**
 74  
      * Used to set the display order of attributes encountered in parsing runs during the lifetime of this object
 75  
      */
 76  0
     private int templateAttributeCounter = 0;
 77  
 
 78  
     public List<RuleTemplate> parseRuleTemplates(InputStream input) throws IOException, InvalidXmlException {
 79  
 
 80  
         try {
 81  0
             Document doc = XmlHelper.trimSAXXml(input);
 82  0
             Element root = doc.getRootElement();
 83  0
             return parseRuleTemplates(root);
 84  0
         } catch (JDOMException e) {
 85  0
             throw new InvalidXmlException("Parse error.", e);
 86  0
         } catch (SAXException e) {
 87  0
             throw new InvalidXmlException("Parse error.", e);
 88  0
         } catch (ParserConfigurationException e) {
 89  0
             throw new InvalidXmlException("Parse error.", e);
 90  
         }
 91  
     }
 92  
 
 93  
     public List<RuleTemplate> parseRuleTemplates(Element element) throws InvalidXmlException {
 94  0
         List<RuleTemplate> ruleTemplates = new ArrayList<RuleTemplate>();
 95  
 
 96  
         // iterate over any RULE_TEMPLATES elements
 97  0
         Vector ruleTemplatesElements = XmlHelper.findElements(element, RULE_TEMPLATES);
 98  0
         Iterator ruleTemplatesIterator = ruleTemplatesElements.iterator();
 99  0
         while (ruleTemplatesIterator.hasNext()) {
 100  0
             Element ruleTemplatesElement = (Element) ruleTemplatesIterator.next();
 101  0
             Vector ruleTemplateElements = XmlHelper.findElements(ruleTemplatesElement, RULE_TEMPLATE);
 102  0
             for (Iterator iterator = ruleTemplateElements.iterator(); iterator.hasNext();) {
 103  0
                 ruleTemplates.add(parseRuleTemplate((Element) iterator.next(), ruleTemplates));
 104  
             }
 105  0
         }
 106  0
         return ruleTemplates;
 107  
     }
 108  
 
 109  
     private RuleTemplate parseRuleTemplate(Element element, List<RuleTemplate> ruleTemplates) throws InvalidXmlException {
 110  0
         String name = element.getChildText(NAME, RULE_TEMPLATE_NAMESPACE);
 111  0
         String description = element.getChildText(DESCRIPTION, RULE_TEMPLATE_NAMESPACE);
 112  0
         Attribute allowOverwriteAttrib = element.getAttribute("allowOverwrite");
 113  
 
 114  0
         boolean allowOverwrite = false;
 115  0
         if (allowOverwriteAttrib != null) {
 116  0
             allowOverwrite = Boolean.valueOf(allowOverwriteAttrib.getValue()).booleanValue();
 117  
         }
 118  0
         if (Utilities.isEmpty(name)) {
 119  0
             throw new InvalidXmlException("RuleTemplate must have a name");
 120  
         }
 121  0
         if (Utilities.isEmpty(description)) {
 122  0
             throw new InvalidXmlException("RuleTemplate must have a description");
 123  
         }
 124  
 
 125  
         // look up the rule template by name first
 126  0
         RuleTemplate ruleTemplate = KEWServiceLocator.getRuleTemplateService().findByRuleTemplateName(name);
 127  
 
 128  0
         if (ruleTemplate == null) {
 129  
             // if it does not exist create a new one
 130  0
             ruleTemplate = new RuleTemplate();
 131  
         } else {
 132  
             // if it does exist, update it, only if allowOverwrite is set
 133  0
             if (!allowOverwrite) {
 134  0
                 throw new RuntimeException("Attempting to overwrite template " + name + " without allowOverwrite set");
 135  
             }
 136  
 
 137  
             // the name should be equal if one was actually found
 138  0
             assert(name.equals(ruleTemplate.getName())) : "Existing template definition name does not match incoming definition name";
 139  
         } 
 140  
 
 141  
         // overwrite simple properties
 142  0
         ruleTemplate.setName(name);
 143  0
         ruleTemplate.setDescription(description);
 144  
 
 145  
         // update the delegation template
 146  0
         updateDelegationTemplate(element, ruleTemplate, ruleTemplates);
 147  
 
 148  
         // update the attribute relationships
 149  0
         updateRuleTemplateAttributes(element, ruleTemplate);
 150  
 
 151  
         // save the rule template first so that the default/template rule that is generated
 152  
         // in the process of setting defaults is associated properly with this rule template
 153  0
         KEWServiceLocator.getRuleTemplateService().save(ruleTemplate);
 154  
 
 155  
         // update the default options
 156  0
         updateRuleTemplateDefaultOptions(element, ruleTemplate);
 157  
 
 158  0
         KEWServiceLocator.getRuleTemplateService().save(ruleTemplate);
 159  
 
 160  0
         return ruleTemplate;
 161  
     }
 162  
 
 163  
     /**
 164  
      * Updates the rule template default options.  Updates any existing options, removes any omitted ones.
 165  
      * @param ruleTemplateElement the rule template XML element
 166  
      * @param updatedRuleTemplate the RuleTemplate being updated
 167  
      * @throws InvalidXmlException
 168  
      */
 169  
     /*
 170  
      <element name="description" type="c:LongStringType"/>
 171  
      <element name="fromDate" type="c:ShortStringType" minOccurs="0"/>
 172  
      <element name="toDate" type="c:ShortStringType" minOccurs="0"/>
 173  
      <element name="forceAction" type="boolean"/>
 174  
      <element name="active" type="boolean"/>
 175  
      <element name="defaultActionRequested" type="c:ShortStringType"/>
 176  
      <element name="supportsComplete" type="boolean" default="true"/>
 177  
      <element name="supportsApprove" type="boolean" default="true"/>
 178  
      <element name="supportsAcknowledge" type="boolean" default="true"/>
 179  
      <element name="supportsFYI" type="boolean" default="true"/>
 180  
     */
 181  
     protected void updateRuleTemplateDefaultOptions(Element ruleTemplateElement, RuleTemplate updatedRuleTemplate) throws InvalidXmlException {
 182  0
         Element defaultsElement = ruleTemplateElement.getChild(RULE_DEFAULTS, RULE_TEMPLATE_NAMESPACE);
 183  
 
 184  
         // update the rule defaults; this yields whether or not this is a delegation rule template
 185  0
         boolean isDelegation = updateRuleDefaults(defaultsElement, updatedRuleTemplate);
 186  
 
 187  
         // update the rule template options
 188  0
         updateRuleTemplateOptions(defaultsElement, updatedRuleTemplate, isDelegation);
 189  
 
 190  0
     }
 191  
 
 192  
     /**
 193  
      * Updates the rule template defaults options with those in the defaults element
 194  
      * @param defaultsElement the ruleDefaults element
 195  
      * @param updatedRuleTemplate the Rule Template being updated
 196  
      */
 197  
     protected void updateRuleTemplateOptions(Element defaultsElement, RuleTemplate updatedRuleTemplate, boolean isDelegation) throws InvalidXmlException {
 198  
         // the possible defaults options
 199  
         // NOTE: the current implementation will remove any existing RuleTemplateOption records for any values which are null, i.e. not set in the incoming XML.
 200  
         // to pro-actively set default values for omitted options, simply set those values here, and records will be added if not present
 201  0
         String defaultActionRequested = null;
 202  0
         Boolean supportsComplete = null;
 203  0
         Boolean supportsApprove = null;
 204  0
         Boolean supportsAcknowledge = null;
 205  0
         Boolean supportsFYI = null;
 206  
         
 207  
         // remove any RuleTemplateOptions the template may have but that we know we aren't going to update/reset
 208  
         // (not sure if this case even exists...does anything else set rule template options?)
 209  0
         updatedRuleTemplate.removeNonDefaultOptions();
 210  
         
 211  
         // read in new settings
 212  0
         if (defaultsElement != null) {
 213  
 
 214  0
                 defaultActionRequested = defaultsElement.getChildText(DEFAULT_ACTION_REQUESTED, RULE_TEMPLATE_NAMESPACE);
 215  0
             supportsComplete = BooleanUtils.toBooleanObject(defaultsElement.getChildText(SUPPORTS_COMPLETE, RULE_TEMPLATE_NAMESPACE));
 216  0
             supportsApprove = BooleanUtils.toBooleanObject(defaultsElement.getChildText(SUPPORTS_APPROVE, RULE_TEMPLATE_NAMESPACE));
 217  0
             supportsAcknowledge = BooleanUtils.toBooleanObject(defaultsElement.getChildText(SUPPORTS_ACKNOWLEDGE, RULE_TEMPLATE_NAMESPACE));
 218  0
             supportsFYI = BooleanUtils.toBooleanObject(defaultsElement.getChildText(SUPPORTS_FYI, RULE_TEMPLATE_NAMESPACE));
 219  
         }
 220  
 
 221  0
         if (!isDelegation) {
 222  
             // if this is not a delegation template, store the template options that govern rule action constraints
 223  
             // in the RuleTemplateOptions of the template
 224  
             // we have two options for this behavior:
 225  
             // 1) conditionally parse above, and then unconditionally set/unset the properties; this will have the effect of REMOVING
 226  
             //    any of these previously specified rule template options (and is arguably the right thing to do)
 227  
             // 2) unconditionally parse above, and then conditionally set/unset the properties; this will have the effect of PRESERVING
 228  
             //    the existing rule template options on this template if it is a delegation template (which of course will be overwritten
 229  
             //    by this very same code if they subsequently upload without the delegation flag)
 230  
             // This is a minor point, but the second implementation is chosen as it preserved the current behavior
 231  0
             updateOrDeleteRuleTemplateOption(updatedRuleTemplate, KEWConstants.ACTION_REQUEST_DEFAULT_CD, defaultActionRequested);
 232  0
             updateOrDeleteRuleTemplateOption(updatedRuleTemplate, KEWConstants.ACTION_REQUEST_APPROVE_REQ, supportsApprove);
 233  0
             updateOrDeleteRuleTemplateOption(updatedRuleTemplate, KEWConstants.ACTION_REQUEST_ACKNOWLEDGE_REQ, supportsAcknowledge);
 234  0
             updateOrDeleteRuleTemplateOption(updatedRuleTemplate, KEWConstants.ACTION_REQUEST_FYI_REQ, supportsFYI);
 235  0
             updateOrDeleteRuleTemplateOption(updatedRuleTemplate, KEWConstants.ACTION_REQUEST_COMPLETE_REQ, supportsComplete);
 236  
         }
 237  
 
 238  0
     }
 239  
     
 240  
     /**
 241  
      * 
 242  
      * Updates the default/template rule options with those in the defaults element
 243  
      * @param defaultsElement the ruleDefaults element
 244  
      * @param updatedRuleTemplate the Rule Template being updated
 245  
      * @return whether this is a delegation rule template
 246  
      */
 247  
     protected boolean updateRuleDefaults(Element defaultsElement, RuleTemplate updatedRuleTemplate) throws InvalidXmlException {
 248  
         // NOTE: implementation detail: in contrast with the other options, the delegate template, and the rule attributes,
 249  
         // we unconditionally blow away the default rule and re-create it (we don't update the existing one, if there is one)
 250  0
         if (updatedRuleTemplate.getRuleTemplateId() != null) {
 251  0
             RuleBaseValues ruleDefaults = KEWServiceLocator.getRuleService().findDefaultRuleByRuleTemplateId(updatedRuleTemplate.getRuleTemplateId());
 252  0
             if (ruleDefaults != null) {
 253  0
                 List ruleDelegationDefaults = KEWServiceLocator.getRuleDelegationService().findByDelegateRuleId(ruleDefaults.getRuleBaseValuesId());
 254  
                 // delete the rule
 255  0
                 KEWServiceLocator.getRuleService().delete(ruleDefaults.getRuleBaseValuesId());
 256  
                 // delete the associated rule delegation defaults
 257  0
                 for (Iterator iterator = ruleDelegationDefaults.iterator(); iterator.hasNext();) {
 258  0
                     RuleDelegation ruleDelegation = (RuleDelegation) iterator.next();
 259  0
                     KEWServiceLocator.getRuleDelegationService().delete(ruleDelegation.getRuleDelegationId());
 260  0
                 }
 261  
             }
 262  
         }
 263  
 
 264  0
         boolean isDelegation = false;
 265  
 
 266  0
         if (defaultsElement != null) {
 267  0
             String delegationType = defaultsElement.getChildText(DELEGATION_TYPE, RULE_TEMPLATE_NAMESPACE);
 268  0
             isDelegation = !Utilities.isEmpty(delegationType);
 269  
 
 270  0
             String description = defaultsElement.getChildText(DESCRIPTION, RULE_TEMPLATE_NAMESPACE);
 271  
             
 272  
             // would normally be validated via schema but might not be present if invoking RuleXmlParser directly
 273  0
             if (description == null) {
 274  0
                 throw new InvalidXmlException("Description must be specified in rule defaults");
 275  
             }
 276  
             
 277  0
             String fromDate = defaultsElement.getChildText(FROM_DATE, RULE_TEMPLATE_NAMESPACE);
 278  0
             String toDate = defaultsElement.getChildText(TO_DATE, RULE_TEMPLATE_NAMESPACE);
 279  
             // toBooleanObject ensures that if the value is null (not set) that the Boolean object will likewise be null (will not default to a value)
 280  0
             Boolean forceAction = BooleanUtils.toBooleanObject(defaultsElement.getChildText(FORCE_ACTION, RULE_TEMPLATE_NAMESPACE));
 281  0
             Boolean active = BooleanUtils.toBooleanObject(defaultsElement.getChildText(ACTIVE, RULE_TEMPLATE_NAMESPACE));
 282  
 
 283  0
             if (isDelegation && !KEWConstants.DELEGATION_PRIMARY.equals(delegationType) && !KEWConstants.DELEGATION_SECONDARY.equals(delegationType)) {
 284  0
                 throw new InvalidXmlException("Invalid delegation type '" + delegationType + "'." + "  Expected one of: "
 285  
                         + KEWConstants.DELEGATION_PRIMARY + "," + KEWConstants.DELEGATION_SECONDARY);
 286  
             }
 287  
     
 288  
             // create our "default rule" which encapsulates the defaults for the rule
 289  0
             RuleBaseValues ruleDefaults = new RuleBaseValues();
 290  
     
 291  
             // set simple values
 292  0
             ruleDefaults.setRuleTemplate(updatedRuleTemplate);
 293  0
             ruleDefaults.setDocTypeName(DUMMY_DOCUMENT_TYPE);
 294  0
             ruleDefaults.setTemplateRuleInd(Boolean.TRUE);
 295  0
             ruleDefaults.setCurrentInd(Boolean.TRUE);
 296  0
             ruleDefaults.setVersionNbr(new Integer(0));
 297  0
             ruleDefaults.setDescription(description);
 298  
     
 299  
             // these are non-nullable fields, so default them if they were not set in the defaults section
 300  0
             ruleDefaults.setForceAction(Boolean.valueOf(BooleanUtils.isTrue(forceAction)));
 301  0
             ruleDefaults.setActiveInd(Boolean.valueOf(BooleanUtils.isTrue(active)));
 302  
         
 303  0
             if (ruleDefaults.getActivationDate() == null) {
 304  0
                 ruleDefaults.setActivationDate(new Timestamp(System.currentTimeMillis()));
 305  
             }
 306  
     
 307  0
             ruleDefaults.setFromDate(formatDate("fromDate",fromDate));
 308  0
             ruleDefaults.setToDate(formatDate("toDate", toDate));
 309  
             
 310  
             // ok, if this is a "Delegate Template", then we need to set this other RuleDelegation object which contains
 311  
             // some delegation-related info
 312  0
             RuleDelegation ruleDelegationDefaults = null;
 313  0
             if (isDelegation) {
 314  0
                 ruleDelegationDefaults = new RuleDelegation();
 315  0
                 ruleDelegationDefaults.setDelegationRuleBaseValues(ruleDefaults);
 316  0
                 ruleDelegationDefaults.setDelegationType(delegationType);
 317  0
                 ruleDelegationDefaults.setResponsibilityId(new Long(-1));
 318  
             }
 319  
 
 320  
             // explicitly save the new rule delegation defaults and default rule
 321  0
             KEWServiceLocator.getRuleTemplateService().saveRuleDefaults(ruleDelegationDefaults, ruleDefaults);
 322  
         } else {
 323  
             // do nothing, rule defaults will be deleted if ruleDefaults element is omitted
 324  
         }
 325  
         
 326  0
         return isDelegation;
 327  
     }
 328  
 
 329  
 
 330  
     /**
 331  
      * Updates or deletes a specified rule template option on the rule template
 332  
      * @param updatedRuleTemplate the RuleTemplate being updated
 333  
      * @param key the option key
 334  
      * @param value the option value
 335  
      */
 336  
     protected void updateOrDeleteRuleTemplateOption(RuleTemplate updatedRuleTemplate, String key, Object value) {
 337  0
         if (value != null) {
 338  
             // if the option exists and the incoming value is non-null (it's set), update it
 339  0
             RuleTemplateOption option = updatedRuleTemplate.getRuleTemplateOption(key);
 340  0
             if (option != null) {
 341  0
                 option.setValue(value.toString());
 342  
             } else {
 343  0
                 updatedRuleTemplate.getRuleTemplateOptions().add(new RuleTemplateOption(key, value.toString()));
 344  
             }
 345  0
         } else {
 346  
             // otherwise if the incoming value IS null (not set), then explicitly remove the entry (if it exists)
 347  0
             Iterator<RuleTemplateOption> options = updatedRuleTemplate.getRuleTemplateOptions().iterator();
 348  0
             while (options.hasNext()) {
 349  0
                 RuleTemplateOption opt = options.next();
 350  0
                 if (key.equals(opt.getKey())) {
 351  0
                     options.remove();
 352  0
                     break;
 353  
                 }
 354  0
             }
 355  
         }
 356  0
     }
 357  
 
 358  
     /**
 359  
      * Updates the rule template delegation template with the one specified in the XML (if any)
 360  
      * @param ruleTemplateElement the XML ruleTemplate element
 361  
      * @param updatedRuleTemplate the rule template to update
 362  
      * @param parsedRuleTemplates the rule templates parsed in this parsing run
 363  
      * @throws InvalidXmlException if a delegation template was specified but could not be found
 364  
      */
 365  
     protected void updateDelegationTemplate(Element ruleTemplateElement, RuleTemplate updatedRuleTemplate, List<RuleTemplate> parsedRuleTemplates) throws InvalidXmlException {
 366  0
         String delegateTemplateName = ruleTemplateElement.getChildText(DELEGATION_TEMPLATE, RULE_TEMPLATE_NAMESPACE);
 367  
 
 368  0
         if (delegateTemplateName != null) {
 369  
             // if a delegateTemplate was set in the XML, then look it up and set it on the RuleTemplate object
 370  
             // first try looking up an existing delegateTemplate in the system
 371  0
             RuleTemplate delegateTemplate = KEWServiceLocator.getRuleTemplateService().findByRuleTemplateName(delegateTemplateName);
 372  
 
 373  
             // if not found, try the list of templates currently parsed
 374  0
             if (delegateTemplate == null) {
 375  0
                 for (RuleTemplate rt: parsedRuleTemplates) {
 376  0
                     if (delegateTemplateName.equalsIgnoreCase(rt.getName())) {
 377  
                         // set the expected next rule template id on the target delegateTemplate
 378  0
                         Long ruleTemplateId = KEWServiceLocator.getRuleTemplateService().getNextRuleTemplateId();
 379  0
                         rt.setRuleTemplateId(ruleTemplateId);
 380  0
                         delegateTemplate = rt;
 381  0
                         break;
 382  
                     }
 383  
                 }
 384  
             }
 385  
 
 386  0
             if (delegateTemplate == null) {
 387  0
                 throw new InvalidXmlException("Cannot find delegation template " + delegateTemplateName);
 388  
             }
 389  
 
 390  0
             updatedRuleTemplate.setDelegationTemplateId(delegateTemplate.getDelegationTemplateId());
 391  0
             updatedRuleTemplate.setDelegationTemplate(delegateTemplate);           
 392  
         } else {
 393  
             // the previously referenced template is left in the system
 394  
         }
 395  0
     }
 396  
 
 397  
     /**
 398  
      * Updates the attributes set on the RuleTemplate
 399  
      * @param ruleTemplateElement the XML ruleTemplate element
 400  
      * @param updatedRuleTemplate the RuleTemplate being updated
 401  
      * @throws InvalidXmlException if there was a problem parsing the rule template attributes
 402  
      */
 403  
     protected void updateRuleTemplateAttributes(Element ruleTemplateElement, RuleTemplate updatedRuleTemplate) throws InvalidXmlException {
 404  
         // add any newly defined rule template attributes to the rule template,
 405  
         // update the active and required flags of any existing ones.
 406  
         // if this is an update of an existing rule template, related attribute objects will be present in this rule template object,
 407  
         // otherwise none will be present (so they'll all be new)
 408  
 
 409  0
         Element attributesElement = ruleTemplateElement.getChild(ATTRIBUTES, RULE_TEMPLATE_NAMESPACE);
 410  0
         List<RuleTemplateAttribute> incomingAttributes = new ArrayList<RuleTemplateAttribute>();
 411  0
         if (attributesElement != null) {
 412  0
             incomingAttributes.addAll(parseRuleTemplateAttributes(attributesElement, updatedRuleTemplate));
 413  
         }
 414  
 
 415  
         // inactivate all current attributes
 416  0
         for (RuleTemplateAttribute currentRuleTemplateAttribute: updatedRuleTemplate.getRuleTemplateAttributes()) {
 417  0
             String ruleAttributeName = (currentRuleTemplateAttribute.getRuleAttribute() != null) ? currentRuleTemplateAttribute.getRuleAttribute().getName() : "(null)";
 418  0
             LOG.debug("Inactivating rule template attribute with id " + currentRuleTemplateAttribute.getRuleTemplateAttributeId() + " and rule attribute with name " + ruleAttributeName);
 419  0
             currentRuleTemplateAttribute.setActive(Boolean.FALSE);
 420  0
         }
 421  
         // NOTE: attributes are deactivated, not removed
 422  
 
 423  
         // add/update any new attributes
 424  0
         for (RuleTemplateAttribute ruleTemplateAttribute: incomingAttributes) {
 425  0
             RuleTemplateAttribute potentialExistingTemplateAttribute = updatedRuleTemplate.getRuleTemplateAttribute(ruleTemplateAttribute);
 426  0
             if (potentialExistingTemplateAttribute != null) {
 427  
                 // template attribute exists on rule template already; update the options
 428  0
                 potentialExistingTemplateAttribute.setActive(ruleTemplateAttribute.getActive());
 429  0
                 potentialExistingTemplateAttribute.setRequired(ruleTemplateAttribute.getRequired());
 430  
             } else {
 431  
                 // template attribute does not yet exist on template so add it
 432  0
                 updatedRuleTemplate.getRuleTemplateAttributes().add(ruleTemplateAttribute);
 433  
             }
 434  0
         }
 435  0
     }
 436  
 
 437  
     /**
 438  
      * Parses the RuleTemplateAttributes defined on the rule template element
 439  
      * @param attributesElement the jdom Element object for the Rule Template attributes
 440  
      * @param ruleTemplate the RuleTemplate object
 441  
      * @return the RuleTemplateAttributes defined on the rule template element
 442  
      * @throws InvalidXmlException
 443  
      */
 444  
     private List<RuleTemplateAttribute> parseRuleTemplateAttributes(Element attributesElement, RuleTemplate ruleTemplate) throws InvalidXmlException {
 445  0
         List<RuleTemplateAttribute> ruleTemplateAttributes = new ArrayList<RuleTemplateAttribute>();
 446  0
         Vector attributeElements = XmlHelper.findElements(attributesElement, ATTRIBUTE);
 447  0
         for (Iterator iterator = attributeElements.iterator(); iterator.hasNext();) {
 448  0
             ruleTemplateAttributes.add(parseRuleTemplateAttribute((Element) iterator.next(), ruleTemplate));
 449  
         }
 450  0
         return ruleTemplateAttributes;
 451  
     }
 452  
 
 453  
     /**
 454  
      * Parses a rule template attribute
 455  
      * @param element the attribute XML element
 456  
      * @param ruleTemplate the ruleTemplate to update
 457  
      * @return a parsed rule template attribute
 458  
      * @throws InvalidXmlException if the attribute does not exist
 459  
      */
 460  
     private RuleTemplateAttribute parseRuleTemplateAttribute(Element element, RuleTemplate ruleTemplate) throws InvalidXmlException {
 461  0
         String attributeName = element.getChildText(NAME, RULE_TEMPLATE_NAMESPACE);
 462  0
         String requiredValue = element.getChildText(REQUIRED, RULE_TEMPLATE_NAMESPACE);
 463  0
         String activeValue = element.getChildText(ACTIVE, RULE_TEMPLATE_NAMESPACE);
 464  0
         if (Utilities.isEmpty(attributeName)) {
 465  0
             throw new InvalidXmlException("Attribute name must be non-empty");
 466  
         }
 467  0
         boolean required = DEFAULT_ATTRIBUTE_REQUIRED;
 468  0
         if (requiredValue != null) {
 469  0
             required = Boolean.parseBoolean(requiredValue);
 470  
         }
 471  0
         boolean active = DEFAULT_ATTRIBUTE_ACTIVE;
 472  0
         if (activeValue != null) {
 473  0
             active = Boolean.parseBoolean(activeValue);
 474  
         }
 475  0
         RuleAttribute ruleAttribute = KEWServiceLocator.getRuleAttributeService().findByName(attributeName);
 476  0
         if (ruleAttribute == null) {
 477  0
             throw new InvalidXmlException("Could not locate rule attribute for name '" + attributeName + "'");
 478  
         }
 479  0
         RuleTemplateAttribute templateAttribute = new RuleTemplateAttribute();
 480  0
         templateAttribute.setRuleAttribute(ruleAttribute);
 481  0
         templateAttribute.setRuleAttributeId(ruleAttribute.getRuleAttributeId());
 482  0
         templateAttribute.setRuleTemplate(ruleTemplate);
 483  0
         templateAttribute.setRequired(Boolean.valueOf(required));
 484  0
         templateAttribute.setActive(Boolean.valueOf(active));
 485  0
         templateAttribute.setDisplayOrder(new Integer(templateAttributeCounter++));
 486  0
         return templateAttribute;
 487  
     }
 488  
     
 489  
     public Timestamp formatDate(String dateLabel, String dateString) throws InvalidXmlException {
 490  0
             if (StringUtils.isBlank(dateString)) {
 491  0
                     return null;
 492  
             }
 493  
             try {
 494  0
                     return new Timestamp(RiceConstants.getDefaultDateFormat().parse(dateString).getTime());
 495  0
             } catch (ParseException e) {
 496  0
                     throw new InvalidXmlException(dateLabel + " is not in the proper format.  Should have been: " + RiceConstants.DEFAULT_DATE_FORMAT_PATTERN);
 497  
             }
 498  
     }
 499  
 
 500  
 }