AspectJ建议没有通过单元测试执行

时间:2013-06-06 20:43:10

标签: java spring maven aspectj

我很难过。我正在尝试测试一个AspectJ类。当我运行我的应用程序时,我的Aspect类被完美地拾取。但是,我似乎无法让任何Aspect类拦截测试中的任何方法。

我正在使用Spring 3.2.2,AspectJ 1.7.2和Maven 4.

以下是我正在使用的简单测试:

测试AspectJ类

package my.package.path.config;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Aspect
public class TestAOP {
    private String message;

    public TestAOP() {
    }

    @Pointcut("execution(* my.package.path.TestAOPClient.relayMessage(..))")
    public void aopPointcut() {
    }

    @Around("aopPointcut()")
    public String monitor(ProceedingJoinPoint pjp) throws Throwable {
        String msg = (String)pjp.proceed();
        this.setMessage(msg);
        return msg;
    }

}

正在拦截其方法的类

package my.package.path.config;

public class TestAOPClient {
    public String relayMessage(String msg) {
        return msg;
    }
}

测试类

package my.package.path.config;

import static org.hamcrest.Matchers.equalTo;
import static org.junit.Assert.assertThat;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.AbstractJUnit4SpringContextTests;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.web.WebAppConfiguration;

@Configuration
@ContextConfiguration(classes={WebConfig.class})
@RunWith(SpringJUnit4ClassRunner.class)
@WebAppConfiguration("src/main/java")
public class AopConfigTest extends AbstractJUnit4SpringContextTests {

     @Bean
     public TestAOP testAop() throws Exception {
        return new TestAOP();
     }

     @Test
     public void assertTestConfigIsActive() {
        TestAOPClient client = new TestAOPClient();
        client.relayMessage("hello");
        assertThat(((TestAOP)applicationContext.getBean("testAop")).getMessage(), equalTo("hello"));

    }
}

WebConfig文件

package my.package.path.web.context;
@Configuration
@EnableWebMvc
@EnableAspectJAutoProxy(proxyTargetClass=false)
@ComponentScan(value={"my.package.path.config", "my.package.path.web"})
public class WebConfig {

}

总是,我会得到断言错误

 Expected: "hello" but: was null

我的WebApplicationContext似乎被选中了,因为在运行时,如果我指定一个对于我的Aspect切入点不存在的类,我将得到一个ApplicationContext无法加载错误。

我错过了什么?

2 个答案:

答案 0 :(得分:5)

您使用单位测试作为@Configuration来源有点奇怪。

您应该从单元测试中删除@Configuration注释,并将testAOP() bean定义移至WebConfig。但最重要的是,建议使用的bean 不能手动创建,而是由Spring 创建:

@ContextConfiguration(classes={WebConfig.class})
@WebAppConfiguration("src/main/java")
public class AopConfigTest extends AbstractJUnit4SpringContextTests {

     @Autowired
     private TestAOP testAop;

     @Autowired
     private TestAOPClient client;

     @Test
     public void assertTestConfigIsActive() {
        client.relayMessage("hello");
        assertThat(((TestAOP)applicationContext.getBean("testAop")).getMessage(), 
            equalTo("hello"));
    }

}

使用bean定义更新了配置:

@Configuration
@EnableWebMvc
@EnableAspectJAutoProxy(proxyTargetClass=false)
@ComponentScan(value={"my.package.path.config", "my.package.path.web"})
public class WebConfig {

    @Bean
    public TestAOP testAop() throws Exception {
        return new TestAOP();
    }

    @Bean
    public TestAOPClient testAopClient() throws Exception {
        return new TestAOPClient();
    }

}

如果您的目标是测试AOP配置是否有效且TestAOP是否真的是测试bean(不仅仅是此问题的虚拟名称),您可以创建一个特殊的TestConfig配置类,移动那里的bean定义,并从测试@ContextConfiguration(classes={WebConfig.class,TestConfig.class})中使用它。

答案 1 :(得分:1)

您的切入点配置错误(包应该是my.package.path.config以匹配您的测试客户端)

@Pointcut("execution(* my.package.path.TestAOPClient.relayMessage(..))")

您的客户

package my.package.path.config;

public class TestAOPClient {
...

将其更改为此

@Pointcut("execution(* my.package.path.config.TestAOPClient.relayMessage(..))")

应该有用。