org.jboss.arquillian.protocol.jmx.JMXTestRunner Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of arquillian-protocol-jmx Show documentation
Show all versions of arquillian-protocol-jmx Show documentation
Protocol handler for communicating via JMX
/*
* JBoss, Home of Professional Open Source
* Copyright 2009 Red Hat Inc. and/or its affiliates and other contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jboss.arquillian.protocol.jmx;
import java.lang.reflect.Method;
import java.security.AccessController;
import java.security.PrivilegedExceptionAction;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.management.JMException;
import javax.management.MBeanServer;
import javax.management.Notification;
import javax.management.NotificationBroadcasterSupport;
import javax.management.ObjectName;
import org.jboss.arquillian.container.test.spi.TestRunner;
import org.jboss.arquillian.container.test.spi.command.Command;
import org.jboss.arquillian.container.test.spi.util.TestRunners;
import org.jboss.arquillian.test.spi.TestResult;
import org.jboss.arquillian.test.spi.TestResult.Status;
/**
* An MBean to run test methods in container.
*
* @author [email protected]
*/
public class JMXTestRunner extends NotificationBroadcasterSupport implements JMXTestRunnerMBean {
// package shared MBeanServer with JMXCommandService
static MBeanServer localMBeanServer;
// Provide logging
private static Logger log = Logger.getLogger(JMXTestRunner.class.getName());
private final String objectName;
private ConcurrentHashMap> events;
private ThreadLocal currentCall;
// Notification Sequence number
private AtomicInteger integer = new AtomicInteger();
// TestRunner to used for testing
private TestRunner mockTestRunner;
private TestClassLoader testClassLoader;
public JMXTestRunner(TestClassLoader classLoader) {
this(classLoader, JMXTestRunnerMBean.OBJECT_NAME);
}
public JMXTestRunner(TestClassLoader classLoader, String objectName) {
this.testClassLoader = classLoader;
// Initialize the default TestClassLoader
if (testClassLoader == null) {
testClassLoader = new TestClassLoader() {
public Class> loadTestClass(String className) throws ClassNotFoundException {
ClassLoader classLoader = JMXTestRunner.class.getClassLoader();
return classLoader.loadClass(className);
}
};
}
events = new ConcurrentHashMap>();
currentCall = new ThreadLocal();
this.objectName = objectName;
}
public ObjectName registerMBean(MBeanServer mbeanServer) throws JMException {
ObjectName oname = new ObjectName(this.objectName);
mbeanServer.registerMBean(this, oname);
log.fine("JMXTestRunner registered: " + oname);
localMBeanServer = mbeanServer;
return oname;
}
public void unregisterMBean(MBeanServer mbeanServer) throws JMException {
ObjectName oname = new ObjectName(this.objectName);
if (mbeanServer.isRegistered(oname)) {
mbeanServer.unregisterMBean(oname);
log.fine("JMXTestRunner unregistered: " + oname);
}
localMBeanServer = null;
}
@Override
public byte[] runTestMethod(String className, String methodName) {
TestResult result = runTestMethodInternal(className, methodName, new HashMap());
return Serializer.toByteArray(result);
}
@Override
public byte[] runTestMethod(String className, String methodName, Map protocolProps) {
// detect if deprecated method is overridden in sub class, if so call it instead
try {
final Class> impl = this.getClass();
Method m = AccessController.doPrivileged(new PrivilegedExceptionAction() {
public Method run() throws NoSuchMethodException {
return impl.getMethod("runTestMethod", String.class, String.class);
}
});
if (m.getDeclaringClass() != JMXTestRunner.class) {
return runTestMethod(className, methodName);
}
} catch (Exception e) {
}
TestResult result = runTestMethodInternal(className, methodName, protocolProps);
return Serializer.toByteArray(result);
}
private TestResult runTestMethodInternal(String className, String methodName, Map protocolProps) {
currentCall.set(className + methodName);
TestResult result = new TestResult();
try {
TestRunner runner = mockTestRunner;
if (runner == null) {
runner = TestRunners.getTestRunner(getClass().getClassLoader());
}
log.fine("Load test class: " + className);
Class> testClass = testClassLoader.loadTestClass(className);
log.fine("Test class loaded from: " + testClass.getClassLoader());
log.fine("Execute: " + className + "." + methodName);
result = doRunTestMethod(runner, testClass, methodName, protocolProps);
} catch (Throwable th) {
result.setStatus(Status.FAILED);
result.setEnd(System.currentTimeMillis());
result.setThrowable(th);
} finally {
log.fine("Result: " + result);
if (result.getStatus() == Status.FAILED) {
log.log(Level.SEVERE, "Failed: " + className + "." + methodName, result.getThrowable());
}
}
return result;
}
protected TestResult doRunTestMethod(TestRunner runner, Class> testClass, String methodName,
Map protocolProps) {
return runner.execute(testClass, methodName);
}
@Override
public void send(Command> command) {
Notification notification = new Notification("arquillian-command", this, integer.incrementAndGet(),
currentCall.get());
notification.setUserData(Serializer.toByteArray(command));
sendNotification(notification);
}
@Override
public Command> receive() {
return events.remove(currentCall.get());
}
@Override
public void push(String eventId, byte[] command) {
events.put(eventId, Serializer.toObject(Command.class, command));
}
/**
* @return the currentCall
*/
protected String getCurrentCall() {
return currentCall.get();
}
/*
* Internal Helpers for Test
*/
protected void setCurrentCall(String current) {
currentCall.set(current);
}
void setExposedTestRunnerForTest(TestRunner mockTestRunner) {
this.mockTestRunner = mockTestRunner;
}
public interface TestClassLoader {
Class> loadTestClass(String className) throws ClassNotFoundException;
}
}