gumtree.spoon.diff.DiffImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gumtree-spoon-ast-diff Show documentation
Show all versions of gumtree-spoon-ast-diff Show documentation
Computes the AST difference between two Spoon abstract syntax trees using the Gumtree algorithm.
package gumtree.spoon.diff;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.Optional;
import java.util.stream.Collectors;
import com.github.gumtreediff.actions.ActionGenerator;
import com.github.gumtreediff.actions.model.Action;
import com.github.gumtreediff.actions.model.Delete;
import com.github.gumtreediff.actions.model.Insert;
import com.github.gumtreediff.actions.model.Move;
import com.github.gumtreediff.actions.model.Update;
import com.github.gumtreediff.matchers.CompositeMatchers;
import com.github.gumtreediff.matchers.MappingStore;
import com.github.gumtreediff.matchers.Matcher;
import com.github.gumtreediff.tree.ITree;
import com.github.gumtreediff.tree.TreeContext;
import gumtree.spoon.builder.SpoonGumTreeBuilder;
import gumtree.spoon.diff.operations.DeleteOperation;
import gumtree.spoon.diff.operations.InsertOperation;
import gumtree.spoon.diff.operations.MoveOperation;
import gumtree.spoon.diff.operations.Operation;
import gumtree.spoon.diff.operations.OperationKind;
import gumtree.spoon.diff.operations.UpdateOperation;
import spoon.reflect.declaration.CtElement;
/**
* @author Matias Martinez, [email protected]
*/
public class DiffImpl implements Diff {
/**
* Actions over all tree nodes (CtElements)
*/
private final List allOperations;
/**
* Actions over the changes roots.
*/
private final List rootOperations;
/**
* the mapping of this diff
*/
private final MappingStore _mappingsComp;
/**
* Context of the spoon diff.
*/
private final TreeContext context;
public DiffImpl(TreeContext context, ITree rootSpoonLeft, ITree rootSpoonRight) {
final MappingStore mappingsComp = new MappingStore();
final Matcher matcher = new CompositeMatchers.ClassicGumtree(rootSpoonLeft, rootSpoonRight, mappingsComp);
matcher.match();
final ActionGenerator actionGenerator = new ActionGenerator(rootSpoonLeft, rootSpoonRight,
matcher.getMappings());
actionGenerator.generate();
ActionClassifier actionClassifier = new ActionClassifier(matcher.getMappingsAsSet(),
actionGenerator.getActions());
// Bugfix: the Action classifier must be executed *BEFORE* the convertToSpoon
// because it writes meta-data on the trees
this.rootOperations = convertToSpoon(actionClassifier.getRootActions());
this.allOperations = convertToSpoon(actionGenerator.getActions());
this._mappingsComp = mappingsComp;
this.context = context;
for (int i = 0; i < this.getAllOperations().size(); i++) {
Operation operation = this.getAllOperations().get(i);
if (operation instanceof MoveOperation) {
operation.getSrcNode().putMetadata("isMoved", true);
operation.getDstNode().putMetadata("isMoved", true);
}
}
}
private List convertToSpoon(List actions) {
return actions.stream().map(action -> {
if (action instanceof Insert) {
return new InsertOperation((Insert) action);
} else if (action instanceof Delete) {
return new DeleteOperation((Delete) action);
} else if (action instanceof Update) {
return new UpdateOperation((Update) action);
} else if (action instanceof Move) {
return new MoveOperation((Move) action);
} else {
throw new IllegalArgumentException("Please support the new type " + action.getClass());
}
}).collect(Collectors.toList());
}
@Override
public List getAllOperations() {
return Collections.unmodifiableList(allOperations);
}
@Override
public List getRootOperations() {
return Collections.unmodifiableList(rootOperations);
}
@Override
public List getOperationChildren(Operation operationParent, List rootOperations) {
return rootOperations.stream() //
.filter(operation -> operation.getNode().getParent().equals(operationParent)) //
.collect(Collectors.toList());
}
@Override
public CtElement changedNode() {
if (rootOperations.size() != 1) {
throw new IllegalArgumentException("Should have only one root action.");
}
return commonAncestor();
}
@Override
public CtElement commonAncestor() {
final List copy = new ArrayList<>();
for (Operation operation : rootOperations) {
CtElement el = operation.getNode();
if (operation instanceof InsertOperation) {
// we take the corresponding node in the source tree
el = (CtElement) _mappingsComp.getSrc(operation.getAction().getNode().getParent())
.getMetadata(SpoonGumTreeBuilder.SPOON_OBJECT);
}
copy.add(el);
}
while (copy.size() >= 2) {
CtElement first = copy.remove(0);
CtElement second = copy.remove(0);
copy.add(commonAncestor(first, second));
}
return copy.get(0);
}
private CtElement commonAncestor(CtElement first, CtElement second) {
while (first != null) {
CtElement el = second;
while (el != null) {
if (first == el) {
return first;
}
el = el.getParent();
}
first = first.getParent();
}
return null;
}
@Override
public CtElement changedNode(Class extends Operation> operationWanted) {
final Optional firstNode = rootOperations.stream() //
.filter(operation -> operationWanted.isAssignableFrom(operation.getClass())) //
.findFirst();
if (firstNode.isPresent()) {
return firstNode.get().getNode();
}
throw new NoSuchElementException();
}
@Override
public boolean containsOperation(OperationKind kind, String nodeKind) {
return rootOperations.stream() //
.anyMatch(operation -> operation.getAction().getClass().getSimpleName().equals(kind.name()) //
&& context.getTypeLabel(operation.getAction().getNode()).equals(nodeKind));
}
@Override
public boolean containsOperation(OperationKind kind, String nodeKind, String nodeLabel) {
return containsOperations(getRootOperations(), kind, nodeKind, nodeLabel);
}
@Override
public boolean containsOperations(List operations, OperationKind kind, String nodeKind,
String nodeLabel) {
return operations.stream()
.anyMatch(operation -> operation.getAction().getClass().getSimpleName().equals(kind.name()) //
&& context.getTypeLabel(operation.getAction().getNode()).equals(nodeKind)
&& operation.getAction().getNode().getLabel().equals(nodeLabel));
}
@Override
public void debugInformation() {
System.err.println(toDebugString());
}
private String toDebugString() {
String result = "";
for (Operation operation : rootOperations) {
ITree node = operation.getAction().getNode();
final CtElement nodeElement = operation.getNode();
String label = "\"" + node.getLabel() + "\"";
if (operation instanceof UpdateOperation) {
label += " to \"" + ((Update) operation.getAction()).getValue() + "\"";
}
String nodeType = context.getTypeLabel(node.getType());
if (nodeElement != null) {
nodeType += "(" + nodeElement.getClass().getSimpleName() + ")";
}
result += "\"" + operation.getAction().getClass().getSimpleName() + "\", \"" + nodeType + "\", " + label
+ " (size: " + node.getDescendants().size() + ")" + node.toTreeString();
}
return result;
}
@Override
public String toString() {
if (rootOperations.size() == 0) {
return "no AST change";
}
final StringBuilder stringBuilder = new StringBuilder();
final CtElement ctElement = commonAncestor();
for (Operation operation : rootOperations) {
stringBuilder.append(operation.toString());
// if all actions are applied on the same node print only the first action
if (operation.getSrcNode().equals(ctElement) && operation instanceof UpdateOperation) {
break;
}
}
return stringBuilder.toString();
}
public TreeContext getContext() {
return context;
}
public MappingStore getMappingsComp() {
return _mappingsComp;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy