All Downloads are FREE. Search and download functionalities are using the official Maven repository.

eu.stamp.project.assertfixer.asserts.AssertFixer Maven / Gradle / Ivy

There is a newer version: 1.0.10
Show newest version
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