在单元测试运行期间禁用某些方面

时间:2011-09-15 15:45:24

标签: java unit-testing spring aspectj

我有集成测试(加载上下文)和单元测试一起运行。我的代码使用spring进行编译时编织。

我的问题是,在我的一些单元测试中,我声明的建议也会运行。这会破坏单元测试的概念,这就是我想禁用它们的原因。

我是否可以在切入点声明上添加一些内容,我可以调用一些方法,一些弹簧配置或maven命令禁用这些建议,例如所有* UnitTest.java?

感谢您的帮助。


示例:

我有以下单元测试:

@RunWith(MockitoJUnitRunner.class)
public class CompanyServiceImplTest {
    @Test
    public void createCampaignTest() throws Exception {
        when(companyDaoMock.saveCompany(any(Campaign.class))).thenReturn(77L);

        Long campaignId = companyService.createCampaign(campaignMock);

        assertEquals(Long.valueOf(77L), Long.valueOf(campaignId));
    }
}

并遵循服务方法:

@Override
@Transactional
@EventJournal(type = EventType.CAMPAIGN_CREATE, owner = EventOwner.TERMINAL_USER)
public Long createCampaign(Campaign campaign) {
    return companyDao.saveCompany(campaign);
}

方面:

@Aspect
public class EventJournalAspect {

    @Autowired
    private EventJournalService eventJournalService;

    @Pointcut(value="execution(public * *(..))")
    public void anyPublicMethod() {}

    @Pointcut("within(com.terminal.service..*)")
    private void inService() {}

    @AfterReturning(pointcut = "anyPublicMethod() && inService() && @annotation(eventJournal) && args(entity,..)", returning = "id")
    public void process(Object id, EventJournal eventJournal, AbstractDomainEntity entity)
            throws Throwable {
        if (eventJournal.type() != EventType.CAMPAIGN_PAYMENT || id != null) {
            saveEvent(eventJournal, EventStatus.SUCCESS, entity, (Long) id);
        }
    }

    @AfterThrowing(pointcut = "anyPublicMethod() && inService() && @annotation(eventJournal) && args(entity,..)", throwing="ex")
    public void processException(EventJournal eventJournal, AbstractDomainEntity entity, Exception ex) throws Throwable {
        saveEvent(eventJournal, EventStatus.FAILURE, entity, null);
    }

    private void saveEvent(EventJournal eventJournal, EventStatus status, AbstractDomainEntity entity, Long persistentId)   {
        EventType type = eventJournal.type();
        EventOwner owner = eventJournal.owner();
        eventJournalService.saveEvent(type, owner, EventStatus.SUCCESS, entity, persistentId);
    }

}

执行测试时 - eventJournalService为空。因此,我看到NullPointerException

4 个答案:

答案 0 :(得分:6)

答案很简单:您想使用if() pointcut expression


更新(问题也已更新后):上面提供的原始链接应该包含足够的信息,但是对于它的价值,一个简短的解释和一个简单的例子:

if()切入点是static方面方法,返回boolean。如果返回值为true,则表示只要myPointcut() && if()匹配,任何组合切入点(如myPointcut())都会匹配。对于返回值false,整个组合切入点不匹配,从而有效地停用与切入点相关的任何建议。

那么你可以在静态if()切入点中做什么?

  • 评估某些工具类的静态布尔成员,例如TestMode.ACTIVE,仅在单元测试或集成测试期间为真
  • 评估仅在测试期间设置的环境变量
  • 评估仅在测试期间设置的Java系统属性
  • 还有很多东西

如果你想做更好的(并且更棘手)并且性能不是那么重要,你也可以尝试动态确定自动连线方面成员变量是否等于null,并且只有在注入的对象是的时才激活你的切入点实际存在。这里唯一的问题是如何从静态方法确定成员变量。我不知道Spring AOP,但在简单的AspectJ中,有一个辅助类Aspects,其中包含几个名为aspectOf(..)的重载方法。假设您的方面被实例化为单例,您可以执行以下操作:

@Pointcut("if()")
public static boolean isActive() {
    return Aspects.aspectOf(PerformanceMonitorAspect.class).eventJournalService != null;
}

// ...

@AfterReturning(pointcut = "isActive() && anyPublicMethod() && inService() && @annotation(eventJournal) && args(entity,..)", returning = "id")
// ...

@AfterThrowing(pointcut = "isActive() && anyPublicMethod() && inService() && @annotation(eventJournal) && args(entity,..)", throwing="ex")
// ...

答案 1 :(得分:0)

我只能猜测: 首先要有一个单独的Spring applicationContext-test.xml, 没有组件扫描; 在maven中,您可以添加一个阶段运行时,不包括用于测试的编织罐。

答案 2 :(得分:0)

编译时编织将在您拥有的切入点所标识的目标方法中内联建议调用。我个人觉得在编译时编织时进行单元测试是好的,因为在运行时你的单元确实包含带有内联建议的类?

我不得不包括建议的想法是有两个不同的编译目标,一个编译时编织,一个没有,你应该能够通过maven配置文件,一个dev配置文件,不编织建议和prod profile to weave aspects in。

答案 3 :(得分:0)

如果使用JUnit框架启动当前执行,您可以编写一个返回的方法。

该方法可以使用Thread.currentThread()。getStackTrace()检查堆栈跟踪并搜索MockitoJUnitRunner存在。

我使用SpringJUnit4ClassRunner测试了这个解决方案,但我认为可以使用MockitoJUnitRunner。

此外,您可以获得一个静态布尔字段,如:

private static boolean TEST_ENVIRONMENT = false;

在项目中存在的类中(不在测试中)并检查控制方法中的值而不是使用堆栈跟踪。

运行测试时,可以使用@BeforeClass注释设置TEST_ENVIRONMENT = true。

此解决方案仅为您提供了一种了解代码是否从测试运行的方法。