001    /**
002     * Copyright 2005-2014 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.rule;
017    
018    import java.util.ArrayList;
019    import java.util.Collections;
020    import java.util.List;
021    import java.util.Map;
022    
023    import javax.xml.xpath.XPathExpressionException;
024    
025    import org.apache.commons.lang.ObjectUtils;
026    import org.apache.log4j.Logger;
027    import org.kuali.rice.kew.api.rule.RuleExtension;
028    import org.kuali.rice.kew.routeheader.DocumentContent;
029    
030    
031    /**
032     * Generic base class that implements common functionality to simplify implementing
033     * a WorkflowAttribute.  This includes simplified template methods, as well as a generic
034     * attribute content model.
035     * 
036     * <p>Control flow (for isMatch):</p>
037     * 
038     * <ol>
039     *   <li>{@link #isMatch(DocumentContent, List)}
040     *     <ol>
041     *       <li>{@link #isMatch(List, List)}
042     *         <ol>
043     *           <li>{@link #isMatch(Map, List)}</li>
044     *         </ol>
045     *       </li>
046     *     </ol>
047     *   </li>
048     * </ol>
049     * 
050     * The default matching algorithm will match:
051     * <blockquote><i>if any single attribute's properties are a match for all rule extension values</i></blockquote>
052     * This implementation does not (yet!) implement a generic internal map of properties, so it is up to subclasses
053     * to expose specific named getters/setters to set data on an attribute of this ancestry.
054     * 
055     * @author Kuali Rice Team (rice.collab@kuali.org)
056     */
057    public abstract class GenericWorkflowAttribute extends AbstractWorkflowAttribute {
058        protected final Logger log = Logger.getLogger(getClass());
059        protected final String attributeName;
060        protected final GenericAttributeContent content;
061        
062        public GenericWorkflowAttribute() {
063            this(null); // can't do getClass().getName() so we'll have to pass null...shame
064        }
065    
066        public GenericWorkflowAttribute(String uniqueName) {
067            if (uniqueName != null) {
068                this.attributeName = uniqueName;
069            } else {
070                this.attributeName = getClass().getName();
071            }
072            content = new GenericAttributeContent(attributeName);
073        }
074    
075        /**
076         * Template method for subclasses to override to expose attribute state
077         * @return map exposing attribute state
078         */
079        public abstract Map<String, String> getProperties();
080    
081        /**
082         * Simply defers to GenericAttributeContent to generate suitable XML content in a standard fashion
083         */
084        public String getDocContent() {
085            String dc = content.generateContent(getProperties());
086            //log.info("Generating doc content: " + dc, new Exception("Dummy exception"));
087            return dc;
088        }
089    
090        public boolean isMatch(DocumentContent docContent, List<RuleExtension> ruleExtensions) {
091            log.info("isMatch: " + docContent + " " + ruleExtensions);
092            try {
093                // could be multiple attributes on the incoming doc content!
094                List<Map<String, String>> propertiesList = content.parseContent(docContent.getAttributeContent());
095                
096                return isMatch(propertiesList, ruleExtensions);
097            } catch (XPathExpressionException xpee) {
098                String message = "Error parsing attribute '" + attributeName + "' content: " + docContent.getDocContent();
099                log.error(message, xpee);
100                throw new RuntimeException(xpee);
101            }
102        }
103    
104        /**
105         * Returns true if any single incoming attribute's properties are a match for all rule extension values
106         * @param propertiesList the list of incoming attributes' properties
107         * @param ruleExtensions the rule extensions
108         * @return true if any single attribute's properties are a match for all rule extension values
109         */
110        protected boolean isMatch(List<Map<String, String>> propertiesList, List<RuleExtension> ruleExtensions) {
111            for (Map<String, String> properties: propertiesList) {
112                return isMatch(properties, ruleExtensions);
113            }
114            return false;
115        }
116    
117        /**
118         * Returns true if all key/value pairs defined by the specified rule extensions are present in the incoming attribute's
119         * properties
120         * @param properties incoming attribute's properties
121         * @param ruleExtensions list of rule extensions
122         * @return true if all key/value pairs defined by the specified rule extensions are present in the incoming attribute's
123         */
124        protected boolean isMatch(Map<String, String> properties, List<RuleExtension> ruleExtensions) {
125            for (RuleExtension ruleExtension: ruleExtensions) {
126                for (Map.Entry<String, String> ruleExtensionValue: ruleExtension.getExtensionValuesMap().entrySet()) {
127                    if (!ObjectUtils.equals(ruleExtensionValue.getValue(), properties.get(ruleExtensionValue.getKey()))) {
128                        return false;
129                    }
130                }
131            }
132            return true;
133        }
134    
135        /**
136         * These guys should probably be implemented to set the parameters on an internal member property map this attribute
137         * should use to contain all properties set on it, like StandardGenericXmlAttribute.
138         * @see #getProperties()
139         * TODO: implement me!
140         */
141        public List validateRoutingData(Map paramMap) {
142            return Collections.EMPTY_LIST;
143        }
144        public List validateRuleData(Map paramMap) {
145            return Collections.EMPTY_LIST;
146        }
147    
148        //public List validateClientRouting....
149    
150        /**
151         * I think the job of this method is to marshal the current state of the attribute into a representative list of rule extension
152         * values.  On that assumption, this method should simply create a list of RuleExtensionValues based on the the property map
153         * this attribute uses to hold property values.
154         * 
155         * TODO: this is not fully implemented! e.g. generic property map like StandardGenericXmlAttribute
156         */
157        public List<RuleExtensionValue> getRuleExtensionValues() {
158            log.info("getRuleExtensionValues");
159            List<RuleExtensionValue> exts = new ArrayList<RuleExtensionValue>();
160            Map<String, String> props = getProperties();
161            if (props != null) {
162                for (Map.Entry<String, String> entry: props.entrySet()) {
163                    if (entry.getValue() != null) {
164                        RuleExtensionValue ruleVal = new RuleExtensionValue();
165                        ruleVal.setKey(entry.getKey());
166                        ruleVal.setValue(entry.getValue());
167                        exts.add(ruleVal);
168                    }
169                }
170            }
171            return exts;
172        }
173    }