eu.stamp.project.assertfixer.asserts.AssertFixer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of assert-fixer Show documentation
Show all versions of assert-fixer Show documentation
This tool aims at repairing the assertion inside JUnit tests.
package eu.stamp.project.assertfixer.asserts;
import com.github.difflib.DiffUtils;
import com.github.difflib.UnifiedDiffUtils;
import com.github.difflib.algorithm.DiffException;
import com.github.difflib.patch.Patch;
import eu.stamp.project.assertfixer.AssertFixerResult;
import eu.stamp.project.assertfixer.Configuration;
import eu.stamp.project.assertfixer.asserts.log.Logger;
import eu.stamp.project.assertfixer.test.TestRunner;
import eu.stamp.project.assertfixer.util.Counter;
import eu.stamp.project.assertfixer.util.Util;
import eu.stamp_project.testrunner.EntryPoint;
import eu.stamp_project.testrunner.listener.TestResult;
import eu.stamp_project.testrunner.runner.Failure;
import org.apache.commons.io.FileUtils;
import org.apache.commons.lang3.StringUtils;
import spoon.Launcher;
import spoon.OutputType;
import spoon.SpoonModelBuilder;
import spoon.reflect.code.CtBlock;
import spoon.reflect.code.CtCatch;
import spoon.reflect.code.CtInvocation;
import spoon.reflect.code.CtTry;
import spoon.reflect.declaration.CtClass;
import spoon.reflect.declaration.CtMethod;
import spoon.reflect.declaration.CtPackage;
import spoon.reflect.visitor.filter.TypeFilter;
import java.io.File;
import java.net.MalformedURLException;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.TimeoutException;
/**
* Created by Benjamin DANGLOT
* [email protected]
* on 23/05/17
*/
public class AssertFixer {
public static AssertFixerResult fixAssert(Configuration configuration, Launcher spoon, CtClass originalClass, String testCaseName, Failure failure, String cp) {
final CtClass> classTestToBeFixed = originalClass.clone();
final String originalClassStr = originalClass.toString();
final String filePath = originalClass.getPosition().getFile().getPath();
final String basePath = new File(".").getPath();
String relativeFilePath = new File(basePath).toURI().relativize(new File(filePath).toURI()).getPath();
final CtPackage parentPackage = originalClass.getPackage();
// switch the original class and the clone from the model
parentPackage.removeType(originalClass);
parentPackage.addType(classTestToBeFixed);
String testClassName = originalClass.getQualifiedName();
AssertFixerResult result = new AssertFixerResult(testClassName, testCaseName);
result.setFilePath(originalClass.getPosition().getFile().getPath());
AssertFixerResult.RepairType repairType = AssertFixerResult.RepairType.NoRepair;
CtMethod> testCaseToBeFix = classTestToBeFixed.getMethodsByName(testCaseName).get(0);
Counter.addNumberOfAssertionInTests(testCaseToBeFix.getElements(new TypeFilter(CtInvocation.class) {
@Override
public boolean matches(CtInvocation element) {
return Util.isAssert.test(element);
}
}).size());
if (Util.assertionErrors.contains(failure.fullQualifiedNameOfException)) {
Counter.incNumberOfFailingAssertion();
Counter.incNumberOfFailingTestFromAssertion();
CtMethod> clone = testCaseToBeFix.clone();
classTestToBeFixed.removeMethod(testCaseToBeFix);
classTestToBeFixed.addMethod(clone);
if ((failure.messageOfFailure != null &&
failure.messageOfFailure.startsWith(TryCatchFixer.PREFIX_MESSAGE_EXPECTED_EXCEPTION)
&& failure.messageOfFailure.endsWith("Exception"))) {
removeExpectedException(configuration, spoon, testClassName, testCaseName, cp, clone);//TODO this remove the fail failure but there is no more oracle
repairType = AssertFixerResult.RepairType.RemoveException;
} else if (failure.messageOfFailure != null && !failure.messageOfFailure.contains("expected")) {
String message = "AssertFixer cannot fix this assertion. Message of failure: " + failure.messageOfFailure;
result.setSuccess(false);
result.setExceptionMessage(message);
} else {
// replace assertion
final List indexToLog = AssertionReplacer.replaceByLogStatement(clone);
// run tests
TestRunner.runTestWithLogger(configuration, spoon, cp, testClassName, testCaseName);
Logger.load();
// replace wrong value
classTestToBeFixed.removeMethod(clone);
classTestToBeFixed.addMethod(testCaseToBeFix);
AssertionsFixer.fixAssertion(spoon.getFactory(), testCaseToBeFix, indexToLog);
repairType = AssertFixerResult.RepairType.AssertRepair;
}
} else {
Counter.incNumberOfFailingTestFromException();
final List catches = testCaseToBeFix.getElements(new TypeFilter<>(CtCatch.class));
if (!catches.isEmpty()) {
TryCatchFixer.fixTryCatchFailAssertion(spoon, testCaseToBeFix, failure, catches);
} else {
if (configuration.isGenTryCatch()) {
TryCatchFixer.addTryCatchFailAssertion(spoon, testCaseToBeFix, failure);
} else {
result.setSuccess(false);
result.setExceptionMessage("addTryCatchFailAssertion skipped");
}
}
repairType = AssertFixerResult.RepairType.TryCatchRepair;
}
// we generate the source on disk
SpoonModelBuilder compiler = spoon.createCompiler();
compiler.generateProcessedSourceFiles(OutputType.CLASSES);
// we compile the sources in a fresh spoon
Launcher spoon2 = new Launcher();
spoon2.addInputResource(configuration.getSourceOutputDirectory());
spoon2.getEnvironment().setSourceClasspath(configuration.getClasspath().split(Util.PATH_SEPARATOR));
spoon2.setBinaryOutputDirectory(new File(configuration.getBinaryOutputDirectory()));
spoon2.getEnvironment().setShouldCompile(true);
spoon2.run();
boolean success = false;
TestResult testResult;
try {
testResult = EntryPoint.runTests(
configuration.getBinaryOutputDirectory() +
Util.PATH_SEPARATOR + configuration.getClasspath(),
testClassName,
testCaseName
);
success = testResult.getFailingTests().isEmpty();
} catch (TimeoutException e) {
throw new RuntimeException(e);
}
result.setSuccess(success);
if (success) {
String diff = computeDiff(originalClassStr, classTestToBeFixed.toString(), relativeFilePath);
result.setDiff(diff);
result.setRepairType(repairType);
} else {
System.err.println(testResult.getFailingTests());
}
// switch back the clone and the original class in the model for other changes
parentPackage.removeType(classTestToBeFixed);
parentPackage.addType(originalClass);
return result;
}
private static String computeDiff(String oldClassStr, String newClassStr, String filePath) {
List oldClassLines = Arrays.asList(oldClassStr.split("\\n"));
List newClassLines = Arrays.asList(newClassStr.split("\\n"));
try {
Patch patch = DiffUtils.diff(oldClassLines, newClassLines);
List strings = UnifiedDiffUtils.generateUnifiedDiff(filePath, filePath, oldClassLines, patch, 0);
return StringUtils.join(strings, "\n");
} catch (DiffException e) {
e.printStackTrace();
}
return "";
}
private static void removeExpectedException(Configuration configuration, Launcher spoon, String fullQualifiedName, String testCaseName, String cp, CtMethod> clone) {
final CtTry ctTry = clone.getElements(new TypeFilter<>(CtTry.class)).get(0);
ctTry.replace(ctTry.getBody());
final CtInvocation failToRemove = clone.getElements(new TypeFilter<>(CtInvocation.class)).stream()
.filter(Util.isFail)
.findFirst()
.orElseThrow(RuntimeException::new);
final boolean remove = failToRemove.getParent(CtBlock.class).getStatements().remove(failToRemove);
if (!remove) {
throw new RuntimeException();
}
TestRunner.runTest(configuration, spoon, fullQualifiedName, testCaseName);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy