Skip to contentMethod: AnalyzeInheritanceTreesTask.InheritancePath(AnalyzeInheritanceTreesTask)
      1: package de.fhdw.wtf.walker.tasks;
2: 
3: import java.util.HashMap;
4: import java.util.Iterator;
5: import java.util.LinkedList;
6: import java.util.List;
7: import java.util.Map;
8: import java.util.Map.Entry;
9: 
10: import de.fhdw.wtf.common.ast.Attribute;
11: import de.fhdw.wtf.common.ast.ConstructorOrOperation;
12: import de.fhdw.wtf.common.ast.Group;
13: import de.fhdw.wtf.common.ast.Model;
14: import de.fhdw.wtf.common.ast.type.AtomicType;
15: import de.fhdw.wtf.common.ast.type.ClassType;
16: import de.fhdw.wtf.common.ast.type.CompositeType;
17: import de.fhdw.wtf.common.ast.type.ListType;
18: import de.fhdw.wtf.common.ast.type.MapType;
19: import de.fhdw.wtf.common.ast.type.ProductType;
20: import de.fhdw.wtf.common.ast.type.SumType;
21: import de.fhdw.wtf.common.ast.type.ThrownType;
22: import de.fhdw.wtf.common.ast.type.Type;
23: import de.fhdw.wtf.common.ast.type.TypeProxy;
24: import de.fhdw.wtf.common.ast.visitor.CompositeTypeVisitorReturn;
25: import de.fhdw.wtf.common.ast.visitor.TypeVisitorReturn;
26: import de.fhdw.wtf.common.exception.walker.TaskException;
27: import de.fhdw.wtf.common.task.TaskExecutor;
28: import de.fhdw.wtf.walker.walker.SimpleWalkerTask;
29: 
30: /**
31:  * Analyzes the inheritance trees of the given model. Enriches the model with useful information regarding necessary
32:  * constructor calls.
33:  */
34: public final class AnalyzeInheritanceTreesTask extends SimpleWalkerTask {
35:         
36:         /**
37:          * Used to identify the type Anything.
38:          */
39:         private static final String ANYTHING = "Anything";
40:         
41:         /**
42:          * Returns a new instance of {@link AnalyzeInheritanceTreesTask}.
43:          * 
44:          * @param model
45:          *            The model to perform this task on.
46:          * @param taskmanager
47:          *            The executor which performs this task.
48:          * @return a new instance of {@link AnalyzeInheritanceTreesTask}
49:          */
50:         public static AnalyzeInheritanceTreesTask create(final Model model, final TaskExecutor taskmanager) {
51:                 return new AnalyzeInheritanceTreesTask(model, taskmanager);
52:         }
53:         
54:         /**
55:          * Instantiates a new instance of {@link AnalyzeInheritanceTreesTask}.
56:          * 
57:          * @param model
58:          *            The model to perform this task on.
59:          * @param taskmanager
60:          *            The executor which performs this task.
61:          */
62:         private AnalyzeInheritanceTreesTask(final Model model, final TaskExecutor taskmanager) {
63:                 super(model, taskmanager);
64:         }
65:         
66:         /**
67:          * Helper class which represents one path along the inheritance-hierarchy of a root class.
68:          */
69:         private class InheritancePath {
70:                 /**
71:                  * Types ordered along this InheritancePath.
72:                  */
73:                 private final List<Type> pathElements;
74:                 
75:                 /**
76:                  * Constructs an empty InheritancePath.
77:                  */
78:                 private InheritancePath() {
79:                         this.pathElements = new LinkedList<>();
80:                 }
81:                 
82:                 /**
83:                  * Constructs an InheritancePath with first as its root.
84:                  * 
85:                  * @param first
86:                  *            the first element of this path.
87:                  */
88:                 InheritancePath(final Type first) {
89:                         this();
90:                         this.addElement(first);
91:                 }
92:                 
93:                 /**
94:                  * Constructs an InheritancePath with firsts as its path elements.
95:                  * 
96:                  * @param firsts
97:                  *            the path elements for this.
98:                  */
99:                 InheritancePath(final List<Type> firsts) {
100:                         this();
101:                         this.addElements(firsts);
102:                 }
103:                 
104:                 /**
105:                  * Constructs an InheritancePath which is identical to firsts.
106:                  * 
107:                  * @param firsts
108:                  *            the path to copy.
109:                  */
110:                 InheritancePath(final InheritancePath firsts) {
111:                         this(firsts.pathElements);
112:                 }
113:                 
114:                 /**
115:                  * Adds element to the end of this path.
116:                  * 
117:                  * @param element
118:                  *            the new end of this path.
119:                  */
120:                 void addElement(final Type element) {
121:                         this.pathElements.add(element);
122:                 }
123:                 
124:                 /**
125:                  * Adds elements to this path so that these elements remain in the order provided in elements.
126:                  * 
127:                  * @param elements
128:                  *            the elements added to this.
129:                  */
130:                 void addElements(final List<Type> elements) {
131:                         this.pathElements.addAll(elements);
132:                 }
133:                 
134:                 /**
135:                  * Returns true if element is also an element of this. False otherwise.
136:                  * 
137:                  * @param element
138:                  *            to check.
139:                  * @return true if element is part of this. False otherwise.
140:                  */
141:                 boolean contains(final Type element) {
142:                         return this.pathElements.contains(element);
143:                 }
144:                 
145:                 /**
146:                  * Returns all elements that this and other have in common. The common elements will be provided the order
147:                  * provided by this.
148:                  * 
149:                  * @param other
150:                  *            the path to intersect with.
151:                  * @return common elements in the order of their existence in this path.
152:                  */
153:                 List<Type> intersect(final InheritancePath other) {
154:                         final List<Type> result = new LinkedList<>();
155:                         final Iterator<Type> myElements = this.pathElements.iterator();
156:                         while (myElements.hasNext()) {
157:                                 final Type myElement = myElements.next();
158:                                 if (other.contains(myElement)) {
159:                                         result.add(myElement);
160:                                 }
161:                         }
162:                         return result;
163:                 }
164:                 
165:                 /**
166:                  * Makes one step in the inheritance-tree. Alternative paths will be returned.
167:                  * 
168:                  * @return alternative paths if any exist. An empty llist otherwise.
169:                  */
170:                 List<InheritancePath> step() {
171:                         final List<InheritancePath> alternativePaths = new LinkedList<>();
172:                         final Type lastElement = this.pathElements.get(this.pathElements.size() - 1);
173:                         final Iterator<Type> followers = lastElement.getSuperTypes().iterator();
174:                         if (followers.hasNext()) {
175:                                 final Type firstFollower = followers.next();
176:                                 while (followers.hasNext()) {
177:                                         final Type follower = followers.next();
178:                                         if (!follower.toString().equals(ANYTHING)) {
179:                                                 final InheritancePath alternativePath = new InheritancePath(this);
180:                                                 alternativePath.addElement(follower);
181:                                                 alternativePaths.add(alternativePath);
182:                                         }
183:                                 }
184:                                 if (!firstFollower.toString().equals(ANYTHING)) {
185:                                         this.addElement(firstFollower);
186:                                 }
187:                         }
188:                         return alternativePaths;
189:                 }
190:                 
191:                 /**
192:                  * Calculates all inheritance paths reachable from the end of this path and returns them.
193:                  * 
194:                  * @return a list with all paths reachable from the last element of this.
195:                  */
196:                 List<InheritancePath> run() {
197:                         final List<InheritancePath> result = new LinkedList<>();
198:                         final List<InheritancePath> preResult = new LinkedList<>();
199:                         int oldSize = this.pathElements.size();
200:                         List<InheritancePath> alternativePaths = this.step();
201:                         while (this.pathElements.size() != oldSize || !alternativePaths.isEmpty()) {
202:                                 preResult.addAll(alternativePaths);
203:                                 oldSize = this.pathElements.size();
204:                                 alternativePaths = this.step();
205:                         }
206:                         final Iterator<InheritancePath> paths = preResult.iterator();
207:                         while (paths.hasNext()) {
208:                                 result.addAll(paths.next().run());
209:                         }
210:                         result.add(this);
211:                         return result;
212:                 }
213:                 
214:                 @Override
215:                 public String toString() {
216:                         return this.pathElements.toString();
217:                 }
218:                 
219:                 @Override
220:                 public boolean equals(final Object o) {
221:                         if (o instanceof InheritancePath) {
222:                                 final InheritancePath oAsInheritancePath = (InheritancePath) o;
223:                                 return this.pathElements.equals(oAsInheritancePath.pathElements);
224:                         }
225:                         return false;
226:                 }
227:                 
228:                 @Override
229:                 public int hashCode() {
230:                         return this.pathElements.hashCode();
231:                 }
232:                 
233:         }
234:         
235:         @Override
236:         public void handleClass(final ClassType c) throws TaskException {
237:                 final List<InheritancePath> allPaths = new LinkedList<>();
238:                 final Iterator<Type> superTypes = c.getSuperTypes().iterator();
239:                 while (superTypes.hasNext()) {
240:                         final Type current = superTypes.next();
241:                         if (!current.toString().equals(ANYTHING)) {
242:                                 allPaths.addAll(new InheritancePath(current).run());
243:                         }
244:                 }
245:                 this.mapCalls(c, allPaths);
246:         }
247:         
248:         /**
249:          * Maps c to all types it has to instantiate during runtime. Uses the map "calls" for this purpose. Maps to an empty
250:          * list if c has no responsibilities in the regard of instantiating super types.
251:          * 
252:          * @param c
253:          *            the root of the inheritance tree to map types from.
254:          * @param allPaths
255:          *            The inheritance tree with c as root in form of all paths that have the direct super types of c as
256:          *            root.
257:          */
258:         private void mapCalls(final ClassType c, final List<InheritancePath> allPaths) {
259:                 final Map<Type, Integer> callPriorities = new HashMap<>();
260:                 final List<Type> definitiveResponsibilities = new LinkedList<>();
261:                 final List<InheritancePath> copy = new LinkedList<>();
262:                 copy.addAll(allPaths);
263:                 final Iterator<InheritancePath> copyPaths = copy.iterator();
264:                 while (copyPaths.hasNext()) {
265:                         final InheritancePath copyPath = copyPaths.next();
266:                         final Iterator<InheritancePath> paths = allPaths.iterator();
267:                         while (paths.hasNext()) {
268:                                 final InheritancePath path = paths.next();
269:                                 final List<Type> interception = path.intersect(copyPath);
270:                                 this.gatherResponsibilities(interception, callPriorities, definitiveResponsibilities);
271:                                 
272:                         }
273:                         copyPaths.remove();
274:                 }
275:                 this.addToCalls(c, definitiveResponsibilities, callPriorities);
276:         }
277:         
278:         /**
279:          * Adds all types in definitiveResponsibilities and direct super types of c as list values to the calls map and uses
280:          * c as key. Additionally ensures that the order of the elements in the list will be a valid constructor-call-order
281:          * when instantiating c.
282:          * 
283:          * @param c
284:          *            the key to use.
285:          * @param definitiveResponsibilities
286:          *            those classes that c is definitely responsible to call. Every type in this list must also be a key in
287:          *            callPriorities with a valid value != null;
288:          * @param callPriorities
289:          *            the priorities which are used to determine the call order of the super constructors for c. Will be
290:          *            empty after a call to this method!
291:          */
292:         private void addToCalls(final ClassType c,
293:                         final List<Type> definitiveResponsibilities,
294:                         final Map<Type, Integer> callPriorities) {
295:                 final List<Type> callList = new LinkedList<>();
296:                 if (!definitiveResponsibilities.isEmpty()) {
297:                         while (!callPriorities.isEmpty()) {
298:                                 final Type max = this.findAndRemoveHighestPathCountType(callPriorities);
299:                                 if (definitiveResponsibilities.contains(max)) {
300:                                         callList.add(max);
301:                                 }
302:                         }
303:                 }
304:                 final Iterator<Type> directSuperTypes = c.getSuperTypes().iterator();
305:                 while (directSuperTypes.hasNext()) {
306:                         final Type superType = directSuperTypes.next();
307:                         if (!superType.toString().equals(ANYTHING) && !definitiveResponsibilities.contains(superType)) {
308:                                 callList.add(superType);
309:                         }
310:                 }
311:                 this.getModel().putConstructorCallDependency(c, callList);
312:         }
313:         
314:         /**
315:          * Returns true if c is part of the given type when type is a sum type. Returns false otherwise.
316:          * 
317:          * @param c
318:          *            a ClassType.
319:          * @param type
320:          *            some type.
321:          * @return true if c is part of the given type when type is a sum type. Returns false otherwise.
322:          */
323:         private boolean isSumTypeAndContainedInSumType(final ClassType c, final Type type) {
324:                 return type.accept(new TypeVisitorReturn<Boolean>() {
325:                         
326:                         @Override
327:                         public Boolean handle(final AtomicType atomicType) {
328:                                 return false;
329:                         }
330:                         
331:                         @Override
332:                         public Boolean handle(final CompositeType compositeType) {
333:                                 return compositeType.accept(new CompositeTypeVisitorReturn<Boolean>() {
334:                                         
335:                                         @Override
336:                                         public Boolean handle(final ListType list) {
337:                                                 return false;
338:                                         }
339:                                         
340:                                         @Override
341:                                         public Boolean handle(final MapType map) {
342:                                                 return false;
343:                                         }
344:                                         
345:                                         @Override
346:                                         public Boolean handle(final ProductType product) {
347:                                                 return false;
348:                                         }
349:                                         
350:                                         @Override
351:                                         public Boolean handle(final SumType sum) {
352:                                                 return sum.getElements().contains(c);
353:                                         }
354:                                         
355:                                         @Override
356:                                         public Boolean handle(final ThrownType thrownType) {
357:                                                 return false;
358:                                         }
359:                                         
360:                                 });
361:                         }
362:                         
363:                         @Override
364:                         public Boolean handle(final TypeProxy typeProxy) {
365:                                 return AnalyzeInheritanceTreesTask.this.isSumTypeAndContainedInSumType(c, typeProxy.getTarget());
366:                         }
367:                         
368:                 });
369:         }
370:         
371:         /**
372:          * Removes the key with the highest integer value as value from callPriorities and returns that key. removes and
373:          * returns the first key found if more than one value is at the current maximum. Returns null if callPriorities is
374:          * empty.
375:          * 
376:          * @param callPriorities
377:          *            the map to remove a key from.
378:          * @return null if callPriorities is empty. The first key with the highest integer as value found in callPriorities.
379:          */
380:         private Type findAndRemoveHighestPathCountType(final Map<Type, Integer> callPriorities) {
381:                 final Iterator<Entry<Type, Integer>> entries = callPriorities.entrySet().iterator();
382:                 int highestPathCount = 0;
383:                 Type highestPathCountType = null;
384:                 while (entries.hasNext()) {
385:                         final Entry<Type, Integer> entry = entries.next();
386:                         final int pathCount = entry.getValue();
387:                         if (pathCount > highestPathCount) {
388:                                 highestPathCount = pathCount;
389:                                 highestPathCountType = entry.getKey();
390:                         }
391:                 }
392:                 callPriorities.remove(highestPathCountType);
393:                 return highestPathCountType;
394:         }
395:         
396:         /**
397:          * Adds all types in intersection to callPriorities and increments the corresponding value for a key by one. Also
398:          * adds the first type in intersection to definitiveResponsibilities (no duplicates will be added).
399:          * 
400:          * @param intersection
401:          *            an intersection of two InheritancePaths.
402:          * @param callPriorities
403:          *            a map with call priorities.
404:          * @param definitiveResponsibilities
405:          *            a list with types that have to be instantiated by a certain sub type.
406:          */
407:         private void gatherResponsibilities(final List<Type> intersection,
408:                         final Map<Type, Integer> callPriorities,
409:                         final List<Type> definitiveResponsibilities) {
410:                 final Iterator<Type> commonTypes = intersection.iterator();
411:                 if (commonTypes.hasNext()) {
412:                         final Type definitiveOne = commonTypes.next();
413:                         this.addToPriorities(definitiveOne, callPriorities);
414:                         if (!definitiveResponsibilities.contains(definitiveOne)) {
415:                                 definitiveResponsibilities.add(definitiveOne);
416:                         }
417:                 }
418:                 while (commonTypes.hasNext()) {
419:                         this.addToPriorities(commonTypes.next(), callPriorities);
420:                 }
421:         }
422:         
423:         /**
424:          * Adds the pair (type,2) to callPriorities if type is not a key of callPriorities. Increments the the value of type
425:          * by one if type is a key of callPriorities. The initial value is 2 because it represents the amount of paths a
426:          * type has been found in.
427:          * 
428:          * @param type
429:          *            the key to add/ to find.
430:          * @param callPriorities
431:          *            the map too search in.
432:          */
433:         private void addToPriorities(final Type type, final Map<Type, Integer> callPriorities) {
434:                 if (!callPriorities.containsKey(type)) {
435:                         callPriorities.put(type, 1);
436:                 }
437:                 callPriorities.put(type, callPriorities.get(type) + 1);
438:         }
439:         
440:         @Override
441:         public void handleGroup(final Group g) throws TaskException {
442:                 // Nothing to do.
443:         }
444:         
445:         @Override
446:         public void handleAttribute(final Attribute a, final ClassType owner) throws TaskException {
447:                 // Nothing to do.
448:         }
449:         
450:         @Override
451:         public void handleConstructorOrOperation(final ConstructorOrOperation coo, final ClassType owner)
452:                         throws TaskException {
453:                 // Nothing to do.
454:         }
455:         
456:         @Override
457:         public void finalizeTask() throws TaskException {
458:                 // Nothing to do.
459:         }
460:         
461:         @Override
462:         public void beginTask() throws TaskException {
463:                 // Nothing to do.
464:         }
465:         
466: }