1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.kuali.rice.krad.data.provider.util;
17
18 import com.google.common.collect.Sets;
19 import org.apache.commons.collections.CollectionUtils;
20 import org.apache.commons.lang.ArrayUtils;
21 import org.apache.commons.lang.StringUtils;
22 import org.apache.commons.lang3.reflect.FieldUtils;
23 import org.kuali.rice.krad.data.DataObjectService;
24 import org.kuali.rice.krad.data.DataObjectWrapper;
25 import org.kuali.rice.krad.data.link.Link;
26 import org.kuali.rice.krad.data.metadata.DataObjectAttributeRelationship;
27 import org.kuali.rice.krad.data.metadata.DataObjectCollection;
28 import org.kuali.rice.krad.data.metadata.DataObjectMetadata;
29 import org.kuali.rice.krad.data.metadata.DataObjectRelationship;
30 import org.kuali.rice.krad.data.metadata.MetadataChild;
31 import org.slf4j.Logger;
32 import org.slf4j.LoggerFactory;
33 import org.springframework.beans.PropertyAccessor;
34 import org.springframework.beans.PropertyAccessorUtils;
35 import org.springframework.core.annotation.AnnotationUtils;
36
37 import java.lang.reflect.Field;
38 import java.util.ArrayList;
39 import java.util.Arrays;
40 import java.util.HashMap;
41 import java.util.HashSet;
42 import java.util.List;
43 import java.util.Map;
44 import java.util.Set;
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67 public class ReferenceLinker {
68
69 private static final Logger LOG = LoggerFactory.getLogger(ReferenceLinker.class);
70
71 private DataObjectService dataObjectService;
72
73
74
75
76
77
78 public DataObjectService getDataObjectService() {
79 return dataObjectService;
80 }
81
82
83
84
85
86
87
88
89 public void setDataObjectService(DataObjectService dataObjectService) {
90 this.dataObjectService = dataObjectService;
91 }
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107 public void linkChanges(Object rootObject, Set<String> changedPropertyPaths) {
108 if (rootObject == null || CollectionUtils.isEmpty(changedPropertyPaths)) {
109 return;
110 }
111 Class<?> type = rootObject.getClass();
112 if (!dataObjectService.supports(type)) {
113 LOG.info("Object supplied for linking is not a supported data object type: " + type);
114 return;
115 }
116 if (LOG.isInfoEnabled()) {
117 LOG.info("Performing linking on instance of " + type + " with the following changed property paths: " +
118 Arrays.toString(changedPropertyPaths.toArray()));
119 }
120 Map<String, Set<String>> decomposedPaths = decomposePropertyPaths(changedPropertyPaths);
121 linkChangesInternal(rootObject, decomposedPaths, new HashSet<Object>());
122 }
123
124
125
126
127
128
129
130
131
132
133
134 protected void linkChangesInternal(Object object, Map<String, Set<String>> changedPropertyPaths,
135 Set<Object> linked) {
136 if (object == null || linked.contains(object) || !dataObjectService.supports(object.getClass()) ||
137 changedPropertyPaths.isEmpty()) {
138 return;
139 }
140 linked.add(object);
141 DataObjectWrapper<?> wrapped = dataObjectService.wrap(object);
142
143
144 linkRelationshipChanges(wrapped, changedPropertyPaths, linked);
145 linkCollectionChanges(wrapped, changedPropertyPaths, linked);
146 cascadeLinkingAnnotations(wrapped, changedPropertyPaths, linked);
147 }
148
149
150
151
152
153
154
155
156 protected void linkRelationshipChanges(DataObjectWrapper<?> wrapped, Map<String, Set<String>> decomposedPaths,
157 Set<Object> linked) {
158 List<DataObjectRelationship> relationships = wrapped.getMetadata().getRelationships();
159 for (DataObjectRelationship relationship : relationships) {
160 String relationshipName = relationship.getName();
161
162
163 if (relationship.isSavedWithParent() && decomposedPaths.containsKey(relationshipName)) {
164 Object value = wrapped.getPropertyValue(relationshipName);
165 Map<String, Set<String>> nextPropertyPaths =
166 decomposePropertyPaths(decomposedPaths.get(relationshipName));
167 linkChangesInternal(value, nextPropertyPaths, linked);
168 }
169
170
171
172 List<DataObjectAttributeRelationship> attributeRelationships = relationship.getAttributeRelationships();
173 boolean modifiedAttributeRelationship = false;
174 for (DataObjectAttributeRelationship attributeRelationship : attributeRelationships) {
175 if (decomposedPaths.containsKey(attributeRelationship.getParentAttributeName())) {
176 modifiedAttributeRelationship = true;
177 break;
178 }
179 }
180 if (modifiedAttributeRelationship) {
181
182 wrapped.fetchRelationship(relationshipName, true, true);
183 } else if (decomposedPaths.containsKey(relationshipName)) {
184
185 Class<?> targetType = relationship.getRelatedType();
186 DataObjectMetadata targetMetadata =
187 dataObjectService.getMetadataRepository().getMetadata(targetType);
188 Set<String> modifiedPropertyPaths = decomposedPaths.get(relationshipName);
189 if (isPrimaryKeyModified(targetMetadata, modifiedPropertyPaths)) {
190
191
192 wrapped.fetchRelationship(relationshipName, false, false);
193 } else {
194
195
196
197 wrapped.linkForeignKeys(relationshipName, false);
198 }
199 }
200 }
201 }
202
203
204
205
206
207
208
209
210 protected void linkCollectionChanges(DataObjectWrapper<?> wrapped, Map<String, Set<String>> decomposedPaths,
211 Set<Object> linked) {
212 List<DataObjectCollection> collections = wrapped.getMetadata().getCollections();
213 for (DataObjectCollection collectionMetadata : collections) {
214
215 if (collectionMetadata.isSavedWithParent()) {
216 Set<Integer> modifiedIndicies = extractModifiedIndicies(collectionMetadata, decomposedPaths);
217 if (!modifiedIndicies.isEmpty()) {
218 Object collectionValue = wrapped.getPropertyValue(collectionMetadata.getName());
219 if (collectionValue instanceof Iterable<?>) {
220 int index = 0;
221
222 for (Object element : (Iterable<?>)collectionValue) {
223
224
225 if (modifiedIndicies.contains(Integer.valueOf(Integer.MAX_VALUE)) ||
226 modifiedIndicies.contains(Integer.valueOf(index))) {
227
228 String pathKey = collectionMetadata.getName() + "[" + index + "]";
229 if (decomposedPaths.containsKey(pathKey)) {
230 Map<String, Set<String>> nextPropertyPaths =
231 decomposePropertyPaths(decomposedPaths.get(pathKey));
232 linkChangesInternal(element, nextPropertyPaths, linked);
233 }
234 if (dataObjectService.supports(element.getClass())) {
235 DataObjectWrapper<?> elementWrapper = dataObjectService.wrap(element);
236 linkBiDirectionalCollection(wrapped, elementWrapper, collectionMetadata);
237 }
238 }
239 index++;
240 }
241 }
242 }
243 }
244 }
245 }
246
247
248
249
250
251 protected void linkBiDirectionalCollection(DataObjectWrapper<?> parentWrapper,
252 DataObjectWrapper<?> elementWrapper, DataObjectCollection collectionMetadata) {
253 MetadataChild inverseRelationship = collectionMetadata.getInverseRelationship();
254 if (inverseRelationship != null) {
255
256 elementWrapper.setPropertyValue(inverseRelationship.getName(), parentWrapper.getWrappedInstance());
257
258
259 elementWrapper.linkForeignKeys(inverseRelationship.getName(), false);
260 }
261 }
262
263
264
265
266
267
268 private Set<Integer> extractModifiedIndicies(DataObjectCollection collectionMetadata,
269 Map<String, Set<String>> decomposedPaths) {
270 String relationshipName = collectionMetadata.getName();
271 Set<Integer> modifiedIndicies = Sets.newHashSet();
272
273 if (decomposedPaths.containsKey(relationshipName)) {
274 modifiedIndicies.add(Integer.valueOf(Integer.MAX_VALUE));
275 }
276 for (String propertyName : decomposedPaths.keySet()) {
277 if (relationshipName.equals(PropertyAccessorUtils.getPropertyName(relationshipName))) {
278 Integer index = extractIndex(propertyName);
279 if (index != null) {
280 modifiedIndicies.add(index);
281 }
282 }
283 }
284 return modifiedIndicies;
285 }
286
287 private Integer extractIndex(String propertyName) {
288 int firstIndex = propertyName.indexOf(PropertyAccessor.PROPERTY_KEY_PREFIX_CHAR);
289 int lastIndex = propertyName.lastIndexOf(PropertyAccessor.PROPERTY_KEY_SUFFIX_CHAR);
290 if (firstIndex != -1 && lastIndex != -1) {
291 String indexValue = propertyName.substring(firstIndex + 1, lastIndex);
292 try {
293 int index = Integer.parseInt(indexValue);
294 return Integer.valueOf(index);
295 } catch (NumberFormatException e) {
296
297 }
298 }
299 return null;
300 }
301
302 protected boolean isPrimaryKeyModified(DataObjectMetadata metadata, Set<String> modifiedPropertyPaths) {
303 Set<String> primaryKeyAttributeNames = new HashSet<String>(metadata.getPrimaryKeyAttributeNames());
304 for (String modifiedPropertyPath : modifiedPropertyPaths) {
305 if (primaryKeyAttributeNames.contains(modifiedPropertyPath)) {
306 return true;
307 }
308 }
309 return false;
310 }
311
312 protected void cascadeLinkingAnnotations(DataObjectWrapper<?> wrapped, Map<String, Set<String>> decomposedPaths,
313 Set<Object> linked) {
314 Field[] fields = FieldUtils.getAllFields(wrapped.getWrappedClass());
315 Map<String, Field> modifiedFieldMap = new HashMap<String, Field>();
316 for (Field field : fields) {
317 if (decomposedPaths.containsKey(field.getName())) {
318 modifiedFieldMap.put(field.getName(), field);
319 }
320 }
321 for (String modifiedFieldName : modifiedFieldMap.keySet()) {
322 Field modifiedField = modifiedFieldMap.get(modifiedFieldName);
323 Link link = modifiedField.getAnnotation(Link.class);
324 if (link == null) {
325
326 link = AnnotationUtils.findAnnotation(modifiedField.getType(), Link.class);
327 }
328 if (link != null && link.cascade()) {
329 List<String> linkingPaths = assembleLinkingPaths(link);
330 for (String linkingPath : linkingPaths) {
331 Map<String, Set<String>> decomposedLinkingPath =
332 decomposePropertyPaths(decomposedPaths.get(modifiedFieldName), linkingPath);
333 String valuePath = modifiedFieldName;
334 if (StringUtils.isNotBlank(linkingPath)) {
335 valuePath = valuePath + "." + link.path();
336 }
337 Object linkRootObject = wrapped.getPropertyValueNullSafe(valuePath);
338 linkChangesInternal(linkRootObject, decomposedLinkingPath, linked);
339 }
340 }
341 }
342 }
343
344 protected List<String> assembleLinkingPaths(Link link) {
345 List<String> linkingPaths = new ArrayList<String>();
346 if (ArrayUtils.isEmpty(link.path())) {
347 linkingPaths.add("");
348 } else {
349 for (String path : link.path()) {
350 linkingPaths.add(path);
351 }
352 }
353 return linkingPaths;
354 }
355
356 protected Map<String, Set<String>> decomposePropertyPaths(Set<String> changedPropertyPaths) {
357 return decomposePropertyPaths(changedPropertyPaths, "");
358 }
359
360 protected Map<String, Set<String>> decomposePropertyPaths(Set<String> changedPropertyPaths, String prefix) {
361
362 Set<String> processedProperties = new HashSet<String>();
363 if (StringUtils.isNotBlank(prefix) && changedPropertyPaths != null) {
364 for (String changedPropertyPath : changedPropertyPaths) {
365 if (changedPropertyPath.startsWith(prefix)) {
366 processedProperties.add(changedPropertyPath.substring(prefix.length() + 1));
367 }
368 }
369 } else {
370 processedProperties = changedPropertyPaths;
371 }
372 Map<String, Set<String>> decomposedPropertyPaths = new HashMap<String, Set<String>>();
373 for (String changedPropertyPath : processedProperties) {
374 int index = PropertyAccessorUtils.getFirstNestedPropertySeparatorIndex(changedPropertyPath);
375 if (index == -1) {
376 decomposedPropertyPaths.put(changedPropertyPath, new HashSet<String>());
377 } else {
378 String pathEntry = changedPropertyPath.substring(0, index);
379 Set<String> remainingPaths = decomposedPropertyPaths.get(pathEntry);
380 if (remainingPaths == null) {
381 remainingPaths = new HashSet<String>();
382 decomposedPropertyPaths.put(pathEntry, remainingPaths);
383 }
384 String remainingPath = changedPropertyPath.substring(index + 1);
385 remainingPaths.add(remainingPath);
386 }
387 }
388 return decomposedPropertyPaths;
389 }
390
391 }