001    /*
002     * Copyright 2007 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 java.util.ArrayList;
019    import java.util.List;
020    import java.util.Map;
021    import java.util.Properties;
022    
023    import javax.xml.xpath.XPath;
024    import javax.xml.xpath.XPathExpressionException;
025    import javax.xml.xpath.XPathFactory;
026    
027    import org.apache.commons.lang.StringUtils;
028    import org.kuali.rice.kew.engine.RouteContext;
029    import org.kuali.rice.kew.rule.xmlrouting.WorkflowFunctionResolver;
030    import org.kuali.rice.kew.rule.xmlrouting.WorkflowNamespaceContext;
031    import org.kuali.rice.kns.util.FieldUtils;
032    import org.kuali.rice.kns.web.ui.Field;
033    import org.kuali.rice.kns.web.ui.Row;
034    import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
035    import org.kuali.rice.krad.util.KRADConstants;
036    import org.kuali.rice.krad.util.KRADPropertyConstants;
037    import org.kuali.rice.krad.util.UrlFactory;
038    import org.w3c.dom.Document;
039    
040    
041    public final class WorkflowUtils {
042        private static final String XPATH_ROUTE_CONTEXT_KEY = "_xpathKey";
043        public static final String XSTREAM_SAFE_PREFIX = "wf:xstreamsafe('";
044        public static final String XSTREAM_SAFE_SUFFIX = "')";
045        public static final String XSTREAM_MATCH_ANYWHERE_PREFIX = "//";
046        public static final String XSTREAM_MATCH_RELATIVE_PREFIX = "./";
047    
048            private WorkflowUtils() {
049                    throw new UnsupportedOperationException("do not call");
050            }
051        
052        /**
053         *
054         * This method sets up the XPath with the correct workflow namespace and resolver initialized. This ensures that the XPath
055         * statements can use required workflow functions as part of the XPath statements.
056         *
057         * @param document - document
058         * @return a fully initialized XPath instance that has access to the workflow resolver and namespace.
059         *
060         */
061        public final static XPath getXPath(Document document) {
062            XPath xpath = getXPath(RouteContext.getCurrentRouteContext());
063            xpath.setNamespaceContext(new WorkflowNamespaceContext());
064            WorkflowFunctionResolver resolver = new WorkflowFunctionResolver();
065            resolver.setXpath(xpath);
066            resolver.setRootNode(document);
067            xpath.setXPathFunctionResolver(resolver);
068            return xpath;
069        }
070    
071        public final static XPath getXPath(RouteContext routeContext) {
072            if (routeContext == null) {
073                return XPathFactory.newInstance().newXPath();
074            }
075            if (!routeContext.getParameters().containsKey(XPATH_ROUTE_CONTEXT_KEY)) {
076                routeContext.getParameters().put(XPATH_ROUTE_CONTEXT_KEY, XPathFactory.newInstance().newXPath());
077            }
078            return (XPath) routeContext.getParameters().get(XPATH_ROUTE_CONTEXT_KEY);
079        }
080    
081        /**
082         * This method will do a simple XPath.evaluate, while wrapping your xpathExpression with the xstreamSafe function. It assumes a
083         * String result, and will return such. If an XPathExpressionException is thrown, this will be re-thrown within a
084         * RuntimeException.
085         *
086         * @param xpath A correctly initialized XPath instance.
087         * @param xpathExpression Your XPath Expression that needs to be wrapped in an xstreamSafe wrapper and run.
088         * @param item The document contents you will be searching within.
089         * @return The string value of the xpath.evaluate().
090         */
091        public static final String xstreamSafeEval(XPath xpath, String xpathExpression, Object item) {
092            String xstreamSafeXPath = xstreamSafeXPath(xpathExpression);
093            String evalResult = "";
094            try {
095                evalResult = xpath.evaluate(xstreamSafeXPath, item);
096            }
097            catch (XPathExpressionException e) {
098                throw new RuntimeException("XPathExpressionException occurred on xpath: " + xstreamSafeXPath, e);
099            }
100            return evalResult;
101        }
102    
103        /**
104         * This method wraps the passed-in XPath expression in XStream Safe wrappers, so that XStream generated reference links will be
105         * handled correctly.
106         *
107         * @param xpathExpression The XPath Expression you wish to use.
108         * @return Your XPath Expression wrapped in the XStreamSafe wrapper.
109         */
110        public static final String xstreamSafeXPath(String xpathExpression) {
111            return new StringBuilder(XSTREAM_SAFE_PREFIX).append(xpathExpression).append(XSTREAM_SAFE_SUFFIX).toString();
112        }
113    
114        /**
115         * This method is for use by WorkflowLookupableImpl and WorkflowAttribute implementations to derive the fieldHelpUrl for use on
116         * org.kuali.rice.krad.web.ui.Fieldss.
117         *
118         * @param field The kuali field that we need to derive a help url for. @ return Returns the help url for the field.
119         */
120        public static String getHelpUrl(Field field) {
121            Properties params = new Properties();
122            params.put(KRADConstants.DISPATCH_REQUEST_PARAMETER, "getAttributeHelpText");
123            params.put(KRADConstants.BUSINESS_OBJECT_CLASS_ATTRIBUTE, field.getBusinessObjectClassName());
124            params.put(KRADPropertyConstants.ATTRIBUTE_NAME, field.getPropertyName());
125            String baseUrl = KRADServiceLocatorWeb.getRiceApplicationConfigurationMediationService().getBaseHelpUrl(field.getBusinessObjectClassName());
126            if (baseUrl == null) {
127                    return null;
128            }
129            return UrlFactory.parameterizeUrl(baseUrl, params);
130        }
131    
132        /**
133         * This method returns a label from the data dictionary service
134         *
135         * @param businessObjectClass - class where the label should come from
136         * @param attributeName - name of the attribute you need the label for
137         * @return the label from the data dictionary for the given Class and attributeName or null if not found
138         */
139        public static final String getBusinessObjectAttributeLabel(Class businessObjectClass, String attributeName) {
140            return KRADServiceLocatorWeb.getDataDictionaryService().getAttributeLabel(businessObjectClass, attributeName);
141        }
142    
143    
144        /**
145         * This method builds a workflow-lookup-screen Row of type TEXT, with no quickfinder/lookup.
146         *
147         * @param propertyClass The Class of the BO that this row is based on. For example, Account.class for accountNumber.
148         * @param boPropertyName The property name on the BO that this row is based on. For example, accountNumber for
149         *        Account.accountNumber.
150         * @param workflowPropertyKey The workflow-lookup-screen property key. For example, account_nbr for Account.accountNumber. This
151         *        key can be anything, but needs to be consistent with what is used for the row/field key on the java attribute, so
152         *        everything links up correctly.
153         * @return A populated and ready-to-use workflow lookupable.Row.
154         */
155        public static Row buildTextRow(Class propertyClass, String boPropertyName, String workflowPropertyKey) {
156            if (propertyClass == null) {
157                throw new IllegalArgumentException("Method parameter 'propertyClass' was passed a NULL value.");
158            }
159            if (StringUtils.isBlank(boPropertyName)) {
160                throw new IllegalArgumentException("Method parameter 'boPropertyName' was passed a NULL or blank value.");
161            }
162            if (StringUtils.isBlank(workflowPropertyKey)) {
163                throw new IllegalArgumentException("Method parameter 'workflowPropertyKey' was passed a NULL or blank value.");
164            }
165            List<Field> fields = new ArrayList<Field>();
166            Field field;
167            field = FieldUtils.getPropertyField(propertyClass, boPropertyName, false);
168            fields.add(field);
169            return new Row(fields);
170        }
171    
172        /**
173         * This method builds a workflow-lookup-screen Row of type TEXT, with the attached lookup icon and functionality.
174         *
175         * @param propertyClass The Class of the BO that this row is based on. For example, Account.class for accountNumber.
176         * @param boPropertyName The property name on the BO that this row is based on. For example, accountNumber for
177         *        Account.accountNumber.
178         * @param workflowPropertyKey The workflow-lookup-screen property key. For example, account_nbr for Account.accountNumber. This
179         *        key can be anything, but needs to be consistent with what is used for the row/field key on the java attribute, so
180         *        everything links up correctly.
181         * @return A populated and ready-to-use workflow lookupable.Row, which includes both the property field and the lookup icon.
182         */
183        public static Row buildTextRowWithLookup(Class propertyClass, String boPropertyName, String workflowPropertyKey) {
184            return buildTextRowWithLookup(propertyClass, boPropertyName, workflowPropertyKey, null);
185        }
186    
187        /**
188         * This method builds a workflow-lookup-screen Row of type TEXT, with the attached lookup icon and functionality.
189         *
190         * @param propertyClass The Class of the BO that this row is based on. For example, Account.class for accountNumber.
191         * @param boPropertyName The property name on the BO that this row is based on. For example, accountNumber for
192         *        Account.accountNumber.
193         * @param workflowPropertyKey The workflow-lookup-screen property key. For example, account_nbr for Account.accountNumber. This
194         *        key can be anything, but needs to be consistent with what is used for the row/field key on the java attribute, so
195         *        everything links up correctly.
196         * @param fieldConversionsByBoPropertyName A list of extra field conversions where the key is the business object property name
197         *        and the value is the workflow property key
198         * @return A populated and ready-to-use workflow lookupable.Row, which includes both the property field and the lookup icon.
199         */
200        public static Row buildTextRowWithLookup(Class propertyClass, String boPropertyName, String workflowPropertyKey, Map fieldConversionsByBoPropertyName) {
201            if (propertyClass == null) {
202                throw new IllegalArgumentException("Method parameter 'propertyClass' was passed a NULL value.");
203            }
204            if (StringUtils.isBlank(boPropertyName)) {
205                throw new IllegalArgumentException("Method parameter 'boPropertyName' was passed a NULL or blank value.");
206            }
207            if (StringUtils.isBlank(workflowPropertyKey)) {
208                throw new IllegalArgumentException("Method parameter 'workflowPropertyKey' was passed a NULL or blank value.");
209            }
210            Field field;
211            field = FieldUtils.getPropertyField(propertyClass, boPropertyName, false);
212    
213            List<Field> fields = new ArrayList<Field>();
214            fields.add(field);
215            return new Row(fields);
216        }
217    
218        /**
219         * This method builds a workflow-lookup-screen Row of type DROPDOWN.
220         *
221         * @param propertyClass The Class of the BO that this row is based on. For example, Account.class for accountNumber.
222         * @param boPropertyName The property name on the BO that this row is based on. For example, accountNumber for
223         *        Account.accountNumber.
224         * @param workflowPropertyKey The workflow-lookup-screen property key. For example, account_nbr for Account.accountNumber. This
225         *        key can be anything, but needs to be consistent with what is used for the row/field key on the java attribute, so
226         *        everything links up correctly.
227         * @param optionMap The map of value, text pairs that will be used to constuct the dropdown list.
228         * @return A populated and ready-to-use workflow lookupable.Row.
229         */
230        public static Row buildDropdownRow(Class propertyClass, String boPropertyName, String workflowPropertyKey, Map<String, String> optionMap, boolean addBlankRow) {
231            if (propertyClass == null) {
232                throw new IllegalArgumentException("Method parameter 'propertyClass' was passed a NULL value.");
233            }
234            if (StringUtils.isBlank(boPropertyName)) {
235                throw new IllegalArgumentException("Method parameter 'boPropertyName' was passed a NULL or blank value.");
236            }
237            if (StringUtils.isBlank(workflowPropertyKey)) {
238                throw new IllegalArgumentException("Method parameter 'workflowPropertyKey' was passed a NULL or blank value.");
239            }
240            if (optionMap == null) {
241                throw new IllegalArgumentException("Method parameter 'optionMap' was passed a NULL value.");
242            }
243            List<Field> fields = new ArrayList<Field>();
244            Field field;
245            field = FieldUtils.getPropertyField(propertyClass, boPropertyName, false);
246            fields.add(field);
247            return new Row(fields);
248        }
249    }