我有集成测试(加载上下文)和单元测试一起运行。我的代码使用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
答案 0 :(得分:6)
答案很简单:您想使用if()
pointcut expression。
更新(问题也已更新后):上面提供的原始链接应该包含足够的信息,但是对于它的价值,一个简短的解释和一个简单的例子:
if()
切入点是static
方面方法,返回boolean
。如果返回值为true
,则表示只要myPointcut() && if()
匹配,任何组合切入点(如myPointcut()
)都会匹配。对于返回值false
,整个组合切入点不匹配,从而有效地停用与切入点相关的任何建议。
那么你可以在静态if()
切入点中做什么?
TestMode.ACTIVE
,仅在单元测试或集成测试期间为真如果你想做更好的(并且更棘手)并且性能不是那么重要,你也可以尝试动态确定自动连线方面成员变量是否等于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。
此解决方案仅为您提供了一种了解代码是否从测试运行的方法。