blob: c3f15f396c48272343d13299f6b7eae8e1b4e15f [file] [log] [blame]
/*
* Copyright (c) 2018 Mockito contributors
* This program is made available under the terms of the MIT License.
*/
package org.mockito.internal.framework;
import org.mockito.Mockito;
import org.mockito.MockitoAnnotations;
import org.mockito.MockitoSession;
import org.mockito.exceptions.misusing.RedundantListenerException;
import org.mockito.internal.exceptions.Reporter;
import org.mockito.internal.junit.TestFinishedEvent;
import org.mockito.internal.junit.UniversalTestListener;
import org.mockito.internal.util.MockitoLogger;
import org.mockito.quality.Strictness;
import java.util.List;
public class DefaultMockitoSession implements MockitoSession {
private final String name;
private final UniversalTestListener listener;
public DefaultMockitoSession(List<Object> testClassInstances, String name, Strictness strictness, MockitoLogger logger) {
this.name = name;
listener = new UniversalTestListener(strictness, logger);
try {
//So that the listener can capture mock creation events
Mockito.framework().addListener(listener);
} catch (RedundantListenerException e) {
Reporter.unfinishedMockingSession();
}
try {
for (Object testClassInstance : testClassInstances) {
MockitoAnnotations.initMocks(testClassInstance);
}
} catch (RuntimeException | Error e) {
try {
// TODO: ideally this scenario should be tested on DefaultMockitoSessionBuilderTest,
// but we don't have any Android.bp project to run it.
// Besides, the latest Mockito code (https://github.com/mockito/mockito/blob/main/src/main/java/org/mockito/internal/framework/DefaultMockitoSession.java
// at the time this patch was merged) has a different workflow, where the listener
// is marked as dirty when an exception is thrown, so we're forking the solution.
Mockito.framework().removeListener(listener);
} catch (RuntimeException | Error e2) {
// Ignore it, as the real failure is e, thrown at the end
System.err.println("DefaultMockitoSession: ignoring exception thrown when removing "
+ "listener " + listener);
e2.printStackTrace(System.err);
}
throw e;
}
}
@Override
public void setStrictness(Strictness strictness) {
listener.setStrictness(strictness);
}
@Override
public void finishMocking() {
finishMocking(null);
}
@Override
public void finishMocking(final Throwable failure) {
//Cleaning up the state, we no longer need the listener hooked up
//The listener implements MockCreationListener and at this point
//we no longer need to listen on mock creation events. We are wrapping up the session
Mockito.framework().removeListener(listener);
//Emit test finished event so that validation such as strict stubbing can take place
listener.testFinished(new TestFinishedEvent() {
@Override
public Throwable getFailure() {
return failure;
}
@Override
public String getTestName() {
return name;
}
});
//Validate only when there is no test failure to avoid reporting multiple problems
if (failure == null) {
//Finally, validate user's misuse of Mockito framework.
Mockito.validateMockitoUsage();
}
}
}