View Javadoc
1   /**
2    * Copyright 2005-2014 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.krad.lookup;
17  
18  import org.apache.commons.beanutils.PropertyUtils;
19  import org.apache.commons.lang.StringUtils;
20  import org.kuali.rice.core.api.CoreApiServiceLocator;
21  import org.kuali.rice.core.api.encryption.EncryptionService;
22  import org.kuali.rice.core.api.search.SearchOperator;
23  import org.kuali.rice.coreservice.framework.CoreFrameworkServiceLocator;
24  import org.kuali.rice.krad.bo.ExternalizableBusinessObject;
25  import org.kuali.rice.krad.data.KradDataServiceLocator;
26  import org.kuali.rice.krad.datadictionary.RelationshipDefinition;
27  import org.kuali.rice.krad.datadictionary.exception.UnknownBusinessClassAttributeException;
28  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
29  import org.kuali.rice.krad.service.ModuleService;
30  import org.kuali.rice.krad.uif.UifConstants;
31  import org.kuali.rice.krad.uif.UifPropertyPaths;
32  import org.kuali.rice.krad.uif.lifecycle.ComponentPostMetadata;
33  import org.kuali.rice.krad.uif.lifecycle.ViewPostMetadata;
34  import org.kuali.rice.krad.uif.util.ObjectPropertyUtils;
35  import org.kuali.rice.krad.util.ExternalizableBusinessObjectUtils;
36  import org.kuali.rice.krad.util.KRADConstants;
37  import org.kuali.rice.krad.util.KRADPropertyConstants;
38  import org.kuali.rice.krad.util.KRADUtils;
39  import org.kuali.rice.krad.web.form.UifFormBase;
40  import org.springframework.beans.PropertyAccessorUtils;
41  
42  import javax.servlet.http.HttpServletRequest;
43  
44  import java.security.GeneralSecurityException;
45  import java.sql.Date;
46  import java.sql.Timestamp;
47  import java.text.ParseException;
48  import java.util.ArrayList;
49  import java.util.Calendar;
50  import java.util.Collections;
51  import java.util.HashMap;
52  import java.util.HashSet;
53  import java.util.List;
54  import java.util.Map;
55  import java.util.Set;
56  
57  /**
58   * Provides static utility methods for use within the lookup framework.
59   *
60   * @author Kuali Rice Team (rice.collab@kuali.org)
61   */
62  public class LookupUtils {
63      private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(LookupUtils.class);
64      private static final String[] searchList = new String[SearchOperator.QUERY_CHARACTERS.size()];
65  
66      static {
67          int index = 0;
68          for (SearchOperator operator : SearchOperator.QUERY_CHARACTERS) {
69              searchList[index++] = operator.op();
70          }
71      }
72  
73      private static final String[] replacementList = Collections.nCopies(searchList.length, "").toArray(new String[0]);
74  
75      private LookupUtils() {}
76  
77      /**
78       * Retrieves the value for the given parameter name to send as a lookup parameter.
79       *
80       * @param form form instance to retrieve values from
81       * @param request request object to retrieve parameters from
82       * @param lookupObjectClass data object class associated with the lookup, used to check whether the
83       * value needs to be encyrpted
84       * @param propertyName name of the property associated with the parameter, used to check whether the
85       * value needs to be encrypted
86       * @param parameterName name of the parameter to retrieve the value for
87       * @return String parameter value or empty string if no value was found
88       */
89      public static String retrieveLookupParameterValue(UifFormBase form, HttpServletRequest request,
90              Class<?> lookupObjectClass, String propertyName, String parameterName) {
91          String parameterValue = "";
92  
93          // get literal parameter values first
94          if (StringUtils.startsWith(parameterName, "'") && StringUtils.endsWith(parameterName, "'")) {
95              parameterValue = StringUtils.substringBetween(parameterName, "'");
96          } else if (parameterValue.startsWith(KRADConstants.LOOKUP_PARAMETER_LITERAL_PREFIX
97                  + KRADConstants.LOOKUP_PARAMETER_LITERAL_DELIMITER)) {
98              parameterValue = StringUtils.removeStart(parameterValue, KRADConstants.LOOKUP_PARAMETER_LITERAL_PREFIX
99                      + KRADConstants.LOOKUP_PARAMETER_LITERAL_DELIMITER);
100         }
101         // check if parameter is in request
102         else if (request.getParameterMap().containsKey(parameterName)) {
103             parameterValue = request.getParameter(parameterName);
104         }
105         // get parameter value from form object
106         else {
107             parameterValue = ObjectPropertyUtils.getPropertyValue(form, parameterName);
108             // TODO: should go through property editor for display
109         }
110 
111         if (parameterValue != null && lookupObjectClass != null
112                 && KRADServiceLocatorWeb.getDataObjectAuthorizationService()
113                 .attributeValueNeedsToBeEncryptedOnFormsAndLinks(lookupObjectClass, propertyName)) {
114             try {
115                 if (CoreApiServiceLocator.getEncryptionService().isEnabled()) {
116                     parameterValue = CoreApiServiceLocator.getEncryptionService().encrypt(parameterValue)
117                             + EncryptionService.ENCRYPTION_POST_PREFIX;
118                 }
119             } catch (GeneralSecurityException e) {
120                 LOG.error("Unable to encrypt value for property name: " + propertyName);
121                 throw new RuntimeException(e);
122             }
123         }
124 
125         return parameterValue;
126     }
127 
128     /**
129      * Retrieves the default KRAD base lookup URL, used to build lookup URLs in code
130      *
131      * @return String base lookup URL (everything except query string)
132      */
133     public static String getBaseLookupUrl() {
134         return CoreApiServiceLocator.getKualiConfigurationService().getPropertyValueAsString(
135                 KRADConstants.KRAD_LOOKUP_URL_KEY);
136     }
137 
138     /**
139      * Uses the DataDictionary to determine whether to force uppercase the value, and if it should, then it does the
140      * uppercase, and returns the upper-cased value.
141      *
142      * @param dataObjectClass parent DO class that the fieldName is a member of
143      * @param fieldName name of the field to be forced to uppercase
144      * @param fieldValue value of the field that may be uppercased
145      * @return the correctly uppercased fieldValue if it should be uppercased, otherwise fieldValue is returned
146      *         unchanged
147      */
148     public static String forceUppercase(Class<?> dataObjectClass, String fieldName, String fieldValue) {
149         // short-circuit to exit if there isnt enough information to do the forceUppercase
150         if (StringUtils.isBlank(fieldValue)) {
151             return fieldValue;
152         }
153 
154         // parameter validation
155         if (dataObjectClass == null) {
156             throw new IllegalArgumentException("Parameter dataObjectClass passed in with null value.");
157         }
158 
159         if (StringUtils.isBlank(fieldName)) {
160             throw new IllegalArgumentException("Parameter fieldName passed in with empty value.");
161         }
162 
163         if (!KRADServiceLocatorWeb.getDataDictionaryService().isAttributeDefined(dataObjectClass, fieldName)
164                 .booleanValue()) {
165             return fieldValue;
166         }
167 
168         boolean forceUpperCase = false;
169         try {
170             forceUpperCase = KRADServiceLocatorWeb.getDataDictionaryService()
171                     .getAttributeForceUppercase(dataObjectClass, fieldName).booleanValue();
172         } catch (UnknownBusinessClassAttributeException ubae) {
173             // do nothing, don't alter the fieldValue
174         }
175 
176         if (forceUpperCase && !fieldValue.endsWith(EncryptionService.ENCRYPTION_POST_PREFIX)) {
177             return fieldValue.toUpperCase();
178         }
179 
180         return fieldValue;
181     }
182 
183     /**
184      * Uses the DataDictionary to determine whether to force uppercase the values, and if it should, then it does the
185      * uppercase, and returns the upper-cased Map of fieldname/fieldValue pairs.
186      *
187      * @param dataObjectClass parent DO class that the fieldName is a member of
188      * @param fieldValues a Map<String,String> where the key is the fieldName and the value is the fieldValue
189      * @return the same Map is returned, with the appropriate values uppercased (if any)
190      */
191     public static Map<String, String> forceUppercase(Class<?> dataObjectClass, Map<String, String> fieldValues) {
192         if (dataObjectClass == null) {
193             throw new IllegalArgumentException("Parameter boClass passed in with null value.");
194         }
195 
196         if (fieldValues == null) {
197             throw new IllegalArgumentException("Parameter fieldValues passed in with null value.");
198         }
199 
200         for (String fieldName : fieldValues.keySet()) {
201             fieldValues.put(fieldName, forceUppercase(dataObjectClass, fieldName, fieldValues.get(fieldName)));
202         }
203 
204         return fieldValues;
205     }
206 
207     /**
208      * Parses and returns the lookup result set limit, checking first for the limit for the specific view,
209      * then the class being looked up, and then the global application limit if there isn't a limit specific
210      * to this data object class.
211      *
212      * @param dataObjectClass class to get limit for
213      * @param lookupForm lookupForm to use.  May be null if the form is unknown. If lookupForm is null, only the
214      * dataObjectClass will be used to find the search results set limit
215      * @return result set limit
216      */
217     public static Integer getSearchResultsLimit(Class dataObjectClass, LookupForm lookupForm) {
218         Integer limit = KRADServiceLocatorWeb.getViewDictionaryService().getResultSetLimitForLookup(dataObjectClass,
219                 lookupForm);
220         if (limit == null) {
221             limit = getApplicationSearchResultsLimit();
222         }
223 
224         return limit;
225     }
226 
227     /**
228      * Retrieves the default application search limit configured through a system parameter.
229      *
230      * @return default result set limit of the application
231      */
232     public static Integer getApplicationSearchResultsLimit() {
233         String limitString = CoreFrameworkServiceLocator.getParameterService()
234                 .getParameterValueAsString(KRADConstants.KRAD_NAMESPACE,
235                         KRADConstants.DetailTypes.LOOKUP_PARM_DETAIL_TYPE,
236                         KRADConstants.SystemGroupParameterNames.LOOKUP_RESULTS_LIMIT);
237         if (limitString != null) {
238             return Integer.valueOf(limitString);
239         }
240 
241         return null;
242     }
243 
244     /**
245      * Retrieves the default application multiple value search limit configured through a system parameter.
246      *
247      * @return default multiple value result set limit of the application
248      */
249     public static Integer getApplicationMultipleValueSearchResultsLimit() {
250         String limitString = CoreFrameworkServiceLocator.getParameterService()
251                 .getParameterValueAsString(KRADConstants.KRAD_NAMESPACE,
252                         KRADConstants.DetailTypes.LOOKUP_PARM_DETAIL_TYPE,
253                         KRADConstants.SystemGroupParameterNames.MULTIPLE_VALUE_LOOKUP_RESULTS_LIMIT);
254         if (limitString != null) {
255             return Integer.valueOf(limitString);
256         }
257 
258         return null;
259     }
260 
261     /**
262      * Determines what Timestamp should be used for active queries on effective dated records. Determination made as
263      * follows:
264      *
265      * <ul>
266      * <li>Use activeAsOfDate value from search values Map if value is not empty</li>
267      * <li>If search value given, try to convert to sql date, if conversion fails, try to convert to Timestamp</li>
268      * <li>If search value empty, use current Date</li>
269      * <li>If Timestamp value not given, create Timestamp from given Date setting the time as 1 second before midnight
270      * </ul>
271      *
272      * @param searchValues map containing search key/value pairs
273      * @return timestamp to be used for active criteria
274      */
275     public static Timestamp getActiveDateTimestampForCriteria(Map searchValues) {
276         Date activeDate = CoreApiServiceLocator.getDateTimeService().getCurrentSqlDate();
277 
278         Timestamp activeTimestamp = null;
279         if (searchValues.containsKey(KRADPropertyConstants.ACTIVE_AS_OF_DATE)) {
280             String activeAsOfDate = (String) searchValues.get(KRADPropertyConstants.ACTIVE_AS_OF_DATE);
281             if (StringUtils.isNotBlank(activeAsOfDate)) {
282                 try {
283                     activeDate = CoreApiServiceLocator.getDateTimeService()
284                             .convertToSqlDate(KRADUtils.clean(activeAsOfDate));
285                 } catch (ParseException e) {
286                     // try to parse as timestamp
287                     try {
288                         activeTimestamp = CoreApiServiceLocator.getDateTimeService()
289                                 .convertToSqlTimestamp(KRADUtils.clean(activeAsOfDate));
290                     } catch (ParseException e1) {
291                         throw new RuntimeException("Unable to convert date: " + KRADUtils.clean(activeAsOfDate));
292                     }
293                 }
294             }
295         }
296 
297         // if timestamp not given set to 1 second before midnight on the given date
298         if (activeTimestamp == null) {
299             Calendar cal = Calendar.getInstance();
300 
301             cal.setTime(activeDate);
302             cal.set(Calendar.HOUR, cal.getMaximum(Calendar.HOUR));
303             cal.set(Calendar.MINUTE, cal.getMaximum(Calendar.MINUTE));
304             cal.set(Calendar.SECOND, cal.getMaximum(Calendar.SECOND));
305 
306             activeTimestamp = new Timestamp(cal.getTime().getTime());
307         }
308 
309         return activeTimestamp;
310     }
311 
312     /**
313      * Changes from/to dates into the range operators the lookupable dao expects ("..",">" etc) this method modifies
314      * the passed in map and returns an updated search criteria map.
315      *
316      * @param searchCriteria map of criteria currently set for which the date criteria will be adjusted
317      * @return map updated search criteria
318      */
319     public static Map<String, String> preprocessDateFields(Map<String, String> searchCriteria) {
320         Map<String, String> fieldsToUpdate = new HashMap<String, String>();
321         Map<String, String> searchCriteriaUpdated = new HashMap<String, String>(searchCriteria);
322 
323         Set<String> fieldsForLookup = searchCriteria.keySet();
324         for (String propName : fieldsForLookup) {
325             if (propName.startsWith(KRADConstants.LOOKUP_RANGE_LOWER_BOUND_PROPERTY_PREFIX)) {
326                 String from_DateValue = searchCriteria.get(propName);
327                 String dateFieldName =
328                         StringUtils.remove(propName, KRADConstants.LOOKUP_RANGE_LOWER_BOUND_PROPERTY_PREFIX);
329                 String to_DateValue = searchCriteria.get(dateFieldName);
330                 String newPropValue = to_DateValue;
331 
332                 if (StringUtils.isNotEmpty(from_DateValue) && StringUtils.isNotEmpty(to_DateValue)) {
333                     newPropValue = from_DateValue + SearchOperator.BETWEEN + to_DateValue;
334                 } else if (StringUtils.isNotEmpty(from_DateValue) && StringUtils.isEmpty(to_DateValue)) {
335                     newPropValue = SearchOperator.GREATER_THAN_EQUAL.op() + from_DateValue;
336                 } else if (StringUtils.isNotEmpty(to_DateValue) && StringUtils.isEmpty(from_DateValue)) {
337                     newPropValue = SearchOperator.LESS_THAN_EQUAL.op() + to_DateValue;
338                 } // could optionally continue on else here
339 
340                 fieldsToUpdate.put(dateFieldName, newPropValue);
341             }
342         }
343 
344         // update lookup values from found date values to update
345         Set<String> keysToUpdate = fieldsToUpdate.keySet();
346         for (String updateKey : keysToUpdate) {
347             searchCriteriaUpdated.put(updateKey, fieldsToUpdate.get(updateKey));
348         }
349 
350         return searchCriteriaUpdated;
351     }
352 
353     /**
354      * Checks whether any of the fieldValues being passed refer to a property within an ExternalizableBusinessObject.
355      *
356      * @param boClass business object class of the lookup
357      * @param fieldValues map of the lookup criteria values
358      * @return true if externalizable business object are contained, false otherwise
359      * @throws IllegalAccessException
360      * @throws InstantiationException
361      */
362     public static boolean hasExternalBusinessObjectProperty(Class<?> boClass,
363             Map<String, String> fieldValues) throws IllegalAccessException, InstantiationException {
364         Object sampleBo = boClass.newInstance();
365         for (String key : fieldValues.keySet()) {
366             if (isExternalBusinessObjectProperty(sampleBo, key)) {
367                 return true;
368             }
369         }
370 
371         return false;
372     }
373 
374     /**
375      * Check whether the given property represents a property within an EBO starting with the sampleBo object given.
376      * This is used to determine if a criteria needs to be applied to the EBO first,
377      * before sending to the normal lookup DAO.
378      *
379      * @param sampleBo business object of the property to be tested
380      * @param propertyName property name to be tested
381      * @return true if the property is within an externalizable business object.
382      */
383     public static boolean isExternalBusinessObjectProperty(Object sampleBo, String propertyName) {
384         if (propertyName.indexOf(".") > 0 && !StringUtils.contains(propertyName, "add.")) {
385             Class<?> propertyClass =
386                     ObjectPropertyUtils.getPropertyType(sampleBo, StringUtils.substringBeforeLast(propertyName, "."));
387             if (propertyClass != null) {
388                 return ExternalizableBusinessObjectUtils.isExternalizableBusinessObjectInterface(propertyClass);
389             }
390         }
391 
392         return false;
393     }
394 
395     /**
396      * Returns a map stripped of any properties which refer to ExternalizableBusinessObjects. These values may not be
397      * passed into the lookup service, since the objects they refer to are not in the
398      * local database.
399      *
400      * @param boClass business object class of the lookup
401      * @param fieldValues map of lookup criteria from which to remove the externalizable business objects
402      * @return map of lookup criteria without externalizable business objects
403      */
404     public static Map<String, String> removeExternalizableBusinessObjectFieldValues(Class<?> boClass,
405             Map<String, String> fieldValues) throws IllegalAccessException, InstantiationException {
406         Map<String, String> eboFieldValues = new HashMap<String, String>();
407         Object sampleBo = boClass.newInstance();
408         for (String key : fieldValues.keySet()) {
409             if (!isExternalBusinessObjectProperty(sampleBo, key)) {
410                 eboFieldValues.put(key, fieldValues.get(key));
411             }
412         }
413 
414         return eboFieldValues;
415     }
416 
417     /**
418      * Return the EBO fieldValue entries explicitly for the given eboPropertyName. (I.e., any properties with the given
419      * property name as a prefix.
420      *
421      * @param eboPropertyName the externalizable business object property name to retrieve
422      * @param fieldValues map of lookup criteria
423      * return map of lookup criteria for the given eboPropertyName
424      */
425     public static Map<String, String> getExternalizableBusinessObjectFieldValues(String eboPropertyName,
426             Map<String, String> fieldValues) {
427         Map<String, String> eboFieldValues = new HashMap<String, String>();
428         for (String key : fieldValues.keySet()) {
429             if (key.startsWith(eboPropertyName + ".")) {
430                 eboFieldValues.put(StringUtils.substringAfterLast(key, "."), fieldValues.get(key));
431             }
432         }
433 
434         return eboFieldValues;
435     }
436 
437     /**
438      * Get the complete list of all properties referenced in the fieldValues that are ExternalizableBusinessObjects.
439      *
440      * <p>
441      * This is a list of the EBO object references themselves, not of the properties within them.
442      * </p>
443      *
444      * @param boClass business object class of the lookup
445      * @param fieldValues map of lookup criteria from which to return the externalizable business objects
446      * @return map of lookup criteria that are externalizable business objects
447      * @throws IllegalAccessException
448      * @throws InstantiationException
449      */
450     public static List<String> getExternalizableBusinessObjectProperties(Class<?> boClass,
451             Map<String, String> fieldValues) throws IllegalAccessException, InstantiationException {
452         Set<String> eboPropertyNames = new HashSet<String>();
453 
454         Object sampleBo = boClass.newInstance();
455         for (String key : fieldValues.keySet()) {
456             if (isExternalBusinessObjectProperty(sampleBo, key)) {
457                 eboPropertyNames.add(StringUtils.substringBeforeLast(key, "."));
458             }
459         }
460 
461         return new ArrayList<String>(eboPropertyNames);
462     }
463 
464     /**
465      * Given an property on the main BO class, return the defined type of the ExternalizableBusinessObject. This will
466      * be used by other code to determine the correct module service to call for the lookup.
467      *
468      * @param boClass business object class of the lookup
469      * @param propertyName property of which the externalizable business object type is to be determined
470      * @return externalizable business object type
471      * @throws IllegalAccessException
472      * @throws InstantiationException
473      */
474     public static Class<? extends ExternalizableBusinessObject> getExternalizableBusinessObjectClass(Class<?> boClass,
475             String propertyName) throws IllegalAccessException, InstantiationException {
476         return (Class<? extends ExternalizableBusinessObject>) ObjectPropertyUtils
477                 .getPropertyType(boClass.newInstance(), StringUtils.substringBeforeLast(propertyName, "."));
478     }
479 
480     /**
481      * Looks for criteria against nested EBOs and performs a search against that EBO and updates the criteria.
482      *
483      * @param searchCriteria map of criteria currently set
484      * @param unbounded indicates whether the complete result should be returned.  When set to false the result is
485      * limited (if necessary) to the max search result limit configured.
486      * @return Map of adjusted criteria for nested EBOs
487      * @throws InstantiationException
488      * @throws IllegalAccessException
489      */
490     public static Map<String, String> adjustCriteriaForNestedEBOs(Class<?> dataObjectClass,
491             Map<String, String> searchCriteria,
492             boolean unbounded) throws InstantiationException, IllegalAccessException {
493         // remove the EBO criteria
494         Map<String, String> nonEboFieldValues = removeExternalizableBusinessObjectFieldValues(
495                 dataObjectClass, searchCriteria);
496         if (LOG.isDebugEnabled()) {
497             LOG.debug("Non EBO properties removed: " + nonEboFieldValues);
498         }
499 
500         // get the list of EBO properties attached to this object
501         List<String> eboPropertyNames = getExternalizableBusinessObjectProperties(dataObjectClass, searchCriteria);
502         if (LOG.isDebugEnabled()) {
503             LOG.debug("EBO properties: " + eboPropertyNames);
504         }
505 
506         // loop over those properties
507         for (String eboPropertyName : eboPropertyNames) {
508             // extract the properties as known to the EBO
509             Map<String, String> eboFieldValues = LookupUtils.getExternalizableBusinessObjectFieldValues(eboPropertyName,
510                     searchCriteria);
511             if (LOG.isDebugEnabled()) {
512                 LOG.debug("EBO properties for master EBO property: " + eboPropertyName);
513                 LOG.debug("properties: " + eboFieldValues);
514             }
515 
516             // run search against attached EBO's module service
517             ModuleService eboModuleService = KRADServiceLocatorWeb.getKualiModuleService().getResponsibleModuleService(
518                     getExternalizableBusinessObjectClass(dataObjectClass, eboPropertyName));
519 
520             // KULRICE-4401 made eboResults an empty list and only filled if service is found.
521             List<?> eboResults = Collections.emptyList();
522             if (eboModuleService != null) {
523                 eboResults = eboModuleService.getExternalizableBusinessObjectsListForLookup(
524                         getExternalizableBusinessObjectClass(dataObjectClass, eboPropertyName),
525                         (Map) eboFieldValues, unbounded);
526             } else {
527                 LOG.debug("EBO ModuleService is null: " + eboPropertyName);
528             }
529 
530             // get the parent property type
531             Class<?> eboParentClass;
532             String eboParentPropertyName;
533             if (PropertyAccessorUtils.isNestedOrIndexedProperty(eboPropertyName)) {
534                 eboParentPropertyName = StringUtils.substringBeforeLast(eboPropertyName, ".");
535                 try {
536                     eboParentClass = KradDataServiceLocator.getDataObjectService().wrap(dataObjectClass.newInstance()).getPropertyType(
537                             eboParentPropertyName);
538                 } catch (Exception ex) {
539                     throw new RuntimeException(
540                             "Unable to create an instance of the business object class: " + dataObjectClass
541                                     .getName(), ex);
542                 }
543             } else {
544                 eboParentClass = dataObjectClass;
545                 eboParentPropertyName = null;
546             }
547 
548             if (LOG.isDebugEnabled()) {
549                 LOG.debug("determined EBO parent class/property name: " + eboParentClass + "/" + eboParentPropertyName);
550             }
551 
552             // look that up in the DD (BOMDS) find the appropriate relationship
553             // CHECK THIS: what if eboPropertyName is a nested attribute - need to strip off the
554             // eboParentPropertyName if not null
555             RelationshipDefinition rd = KRADServiceLocatorWeb.getLegacyDataAdapter().getDictionaryRelationship(
556                     eboParentClass, eboPropertyName);
557             if (LOG.isDebugEnabled()) {
558                 LOG.debug("Obtained RelationshipDefinition for " + eboPropertyName);
559                 LOG.debug(rd);
560             }
561 
562             // copy the needed properties (primary only) to the field values KULRICE-4446 do
563             // so only if the relationship definition exists
564             // NOTE: this will work only for single-field PK unless the ORM
565             // layer is directly involved
566             // (can't make (field1,field2) in ( (v1,v2),(v3,v4) ) style
567             // queries in the lookup framework
568             if (KRADUtils.isNotNull(rd)) {
569                 if (rd.getPrimitiveAttributes().size() > 1) {
570                     throw new RuntimeException(
571                             "EBO Links don't work for relationships with multiple-field primary keys.");
572                 }
573                 String boProperty = rd.getPrimitiveAttributes().get(0).getSourceName();
574                 String eboProperty = rd.getPrimitiveAttributes().get(0).getTargetName();
575                 StringBuffer boPropertyValue = new StringBuffer();
576 
577                 // loop over the results, making a string that the lookup DAO will convert into an
578                 // SQL "IN" clause
579                 for (Object ebo : eboResults) {
580                     if (boPropertyValue.length() != 0) {
581                         boPropertyValue.append(SearchOperator.OR.op());
582                     }
583                     try {
584                         boPropertyValue.append(PropertyUtils.getProperty(ebo, eboProperty).toString());
585                     } catch (Exception ex) {
586                         LOG.warn("Unable to get value for " + eboProperty + " on " + ebo);
587                     }
588                 }
589 
590                 if (eboParentPropertyName == null) {
591                     // non-nested property containing the EBO
592                     nonEboFieldValues.put(boProperty, boPropertyValue.toString());
593                 } else {
594                     // property nested within the main searched-for BO that contains the EBO
595                     nonEboFieldValues.put(eboParentPropertyName + "." + boProperty, boPropertyValue.toString());
596                 }
597             }
598         }
599 
600         return nonEboFieldValues;
601     }
602 
603     /**
604      * Removes query characters (such as wildcards) from the given string value.
605      *
606      * @param criteriaValue string to clean
607      * @return string with query characters removed
608      */
609     public static String scrubQueryCharacters(String criteriaValue) {
610         return StringUtils.replaceEach(criteriaValue, searchList, replacementList);
611     }
612 
613     /**
614      * Generates a key string in case of multivalue return. The values are extracted
615      * from the list of properties on the lineDataObject.
616      *
617      * If fieldConversionKeys is empty return the identifier string for the lineDataObject
618      *
619      * @param lineDataObject   Object from which to extract values
620      * @param fieldConversionKeys List of keys whose values have to be concatenated
621      * @return string representing the multivalue key 
622      */
623     public static String generateMultiValueKey(Object lineDataObject, List<String> fieldConversionKeys) {
624         String lineIdentifier = "";
625 
626         if(fieldConversionKeys == null || fieldConversionKeys.isEmpty()) {
627             lineIdentifier =
628                     KRADServiceLocatorWeb.getLegacyDataAdapter().getDataObjectIdentifierString(lineDataObject);
629         } else {
630             Collections.sort(fieldConversionKeys);
631             for (String fromFieldName : fieldConversionKeys) {
632                 Object fromFieldValue = ObjectPropertyUtils.getPropertyValue(lineDataObject, fromFieldName);
633 
634                 if (fromFieldValue != null) {
635                     lineIdentifier += fromFieldValue;
636                 }
637 
638                 lineIdentifier += ":";
639             }
640             lineIdentifier = StringUtils.removeEnd(lineIdentifier, ":");
641         }
642 
643         return lineIdentifier;
644     }
645 
646     /**
647      * Merges the lookup result selections that are part of the request with the selectedLookupResultsCache maintained in
648      * the session.
649      *
650      * @param form lookup form instance containing the selected results and lookup configuration
651      */
652     public static void refreshLookupResultSelections(LookupForm form) {
653 
654         int displayStart = 0;
655         int displayLength = 0;
656 
657         // avoid blowing the stack if the session expired
658         ViewPostMetadata viewPostMetadata = form.getViewPostMetadata();
659         if (viewPostMetadata != null) {
660 
661             // only one concurrent request per view please
662             synchronized (viewPostMetadata) {
663                 ComponentPostMetadata oldCollectionGroup = viewPostMetadata.getComponentPostMetadata("uLookupResults");
664                 displayStart = (Integer) oldCollectionGroup.getData(UifConstants.PostMetadata.COLL_DISPLAY_START);
665                 displayLength = (Integer) oldCollectionGroup.getData(UifConstants.PostMetadata.COLL_DISPLAY_LENGTH);
666             }
667         }
668 
669         List<? extends Object> lookupResults = (List<? extends Object>) form.getLookupResults();
670         List<String> fromFieldNames = form.getMultiValueReturnFields();
671 
672         Set<String> selectedLines = form.getSelectedCollectionLines().get(UifPropertyPaths.LOOKUP_RESULTS);
673         Set<String> selectedLookupResultsCache = form.getSelectedLookupResultsCache();
674 
675         selectedLines = (selectedLines == null) ? new HashSet<String>() : selectedLines;
676 
677 
678         for(int i = displayStart; i < displayStart + displayLength; i++ ) {
679             if(i >= form.getLookupResults().size()) break;
680 
681             Object lineItem = lookupResults.get(i);
682             String lineIdentifier = LookupUtils.generateMultiValueKey(lineItem, fromFieldNames);
683 
684             if(!selectedLines.contains(lineIdentifier)) {
685                  selectedLookupResultsCache.remove(lineIdentifier);
686             } else {
687                 selectedLookupResultsCache.add(lineIdentifier);
688             }
689         }
690 
691         selectedLines.addAll( selectedLookupResultsCache );
692 
693         form.getSelectedCollectionLines().put(UifPropertyPaths.LOOKUP_RESULTS, selectedLines);
694     }
695 
696 }