/** * *************************************************************************** * Copyright (c) 2010 Qcadoo Limited * Project: Qcadoo MES * Version: 1.4 * * This file is part of Qcadoo. * * Qcadoo is free software; you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published * by the Free Software Foundation; either version 3 of the License, * or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * *************************************************************************** */ package com.qcadoo.mes.states.aop; import static junit.framework.Assert.assertTrue; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertNotNull; import static org.mockito.BDDMockito.given; import static org.mockito.Mockito.never; import static org.mockito.Mockito.verify; import java.lang.reflect.Method; import java.lang.reflect.Modifier; import java.util.List; import java.util.Map; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut; import org.junit.Assert; import org.junit.Before; import org.junit.Test; import org.mockito.Mockito; import org.mockito.MockitoAnnotations; import com.google.common.collect.Lists; import com.google.common.collect.Maps; import com.qcadoo.mes.states.MockStateChangeDescriber; import com.qcadoo.mes.states.StateChangeContext; import com.qcadoo.mes.states.StateChangeEntityDescriber; import com.qcadoo.mes.states.StateChangeTest; import com.qcadoo.mes.states.annotation.StateChangePhase; import com.qcadoo.mes.states.messages.constants.StateMessageType; import com.qcadoo.mes.states.service.StateChangeService; import com.qcadoo.model.api.Entity; import com.qcadoo.model.api.EntityList; public class StateChangePhaseAspectTest extends StateChangeTest { private static final String TOUCHED_FIELD = "touched"; private static final String TOUCHED_PHASE = "touchedPhase"; private TestStateChangeService stateChangeService; @Test public final void checkPointcutsDefinition() throws NoSuchMethodException { assertEquals("com.qcadoo.mes.states.annotation.StateChangePhase", StateChangePhase.class.getCanonicalName()); assertEquals("com.qcadoo.mes.states.StateChangeContext", StateChangeContext.class.getCanonicalName()); assertEquals("com.qcadoo.mes.states.service.StateChangeService", StateChangeService.class.getCanonicalName()); final Map<Integer, Class<?>> changeStateArgsMap = Maps.newHashMap(); changeStateArgsMap.put(0, StateChangeContext.class); final Method changeStateMethod = findMethod(StateChangeService.class, "changeState", changeStateArgsMap); assertNotNull(changeStateMethod); Assert.assertTrue(Modifier.isPublic(changeStateMethod.getModifiers())); assertEquals(void.class, changeStateMethod.getReturnType()); final Map<Integer, Class<?>> changeStatePhaseArgsMap = changeStateArgsMap; final Method changeStatePhaseMethod = findMethod(StateChangeService.class, "changeState", changeStatePhaseArgsMap); assertNotNull(changeStatePhaseMethod); Assert.assertTrue(Modifier.isPublic(changeStatePhaseMethod.getModifiers())); assertEquals(void.class, changeStatePhaseMethod.getReturnType()); } private Method findMethod(final Class<?> clazz, final String name, final Map<Integer, Class<?>> argsMap) { final List<Method> methods = Lists.newArrayList(clazz.getMethods()); methods.addAll(Lists.newArrayList(clazz.getDeclaredMethods())); for (Method method : methods) { if (method.getName().equals(name) && methodMatchArguments(method, argsMap)) { return method; } } Assert.fail("method not found"); return null; } private boolean methodMatchArguments(final Method method, final Map<Integer, Class<?>> argsMap) { final Class<?>[] arguments = method.getParameterTypes(); for (Map.Entry<Integer, Class<?>> argEntry : argsMap.entrySet()) { final int argIndex = argEntry.getKey(); final Class<?> argType = argEntry.getValue(); if (!argType.isAssignableFrom(arguments[argIndex])) { return false; } } return true; } @Aspect public static class TestStateChangeService extends AbstractStateChangeAspect { @Pointcut("this(TestStateChangeService)") public void selectorPointcut() { } @Override public void changeState(final StateChangeContext stateChangeContext) { final Entity stateChangeEntity = stateChangeContext.getStateChangeEntity(); stateChangeEntity.setField(TOUCHED_FIELD, true); } @Override public StateChangeEntityDescriber getChangeEntityDescriber() { return new MockStateChangeDescriber(); } @Override protected void changeStatePhase(final StateChangeContext stateChangeContext, final int phaseNumber) { final Entity stateChangeEntity = stateChangeContext.getStateChangeEntity(); stateChangeEntity.setField(TOUCHED_PHASE, phaseNumber); } } @Before public void init() { MockitoAnnotations.initMocks(this); stateChangeService = new TestStateChangeService(); stubStateChangeEntity(DESCRIBER); stubStateChangeContext(); } @Test public final void shouldExecutePhaseMethod() { // given List<Entity> messages = Lists.newArrayList(); messages.add(mockMessage(StateMessageType.SUCCESS, "test")); EntityList messagesEntityList = mockEntityList(messages); given(stateChangeEntity.getHasManyField(DESCRIBER.getMessagesFieldName())).willReturn(messagesEntityList); // when stateChangeService.changeState(stateChangeContext); // then verify(stateChangeEntity).setField(TOUCHED_FIELD, true); } @Test public final void shouldNotExecutePhaseMethod() { // given List<Entity> messages = Lists.newArrayList(); messages.add(mockMessage(StateMessageType.FAILURE, "test")); EntityList messagesEntityList = mockEntityList(messages); given(stateChangeEntity.getHasManyField(DESCRIBER.getMessagesFieldName())).willReturn(messagesEntityList); // when stateChangeService.changeState(stateChangeContext); // then verify(stateChangeEntity, never()).setField(TOUCHED_FIELD, true); } @Test public final void shouldThrowExceptionIfEntityDoesNotHaveMessagesField() { // given given(stateChangeEntity.getHasManyField(DESCRIBER.getMessagesFieldName())).willReturn(null); boolean exceptionWasThrown = false; // when try { stateChangeService.changeState(stateChangeContext); } catch (Exception e) { exceptionWasThrown = true; } // then assertTrue(exceptionWasThrown); } @Test public final void shouldNotExecutePhaseMethodIfValidationErrorOccured() { // given List<Entity> messages = Lists.newArrayList(); Entity validationErrorMessage = mockMessage(StateMessageType.VALIDATION_ERROR, "test"); messages.add(validationErrorMessage); EntityList messagesEntityList = mockEntityList(messages); given(stateChangeEntity.getHasManyField(DESCRIBER.getMessagesFieldName())).willReturn(messagesEntityList); // when stateChangeService.changeState(stateChangeContext); // then verify(stateChangeEntity, never()).setField(TOUCHED_FIELD, true); verify(stateChangeEntity, never()).setField(Mockito.eq(TOUCHED_PHASE), Mockito.anyInt()); } }