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

mockit.integration.testng.TestNGRunnerDecorator Maven / Gradle / Ivy

/*
 * Copyright (c) 2006 JMockit developers
 * This file is subject to the terms of the MIT license (see LICENSE.txt).
 */
package mockit.integration.testng;

import static mockit.internal.util.StackTrace.filterStackTrace;

import edu.umd.cs.findbugs.annotations.NonNull;
import edu.umd.cs.findbugs.annotations.Nullable;

import java.lang.reflect.Method;

import mockit.Expectations;
import mockit.coverage.testRedundancy.TestCoverage;
import mockit.integration.TestRunnerDecorator;
import mockit.internal.state.SavePoint;
import mockit.internal.state.TestRun;

import org.testng.IExecutionListener;
import org.testng.IInvokedMethod;
import org.testng.IInvokedMethodListener;
import org.testng.ITestNGMethod;
import org.testng.ITestResult;
import org.testng.TestException;
import org.testng.annotations.Test;

/**
 * Provides callbacks to be called by the TestNG 6.2+ test runner for each test execution. JMockit will then assert any
 * expectations recorded in {@link Expectations} subclasses during the test.
 * 

* This class is not supposed to be accessed from user code; it will be automatically loaded at startup. */ public final class TestNGRunnerDecorator extends TestRunnerDecorator implements IInvokedMethodListener, IExecutionListener { @NonNull private final ThreadLocal savePoint = new ThreadLocal<>(); @Override public void beforeInvocation(@NonNull IInvokedMethod invokedMethod, @NonNull ITestResult testResult) { ITestNGMethod testNGMethod = testResult.getMethod(); Class testClass = testResult.getTestClass().getRealClass(); TestRun.clearNoMockingZone(); if (!invokedMethod.isTestMethod()) { beforeConfigurationMethod(testNGMethod, testClass); return; } Method method = testNGMethod.getConstructorOrMethod().getMethod(); exportCurrentTestMethodIfApplicable(method); Object testInstance = testResult.getInstance(); if (testInstance == null || testInstance.getClass() != testClass) { // Happens when TestNG is running a JUnit test class, for which "TestResult#getInstance()" erroneously // returns a // org.junit.runner.Description object. return; } TestRun.enterNoMockingZone(); try { updateTestClassState(testInstance, testClass); TestRun.setRunningIndividualTest(testInstance); SavePoint testMethodSavePoint = new SavePoint(); savePoint.set(testMethodSavePoint); if (shouldPrepareForNextTest) { TestRun.prepareForNextTest(); shouldPrepareForNextTest = false; clearTestedObjectsCreatedDuringSetup(); } createInstancesForTestedFieldsFromBaseClasses(testInstance); createInstancesForTestedFields(testInstance); } finally { TestRun.exitNoMockingZone(); } } private static void exportCurrentTestMethodIfApplicable(@Nullable Method testMethod) { TestCoverage testCoverage = TestCoverage.INSTANCE; if (testCoverage != null) { testCoverage.setCurrentTestMethod(testMethod); } } private void beforeConfigurationMethod(@NonNull ITestNGMethod method, @NonNull Class testClass) { TestRun.enterNoMockingZone(); try { updateTestClassState(null, testClass); if (method.isBeforeMethodConfiguration()) { if (shouldPrepareForNextTest) { discardTestLevelMockedTypes(); clearTestedObjectsCreatedDuringSetup(); } Object testInstance = method.getInstance(); updateTestClassState(testInstance, testClass); if (shouldPrepareForNextTest) { prepareForNextTest(); shouldPrepareForNextTest = false; createInstancesForTestedFieldsBeforeSetup(testInstance); } TestRun.setRunningIndividualTest(testInstance); } else if (method.isAfterClassConfiguration()) { TestRun.getExecutingTest().setRecordAndReplay(null); cleanUpMocksFromPreviousTest(); TestRun.clearCurrentTestInstance(); } else if (!method.isAfterMethodConfiguration() && !method.isBeforeClassConfiguration()) { TestRun.getExecutingTest().setRecordAndReplay(null); cleanUpMocksFromPreviousTestClass(); TestRun.clearCurrentTestInstance(); TestRun.setCurrentTestClass(null); } } finally { TestRun.exitNoMockingZone(); } } @Override public void afterInvocation(@NonNull IInvokedMethod invokedMethod, @NonNull ITestResult testResult) { if (!invokedMethod.isTestMethod()) { afterConfigurationMethod(testResult); return; } exportCurrentTestMethodIfApplicable(null); SavePoint testMethodSavePoint = savePoint.get(); if (testMethodSavePoint == null) { return; } TestRun.enterNoMockingZone(); shouldPrepareForNextTest = true; savePoint.remove(); Throwable thrownByTest = testResult.getThrowable(); try { if (thrownByTest == null) { concludeTestExecutionWithNothingThrown(testMethodSavePoint, testResult); } else if (thrownByTest instanceof TestException) { concludeTestExecutionWithExpectedExceptionNotThrown(invokedMethod, testMethodSavePoint, testResult); } else if (testResult.isSuccess()) { concludeTestExecutionWithExpectedExceptionThrown(testMethodSavePoint, testResult, thrownByTest); } else { concludeTestExecutionWithUnexpectedExceptionThrown(testMethodSavePoint, thrownByTest); } } finally { TestRun.finishCurrentTestExecution(); TestRun.clearCurrentTestInstance(); } } private static void afterConfigurationMethod(@NonNull ITestResult testResult) { TestRun.enterNoMockingZone(); try { ITestNGMethod method = testResult.getMethod(); if (method.isAfterMethodConfiguration()) { Throwable thrownAfterTest = testResult.getThrowable(); if (thrownAfterTest != null) { filterStackTrace(thrownAfterTest); } } } finally { TestRun.exitNoMockingZone(); } } private static void concludeTestExecutionWithNothingThrown(@NonNull SavePoint testMethodSavePoint, @NonNull ITestResult testResult) { try { concludeTestMethodExecution(testMethodSavePoint, null, false); } catch (Throwable t) { filterStackTrace(t); testResult.setThrowable(t); testResult.setStatus(ITestResult.FAILURE); } } private static void concludeTestExecutionWithExpectedExceptionNotThrown(@NonNull IInvokedMethod invokedMethod, @NonNull SavePoint testMethodSavePoint, @NonNull ITestResult testResult) { try { concludeTestMethodExecution(testMethodSavePoint, null, false); } catch (Throwable t) { filterStackTrace(t); if (isExpectedException(invokedMethod, t)) { testResult.setThrowable(null); testResult.setStatus(ITestResult.SUCCESS); } else { filterStackTrace(testResult.getThrowable()); } } } private static void concludeTestExecutionWithExpectedExceptionThrown(@NonNull SavePoint testMethodSavePoint, @NonNull ITestResult testResult, @NonNull Throwable thrownByTest) { filterStackTrace(thrownByTest); try { concludeTestMethodExecution(testMethodSavePoint, thrownByTest, true); } catch (Throwable t) { if (t != thrownByTest) { filterStackTrace(t); testResult.setThrowable(t); testResult.setStatus(ITestResult.FAILURE); } } } private static void concludeTestExecutionWithUnexpectedExceptionThrown(@NonNull SavePoint testMethodSavePoint, @NonNull Throwable thrownByTest) { filterStackTrace(thrownByTest); try { concludeTestMethodExecution(testMethodSavePoint, thrownByTest, false); } catch (Throwable ignored) { } } private static boolean isExpectedException(@NonNull IInvokedMethod invokedMethod, @NonNull Throwable thrownByTest) { Method testMethod = invokedMethod.getTestMethod().getConstructorOrMethod().getMethod(); Class[] expectedExceptions = testMethod.getAnnotation(Test.class).expectedExceptions(); Class thrownExceptionType = thrownByTest.getClass(); for (Class expectedException : expectedExceptions) { if (expectedException.isAssignableFrom(thrownExceptionType)) { return true; } } return false; } @Override public void onExecutionStart() { } @Override public void onExecutionFinish() { TestRun.enterNoMockingZone(); try { TestRunnerDecorator.cleanUpAllMocks(); } finally { // Maven Surefire, somehow, runs these methods twice per test run. TestRun.clearNoMockingZone(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy