1
2
3
4
5
6
7
8
9
10
11
12
13
14
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
52
53
54
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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
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
93
94
95
96
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
129
130
131
132
133
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
143
144
145
146
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
158
159
160
161
162
163
164
165
166
167
168
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
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
194
195
196
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
206
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
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
260
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
290
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
306
307
308
309
310
311
312
313
314
315
316
317 public static <T> T unwrap(T obj) {
318 return DelayedCopyableHandler.unwrap(obj);
319 }
320
321
322
323
324
325
326
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
374
375
376
377
378
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
391
392
393
394
395
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
404
405
406
407
408
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
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
426
427
428
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
465
466
467
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
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
489
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
534
535 private void recycle() {
536 queue.clear();
537 cache.clear();
538 RecycleUtils.recycle(this);
539 }
540 }
541
542
543
544
545 private interface CopyReference<T> {
546
547
548
549
550
551
552 Class<T> getTargetClass();
553
554
555
556
557
558
559 String getPath();
560
561
562
563
564
565
566
567 boolean isDelayAvailable();
568
569
570
571
572
573
574 Type getType();
575
576
577
578
579
580
581 Map<String, Type> getTypeVariables();
582
583
584
585
586
587
588
589
590
591
592
593
594 T get();
595
596
597
598
599
600
601
602
603
604
605
606
607 void set(Object value);
608
609
610
611
612 void clean();
613
614 }
615
616
617
618
619 private static <T> void recycle(CopyReference<T> ref) {
620 ref.clean();
621 RecycleUtils.recycle(ref);
622 }
623
624
625
626
627
628 private static class SimpleReference<T> implements CopyReference<T> {
629
630 private T value;
631 private Class<T> targetClass;
632
633
634
635
636
637
638 public Class<T> getTargetClass() {
639 return this.targetClass;
640 }
641
642
643
644
645 @Override
646 public boolean isDelayAvailable() {
647 return false;
648 }
649
650
651
652
653
654
655 @Override
656 public Type getType() {
657 return this.targetClass;
658 }
659
660
661
662
663 @Override
664 public Map<String, Type> getTypeVariables() {
665 return Collections.emptyMap();
666 }
667
668
669
670
671
672
673 @Override
674 public T get() {
675 return value;
676 }
677
678
679
680
681
682
683 @Override
684 public void set(Object value) {
685 this.value = targetClass.cast(value);
686 }
687
688
689
690
691 public String getPath() {
692 return null;
693 }
694
695
696
697
698 @Override
699 public void clean() {
700 this.value = null;
701 this.targetClass = null;
702 }
703 }
704
705
706
707
708
709
710
711
712
713
714
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
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
744
745
746
747 @SuppressWarnings("unchecked")
748 @Override
749 public Class<T> getTargetClass() {
750 return (Class<T>) field.getType();
751 }
752
753
754
755
756 @Override
757 public boolean isDelayAvailable() {
758 return delayAvailable;
759 }
760
761
762
763
764
765
766 @Override
767 public Type getType() {
768 return field.getGenericType();
769 }
770
771
772
773
774 @Override
775 public Map<String, Type> getTypeVariables() {
776 return typeVariables;
777 }
778
779
780
781
782
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
801
802
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
815
816 public String getPath() {
817 return this.path;
818 }
819
820
821
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
837
838
839
840
841
842
843
844
845
846
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
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
918
919
920
921 @SuppressWarnings("unchecked")
922 @Override
923 public Class<T> getTargetClass() {
924 return (Class<T>) source.getClass().getComponentType();
925 }
926
927
928
929
930 @Override
931 public boolean isDelayAvailable() {
932 return delayAvailable;
933 }
934
935
936
937
938
939
940 @Override
941 public Type getType() {
942 return source.getClass().getComponentType();
943 }
944
945
946
947
948 public Map<String, Type> getTypeVariables() {
949 return this.typeVariables;
950 }
951
952
953
954
955
956
957 @SuppressWarnings("unchecked")
958 @Override
959 public T get() {
960 return (T) Array.get(source, index);
961 }
962
963
964
965
966
967
968 @Override
969 public void set(Object value) {
970 Array.set(target, index, value);
971 }
972
973
974
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
993
994
995
996
997
998
999
1000
1001
1002
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
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
1037
1038
1039
1040 @Override
1041 public Class<T> getTargetClass() {
1042 return targetClass;
1043 }
1044
1045
1046
1047
1048 public boolean isDelayAvailable() {
1049 return this.delayAvailable;
1050 }
1051
1052
1053
1054
1055
1056
1057 @Override
1058 public Type getType() {
1059 return type;
1060 }
1061
1062
1063
1064
1065 public Map<String, Type> getTypeVariables() {
1066 return this.typeVariables;
1067 }
1068
1069
1070
1071
1072
1073
1074 @Override
1075 public T get() {
1076 return targetClass.cast(source.get(index));
1077 }
1078
1079
1080
1081
1082
1083
1084 @Override
1085 public void set(Object value) {
1086 target.set(index, targetClass.cast(value));
1087 }
1088
1089
1090
1091
1092 public String getPath() {
1093 return this.path;
1094 }
1095
1096
1097
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
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
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
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
1165
1166
1167
1168 @Override
1169 public Class<T> getTargetClass() {
1170 return targetClass;
1171 }
1172
1173
1174
1175
1176 public boolean isDelayAvailable() {
1177 return this.delayAvailable;
1178 }
1179
1180
1181
1182
1183
1184
1185 @Override
1186 public Type getType() {
1187 return type;
1188 }
1189
1190
1191
1192
1193 public Map<String, Type> getTypeVariables() {
1194 return this.typeVariables;
1195 }
1196
1197
1198
1199
1200
1201
1202 @Override
1203 public T get() {
1204 return sourceEntry.getValue();
1205 }
1206
1207
1208
1209
1210
1211
1212 @Override
1213 public void set(Object value) {
1214 target.put(sourceEntry.getKey(), targetClass.cast(value));
1215 }
1216
1217
1218
1219
1220 public String getPath() {
1221 return this.path;
1222 }
1223
1224
1225
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
1240
1241
1242
1243
1244
1245
1246
1247
1248
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
1277
1278
1279
1280 private static class ClassMetadata {
1281
1282
1283
1284
1285 private final Method cloneMethod;
1286
1287
1288
1289
1290
1291 private final List<Field> cloneFields;
1292
1293
1294
1295
1296
1297 private final Map<Field, Class<?>> collectionTypeByField;
1298
1299
1300
1301
1302 private final Map<Class<?>, Map<String, Annotation>> annotatedFieldsByAnnotationType;
1303
1304
1305
1306
1307
1308
1309 private ClassMetadata(Class<?> targetClass) {
1310 cloneMethod = MethodUtils.getAccessibleMethod(targetClass, "clone", new Class[0]);
1311
1312
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
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
1380
1381 private static final Map<Class<?>, ClassMetadata> CLASS_META_CACHE =
1382 Collections.synchronizedMap(new WeakHashMap<Class<?>, ClassMetadata>());
1383
1384
1385
1386
1387
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 }