/* * JBoss, Home of Professional Open Source * Copyright 2011, Red Hat Middleware LLC, and individual 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.drone.impl; import java.lang.reflect.Method; import java.util.Arrays; import java.util.List; import org.jboss.arquillian.config.descriptor.api.ArquillianDescriptor; import org.jboss.arquillian.container.spi.client.container.DeployableContainer; import org.jboss.arquillian.container.spi.client.deployment.DeploymentDescription; import org.jboss.arquillian.core.api.annotation.ApplicationScoped; import org.jboss.arquillian.core.spi.ServiceLoader; import org.jboss.arquillian.core.spi.context.ApplicationContext; import org.jboss.arquillian.drone.api.annotation.Drone; import org.jboss.arquillian.drone.impl.mockdrone.MockDrone; import org.jboss.arquillian.drone.impl.mockdrone.MockDroneConfiguration; import org.jboss.arquillian.drone.impl.mockdrone.MockDroneFactory; import org.jboss.arquillian.drone.spi.Configurator; import org.jboss.arquillian.drone.spi.Destructor; import org.jboss.arquillian.drone.spi.DroneContext; import org.jboss.arquillian.drone.spi.DronePoint; import org.jboss.arquillian.drone.spi.DroneRegistry; import org.jboss.arquillian.drone.spi.Instantiator; import org.jboss.arquillian.test.spi.TestEnricher; import org.jboss.arquillian.test.spi.context.ClassContext; import org.jboss.arquillian.test.spi.context.SuiteContext; import org.jboss.arquillian.test.spi.context.TestContext; import org.jboss.arquillian.test.spi.event.suite.After; import org.jboss.arquillian.test.spi.event.suite.AfterClass; import org.jboss.arquillian.test.spi.event.suite.Before; import org.jboss.arquillian.test.spi.event.suite.BeforeClass; import org.jboss.arquillian.test.spi.event.suite.BeforeSuite; import org.jboss.arquillian.test.test.AbstractTestTestBase; import org.jboss.shrinkwrap.descriptor.api.Descriptors; import org.junit.Assert; import org.junit.Rule; import org.junit.Test; import org.junit.rules.ExpectedException; import org.junit.runner.RunWith; import org.mockito.Mock; import org.mockito.Mockito; import org.mockito.junit.MockitoJUnitRunner; /** * Tests Configurator precedence and its retrieval chain, uses qualifier as well. * <p/> * Additionally tests DroneTestEnricher * * @author <a href="mailto:kpiwko@redhat.com">Karel Piwko</a> */ @RunWith(MockitoJUnitRunner.Silent.class) public class EnricherTestCase extends AbstractTestTestBase { private static final String DIFFERENT_FIELD = "ArquillianDescriptor @DifferentMock"; private static final String METHOD_ARGUMENT_ONE_FIELD = "ArquillianDescriptor @MethodArgumentOne"; @Rule public ExpectedException exception = ExpectedException.none(); @Mock DeploymentDescription deploymentDescription1; @Mock DeploymentDescription deploymentDescription2; @Mock DeployableContainer deployableContainer; @Mock private ServiceLoader serviceLoader; @Override protected void addExtensions(List<Class<?>> extensions) { extensions.add(DroneLifecycleManager.class); extensions.add(DroneRegistrar.class); extensions.add(DroneConfigurator.class); extensions.add(DroneTestEnricher.class); extensions.add(DroneDestructor.class); } @SuppressWarnings("rawtypes") @org.junit.Before public void setMocks() { ArquillianDescriptor desc = Descriptors.create(ArquillianDescriptor.class).extension("mockdrone-different") .property("field", DIFFERENT_FIELD).extension("mockdrone-methodargumentone") .property("field", METHOD_ARGUMENT_ONE_FIELD); TestEnricher testEnricher = new DroneTestEnricher(); getManager().inject(testEnricher); bind(ApplicationScoped.class, ServiceLoader.class, serviceLoader); bind(ApplicationScoped.class, ArquillianDescriptor.class, desc); Mockito.when(serviceLoader.all(Configurator.class)).thenReturn( Arrays.<Configurator>asList(new MockDroneFactory())); Mockito.when(serviceLoader.all(Instantiator.class)).thenReturn( Arrays.<Instantiator>asList(new MockDroneFactory())); Mockito.when(serviceLoader.all(Destructor.class)).thenReturn(Arrays.<Destructor>asList(new MockDroneFactory())); Mockito.when(serviceLoader.onlyOne(TestEnricher.class)).thenReturn(testEnricher); // These two stubbings are used in DeploymentTestCase -> hence Silent runner Mockito.when(deploymentDescription1.getName()).thenReturn(AnnotationMocks.DEPLOYMENT_1); Mockito.when(deploymentDescription2.getName()).thenReturn(AnnotationMocks.DEPLOYMENT_2); } @Test public void testQualifer() throws Exception { getManager().getContext(ClassContext.class).activate(EnrichedClass.class); DroneContext context = fireAndVerifyBeforeSuiteProcess(); fire(new BeforeClass(EnrichedClass.class)); DronePoint<MockDrone> invalidDronePoint = new DronePointImpl<MockDrone>(MockDrone.class, DronePoint.Lifecycle.CLASS, AnnotationMocks.drone()); DronePoint<MockDrone> dronePoint = new DronePointImpl<MockDrone>(MockDrone.class, DronePoint.Lifecycle.CLASS, AnnotationMocks.drone(), AnnotationMocks.differentQualifier()); MockDroneConfiguration configuration = context.get(dronePoint).getConfigurationAs(MockDroneConfiguration.class); Assert.assertFalse("There is no MockDroneConfiguration with @Default qualifier", context.get(invalidDronePoint).hasConfiguration()); Assert.assertNotNull("MockDroneConfiguration is stored with @Different qualifier", configuration); Assert.assertEquals("MockDrone was configured from @Different configuration", DIFFERENT_FIELD, configuration.getField()); getManager().getContext(ClassContext.class).deactivate(); getManager().getContext(ClassContext.class).destroy(EnrichedClass.class); } @Test public void testMethodQualiferWithCleanup() throws Exception { getManager().getContext(ClassContext.class).activate(MethodEnrichedClass.class); Object instance = new MethodEnrichedClass(); Method testMethod = MethodEnrichedClass.class.getMethod("testMethodEnrichment", MockDrone.class); getManager().getContext(TestContext.class).activate(instance); DroneContext context = fireAndVerifyBeforeSuiteProcess(); fire(new BeforeClass(MethodEnrichedClass.class)); Object[] parameters = enrichClassAndResolveMethod(instance, testMethod); DronePoint<MockDrone> dronePoint = new DronePointImpl<MockDrone>(MockDrone.class, DronePoint.Lifecycle.METHOD, AnnotationMocks.drone(), AnnotationMocks.methodArgumentOneQualifier()); verifyDronePointInstantiated(true, context, dronePoint); testMethod.invoke(instance, parameters); fire(new After(instance, testMethod)); fire(new AfterClass(MethodEnrichedClass.class)); verifyDronePointInstantiated(false, context, dronePoint, true); } @Test(expected = IllegalStateException.class) public void testMethodQualiferUnregistered() throws Exception { getManager().getContext(ClassContext.class).activate(MethodEnrichedClassUnregistered.class); Object instance = new MethodEnrichedClassUnregistered(); Method testMethod = MethodEnrichedClassUnregistered.class.getMethod("testMethodEnrichment", Object.class); getManager().getContext(TestContext.class).activate(instance); DroneContext context = fireAndVerifyBeforeSuiteProcess(); fire(new BeforeClass(MethodEnrichedClassUnregistered.class)); Object[] parameters = enrichClassAndResolveMethod(instance, testMethod); DronePoint<Object> dronePoint = new DronePointImpl<Object>(Object.class, DronePoint.Lifecycle.METHOD, AnnotationMocks.drone()); verifyDronePointInstantiated(true, context, dronePoint); testMethod.invoke(instance, parameters); fire(new After(instance, testMethod)); fire(new AfterClass(MethodEnrichedClassUnregistered.class)); verifyDronePointInstantiated(false, context, dronePoint, true); } @Test public void testClassWithoutArquillianLifecycle() throws Exception { Object instance = new NonArquillianClass(); Method testMethod = NonArquillianClass.class.getMethod("someMethod", MockDrone.class); fire(new BeforeSuite()); DroneContext context = getManager().getContext(ApplicationContext.class).getObjectStore().get(DroneContext.class); Assert.assertNotNull("DroneContext was created in the context", context); TestEnricher testEnricher = serviceLoader.onlyOne(TestEnricher.class); testEnricher.enrich(instance); Object[] parameters = testEnricher.resolve(testMethod); DronePoint<MockDrone> classDronePoint = new DronePointImpl<MockDrone>(MockDrone.class, DronePoint.Lifecycle.CLASS, AnnotationMocks.drone()); verifyDronePointInstantiated(true, context, classDronePoint); DronePoint<MockDrone> methodDronePoint = new DronePointImpl<MockDrone>(MockDrone.class, DronePoint.Lifecycle.METHOD, AnnotationMocks.drone()); verifyDronePointInstantiated(true, context, methodDronePoint); testMethod.invoke(instance, parameters); } /** * Verifies whether the given {@link DronePoint} is instantiated or not and throws an AssertException with an * appropriate message if necessary. It is verified in the context of creation of the instance (not destruction) * * @param shouldInstantiated * Whether the given {@link DronePoint} should be instantiated * @param context * a Drone context the given {@link DronePoint} should belong to * @param dronePoint * a drone point to be verified */ void verifyDronePointInstantiated(boolean shouldInstantiated, DroneContext context, DronePoint<?> dronePoint) { verifyDronePointInstantiated(shouldInstantiated, context, dronePoint, false); } /** * Verifies whether the given {@link DronePoint} is instantiated or not and throws an AssertException with an * appropriate message if necessary. * * @param shouldInstantiated * Whether the given {@link DronePoint} should be instantiated * @param context * a Drone context the given {@link DronePoint} should belong to * @param dronePoint * a drone point to be verified * @param shouldDestroyed * whether the given {@link DronePoint} should be destroyed - based on this parameter only * the message is modified */ void verifyDronePointInstantiated(boolean shouldInstantiated, DroneContext context, DronePoint<?> dronePoint, boolean shouldDestroyed) { if (shouldInstantiated) { Assert.assertTrue("Drone should be created", context.get(dronePoint).isInstantiated()); } else { String message = shouldDestroyed ? "Drone should be destroyed" : "Drone should NOT be created"; Assert.assertFalse(message, context.get(dronePoint).isInstantiated()); } } /** * Fires the {@link BeforeSuite} event, retrieve a {@link DroneContext} and {@link DroneRegistry} instances and * asserts that they are not null. Asserts that the {@link Configurator} and {@link Instantiator} are mock type. * * @return the drone context instance */ DroneContext fireAndVerifyBeforeSuiteProcess() { fire(new BeforeSuite()); DroneContext context = getManager() .getContext(ApplicationContext.class).getObjectStore().get(DroneContext.class); Assert.assertNotNull("DroneContext was created in the context", context); DroneRegistry registry = getManager().getContext(SuiteContext.class).getObjectStore().get(DroneRegistry.class); Assert.assertNotNull("Drone registry was created in the context", registry); Assert.assertTrue("Configurator is of mock type", registry.getEntryFor(MockDrone.class, Configurator.class) instanceof MockDroneFactory); Assert.assertTrue("Instantiator is of mock type", registry.getEntryFor(MockDrone.class, Instantiator.class) instanceof MockDroneFactory); return context; } /** * Enriches the given instance of a test class and resolves the given test method * * @param instance * a test class instance to be enriched * @param testMethod * a test method to be resolved * * @return resolved parameters of the given test method */ Object[] enrichClassAndResolveMethod(Object instance, Method testMethod) { fire(new Before(instance, testMethod)); TestEnricher testEnricher = serviceLoader.onlyOne(TestEnricher.class); testEnricher.enrich(instance); return testEnricher.resolve(testMethod); } static class EnrichedClass { @Drone @Different MockDrone unused; } static class MethodEnrichedClass { public void testMethodEnrichment(@Drone @MethodArgumentOne MockDrone unused) { Assert.assertNotNull("Mock drone instance was created", unused); Assert.assertEquals("MockDroneConfiguration is set via ArquillianDescriptor", METHOD_ARGUMENT_ONE_FIELD, unused.getField()); } } static class MethodEnrichedClassUnregistered { public void testMethodEnrichment(@Drone Object unused) { } } static class NonArquillianClass { @Drone MockDrone classDrone; public void someMethod(@Drone MockDrone methodDrone) { } } }