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.uif.util;
17  
18  import java.lang.annotation.Annotation;
19  import java.lang.reflect.Array;
20  import java.lang.reflect.Field;
21  import java.lang.reflect.InvocationTargetException;
22  import java.lang.reflect.Method;
23  import java.lang.reflect.Modifier;
24  import java.lang.reflect.ParameterizedType;
25  import java.lang.reflect.Type;
26  import java.lang.reflect.TypeVariable;
27  import java.util.ArrayList;
28  import java.util.Collections;
29  import java.util.HashMap;
30  import java.util.HashSet;
31  import java.util.IdentityHashMap;
32  import java.util.LinkedList;
33  import java.util.List;
34  import java.util.Map;
35  import java.util.Map.Entry;
36  import java.util.Queue;
37  import java.util.WeakHashMap;
38  
39  import org.apache.commons.beanutils.MethodUtils;
40  import org.kuali.rice.core.api.config.property.Config;
41  import org.kuali.rice.core.api.config.property.ConfigContext;
42  import org.kuali.rice.krad.datadictionary.Copyable;
43  import org.kuali.rice.krad.uif.component.DelayedCopy;
44  import org.kuali.rice.krad.uif.component.ReferenceCopy;
45  import org.kuali.rice.krad.uif.lifecycle.ViewLifecycle;
46  import org.kuali.rice.krad.util.KRADConstants;
47  import org.slf4j.Logger;
48  import org.slf4j.LoggerFactory;
49  
50  /**
51   * Provides a lightweight "hands-free" copy implementation to replace the need for copyProperties()
52   * in building {@link LifecycleElement} implementations.
53   * 
54   * @author Kuali Rice Team (rice.collab@kuali.org)
55   */
56  public final class CopyUtils {
57  
58      private static Logger LOG = LoggerFactory.getLogger(CopyUtils.class);
59  
60      private static Boolean delay;
61  
62      /**
63       * Determine whether or not to use a delayed copy proxy.
64       * 
65       * <p>
66       * When true, and {@link #isUseClone()} is also true, then deep copy operations will be
67       * truncated where a copyable represented by an interfaces is specified by the field, array,
68       * list or map involved indicated. Rather than copy the object directly, a proxy wrapping the
69       * original will be placed, which when used will invoke the copy operation.
70       * </p>
71       * 
72       * <p>
73       * This value is controlled by the parameter &quot;krad.uif.copyable.delay&quot;. By default,
74       * full deep copy will be used.
75       * </p>
76       *
77       * @return True if deep copy will be truncated with a delayed copy proxy, false for full deep
78       *         copy.
79       */
80      public static boolean isDelay() {
81          if (delay == null) {
82              boolean defaultDelay = false;
83              Config config = ConfigContext.getCurrentContextConfig();
84              delay = config == null ? defaultDelay : config.getBooleanProperty(
85                      KRADConstants.ConfigParameters.KRAD_COPY_DELAY, defaultDelay);
86          }
87  
88          return delay;
89      }
90  
91      /**
92       * Mix-in copy implementation for objects that implement the {@link Copyable} interface}
93       * 
94       * @param <T> copyable type
95       * @param obj The object to copy.
96       * @return A deep copy of the object.
97       */
98      @SuppressWarnings("unchecked")
99      public static <T> T copy(Copyable obj) {
100         if (obj == null) {
101             return null;
102         }
103 
104         if ((obj instanceof LifecycleAwareList) || (obj instanceof LifecycleAwareMap)) {
105             try {
106                 return (T) obj.clone();
107             } catch (CloneNotSupportedException e) {
108                 throw new IllegalStateException("Unexpected error in clone()", e);
109             }
110         }
111 
112         String cid = null;
113         if (ViewLifecycle.isTrace()) {
114             StackTraceElement[] trace = Thread.currentThread().getStackTrace();
115             int i = 3;
116             while (ComponentUtils.class.getName().equals(trace[i].getClassName()))
117                 i++;
118             StackTraceElement caller = trace[i];
119             cid = obj.getClass().getSimpleName() + ":" + caller.getClassName()
120                     + ":" + caller.getMethodName() + ":" + caller.getLineNumber();
121             ProcessLogger.ntrace("deep-copy:", ":" + cid, 1000L, 500L);
122         }
123 
124         return (T) getDeepCopy(obj);
125     }
126 
127     /**
128      * Determine if shallow copying is available on an object.
129      * 
130      * @param <T> copyable type
131      * @param obj The object to check.
132      * @return True if {@link #getShallowCopy(Object)} may be expected to return a shallow copy of
133      *         the object. False if a null return value is expected.
134      */
135     public static <T> boolean isShallowCopyAvailable(T obj) {
136         return obj != null &&
137                 (isDeepCopyAvailable(obj.getClass()) ||
138                 getMetadata(obj.getClass()).cloneMethod != null);
139     }
140 
141     /**
142      * Determine if deep copying is available for a type.
143      * 
144      * @param type The type to check.
145      * @return True if {@link #getDeepCopy(Object)} may be expected to follow references to this
146      *         type. False if the type should not be deeply copied.
147      */
148     public static boolean isDeepCopyAvailable(Class<?> type) {
149         return type != null
150                 && (Copyable.class.isAssignableFrom(type)
151                         || List.class.isAssignableFrom(type)
152                         || Map.class.isAssignableFrom(type)
153                         || (type.isArray() && isDeepCopyAvailable(type.getComponentType())));
154     }
155 
156     /**
157      * Get a shallow copy (clone) of an object.
158      * 
159      * <p>
160      * This method simplifies access to the clone() method.
161      * </p>
162      * 
163      * @param <T> copyable type
164      * @param obj The object to clone.
165      * @return A shallow copy of obj, or null if obj is null.
166      * @throws CloneNotSupportedException If copying is not available on the object, or if thrown by
167      *         clone() itself. When isShallowCopyAvailable() returns true, then this exception is
168      *         not expected and may be considered an internal error.
169      */
170     @SuppressWarnings("unchecked")
171     public static <T> T getShallowCopy(T obj) throws CloneNotSupportedException {
172         if (obj == null) {
173             return null;
174         }
175 
176         if (ViewLifecycle.isTrace()) {
177             ProcessLogger.ntrace("clone:", ":" + obj.getClass().getSimpleName(), 1000);
178         }
179 
180         synchronized (obj) {
181             // Create new mutable, cloneable instances of List/Map to replace wrappers
182             if (!(obj instanceof Cloneable)) {
183                 if (obj instanceof List) {
184                     return (T) new ArrayList<Object>((List<Object>) obj);
185                 } else if (obj instanceof Map) {
186                     return (T) new HashMap<Object, Object>((Map<Object, Object>) obj);
187                 } else {
188                     throw new UnsupportedOperationException(
189                             "Not cloneable, and not a supported collection.  This condition should not be reached.");
190                 }
191             }
192 
193             // Bypass reflection overhead for commonly used types.
194             // There is not need for these checks to be exhaustive - any Cloneable types
195             // that define a public clone method and aren't specifically noted below
196             // will be cloned by reflection.
197             if (obj instanceof Copyable) {
198                 return (T) ((Copyable) obj).clone();
199             }
200 
201             if (obj instanceof Object[]) {
202                 return (T) ((Object[]) obj).clone();
203             }
204 
205             // synchronized on collections/maps below here is to avoid
206             // concurrent modification
207             if (obj instanceof ArrayList) {
208                 synchronized (obj) {
209                     return (T) ((ArrayList<?>) obj).clone();
210                 }
211             }
212 
213             if (obj instanceof LinkedList) {
214                 synchronized (obj) {
215                     return (T) ((LinkedList<?>) obj).clone();
216                 }
217             }
218 
219             if (obj instanceof HashSet) {
220                 synchronized (obj) {
221                     return (T) ((HashSet<?>) obj).clone();
222                 }
223             }
224 
225             if (obj instanceof HashMap) {
226                 synchronized (obj) {
227                     return (T) ((HashMap<?, ?>) obj).clone();
228                 }
229             }
230 
231             // Use reflection to invoke a public clone() method on the object, if available.
232             Method cloneMethod = getMetadata(obj.getClass()).cloneMethod;
233             if (cloneMethod == null) {
234                 throw new CloneNotSupportedException(obj.getClass() + " does not define a public clone() method");
235             } else {
236                 try {
237 
238                     return (T) cloneMethod.invoke(obj);
239 
240                 } catch (IllegalAccessException e) {
241                     throw new IllegalStateException("Access error invoking clone()", e);
242                 } catch (InvocationTargetException e) {
243                     Throwable cause = e.getCause();
244                     if (cause instanceof RuntimeException) {
245                         throw (RuntimeException) cause;
246                     } else if (cause instanceof Error) {
247                         throw (Error) cause;
248                     } else if (cause instanceof CloneNotSupportedException) {
249                         throw (CloneNotSupportedException) cause;
250                     } else {
251                         throw new IllegalStateException("Unexpected error invoking clone()", e);
252                     }
253                 }
254             }
255         }
256     }
257 
258     /**
259      * Helper for {@link #preventModification(Copyable)} and {@link #getDeepCopy(Object)} for
260      * detecting whether or not to queue deep references from the current node.
261      */
262     private static boolean isDeep(CopyReference<?> ref, Object source) {
263         if (!(ref instanceof FieldReference)) {
264             return true;
265         }
266 
267         FieldReference<?> fieldRef = (FieldReference<?>) ref;
268         Field field = fieldRef.field;
269 
270         if (field.isAnnotationPresent(ReferenceCopy.class)) {
271             return false;
272         }
273 
274         if (!(source instanceof Copyable) &&
275                 ((source instanceof Map) || (source instanceof List))) {
276             Class<?> collectionType = getMetadata(fieldRef.source.getClass())
277                     .collectionTypeByField.get(field);
278 
279             if (!Object.class.equals(collectionType)
280                     && !isDeepCopyAvailable(collectionType)) {
281                 return false;
282             }
283         }
284 
285         return true;
286     }
287 
288     /**
289      * Helper for {@link #getDeepCopy(Object)} to detect whether or not to copy the current node or
290      * to keep the cloned reference.
291      */
292     private static boolean isCopy(CopyReference<?> ref) {
293         if (!(ref instanceof FieldReference)) {
294             return true;
295         }
296 
297         FieldReference<?> fieldRef = (FieldReference<?>) ref;
298         Field field = fieldRef.field;
299         ReferenceCopy refCopy = (ReferenceCopy) field.getAnnotation(ReferenceCopy.class);
300 
301         return refCopy == null || refCopy.newCollectionInstance();
302     }
303 
304     /**
305      * Unwrap an object from any wrapper class or proxy it may be decorated with related to the copy
306      * process.
307      * 
308      * <p>
309      * This method is a public utility passthrough for
310      * {@link DelayedCopyableHandler#getDelayedCopy(Copyable)}.
311      * </p>
312      * 
313      * @param obj an object.
314      * @return The non-proxied bean represented by source, copied if needed. When source is not
315      *         copyable, or not proxied, it is returned as-is.
316      */
317     public static <T> T unwrap(T obj) {
318         return DelayedCopyableHandler.unwrap(obj);
319     }
320 
321     /**
322      * Get a deep copy of an object using cloning.
323      * 
324      * @param <T> copyable type
325      * @param obj The object to get a deep copy of.
326      * @return A deep copy of the object.
327      */
328     @SuppressWarnings("unchecked")
329     public static <T> T getDeepCopy(T obj) {
330         CopyState copyState = RecycleUtils.getRecycledInstance(CopyState.class);
331         if (copyState == null) {
332             copyState = new CopyState();
333         }
334 
335         obj = unwrap(obj);
336         
337         SimpleReference<?> topReference = getSimpleReference(obj);
338         try {
339             copyState.queue.offer(topReference);
340 
341             while (!copyState.queue.isEmpty()) {
342                 CopyReference<?> toCopy = copyState.queue.poll();
343                 Object source = toCopy.get();
344 
345                 if (!isShallowCopyAvailable(source) || !isCopy(toCopy)) {
346                     continue;
347                 }
348 
349                 if (source instanceof Copyable) {
350                     source = unwrap(source);
351                 }
352 
353                 if (ViewLifecycle.isTrace()) {
354                     ProcessLogger.ntrace("deep-copy:", ":" + toCopy.getPath(), 10000, 1000);
355                 }
356                 
357                 toCopy.set(copyState.getTarget(source, isDeep(toCopy, source), toCopy));
358 
359                 if (toCopy != topReference) {
360                     recycle(toCopy);
361                 }
362             }
363 
364             return (T) topReference.get();
365 
366         } finally {
367             recycle(topReference);
368             copyState.recycle();
369         }
370     }
371 
372     /**
373      * Retrieves all field names for the given class that have the given annotation
374      *
375      * @param clazz class to find field annotations for
376      * @param annotationClass class for annotation to find
377      * @return map containing the field name that has the annotation as a key and the annotation
378      *         instance as a value
379      */
380     public static Map<String, Annotation> getFieldsWithAnnotation(Class<?> clazz,
381             Class<? extends Annotation> annotationClass) {
382         if (clazz == null) {
383             return Collections.<String, Annotation> emptyMap();
384         }
385         Map<String, Annotation> rv = getMetadata(clazz).annotatedFieldsByAnnotationType.get(annotationClass);
386         return rv == null ? Collections.<String, Annotation> emptyMap() : rv;
387     }
388 
389     /**
390      * Determines whether the field of the given class has the given annotation specified
391      *
392      * @param clazz class containing the field to check
393      * @param fieldName name of the field to check
394      * @param annotationClass class for the annotation to look for
395      * @return true if the named field has the given annotation, false if not
396      */
397     public static boolean fieldHasAnnotation(Class<?> clazz, String fieldName,
398             Class<? extends Annotation> annotationClass) {
399         return getFieldAnnotation(clazz, fieldName, annotationClass) != null;
400     }
401 
402     /**
403      * Returns annotation of the given type for the given field (if present)
404      *
405      * @param clazz class containing the field to check
406      * @param propertyName name of the field to check
407      * @param annotationClass class for the annotation to look for
408      * @return annnotation on field, or null
409      */
410     public static Annotation getFieldAnnotation(Class<?> clazz, String fieldName,
411             Class<? extends Annotation> annotationClass) {
412         Map<String, Annotation> annotationsByField = getFieldsWithAnnotation(clazz, annotationClass);
413         return annotationsByField == null ? null : annotationsByField.get(fieldName);
414     }
415 
416     /**
417      * Holds copy state for use with {@link #getDeepCopy(Object)}.
418      */
419     private static class CopyState {
420 
421         private final Queue<CopyReference<?>> queue = new LinkedList<CopyReference<?>>();
422         private final Map<Object, Object> cache = new IdentityHashMap<Object, Object>();
423 
424         /**
425          * Get a shallow copy of the source object appropriate for the current copy operation.
426          * 
427          * @param source The original source object to copy.
428          * @return A shallow copy of the source object.
429          */
430         private Object getTarget(Object source, boolean queueDeepReferences, CopyReference<?> ref) {
431             boolean useCache = source != Collections.EMPTY_LIST && source != Collections.EMPTY_MAP;
432 
433             Object target = useCache ? cache.get(source) : null;
434 
435             if (target == null) {
436                 Class<?> targetClass = ref.getTargetClass();
437 
438                 if (Copyable.class.isAssignableFrom(targetClass) && targetClass.isInterface()
439                         && ref.isDelayAvailable() && isDelay()) {
440                     target = DelayedCopyableHandler.getDelayedCopy((Copyable) source);
441 
442                 } else {
443 
444                     try {
445                         target = getShallowCopy(source);
446                     } catch (CloneNotSupportedException e) {
447                         throw new IllegalStateException("Unexpected cloning error during shallow copy", e);
448                     }
449 
450                     if (queueDeepReferences) {
451                         queueDeepCopyReferences(source, target, ref);
452                     }
453                 }
454 
455                 if (useCache) {
456                     cache.put(source, target);
457                 }
458             }
459 
460             return target;
461         }
462 
463         /**
464          * Queues references for deep copy after performing the shallow copy.
465          * 
466          * @param source The original source object at the current node.
467          * @param target A shallow copy of the source object, to be analyzed for deep copy.
468          */
469         private void queueDeepCopyReferences(Object source, Object target, CopyReference<?> ref) {
470             Class<?> type = source.getClass();
471             Class<?> targetClass = ref.getTargetClass();
472             if (!isDeepCopyAvailable(type)) {
473                 return;
474             }
475 
476             // Don't queue references if the source has already been seen
477             if (cache.containsKey(source)) {
478                 return;
479             } else if (target == null) {
480                 cache.put(source, source);
481             }
482 
483             if (Copyable.class.isAssignableFrom(type)) {
484                 for (Field field : getMetadata(type).cloneFields) {
485                     queue.offer(getFieldReference(source, target, field, ref));
486                 }
487 
488                 // Used fields for deep copying, even if List or Map is implemented.
489                 // The wrapped list/map should be picked up as a field during deep copy.
490                 return;
491             }
492 
493             if (List.class.isAssignableFrom(targetClass)) {
494                 List<?> sourceList = (List<?>) source;
495                 List<?> targetList = (List<?>) target;
496                 Type componentType = ObjectPropertyUtils.getComponentType(ref.getType());
497 
498                 if (componentType instanceof TypeVariable<?>) {
499                     TypeVariable<?> tvar = (TypeVariable<?>) componentType;
500                     if (ref.getTypeVariables().containsKey(tvar.getName())) {
501                         componentType = ref.getTypeVariables().get(tvar.getName());
502                     }
503                 }
504 
505                 Class<?> componentClass = ObjectPropertyUtils.getUpperBound(componentType);
506 
507                 for (int i = 0; i < sourceList.size(); i++) {
508                     queue.offer(getListReference(sourceList, targetList,
509                             i, componentClass, componentType, ref));
510                 }
511             }
512 
513             if (Map.class.isAssignableFrom(targetClass)) {
514                 Map<?, ?> sourceMap = (Map<?, ?>) source;
515                 Map<?, ?> targetMap = (Map<?, ?>) target;
516                 Type componentType = ObjectPropertyUtils.getComponentType(ref.getType());
517                 Class<?> componentClass = ObjectPropertyUtils.getUpperBound(componentType);
518 
519                 for (Map.Entry<?, ?> sourceEntry : sourceMap.entrySet()) {
520                     queue.offer(getMapReference(sourceEntry, targetMap,
521                             componentClass, componentType, ref));
522                 }
523             }
524 
525             if (targetClass.isArray()) {
526                 for (int i = 0; i < Array.getLength(source); i++) {
527                     queue.offer(getArrayReference(source, target, i, ref));
528                 }
529             }
530         }
531 
532         /**
533          * Clear queue and cache, and recycle this state object.
534          */
535         private void recycle() {
536             queue.clear();
537             cache.clear();
538             RecycleUtils.recycle(this);
539         }
540     }
541 
542     /**
543      * Represents a abstract reference to a targeted value for use during deep copying.
544      */
545     private interface CopyReference<T> {
546 
547         /**
548          * Gets the type this reference refers to.
549          * 
550          * @return the class referred to
551          */
552         Class<T> getTargetClass();
553 
554         /**
555          * This method ...
556          * 
557          * @return
558          */
559         String getPath();
560 
561         /**
562          * Determines whether or not a delayed copy proxy should be considered on this reference.
563          * 
564          * @return True if a delayed copy proxy may be used with this reference, false to always
565          *         perform deep copy.
566          */
567         boolean isDelayAvailable();
568 
569         /**
570          * Gets the generic type this reference refers to.
571          * 
572          * @return the generic type referred to
573          */
574         Type getType();
575 
576         /**
577          * Gets the type variable mapping.
578          * 
579          * @return the type variable mapping.
580          */
581         Map<String, Type> getTypeVariables();
582 
583         /**
584          * Retrieve the targeted value for populating the reference.
585          * 
586          * <p>
587          * This value returned by this method will typically come from a source object, then after
588          * copy operations have been performed {@link #set(Object)} will be called to populate the
589          * target value on the destination object.
590          * </p>
591          * 
592          * @return The targeted value for populating the reference.
593          */
594         T get();
595 
596         /**
597          * Modify the value targeted by the reference.
598          * 
599          * <p>
600          * This value passed to this method will have typically come {@link #get()}. After copy
601          * operations have been performed, this method will be called to populate the target value
602          * on the destination object.
603          * </p>
604          * 
605          * @param value The value to modify the reference as.
606          */
607         void set(Object value);
608 
609         /**
610          * Clean the reference for recycling.
611          */
612         void clean();
613 
614     }
615 
616     /**
617      * Recycle a copy reference for later use, once the copy has been performed.
618      */
619     private static <T> void recycle(CopyReference<T> ref) {
620         ref.clean();
621         RecycleUtils.recycle(ref);
622     }
623 
624     /**
625      * Simple copy reference for holding top-level value to be later inspected and returned. Values
626      * held by this class will be modified in place.
627      */
628     private static class SimpleReference<T> implements CopyReference<T> {
629 
630         private T value;
631         private Class<T> targetClass;
632 
633         /**
634          * Gets the target class.
635          * 
636          * @return target class
637          */
638         public Class<T> getTargetClass() {
639             return this.targetClass;
640         }
641 
642         /**
643          * {@inheritDoc}
644          */
645         @Override
646         public boolean isDelayAvailable() {
647             return false;
648         }
649 
650         /**
651          * Gets the target class.
652          * 
653          * @return target class
654          */
655         @Override
656         public Type getType() {
657             return this.targetClass;
658         }
659 
660         /**
661          * {@inheritDoc}
662          */
663         @Override
664         public Map<String, Type> getTypeVariables() {
665             return Collections.emptyMap();
666         }
667 
668         /**
669          * Gets the value.
670          * 
671          * @return value
672          */
673         @Override
674         public T get() {
675             return value;
676         }
677 
678         /**
679          * Sets the a value.
680          * 
681          * @param value The value to set.
682          */
683         @Override
684         public void set(Object value) {
685             this.value = targetClass.cast(value);
686         }
687 
688         /**
689          * @return the path
690          */
691         public String getPath() {
692             return null;
693         }
694 
695         /**
696          * {@inheritDoc}
697          */
698         @Override
699         public void clean() {
700             this.value = null;
701             this.targetClass = null;
702         }
703     }
704 
705     /**
706      * Get a simple reference for temporary use while deep cloning.
707      * 
708      * <p>
709      * Call {@link #recycle(CopyReference)} when done working with the reference.
710      * </p>
711      * 
712      * @param value The initial object to refer to.
713      * 
714      * @return A simple reference for temporary use while deep cloning.
715      */
716     @SuppressWarnings("unchecked")
717     private static SimpleReference<?> getSimpleReference(Object value) {
718         SimpleReference<Object> ref = RecycleUtils.getRecycledInstance(SimpleReference.class);
719 
720         if (ref == null) {
721             ref = new SimpleReference<Object>();
722         }
723 
724         ref.targetClass = (Class<Object>) value.getClass();
725         ref.value = value;
726 
727         return ref;
728     }
729 
730     /**
731      * Reference implementation for a field on an object.
732      */
733     private static class FieldReference<T> implements CopyReference<T> {
734 
735         private Object source;
736         private Object target;
737         private Field field;
738         private boolean delayAvailable;
739         private Map<String, Type> typeVariables = new HashMap<String, Type>();
740         private String path;
741 
742         /**
743          * Gets the type of the field.
744          * 
745          * {@inheritDoc}
746          */
747         @SuppressWarnings("unchecked")
748         @Override
749         public Class<T> getTargetClass() {
750             return (Class<T>) field.getType();
751         }
752 
753         /**
754          * {@inheritDoc}
755          */
756         @Override
757         public boolean isDelayAvailable() {
758             return delayAvailable;
759         }
760 
761         /**
762          * Gets the generic type of this field.
763          * 
764          * {@inheritDoc}
765          */
766         @Override
767         public Type getType() {
768             return field.getGenericType();
769         }
770 
771         /**
772          * {@inheritDoc}
773          */
774         @Override
775         public Map<String, Type> getTypeVariables() {
776             return typeVariables;
777         }
778 
779         /**
780          * Get a value from the field on the source object.
781          * 
782          * @return The value referred to by the field on the source object.
783          */
784         @SuppressWarnings("unchecked")
785         @Override
786         public T get() {
787             try {
788                 ReferenceCopy ref = field.getAnnotation(ReferenceCopy.class);
789                 if (ref != null && ref.referenceTransient()) {
790                     return null;
791                 }
792 
793                 return (T) field.get(source);
794             } catch (IllegalAccessException e) {
795                 throw new IllegalStateException("Access error attempting to get from " + field, e);
796             }
797         }
798 
799         /**
800          * Set a value for the field on the target object.
801          * 
802          * @param value The value to set for the field on the target object.
803          */
804         @Override
805         public void set(Object value) {
806             try {
807                 field.set(target, value);
808             } catch (IllegalAccessException e) {
809                 throw new IllegalStateException("Access error attempting to set " + field, e);
810             }
811         }
812 
813         /**
814          * @return the path
815          */
816         public String getPath() {
817             return this.path;
818         }
819 
820         /**
821          * {@inheritDoc}
822          */
823         @Override
824         public void clean() {
825             source = null;
826             target = null;
827             field = null;
828             delayAvailable = false;
829             path = null;
830             typeVariables.clear();
831         }
832 
833     }
834 
835     /**
836      * Get a field reference for temporary use while deep cloning.
837      * 
838      * <p>
839      * Call {@link #recycle(CopyReference)} when done working with the reference.
840      * </p>
841      * 
842      * @param source The source object.
843      * @param target The target object.
844      * @param field The field to use as the reference target.
845      * 
846      * @return A field reference for temporary use while deep cloning.
847      */
848     private static <T> FieldReference<T> getFieldReference(Object source, Object target, Field field,
849             CopyReference<T> pref) {
850         @SuppressWarnings("unchecked")
851         FieldReference<T> ref = RecycleUtils.getRecycledInstance(FieldReference.class);
852 
853         if (ref == null) {
854             ref = new FieldReference<T>();
855         }
856 
857         ref.source = source;
858         ref.target = target;
859         ref.field = field;
860 
861         DelayedCopy delayedCopy = field.getAnnotation(DelayedCopy.class);
862         ref.delayAvailable = delayedCopy != null &&
863                 (!delayedCopy.inherit() || pref.isDelayAvailable());
864 
865         Map<String, Type> pTypeVars = pref.getTypeVariables();
866 
867         if (pTypeVars != null && source != null) {
868             Class<?> sourceType = source.getClass();
869             Class<?> targetClass = pref.getTargetClass();
870             Type targetType = ObjectPropertyUtils.findGenericType(sourceType, targetClass);
871             if (targetType instanceof ParameterizedType) {
872                 ParameterizedType parameterizedTargetType = (ParameterizedType) targetType;
873                 Type[] params = parameterizedTargetType.getActualTypeArguments();
874                 for (int j = 0; j < params.length; j++) {
875                     if (params[j] instanceof TypeVariable<?>) {
876                         Type pType = pTypeVars.get(targetClass.getTypeParameters()[j].getName());
877                         ref.typeVariables.put(((TypeVariable<?>) params[j]).getName(), pType);
878                     }
879                 }
880             }
881         }
882 
883         Class<?> rawType = field.getType();
884         Type genericType = field.getGenericType();
885         if (genericType instanceof ParameterizedType) {
886             ParameterizedType parameterizedType = (ParameterizedType) genericType;
887             TypeVariable<?>[] typeParams = rawType.getTypeParameters();
888             Type[] params = parameterizedType.getActualTypeArguments();
889             assert params.length == typeParams.length;
890             for (int i = 0; i < params.length; i++) {
891                 Type paramType = params[i];
892                 if (paramType instanceof TypeVariable<?>) {
893                     Type fType = ref.typeVariables.get(((TypeVariable<?>) paramType).getName());
894                     if (fType != null) {
895                         paramType = fType;
896                     }
897                 }
898                 ref.typeVariables.put(typeParams[i].getName(), paramType);
899             }
900         }
901         return ref;
902     }
903 
904     /**
905      * Reference implementation for an entry in an array.
906      */
907     private static class ArrayReference<T> implements CopyReference<T> {
908 
909         private Object source;
910         private Object target;
911         private int index = -1;
912         private boolean delayAvailable;
913         private String path;
914         private Map<String, Type> typeVariables = new HashMap<String, Type>();
915 
916         /**
917          * Gets the component type of the array.
918          * 
919          * @return component type
920          */
921         @SuppressWarnings("unchecked")
922         @Override
923         public Class<T> getTargetClass() {
924             return (Class<T>) source.getClass().getComponentType();
925         }
926 
927         /**
928          * {@inheritDoc}
929          */
930         @Override
931         public boolean isDelayAvailable() {
932             return delayAvailable;
933         }
934 
935         /**
936          * Gets the component type of the array.
937          * 
938          * @return component type
939          */
940         @Override
941         public Type getType() {
942             return source.getClass().getComponentType();
943         }
944 
945         /**
946          * {@inheritDoc}
947          */
948         public Map<String, Type> getTypeVariables() {
949             return this.typeVariables;
950         }
951 
952         /**
953          * Get the value of the indicated entry in the source array.
954          * 
955          * @return The value of the indicated entry in the source array.
956          */
957         @SuppressWarnings("unchecked")
958         @Override
959         public T get() {
960             return (T) Array.get(source, index);
961         }
962 
963         /**
964          * Modify the value of the indicated entry in the target array.
965          * 
966          * @param value The value to set on the indicated entry in the target array.
967          */
968         @Override
969         public void set(Object value) {
970             Array.set(target, index, value);
971         }
972 
973         /**
974          * @return the path
975          */
976         public String getPath() {
977             return this.path;
978         }
979 
980         @Override
981         public void clean() {
982             source = null;
983             target = null;
984             index = -1;
985             delayAvailable = false;
986             path = null;
987             typeVariables.clear();
988         }
989     }
990 
991     /**
992      * Get an array reference for temporary use while deep cloning.
993      * 
994      * <p>
995      * Call {@link #recycle(CopyReference)} when done working with the reference.
996      * </p>
997      * 
998      * @param source The source array.
999      * @param target The target array.
1000      * @param index The array index.
1001      * 
1002      * @return An array reference for temporary use while deep cloning.
1003      */
1004     private static <T> ArrayReference<T> getArrayReference(
1005             Object source, Object target, int index, CopyReference<?> pref) {
1006         @SuppressWarnings("unchecked")
1007         ArrayReference<T> ref = RecycleUtils.getRecycledInstance(ArrayReference.class);
1008 
1009         if (ref == null) {
1010             ref = new ArrayReference<T>();
1011         }
1012 
1013         ref.source = source;
1014         ref.target = target;
1015         ref.index = index;
1016         ref.delayAvailable = pref.isDelayAvailable();
1017         ref.typeVariables.putAll(pref.getTypeVariables());
1018         return ref;
1019     }
1020 
1021     /**
1022      * Reference implementation for an item in a list.
1023      */
1024     private static class ListReference<T> implements CopyReference<T> {
1025 
1026         private Class<T> targetClass;
1027         private Type type;
1028         private List<T> source;
1029         private List<T> target;
1030         private int index = -1;
1031         private boolean delayAvailable;
1032         private String path;
1033         private Map<String, Type> typeVariables = new HashMap<String, Type>();
1034 
1035         /**
1036          * Gets the item class for the list.
1037          * 
1038          * @return item class
1039          */
1040         @Override
1041         public Class<T> getTargetClass() {
1042             return targetClass;
1043         }
1044 
1045         /**
1046          * {@inheritDoc}
1047          */
1048         public boolean isDelayAvailable() {
1049             return this.delayAvailable;
1050         }
1051 
1052         /**
1053          * Gets the generic item type for the list.
1054          * 
1055          * @return generic item type
1056          */
1057         @Override
1058         public Type getType() {
1059             return type;
1060         }
1061 
1062         /**
1063          * {@inheritDoc}
1064          */
1065         public Map<String, Type> getTypeVariables() {
1066             return this.typeVariables;
1067         }
1068 
1069         /**
1070          * Get the value of the indicated item in the source array.
1071          * 
1072          * @return The value of the indicated item in the source array.
1073          */
1074         @Override
1075         public T get() {
1076             return targetClass.cast(source.get(index));
1077         }
1078 
1079         /**
1080          * Modify the list item.
1081          * 
1082          * @param value The value to modify the list item as.
1083          */
1084         @Override
1085         public void set(Object value) {
1086             target.set(index, targetClass.cast(value));
1087         }
1088 
1089         /**
1090          * @return the path
1091          */
1092         public String getPath() {
1093             return this.path;
1094         }
1095 
1096         /**
1097          * {@inheritDoc}
1098          */
1099         @Override
1100         public void clean() {
1101             targetClass = null;
1102             type = null;
1103             source = null;
1104             target = null;
1105             index = -1;
1106             delayAvailable = false;
1107             typeVariables.clear();
1108         }
1109     }
1110 
1111     /**
1112      * Get a list reference for temporary use while deep cloning.
1113      * 
1114      * <p>
1115      * Call {@link #recycle(CopyReference)} when done working with the reference.
1116      * </p>
1117      * 
1118      * @param source The source list.
1119      * @param target The target list.
1120      * @param index The index of the list item.
1121      * 
1122      * @return A list reference for temporary use while deep cloning.
1123      */
1124     @SuppressWarnings("unchecked")
1125     private static ListReference<?> getListReference(List<?> source, List<?> target, int index,
1126             Class<?> targetClass, Type type, CopyReference<?> pref) {
1127         ListReference<Object> ref = RecycleUtils.getRecycledInstance(ListReference.class);
1128 
1129         if (ref == null) {
1130             ref = new ListReference<Object>();
1131         }
1132 
1133         ref.source = (List<Object>) source;
1134         ref.target = (List<Object>) target;
1135         ref.index = index;
1136         ref.targetClass = (Class<Object>) targetClass;
1137         ref.type = type;
1138         ref.delayAvailable = pref.isDelayAvailable();
1139         ref.typeVariables.putAll(pref.getTypeVariables());
1140 
1141         if (pref == null || pref.getPath() == null) {
1142             ref.path = "[" + index + ']';
1143         } else {
1144             ref.path = pref.getPath() + '[' + index + ']';
1145         }
1146 
1147         return ref;
1148     }
1149 
1150     /**
1151      * Reference implementation for an entry in a map.
1152      */
1153     private static class MapReference<T> implements CopyReference<T> {
1154 
1155         private Class<T> targetClass;
1156         private Type type;
1157         private Map.Entry<Object, T> sourceEntry;
1158         private Map<Object, T> target;
1159         private boolean delayAvailable;
1160         private String path;
1161         private Map<String, Type> typeVariables = new HashMap<String, Type>();
1162 
1163         /**
1164          * Gets the value class for the map.
1165          * 
1166          * @return value class
1167          */
1168         @Override
1169         public Class<T> getTargetClass() {
1170             return targetClass;
1171         }
1172 
1173         /**
1174          * @return the delayAvailable
1175          */
1176         public boolean isDelayAvailable() {
1177             return this.delayAvailable;
1178         }
1179 
1180         /**
1181          * Gets the generic value type for the map.
1182          * 
1183          * @return generic value type
1184          */
1185         @Override
1186         public Type getType() {
1187             return type;
1188         }
1189 
1190         /**
1191          * {@inheritDoc}
1192          */
1193         public Map<String, Type> getTypeVariables() {
1194             return this.typeVariables;
1195         }
1196 
1197         /**
1198          * Get the value of the map entry.
1199          * 
1200          * @return The value of the map entry.
1201          */
1202         @Override
1203         public T get() {
1204             return sourceEntry.getValue();
1205         }
1206 
1207         /**
1208          * Modify the map entry.
1209          * 
1210          * @param value The value to modify the map entry with.
1211          */
1212         @Override
1213         public void set(Object value) {
1214             target.put(sourceEntry.getKey(), targetClass.cast(value));
1215         }
1216 
1217         /**
1218          * @return the path
1219          */
1220         public String getPath() {
1221             return this.path;
1222         }
1223 
1224         /**
1225          * {@inheritDoc}
1226          */
1227         @Override
1228         public void clean() {
1229             targetClass = null;
1230             type = null;
1231             sourceEntry = null;
1232             target = null;
1233             delayAvailable = false;
1234             typeVariables.clear();
1235         }
1236     }
1237 
1238     /**
1239      * Get a map reference for temporary use while deep cloning.
1240      * 
1241      * <p>
1242      * Call {@link #recycle(CopyReference)} when done working with the reference.
1243      * </p>
1244      * 
1245      * @param sourceEntry The source entry.
1246      * @param target The target map.
1247      * 
1248      * @return A map reference for temporary use while deep cloning.
1249      */
1250     @SuppressWarnings("unchecked")
1251     private static MapReference<?> getMapReference(Map.Entry<?, ?> sourceEntry, Map<?, ?> target,
1252             Class<?> targetClass, Type type, CopyReference<?> pref) {
1253         MapReference<Object> ref = RecycleUtils.getRecycledInstance(MapReference.class);
1254 
1255         if (ref == null) {
1256             ref = new MapReference<Object>();
1257         }
1258 
1259         ref.sourceEntry = (Map.Entry<Object, Object>) sourceEntry;
1260         ref.target = (Map<Object, Object>) target;
1261         ref.targetClass = (Class<Object>) targetClass;
1262         ref.type = type;
1263         ref.delayAvailable = pref.isDelayAvailable();
1264         ref.typeVariables.putAll(pref.getTypeVariables());
1265 
1266         if (pref == null || pref.getPath() == null) {
1267             ref.path = "[" + sourceEntry.getKey() + ']';
1268         } else {
1269             ref.path = pref.getPath() + '[' + sourceEntry.getKey() + ']';
1270         }
1271 
1272         return ref;
1273     }
1274 
1275     /**
1276      * Internal field cache meta-data node, for reducing field lookup overhead.
1277      * 
1278      * @author Kuali Rice Team (rice.collab@kuali.org)
1279      */
1280     private static class ClassMetadata {
1281 
1282         /**
1283          * The public clone method, if defined for this type.
1284          */
1285         private final Method cloneMethod;
1286 
1287         /**
1288          * All fields on the class that should have a shallow copy performed during a deep copy
1289          * operation.
1290          */
1291         private final List<Field> cloneFields;
1292 
1293         /**
1294          * Mapping from field to generic collection type, for Map and List fields that should be
1295          * deep copied.
1296          */
1297         private final Map<Field, Class<?>> collectionTypeByField;
1298 
1299         /**
1300          * Mapping from annotation type to field name to annotation mapping.
1301          */
1302         private final Map<Class<?>, Map<String, Annotation>> annotatedFieldsByAnnotationType;
1303 
1304         /**
1305          * Create a new field reference for a target class.
1306          * 
1307          * @param targetClass The class to inspect for meta-data.
1308          */
1309         private ClassMetadata(Class<?> targetClass) {
1310             cloneMethod = MethodUtils.getAccessibleMethod(targetClass, "clone", new Class[0]);
1311 
1312             // Create mutable collections for building meta-data indexes.
1313             List<Field> cloneList = new ArrayList<Field>();
1314             Map<Field, Class<?>> collectionTypeMap = new HashMap<Field, Class<?>>();
1315             Map<Class<?>, Map<String, Annotation>> annotationMap = new HashMap<Class<?>, Map<String, Annotation>>();
1316 
1317             Class<?> currentClass = targetClass;
1318             while (currentClass != Object.class && currentClass != null) {
1319 
1320                 for (Field currentField : currentClass.getDeclaredFields()) {
1321                     if ((currentField.getModifiers() & Modifier.STATIC) == Modifier.STATIC) {
1322                         continue;
1323                     }
1324 
1325                     Annotation[] annotations = currentField.getAnnotations();
1326                     if (annotations != null) {
1327                         for (Annotation annotation : annotations) {
1328                             Class<?> annotationType = annotation.annotationType();
1329                             Map<String, Annotation> amap = annotationMap.get(annotationType);
1330 
1331                             if (amap == null) {
1332                                 amap = new HashMap<String, Annotation>();
1333                                 annotationMap.put(annotationType, amap);
1334                             }
1335 
1336                             amap.put(currentField.getName(), annotation);
1337                         }
1338                     }
1339 
1340                     Class<?> type = currentField.getType();
1341 
1342                     if (type.isArray()
1343                             || isDeepCopyAvailable(type)
1344                             || Cloneable.class.isAssignableFrom(type)) {
1345                         currentField.setAccessible(true);
1346                         cloneList.add(currentField);
1347                     }
1348 
1349                     boolean isList = List.class.isAssignableFrom(type);
1350                     boolean isMap = Map.class.isAssignableFrom(type);
1351                     if (!isList && !isMap) {
1352                         continue;
1353                     }
1354 
1355                     Class<?> collectionType = ObjectPropertyUtils
1356                             .getUpperBound(ObjectPropertyUtils
1357                                     .getComponentType(currentField.getGenericType()));
1358 
1359                     if (collectionType.equals(Object.class) || isDeepCopyAvailable(collectionType)) {
1360                         collectionTypeMap.put(currentField, collectionType);
1361                     }
1362                 }
1363 
1364                 currentClass = currentClass.getSuperclass();
1365             }
1366 
1367             // Seal index collections to prevent external modification.
1368             cloneFields = Collections.unmodifiableList(cloneList);
1369             collectionTypeByField = Collections.unmodifiableMap(collectionTypeMap);
1370 
1371             for (Entry<Class<?>, Map<String, Annotation>> aentry : annotationMap.entrySet()) {
1372                 aentry.setValue(Collections.unmodifiableMap(aentry.getValue()));
1373             }
1374             annotatedFieldsByAnnotationType = Collections.unmodifiableMap(annotationMap);
1375         }
1376     }
1377 
1378     /**
1379      * Static cache for reducing annotated field lookup overhead.
1380      */
1381     private static final Map<Class<?>, ClassMetadata> CLASS_META_CACHE =
1382             Collections.synchronizedMap(new WeakHashMap<Class<?>, ClassMetadata>());
1383 
1384     /**
1385      * Get copy metadata for a class.
1386      * @param targetClass The class.
1387      * @return Copy metadata for the class.
1388      */
1389     private static final ClassMetadata getMetadata(Class<?> targetClass) {
1390         ClassMetadata metadata = CLASS_META_CACHE.get(targetClass);
1391 
1392         if (metadata == null) {
1393             CLASS_META_CACHE.put(targetClass, metadata = new ClassMetadata(targetClass));
1394         }
1395 
1396         return metadata;
1397     }
1398 
1399 }