1
2
3
4
5
6
7
8
9
10
11
12
13
14
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
59
60
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
79
80
81
82
83
84
85
86
87
88
89 public static String retrieveLookupParameterValue(UifFormBase form, HttpServletRequest request,
90 Class<?> lookupObjectClass, String propertyName, String parameterName) {
91 String parameterValue = "";
92
93
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
102 else if (request.getParameterMap().containsKey(parameterName)) {
103 parameterValue = request.getParameter(parameterName);
104 }
105
106 else {
107 parameterValue = ObjectPropertyUtils.getPropertyValue(form, parameterName);
108
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
130
131
132
133 public static String getBaseLookupUrl() {
134 return CoreApiServiceLocator.getKualiConfigurationService().getPropertyValueAsString(
135 KRADConstants.KRAD_LOOKUP_URL_KEY);
136 }
137
138
139
140
141
142
143
144
145
146
147
148 public static String forceUppercase(Class<?> dataObjectClass, String fieldName, String fieldValue) {
149
150 if (StringUtils.isBlank(fieldValue)) {
151 return fieldValue;
152 }
153
154
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
174 }
175
176 if (forceUpperCase && !fieldValue.endsWith(EncryptionService.ENCRYPTION_POST_PREFIX)) {
177 return fieldValue.toUpperCase();
178 }
179
180 return fieldValue;
181 }
182
183
184
185
186
187
188
189
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
209
210
211
212
213
214
215
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
229
230
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
246
247
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
263
264
265
266
267
268
269
270
271
272
273
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
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
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
314
315
316
317
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 }
339
340 fieldsToUpdate.put(dateFieldName, newPropValue);
341 }
342 }
343
344
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
355
356
357
358
359
360
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
376
377
378
379
380
381
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
397
398
399
400
401
402
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
419
420
421
422
423
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
439
440
441
442
443
444
445
446
447
448
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
466
467
468
469
470
471
472
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
482
483
484
485
486
487
488
489
490 public static Map<String, String> adjustCriteriaForNestedEBOs(Class<?> dataObjectClass,
491 Map<String, String> searchCriteria,
492 boolean unbounded) throws InstantiationException, IllegalAccessException {
493
494 Map<String, String> nonEboFieldValues = removeExternalizableBusinessObjectFieldValues(
495 dataObjectClass, searchCriteria);
496 if (LOG.isDebugEnabled()) {
497 LOG.debug("Non EBO properties removed: " + nonEboFieldValues);
498 }
499
500
501 List<String> eboPropertyNames = getExternalizableBusinessObjectProperties(dataObjectClass, searchCriteria);
502 if (LOG.isDebugEnabled()) {
503 LOG.debug("EBO properties: " + eboPropertyNames);
504 }
505
506
507 for (String eboPropertyName : eboPropertyNames) {
508
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
517 ModuleService eboModuleService = KRADServiceLocatorWeb.getKualiModuleService().getResponsibleModuleService(
518 getExternalizableBusinessObjectClass(dataObjectClass, eboPropertyName));
519
520
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
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
553
554
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
563
564
565
566
567
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
578
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
592 nonEboFieldValues.put(boProperty, boPropertyValue.toString());
593 } else {
594
595 nonEboFieldValues.put(eboParentPropertyName + "." + boProperty, boPropertyValue.toString());
596 }
597 }
598 }
599
600 return nonEboFieldValues;
601 }
602
603
604
605
606
607
608
609 public static String scrubQueryCharacters(String criteriaValue) {
610 return StringUtils.replaceEach(criteriaValue, searchList, replacementList);
611 }
612
613
614
615
616
617
618
619
620
621
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
648
649
650
651
652 public static void refreshLookupResultSelections(LookupForm form) {
653
654 int displayStart = 0;
655 int displayLength = 0;
656
657
658 ViewPostMetadata viewPostMetadata = form.getViewPostMetadata();
659 if (viewPostMetadata != null) {
660
661
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 }