// Copyright 2015 ThoughtWorks, Inc.
// This file is part of Gauge-Java.
// This program is free software.
//
// It is dual-licensed under:
// 1) the GNU General Public License as published by the Free Software Foundation,
// either version 3 of the License, or (at your option) any later version;
// or
// 2) the Eclipse Public License v1.0.
//
// You can redistribute it and/or modify it under the terms of either license.
// We would then provide copied of each license in a separate .txt file with the name of the license as the title of the file.
package com.thoughtworks.gauge.refactor;
import com.github.javaparser.ast.CompilationUnit;
import com.github.javaparser.ast.Node;
import com.github.javaparser.ast.body.MethodDeclaration;
import com.github.javaparser.ast.body.Parameter;
import com.github.javaparser.ast.body.VariableDeclaratorId;
import com.github.javaparser.ast.expr.AnnotationExpr;
import com.github.javaparser.ast.expr.BinaryExpr;
import com.github.javaparser.ast.expr.SingleMemberAnnotationExpr;
import com.github.javaparser.ast.expr.StringLiteralExpr;
import com.github.javaparser.ast.type.ClassOrInterfaceType;
import com.github.javaparser.ast.visitor.VoidVisitorAdapter;
import com.thoughtworks.gauge.StepValue;
import gauge.messages.Messages;
import org.apache.commons.lang.StringEscapeUtils;
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;
import java.util.Arrays;
import java.util.List;
public class RefactoringMethodVisitor extends VoidVisitorAdapter {
private StepValue oldStepValue;
private StepValue newStepValue;
private List<Messages.ParameterPosition> paramPositions;
private boolean refactored;
private JavaRefactoringElement javaElement;
public RefactoringMethodVisitor(StepValue oldStepValue, StepValue newStepValue, List<Messages.ParameterPosition> paramPositions) {
this.oldStepValue = oldStepValue;
this.newStepValue = newStepValue;
this.paramPositions = paramPositions;
}
public void visit(MethodDeclaration methodDeclaration, Object arg) {
try {
List<AnnotationExpr> annotations = methodDeclaration.getAnnotations();
if (annotations == null) {
return;
}
for (AnnotationExpr annotationExpr : annotations) {
if (!(annotationExpr instanceof SingleMemberAnnotationExpr)) {
continue;
}
SingleMemberAnnotationExpr annotation = (SingleMemberAnnotationExpr) annotationExpr;
if (annotation.getMemberValue() instanceof BinaryExpr) {
ScriptEngine engine = new ScriptEngineManager().getEngineByName("JavaScript");
try {
Object result = engine.eval(annotation.getMemberValue().toString());
refactor(methodDeclaration, new StringLiteralExpr(result.toString()), annotation);
} catch (ScriptException e) {
continue;
}
}
if (annotation.getMemberValue() instanceof StringLiteralExpr) {
StringLiteralExpr memberValue = (StringLiteralExpr) annotation.getMemberValue();
refactor(methodDeclaration, memberValue, annotation);
}
}
} catch (Exception ignored) {
}
}
private void refactor(MethodDeclaration methodDeclaration, StringLiteralExpr memberValue, SingleMemberAnnotationExpr annotation) {
if (StringEscapeUtils.unescapeJava(memberValue.getValue()).trim().equals(oldStepValue.getStepAnnotationText().trim())) {
List<Parameter> newParameters = Arrays.asList(new Parameter[paramPositions.size()]);
memberValue.setValue(StringEscapeUtils.escapeJava(newStepValue.getStepAnnotationText()));
List<Parameter> parameters = methodDeclaration.getParameters();
for (int i = 0, paramPositionsSize = paramPositions.size(); i < paramPositionsSize; i++) {
if (paramPositions.get(i).getOldPosition() < 0) {
String paramName = Util.getValidJavaIdentifier(Util.convertToCamelCase("arg " + newStepValue.getParameters().get(i)));
if (paramName.equals("arg")) {
paramName += i;
}
newParameters.set(paramPositions.get(i).getNewPosition(), new Parameter(new ClassOrInterfaceType("String"), new VariableDeclaratorId(paramName)));
} else {
newParameters.set(paramPositions.get(i).getNewPosition(), parameters.get(paramPositions.get(i).getOldPosition()));
}
}
methodDeclaration.setParameters(newParameters);
annotation.setMemberValue(memberValue);
this.javaElement = new JavaRefactoringElement(getJavaFileText(methodDeclaration), null);
this.refactored = true;
}
}
private String getJavaFileText(MethodDeclaration methodDeclaration) {
return getFileElement(methodDeclaration).toString();
}
private Node getFileElement(Node node) {
if (node instanceof CompilationUnit) {
return node;
}
return getFileElement(node.getParentNode());
}
public boolean refactored() {
return this.refactored;
}
public JavaRefactoringElement getRefactoredJavaElement() {
return this.javaElement;
}
}