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.
org.testng.internal.MethodInvocationHelper Maven / Gradle / Ivy
package org.testng.internal;
import org.testng.IConfigurable;
import org.testng.IConfigureCallBack;
import org.testng.IHookCallBack;
import org.testng.IHookable;
import org.testng.ITestContext;
import org.testng.ITestNGMethod;
import org.testng.ITestResult;
import org.testng.TestNGException;
import org.testng.internal.annotations.IAnnotationFinder;
import org.testng.internal.collections.ArrayIterator;
import org.testng.internal.collections.OneToTwoDimArrayIterator;
import org.testng.internal.collections.OneToTwoDimIterator;
import org.testng.internal.collections.Pair;
import org.testng.internal.thread.IExecutor;
import org.testng.internal.thread.IFutureResult;
import org.testng.internal.thread.ThreadExecutionException;
import org.testng.internal.thread.ThreadTimeoutException;
import org.testng.internal.thread.ThreadUtil;
import org.testng.xml.XmlSuite;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
/**
* Collections of helper methods to help deal with invocation of TestNG methods
*
* @author Cedric Beust
* @author nullin
*
*/
public class MethodInvocationHelper {
protected static Object invokeMethodNoCheckedException(Method thisMethod, Object instance, List parameters) {
try {
return invokeMethod(thisMethod, instance, parameters);
} catch (InvocationTargetException | IllegalAccessException e) {
// Don't throw TestNGException here or this test won't be reported as a
// skip or failure
throw new RuntimeException(e.getCause());
}
}
protected static void invokeMethodConsideringTimeout(ITestNGMethod tm,
ConstructorOrMethod method,
Object targetInstance,
Object[] params,
ITestResult testResult) throws Throwable {
if (MethodHelper.calculateTimeOut(tm) <= 0) {
MethodInvocationHelper.invokeMethod(method.getMethod(), targetInstance, params);
} else {
MethodInvocationHelper.invokeWithTimeout(tm, targetInstance, params, testResult);
if (!testResult.isSuccess()) {
// A time out happened
Throwable ex = testResult.getThrowable();
testResult.setStatus(ITestResult.FAILURE);
testResult.setThrowable(ex.getCause() == null ? ex : ex.getCause());
throw testResult.getThrowable();
}
}
}
protected static Object invokeMethod(Method thisMethod, Object instance, List parameters)
throws InvocationTargetException, IllegalAccessException {
return invokeMethod(thisMethod, instance, parameters.toArray(new Object[parameters.size()]));
}
protected static Object invokeMethod(Method thisMethod, Object instance, Object[] parameters)
throws InvocationTargetException, IllegalAccessException {
Utils.checkInstanceOrStatic(instance, thisMethod);
// TESTNG-326, allow IObjectFactory to load from non-standard classloader
// If the instance has a different classloader, its class won't match the
// method's class
if (instance != null && !thisMethod.getDeclaringClass().isAssignableFrom(instance.getClass())) {
// for some reason, we can't call this method on this class
// is it static?
boolean isStatic = Modifier.isStatic(thisMethod.getModifiers());
if (!isStatic) {
// not static, so grab a method with the same name and signature in this case
Class clazz = instance.getClass();
try {
thisMethod = clazz.getMethod(thisMethod.getName(), thisMethod.getParameterTypes());
} catch (Exception e) {
// ignore, the method may be private
boolean found = false;
for (; clazz != null; clazz = clazz.getSuperclass()) {
try {
thisMethod = clazz.getDeclaredMethod(thisMethod.getName(),
thisMethod.getParameterTypes());
found = true;
break;
} catch (Exception e2) {
}
}
if (!found) {
// should we assert here? Or just allow it to fail on invocation?
if (thisMethod.getDeclaringClass().equals(instance.getClass())) {
throw new RuntimeException("Can't invoke method " + thisMethod + ", probably due to classloader mismatch");
}
throw new RuntimeException("Can't invoke method " + thisMethod
+ " on this instance of " + instance.getClass() + " due to class mismatch");
}
}
}
}
if (!Modifier.isPublic(thisMethod.getModifiers()) || !thisMethod.isAccessible()) {
try {
thisMethod.setAccessible(true);
} catch (SecurityException e) {
throw new TestNGException(thisMethod.getName() + " must be public", e);
}
}
return thisMethod.invoke(instance, parameters);
}
protected static Iterator invokeDataProvider(Object instance, Method dataProvider,
ITestNGMethod method, ITestContext testContext, Object fedInstance,
IAnnotationFinder annotationFinder) {
List parameters = getParameters(dataProvider, method, testContext, fedInstance, annotationFinder);
Object result = invokeMethodNoCheckedException(dataProvider, instance, parameters);
if (result == null) {
throw new TestNGException("Data Provider " + dataProvider + " returned a null value");
}
// If it returns an Object[][] or Object[], convert it to an Iterator
if (result instanceof Object[][]) {
return new ArrayIterator((Object[][]) result);
} else if (result instanceof Object[]) {
return new OneToTwoDimArrayIterator((Object[]) result);
} else if (result instanceof Iterator) {
Type returnType = dataProvider.getGenericReturnType();
if (returnType instanceof ParameterizedType) {
ParameterizedType contentType = (ParameterizedType) returnType;
Class type = (Class) contentType.getActualTypeArguments()[0];
if (type.isArray()) {
return (Iterator) result;
} else {
return new OneToTwoDimIterator((Iterator) result);
}
} else {
// Raw Iterator, we expect user provides the expected type
return (Iterator) result;
}
}
throw new TestNGException("Data Provider " + dataProvider + " must return"
+ " either Object[][] or Object[] or Iterator or Iterator, not " + dataProvider.getReturnType());
}
private static List getParameters(Method dataProvider, ITestNGMethod method, ITestContext testContext,
Object fedInstance, IAnnotationFinder annotationFinder) {
// Go through all the parameters declared on this Data Provider and
// make sure we have at most one Method and one ITestContext.
// Anything else is an error
List parameters = new ArrayList<>();
Collection>> unresolved = new ArrayList<>();
ConstructorOrMethod com = method.getConstructorOrMethod();
int i = 0;
for (Class cls : dataProvider.getParameterTypes()) {
if (cls.equals(Method.class)) {
parameters.add(com.getMethod());
} else if (cls.equals(Constructor.class)) {
parameters.add(com.getConstructor());
} else if (cls.equals(ConstructorOrMethod.class)) {
parameters.add(com);
} else if (cls.equals(ITestNGMethod.class)) {
parameters.add(method);
} else if (cls.equals(ITestContext.class)) {
parameters.add(testContext);
} else if (cls.equals(Class.class)) {
parameters.add(com.getDeclaringClass());
} else {
boolean isTestInstance = annotationFinder.hasTestInstance(dataProvider, i);
if (isTestInstance) {
parameters.add(fedInstance);
} else {
unresolved.add(new Pair>(i, cls));
}
}
i++;
}
if (!unresolved.isEmpty()) {
StringBuilder sb = new StringBuilder();
sb.append("Some DataProvider ").append(dataProvider).append(" parameters unresolved: ");
for (Pair> pair : unresolved) {
sb.append(" at ").append(pair.first()).append(" type ").append(pair.second()).append("\n");
}
throw new TestNGException(sb.toString());
}
return parameters;
}
/**
* Invokes the run
method of the IHookable
.
*
* @param testInstance
* the instance to invoke the method in
* @param parameters
* the parameters to be passed to IHookCallBack
* @param thisMethod
* the method to be invoked through the IHookCallBack
* @param testResult
* the current ITestResult
passed to
* IHookable.run
* @throws NoSuchMethodException
* @throws IllegalAccessException
* @throws InvocationTargetException
* @throws Throwable
* thrown if the reflective call to
* thisMethod
results in an exception
*/
protected static void invokeHookable(final Object testInstance, final Object[] parameters,
final IHookable hookable, final Method thisMethod,
final ITestResult testResult) throws Throwable {
final Throwable[] error = new Throwable[1];
IHookCallBack callback = new IHookCallBack() {
@Override
public void runTestMethod(ITestResult tr) {
try {
invokeMethod(thisMethod, testInstance, parameters);
} catch (Throwable t) {
error[0] = t;
tr.setThrowable(t); // make Throwable available to IHookable
}
}
@Override
public Object[] getParameters() {
return parameters;
}
};
hookable.run(callback, testResult);
if (error[0] != null) {
throw error[0];
}
}
/**
* Invokes a method on a separate thread in order to allow us to timeout the
* invocation. It uses as implementation an Executor
and a
* CountDownLatch
.
*/
protected static void invokeWithTimeout(ITestNGMethod tm, Object instance,
Object[] parameterValues, ITestResult testResult)
throws InterruptedException, ThreadExecutionException {
invokeWithTimeout(tm, instance, parameterValues, testResult, null);
}
protected static void invokeWithTimeout(ITestNGMethod tm, Object instance,
Object[] parameterValues, ITestResult testResult, IHookable hookable)
throws InterruptedException, ThreadExecutionException {
if (ThreadUtil.isTestNGThread() && testResult.getTestContext().getCurrentXmlTest().getParallel() != XmlSuite.ParallelMode.TESTS) {
// We are already running in our own executor, don't create another one (or we will
// lose the time out of the enclosing executor).
invokeWithTimeoutWithNoExecutor(tm, instance, parameterValues, testResult, hookable);
} else {
invokeWithTimeoutWithNewExecutor(tm, instance, parameterValues, testResult, hookable);
}
}
private static void invokeWithTimeoutWithNoExecutor(ITestNGMethod tm, Object instance,
Object[] parameterValues, ITestResult testResult, IHookable hookable) {
InvokeMethodRunnable imr = new InvokeMethodRunnable(tm, instance, parameterValues, hookable, testResult);
long startTime = System.currentTimeMillis();
long realTimeOut = MethodHelper.calculateTimeOut(tm);
try {
imr.run();
if (System.currentTimeMillis() <= startTime + realTimeOut) {
testResult.setStatus(ITestResult.SUCCESS);
} else {
ThreadTimeoutException exception = new ThreadTimeoutException("Method "
+ tm.getQualifiedName() + "()"
+ " didn't finish within the time-out " + realTimeOut);
testResult.setThrowable(exception);
testResult.setStatus(ITestResult.FAILURE);
}
} catch (Exception ex) {
testResult.setThrowable(ex.getCause());
testResult.setStatus(ITestResult.FAILURE);
}
}
private static void invokeWithTimeoutWithNewExecutor(ITestNGMethod tm, Object instance,
Object[] parameterValues, ITestResult testResult, IHookable hookable)
throws InterruptedException, ThreadExecutionException {
IExecutor exec = ThreadUtil.createExecutor(1, tm.getMethodName());
InvokeMethodRunnable imr = new InvokeMethodRunnable(tm, instance, parameterValues, hookable, testResult);
IFutureResult future = exec.submitRunnable(imr);
exec.shutdown();
long realTimeOut = MethodHelper.calculateTimeOut(tm);
boolean finished = exec.awaitTermination(realTimeOut);
if (!finished) {
exec.stopNow();
ThreadTimeoutException exception = new ThreadTimeoutException("Method "
+ tm.getQualifiedName() + "()"
+ " didn't finish within the time-out " + realTimeOut);
StackTraceElement[][] stacktraces = exec.getStackTraces();
if (stacktraces.length > 0) {
exception.setStackTrace(stacktraces[0]);
}
testResult.setThrowable(exception);
testResult.setStatus(ITestResult.FAILURE);
} else {
Utils.log("Invoker " + Thread.currentThread().hashCode(), 3, "Method " + tm.getMethodName()
+ " completed within the time-out " + tm.getTimeOut());
// We don't need the result from the future but invoking get() on it
// will trigger the exception that was thrown, if any
future.get();
// done.await();
testResult.setStatus(ITestResult.SUCCESS); // if no exception till here
// than SUCCESS
}
}
protected static void invokeConfigurable(final Object instance, final Object[] parameters,
final IConfigurable configurableInstance, final Method thisMethod,
final ITestResult testResult) throws Throwable {
final Throwable[] error = new Throwable[1];
IConfigureCallBack callback = new IConfigureCallBack() {
@Override
public void runConfigurationMethod(ITestResult tr) {
try {
invokeMethod(thisMethod, instance, parameters);
} catch (Throwable t) {
error[0] = t;
tr.setThrowable(t); // make Throwable available to IConfigurable
}
}
@Override
public Object[] getParameters() {
return parameters;
}
};
configurableInstance.run(callback, testResult);
if (error[0] != null) {
throw error[0];
}
}
}