All Downloads are FREE. Search and download functionalities are using the official Maven repository.
Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
mockit.internal.expectations.state.MockedTypeCascade Maven / Gradle / Ivy
Go to download
JMockit is a Java toolkit for automated developer testing.
It contains APIs for the creation of the objects to be tested, for mocking dependencies, and for faking external
APIs; JUnit (4 & 5) and TestNG test runners are supported.
It also contains an advanced code coverage tool.
/*
* Copyright (c) 2006 Rogério Liesenfeld
* This file is subject to the terms of the MIT license (see LICENSE.txt).
*/
package mockit.internal.expectations.state;
import java.lang.reflect.*;
import java.util.*;
import java.util.concurrent.*;
import javax.annotation.*;
import static java.lang.reflect.Modifier.*;
import static java.util.Collections.synchronizedList;
import mockit.internal.*;
import mockit.internal.expectations.mocking.*;
import mockit.internal.reflection.*;
import mockit.internal.state.*;
import mockit.internal.util.*;
import static mockit.external.asm.JavaType.*;
import static mockit.internal.util.Utilities.*;
public final class MockedTypeCascade
{
@Nonnull private static final CascadingTypes CASCADING_TYPES = TestRun.getExecutingTest().getCascadingTypes();
private static final int PUBLIC_INTERFACE = PUBLIC + INTERFACE;
final boolean fromMockField;
@Nonnull private final Type mockedType;
@Nonnull final String mockedTypeDesc;
@Nullable Class mockedClass;
@Nullable private GenericTypeReflection genericReflection;
@Nonnull private final Map cascadedTypesAndMocks;
@Nonnull private final List cascadingInstances;
MockedTypeCascade(boolean fromMockField, @Nonnull Type mockedType, @Nonnull String mockedTypeDesc)
{
this.fromMockField = fromMockField;
this.mockedType = mockedType;
this.mockedTypeDesc = mockedTypeDesc;
cascadedTypesAndMocks = new ConcurrentHashMap(4);
cascadingInstances = synchronizedList(new ArrayList());
}
@Nullable
public static Object getMock(
@Nonnull String mockedTypeDesc, @Nonnull String mockedMethodNameAndDesc, @Nullable Object mockInstance,
@Nonnull String returnTypeDesc, @Nonnull Class returnType)
{
MockedTypeCascade cascade = CASCADING_TYPES.getCascade(mockedTypeDesc, mockInstance);
if (cascade == null) {
return null;
}
String cascadedReturnTypeDesc = getReturnTypeIfCascadingSupportedForIt(returnTypeDesc);
if (cascadedReturnTypeDesc == null) {
return null;
}
return cascade.getCascadedInstance(mockedMethodNameAndDesc, cascadedReturnTypeDesc, returnType);
}
@Nullable
public static Object getMock(
@Nonnull String mockedTypeDesc, @Nonnull String mockedMethodNameAndDesc, @Nullable Object mockInstance,
@Nonnull String returnTypeDesc, @Nullable String genericSignature)
{
char typeCode = returnTypeDesc.charAt(0);
if (typeCode != 'L') {
return null;
}
MockedTypeCascade cascade = CASCADING_TYPES.getCascade(mockedTypeDesc, mockInstance);
if (cascade == null) {
return null;
}
String resolvedReturnTypeDesc = null;
if (genericSignature != null) {
resolvedReturnTypeDesc = cascade.getGenericReturnType(mockedTypeDesc, genericSignature);
}
if (resolvedReturnTypeDesc == null) {
resolvedReturnTypeDesc = getReturnTypeIfCascadingSupportedForIt(returnTypeDesc);
if (resolvedReturnTypeDesc == null) {
return null;
}
}
else if (resolvedReturnTypeDesc.charAt(0) == '[') {
return DefaultValues.computeForArrayType(resolvedReturnTypeDesc);
}
return cascade.getCascadedInstance(mockedMethodNameAndDesc, resolvedReturnTypeDesc, mockInstance);
}
@Nullable
private String getGenericReturnType(@Nonnull String ownerTypeDesc, @Nonnull String genericSignature)
{
String resolvedSignature = getGenericReflection().resolveSignature(ownerTypeDesc, genericSignature);
String returnTypeDesc = resolvedSignature.substring(resolvedSignature.indexOf(')') + 1);
if (returnTypeDesc.charAt(0) == '[') {
return returnTypeDesc;
}
String returnTypeName = returnTypeDesc.substring(1, returnTypeDesc.length() - 1);
return isTypeSupportedForCascading(returnTypeName) ? returnTypeName : null;
}
@Nonnull
private synchronized GenericTypeReflection getGenericReflection()
{
GenericTypeReflection reflection = genericReflection;
if (reflection == null) {
Class ownerClass = getClassWithCalledMethod();
reflection = new GenericTypeReflection(ownerClass, mockedType);
genericReflection = reflection;
}
return reflection;
}
@Nullable
private static String getReturnTypeIfCascadingSupportedForIt(@Nonnull Class returnType)
{
if (MockingFilters.isSubclassOfUnmockable(returnType)) {
return null;
}
String typeName = getInternalName(returnType);
return isTypeSupportedForCascading(typeName) ? typeName : null;
}
@SuppressWarnings("OverlyComplexMethod")
private static boolean isTypeSupportedForCascading(@Nonnull String typeName)
{
//noinspection SimplifiableIfStatement
if (typeName.contains("/Process") || typeName.endsWith("/Runnable")) {
return true;
}
return
(!typeName.startsWith("java/lang/") || typeName.contains("management")) &&
!typeName.startsWith("java/math/") &&
(!typeName.startsWith("java/util/") ||
typeName.endsWith("/Date") || typeName.endsWith("/Callable") || typeName.endsWith("Future") ||
typeName.contains("logging")) &&
!"java/time/Duration".equals(typeName);
}
@Nullable
private static String getReturnTypeIfCascadingSupportedForIt(@Nonnull String typeDesc)
{
String typeName = typeDesc.substring(1, typeDesc.length() - 1);
return isTypeSupportedForCascading(typeName) ? typeName : null;
}
@Nullable
private Object getCascadedInstance(
@Nonnull String methodNameAndDesc, @Nonnull String returnTypeInternalName, @Nonnull Class returnClass)
{
MockedTypeCascade nextLevel = this;
if (!cascadedTypesAndMocks.containsKey(returnTypeInternalName)) {
cascadedTypesAndMocks.put(returnTypeInternalName, returnClass);
nextLevel = CASCADING_TYPES.add(returnTypeInternalName, false, returnClass);
}
return nextLevel.createNewCascadedInstanceOrUseNonCascadedOneIfAvailable(methodNameAndDesc, returnClass);
}
@Nullable
private Object getCascadedInstance(
@Nonnull String methodNameAndDesc, @Nonnull String returnTypeInternalName, @Nullable Object mockInstance)
{
MockedTypeCascade nextLevel = this;
Type returnType = cascadedTypesAndMocks.get(returnTypeInternalName);
Class returnClass;
if (returnType == null) {
Class cascadingClass = getClassWithCalledMethod();
Type genericReturnType = getGenericReturnType(cascadingClass, methodNameAndDesc);
if (genericReturnType == null) {
return null;
}
Class resolvedReturnType = getClassType(genericReturnType);
if (resolvedReturnType.isAssignableFrom(cascadingClass)) {
if (mockInstance != null) {
return mockInstance;
}
returnType = mockedType;
returnClass = cascadingClass;
}
else if (nonPublicTypeReturnedFromPublicInterface(cascadingClass, resolvedReturnType)) {
return null;
}
else {
Object defaultReturnValue = DefaultValues.computeForType(resolvedReturnType);
if (defaultReturnValue != null) {
return defaultReturnValue;
}
cascadedTypesAndMocks.put(returnTypeInternalName, genericReturnType);
nextLevel = CASCADING_TYPES.add(returnTypeInternalName, false, genericReturnType);
returnType = genericReturnType;
returnClass = resolvedReturnType;
}
}
else {
nextLevel = CASCADING_TYPES.getCascade(returnType);
returnClass = getClassType(returnType);
}
if (getReturnTypeIfCascadingSupportedForIt(returnClass) == null) {
return null;
}
return nextLevel.createNewCascadedInstanceOrUseNonCascadedOneIfAvailable(methodNameAndDesc, returnType);
}
private static boolean nonPublicTypeReturnedFromPublicInterface(
@Nonnull Class cascadingClass, @Nonnull Class resolvedReturnType)
{
return
!isPublic(resolvedReturnType.getModifiers()) &&
cascadingClass.getClassLoader() != null &&
(cascadingClass.getModifiers() & PUBLIC_INTERFACE) != 0 &&
!resolvedReturnType.isMemberClass();
}
@Nonnull
private Class getClassWithCalledMethod()
{
if (mockedClass != null) {
return mockedClass;
}
if (mockedType instanceof Class) {
return (Class) mockedType;
}
return (Class) ((ParameterizedType) mockedType).getRawType();
}
@Nullable
private Type getGenericReturnType(@Nonnull Class cascadingClass, @Nonnull String methodNameAndDesc)
{
RealMethodOrConstructor realMethod;
try { realMethod = new RealMethodOrConstructor(cascadingClass, methodNameAndDesc); }
catch (NoSuchMethodException e) { return null; }
Method cascadingMethod = realMethod.getMember();
Type genericReturnType = cascadingMethod.getGenericReturnType();
if (genericReturnType instanceof TypeVariable) {
genericReturnType = getGenericReflection().resolveTypeVariable((TypeVariable) genericReturnType);
}
return genericReturnType == Object.class ? null : genericReturnType;
}
@Nullable
private Object createNewCascadedInstanceOrUseNonCascadedOneIfAvailable(
@Nonnull String methodNameAndDesc, @Nonnull Type mockedReturnType)
{
InstanceFactory instanceFactory = TestRun.mockFixture().findInstanceFactory(mockedReturnType);
if (instanceFactory == null) {
String methodName = methodNameAndDesc.substring(0, methodNameAndDesc.indexOf('('));
CascadingTypeRedefinition typeRedefinition = new CascadingTypeRedefinition(methodName, mockedReturnType);
instanceFactory = typeRedefinition.redefineType();
if (instanceFactory == null) {
return null;
}
}
else {
Object lastInstance = instanceFactory.getLastInstance();
if (lastInstance != null) {
return lastInstance;
}
}
Object cascadedInstance = instanceFactory.create();
instanceFactory.clearLastInstance();
addInstance(cascadedInstance);
TestRun.getExecutingTest().addInjectableMock(cascadedInstance);
return cascadedInstance;
}
void discardCascadedMocks()
{
cascadedTypesAndMocks.clear();
cascadingInstances.clear();
}
void addInstance(@Nonnull Object cascadingInstance) { cascadingInstances.add(cascadingInstance); }
boolean hasInstance(@Nonnull Object cascadingInstance)
{
return containsReference(cascadingInstances, cascadingInstance);
}
}