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.web.bind;
17  
18  import org.apache.commons.lang.ArrayUtils;
19  import org.apache.commons.lang.StringUtils;
20  import org.apache.commons.lang3.reflect.FieldUtils;
21  import org.kuali.rice.krad.data.DataObjectService;
22  import org.kuali.rice.krad.data.DataObjectWrapper;
23  import org.kuali.rice.krad.data.KradDataServiceLocator;
24  import org.kuali.rice.krad.data.util.Link;
25  import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
26  import org.kuali.rice.krad.uif.UifConstants;
27  import org.kuali.rice.krad.uif.UifConstants.ViewType;
28  import org.kuali.rice.krad.uif.UifParameters;
29  import org.kuali.rice.krad.uif.service.ViewService;
30  import org.kuali.rice.krad.uif.view.View;
31  import org.kuali.rice.krad.util.KRADUtils;
32  import org.kuali.rice.krad.web.form.UifFormBase;
33  import org.springframework.core.annotation.AnnotationUtils;
34  import org.springframework.core.convert.ConversionService;
35  import org.springframework.util.Assert;
36  import org.springframework.validation.AbstractPropertyBindingResult;
37  import org.springframework.web.bind.ServletRequestDataBinder;
38  
39  import javax.servlet.ServletRequest;
40  import javax.servlet.http.HttpServletRequest;
41  import java.lang.reflect.Field;
42  import java.util.ArrayList;
43  import java.util.Collections;
44  import java.util.HashSet;
45  import java.util.List;
46  import java.util.Map;
47  import java.util.Set;
48  
49  /**
50   * Override of ServletRequestDataBinder in order to hook in the UifBeanPropertyBindingResult
51   * which instantiates a custom BeanWrapperImpl, and to initialize the view.
52   *
53   * @author Kuali Rice Team (rice.collab@kuali.org)
54   */
55  public class UifServletRequestDataBinder extends ServletRequestDataBinder {
56      protected static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(
57              UifServletRequestDataBinder.class);
58  
59      private UifBeanPropertyBindingResult bindingResult;
60      private ConversionService conversionService;
61      private DataObjectService dataObjectService;
62      private boolean changeTracking = false;
63      private boolean autoLinking = true;
64  
65      public UifServletRequestDataBinder(Object target) {
66          super(target);
67          this.changeTracking = determineChangeTracking(target);
68          setBindingErrorProcessor(new UifBindingErrorProcessor());
69      }
70  
71      public UifServletRequestDataBinder(Object target, String name) {
72          super(target, name);
73          this.changeTracking = determineChangeTracking(target);
74          setBindingErrorProcessor(new UifBindingErrorProcessor());
75      }
76  
77      /**
78       * Return true if the target of this data binder has change tracking enabled.
79       */
80      private static boolean determineChangeTracking(Object target) {
81          ChangeTracking changeTracking = AnnotationUtils.findAnnotation(target.getClass(), ChangeTracking.class);
82          if (changeTracking != null && changeTracking.enabled()) {
83              return true;
84          }
85          return false;
86      }
87  
88      /**
89       * Allows for a custom binding result class.
90       *
91       * @see org.springframework.validation.DataBinder#initBeanPropertyAccess()
92       */
93      @Override
94      public void initBeanPropertyAccess() {
95          Assert.state(this.bindingResult == null,
96                  "DataBinder is already initialized - call initBeanPropertyAccess before other configuration methods");
97  
98          this.bindingResult = new UifBeanPropertyBindingResult(getTarget(), getObjectName(), isAutoGrowNestedPaths(),
99                  getAutoGrowCollectionLimit());
100         this.bindingResult.setChangeTracking(this.changeTracking);
101 
102         if (this.conversionService != null) {
103             this.bindingResult.initConversion(this.conversionService);
104         }
105 
106         if (this.dataObjectService == null) {
107             this.dataObjectService = KradDataServiceLocator.getDataObjectService();
108         }
109     }
110 
111     /**
112      * Allows for the setting attributes to use to find the data dictionary data from Kuali
113      *
114      * @see org.springframework.validation.DataBinder#getInternalBindingResult()
115      */
116     @Override
117     protected AbstractPropertyBindingResult getInternalBindingResult() {
118         if (this.bindingResult == null) {
119             initBeanPropertyAccess();
120         }
121 
122         return this.bindingResult;
123     }
124 
125     /**
126      * Disallows direct field access for Kuali
127      *
128      * @see org.springframework.validation.DataBinder#initDirectFieldAccess()
129      */
130     @Override
131     public void initDirectFieldAccess() {
132         LOG.error("Direct Field access is not allowed in UifServletRequestDataBinder.");
133         throw new RuntimeException("Direct Field access is not allowed in Kuali");
134     }
135 
136     /**
137      * Helper method to facilitate calling super.bind() from {@link #bind(ServletRequest)}.
138      */
139     private void _bind(ServletRequest request) {
140         super.bind(request);
141     }
142 
143     /**
144      * Calls {@link org.kuali.rice.krad.web.form.UifFormBase#preBind(HttpServletRequest)}, Performs data binding
145      * from servlet request parameters to the form, initializes view object, then calls
146      * {@link org.kuali.rice.krad.web.form.UifFormBase#postBind(javax.servlet.http.HttpServletRequest)}
147      *
148      * <p>
149      * The view is initialized by first looking for the {@code viewId} parameter in the request. If found, the view is
150      * retrieved based on this id. If the id is not present, then an attempt is made to find a view by type. In order
151      * to retrieve a view based on type, the view request parameter {@code viewTypeName} must be present. If all else
152      * fails and the viewId is populated on the form (could be populated from a previous request), this is used to
153      * retrieve the view.
154      * </p>
155      *
156      * @param request - HTTP Servlet Request instance
157      */
158     @Override
159     public void bind(ServletRequest request) {
160         UifFormBase form = (UifFormBase) UifServletRequestDataBinder.this.getTarget();
161 
162         request.setAttribute(UifConstants.REQUEST_FORM, form);
163 
164         form.preBind((HttpServletRequest) request);
165 
166         _bind(request);
167 
168         request.setAttribute(UifConstants.PROPERTY_EDITOR_REGISTRY, this.bindingResult.getPropertyEditorRegistry());
169 
170         executeAutomaticLinking(request, form);
171 
172         if (!form.isUpdateNoneRequest()) {
173             // attempt to retrieve a view by unique identifier first, either as request attribute or parameter
174             String viewId = (String) request.getAttribute(UifParameters.VIEW_ID);
175             if (StringUtils.isBlank(viewId)) {
176                 viewId = request.getParameter(UifParameters.VIEW_ID);
177             }
178 
179             View view = null;
180             if (StringUtils.isNotBlank(viewId)) {
181                 view = getViewService().getViewById(viewId);
182             }
183 
184             // attempt to get view instance by type parameters
185             if (view == null) {
186                 view = getViewByType(request, form);
187             }
188 
189             // if view not found attempt to find one based on the cached form
190             if (view == null) {
191                 view = getViewFromPreviousModel(form);
192 
193                 if (view != null) {
194                     LOG.warn("Obtained viewId from cached form, this may not be safe!");
195                 }
196             }
197 
198             if (view != null) {
199                 form.setViewId(view.getId());
200 
201             } else {
202                 form.setViewId(null);
203             }
204 
205             form.setView(view);
206         }
207 
208         // invoke form callback for custom binding
209         form.postBind((HttpServletRequest) request);
210     }
211 
212     /**
213      * Performs automatic reference linking of the given form based on the properties on the form for which linking
214      * is enabled.
215      *
216      * <p>Linking will only be performed if change tracking and auto linking are enabled on this data binder.</p>
217      *
218      * @param request request instance
219      * @param form form instance against which to perform automatic linking
220      */
221     protected void executeAutomaticLinking(ServletRequest request, UifFormBase form) {
222         if (!changeTracking) {
223             LOG.info("Skip automatic linking because change tracking not enabled for this form.");
224             return;
225         }
226 
227         if (!autoLinking) {
228             LOG.info("Skip automatic linking because it has been disabled for this form");
229             return;
230         }
231 
232         Set<String> autoLinkingPaths = determineRootAutoLinkingPaths(form.getClass(), null, new HashSet<Class<?>>());
233         List<AutoLinkTarget> targets = extractAutoLinkTargets(autoLinkingPaths);
234 
235         // perform linking for each target
236         for (AutoLinkTarget target : targets) {
237             if (!dataObjectService.supports(target.getTarget().getClass())) {
238                 LOG.warn("Encountered an auto linking target that is not a valid data object: " + target.getTarget()
239                         .getClass());
240             } else {
241                 DataObjectWrapper<?> wrapped = dataObjectService.wrap(target.getTarget());
242                 wrapped.linkChanges(target.getModifiedPropertyPaths());
243             }
244         }
245     }
246 
247     /**
248      * Determines the root property paths relative to the given root object type against which to perform automatic
249      * linking.
250      *
251      * <p>This will be determined based on the presence of {@link Link} annotations on the given root object type.
252      * This method is invoked recursively as it walks the class structure looking for Link annotations. It uses the
253      * path
254      * and scanned arguments to keep track of how deep into the structure the scanning is and to prevent infinite
255      * recursion.</p>
256      *
257      * @param rootObjectType the root object type from which to perform the scan for auto-linking paths
258      * @param path the current property path relative to the original root object type at which the scan began, if null
259      * then we are scanning from the root-most object type. Each recursive call of this method will append
260      * a new property to this path
261      * @param scanned used to track classes that have already been scanned and prevent infinite recursion
262      * @return a set of property paths that should be auto linked
263      */
264     protected Set<String> determineRootAutoLinkingPaths(Class<?> rootObjectType, String path, Set<Class<?>> scanned) {
265         Set<String> autoLinkingPaths = new HashSet<String>();
266         if (scanned.contains(rootObjectType)) {
267             return autoLinkingPaths;
268         } else {
269             scanned.add(rootObjectType);
270         }
271         Link autoLink = AnnotationUtils.findAnnotation(rootObjectType, Link.class);
272         if (autoLink != null && autoLink.cascade()) {
273             autoLinkingPaths.addAll(assembleAutoLinkingPaths(path, autoLink));
274         } else if (autoLink == null) {
275             Field[] fields = FieldUtils.getAllFields(rootObjectType);
276             for (Field field : fields) {
277                 autoLink = field.getAnnotation(Link.class);
278                 if (autoLink != null) {
279                     if (autoLink.cascade()) {
280                         String fieldPath = appendToPath(path, field.getName());
281                         autoLinkingPaths.addAll(assembleAutoLinkingPaths(fieldPath, autoLink));
282                     }
283                 } else {
284                     autoLinkingPaths.addAll(determineRootAutoLinkingPaths(field.getType(), appendToPath(path,
285                             field.getName()), scanned));
286                 }
287             }
288         }
289         return autoLinkingPaths;
290     }
291 
292     /**
293      * A helper method which simply assembles a set of property paths for the given {@link Link} annotation which
294      * should
295      * be auto linked.
296      *
297      * @param path the property path from the top-most root class to where the Link annotation was found during the
298      * scan
299      * @param autoLink the Link annotation which is being processed
300      * @return a Set of auto linking paths based on the given path parameter, plus the path(s) defined on the
301      * {@link Link} annotation
302      */
303     protected Set<String> assembleAutoLinkingPaths(String path, Link autoLink) {
304         Set<String> autoLinkingPaths = new HashSet<String>();
305         if (ArrayUtils.isEmpty(autoLink.path())) {
306             autoLinkingPaths.add(path);
307         } else {
308             for (String autoLinkingPath : autoLink.path()) {
309                 autoLinkingPaths.add(appendToPath(path, autoLinkingPath));
310             }
311         }
312         return autoLinkingPaths;
313     }
314 
315     /**
316      * Uses the binding result on this data binder to determine the targets on the form that automatic linking should
317      * be performed against.
318      *
319      * <p>Only those property paths for which auto linking is enabled and which were actually modified during the
320      * execution of this data binding will be returned from this method.</p>
321      *
322      * @param autoLinkingPaths a set of paths relative to the form class for which auto-linking has been enabled
323      * @return a list of {@link AutoLinkTarget} objects which contain an object to be linked and which properties on
324      * that object were modified during this data binding execution
325      */
326     protected List<AutoLinkTarget> extractAutoLinkTargets(Set<String> autoLinkingPaths) {
327         List<AutoLinkTarget> targets = new ArrayList<AutoLinkTarget>();
328 
329         for (String autoLinkingPath : autoLinkingPaths) {
330             Object targetObject = getInternalBindingResult().getPropertyAccessor().getPropertyValue(autoLinkingPath);
331             if (targetObject == null) {
332                 continue;
333             }
334 
335             if (targetObject instanceof Map) {
336                 targets.addAll(extractAutoLinkMapTargets(autoLinkingPath, (Map<?, ?>) targetObject));
337 
338                 continue;
339             }
340 
341             if (targetObject instanceof List) {
342                 targets.addAll(extractAutoLinkListTargets(autoLinkingPath, (List<?>) targetObject));
343 
344                 continue;
345             }
346 
347             Set<String> modifiedAutoLinkingPaths = new HashSet<String>();
348 
349             Set<String> modifiedPaths = ((UifBeanPropertyBindingResult) getInternalBindingResult()).getModifiedPaths();
350             for (String modifiedPath : modifiedPaths) {
351                 if (modifiedPath.startsWith(autoLinkingPath)) {
352                     modifiedAutoLinkingPaths.add(modifiedPath.substring(autoLinkingPath.length() + 1));
353                 }
354             }
355 
356             targets.add(new AutoLinkTarget(targetObject, modifiedAutoLinkingPaths));
357         }
358 
359         return targets;
360     }
361 
362     /**
363      * For the map object indicated for linking, iterates through the modified paths and finds paths that match
364      * entries in the map, and if found adds an auto link target.
365      *
366      * @param autoLinkingPath path configured for auto linking
367      * @param targetMap map object for the linking path
368      * @return List of auto linking targets to process
369      */
370     protected List<AutoLinkTarget> extractAutoLinkMapTargets(String autoLinkingPath, Map<?, ?> targetMap) {
371         List<AutoLinkTarget> targets = new ArrayList<AutoLinkTarget>();
372 
373         Set<String> modifiedPaths = ((UifBeanPropertyBindingResult) getInternalBindingResult()).getModifiedPaths();
374 
375         for (Map.Entry<?, ?> targetMapEntry : targetMap.entrySet()) {
376             Set<String> modifiedAutoLinkingPaths = new HashSet<String>();
377 
378             for (String modifiedPath : modifiedPaths) {
379                 String targetPathMatch = autoLinkingPath + "['" + targetMapEntry.getKey() + "']";
380 
381                 if (modifiedPath.startsWith(targetPathMatch)) {
382                     modifiedAutoLinkingPaths.add(modifiedPath.substring(targetPathMatch.length() + 1));
383                 }
384             }
385 
386             if (!modifiedAutoLinkingPaths.isEmpty()) {
387                 targets.add(new AutoLinkTarget(targetMapEntry.getValue(), modifiedAutoLinkingPaths));
388             }
389         }
390 
391         return targets;
392     }
393 
394     /**
395      * For the list object indicated for linking, iterates through the modified paths and finds paths that match
396      * entries in the list, and if found adds an auto link target.
397      *
398      * @param autoLinkingPath path configured for auto linking
399      * @param targetList list object for the linking path
400      * @return List of auto linking targets to process
401      */
402     protected List<AutoLinkTarget> extractAutoLinkListTargets(String autoLinkingPath, List<?> targetList) {
403         List<AutoLinkTarget> targets = new ArrayList<AutoLinkTarget>();
404 
405         Set<String> modifiedPaths = ((UifBeanPropertyBindingResult) getInternalBindingResult()).getModifiedPaths();
406 
407         for (int i = 0; i < targetList.size(); i++) {
408             Set<String> modifiedAutoLinkingPaths = new HashSet<String>();
409 
410             for (String modifiedPath : modifiedPaths) {
411                 String targetPathMatch = autoLinkingPath + "[" + i + "]";
412 
413                 if (modifiedPath.startsWith(targetPathMatch)) {
414                     modifiedAutoLinkingPaths.add(modifiedPath.substring(targetPathMatch.length() + 1));
415                 }
416             }
417 
418             if (!modifiedAutoLinkingPaths.isEmpty()) {
419                 targets.add(new AutoLinkTarget(targetList.get(i), modifiedAutoLinkingPaths));
420             }
421         }
422 
423         return targets;
424     }
425 
426     /**
427      * A utility method which appends two property paths together to create a new nested property path.
428      *
429      * <p>Handles null values for either the path or pathElement. The general output will be path.pathElement
430      * except in situations where either of the two given values are empty or null, in which case only the non-null
431      * value will be returned.</p>
432      *
433      * @param path the prefix of the property path
434      * @param pathElement the suffix of the property path to append to the given path
435      * @return an appended path, appended with a "." between the given path and pathElement (unless one of these is
436      * null)
437      */
438     private String appendToPath(String path, String pathElement) {
439         if (StringUtils.isEmpty(path)) {
440             return pathElement;
441         } else if (StringUtils.isEmpty(pathElement)) {
442             return path;
443         }
444         return path + "." + pathElement;
445     }
446 
447     /**
448      * Attempts to get a view instance by looking for a view type name in the request or the form and querying
449      * that view type with the request parameters
450      *
451      * @param request request instance to pull parameters from
452      * @param form form instance to pull values from
453      * @return View instance if found or null
454      */
455     protected View getViewByType(ServletRequest request, UifFormBase form) {
456         View view = null;
457 
458         String viewTypeName = request.getParameter(UifParameters.VIEW_TYPE_NAME);
459         ViewType viewType = StringUtils.isBlank(viewTypeName) ? form.getViewTypeName() : ViewType.valueOf(viewTypeName);
460 
461         if (viewType != null) {
462             Map<String, String> parameterMap = KRADUtils.translateRequestParameterMap(request.getParameterMap());
463             view = getViewService().getViewByType(viewType, parameterMap);
464         }
465 
466         return view;
467     }
468 
469     /**
470      * Attempts to get a view instance based on the view id stored on the form (which might not be populated
471      * from the request but remaining from session)
472      *
473      * @param form form instance to pull view id from
474      * @return View instance associated with form's view id or null if id or view not found
475      */
476     protected View getViewFromPreviousModel(UifFormBase form) {
477         // maybe we have a view id from the session form
478         if (form.getViewId() != null) {
479             return getViewService().getViewById(form.getViewId());
480         }
481 
482         return null;
483     }
484 
485     public boolean isChangeTracking() {
486         return changeTracking;
487     }
488 
489     public boolean isAutoLinking() {
490         return autoLinking;
491     }
492 
493     public void setAutoLinking(boolean autoLinking) {
494         this.autoLinking = autoLinking;
495     }
496 
497     public ViewService getViewService() {
498         return KRADServiceLocatorWeb.getViewService();
499     }
500 
501     public DataObjectService getDataObjectService() {
502         return this.dataObjectService;
503     }
504 
505     public void setDataObjectService(DataObjectService dataObjectService) {
506         this.dataObjectService = dataObjectService;
507     }
508 
509     /**
510      * Holds an object that will have auto-linking executed against it.
511      *
512      * <p>Also contains a set of property paths (relative to the object) that were modified during the data binding
513      * execution.</p>
514      */
515     private static final class AutoLinkTarget {
516         private final Object target;
517         private final Set<String> modifiedPropertyPaths;
518 
519         AutoLinkTarget(Object target, Set<String> modifiedPropertyPaths) {
520             this.target = target;
521             this.modifiedPropertyPaths = modifiedPropertyPaths;
522         }
523 
524         Object getTarget() {
525             return target;
526         }
527 
528         Set<String> getModifiedPropertyPaths() {
529             return Collections.unmodifiableSet(modifiedPropertyPaths);
530         }
531     }
532 
533 }
534 
535