如何修复测试失败"没有找到ModelAndView"?

时间:2017-03-26 20:04:13

标签: java unit-testing spring-mvc spring-boot applicationcontext

此类位于我的测试层次结构的顶部:

@TestPropertySource("/test.properties")
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest
public abstract class ApplicationAbstractTest {
}

还有更多的测试类:

@WebAppConfiguration
@ActiveProfiles("mysql")
abstract public class AbstractControllerTest extends ApplicationAbstractTest {

    protected MockMvc mockMvc;

    @Autowired
    private WebApplicationContext webApplicationContext;

    @PostConstruct
    private void postConstruct() {
        mockMvc = MockMvcBuilders
                .webAppContextSetup(webApplicationContext)
                .apply(springSecurity())
                .build();
    }
}

JsonUserServiceTest:

@ActiveProfiles("json")
public class JsonUserServiceTest extends ApplicationAbstractTest {
    @Before
    public void setUp() throws Exception {
        ...
    }
}

ContactControllerTest:

public class ContactControllerTest extends AbstractControllerTest {
    @Test
    public void testGet() throws Exception {
        mockMvc.perform(get("/update-" + ID + "-contact")
                .with(userAuth(USER)))
//                .andExpect(status().isOk())
                .andDo(print())
                .andExpect(view().name("details"))
                .andExpect(forwardedUrl("/WEB-INF/jsp/details.jsp"));
    }
}

所以,当我沿着ContactControllerTest运行时 - 它是成功的,而print方法告诉我:

Handler:
             Type = com.telecom.web.ContactController
           Method = public java.lang.String com.myApp.web.ContactController.details(java.lang.Integer,org.springframework.ui.ModelMap)

但是当我运行所有测试时,JsonUserServiceTest首先运行,ContactControllerTest失败。 print显示:

Handler:
             Type = null
...
java.lang.AssertionError: No ModelAndView found

配置有什么问题?或者如何排除故障?

UPD: 与此同时,这样的测试总是很好:

public class UserControllerTest extends AbstractControllerTest {
    @Test
    public void testRegister() throws Exception {
        mockMvc.perform(get("/register"))
                .andDo(print())
                .andExpect(view().name("profile"))
                .andExpect(forwardedUrl("/WEB-INF/jsp/profile.jsp"));
    }
}

UPD: 有控制器的方法我测试:

@GetMapping("/update-{id}-contact")
public String details(@PathVariable Integer id, ModelMap model) {
    Integer userId = AuthorizedUser.id();
    LOG.info("get contact {} for User {}", id, userId);
    Contact contact = service.get(id, userId);
    model.addAttribute("contact", contact);
    return "details";
}

我也有这样的豆子:

@Bean
public InternalResourceViewResolver viewResolver() {
    InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
    viewResolver.setViewClass(JstlView.class);
    viewResolver.setPrefix("/WEB-INF/jsp/");
    viewResolver.setSuffix(".jsp");

    return viewResolver;
}

UPD:我已尝试在单独的类中配置mockMvc:

@Configuration
public class TestConfig {

    @Autowired
    private WebApplicationContext webApplicationContext;

    @Bean
    public MockMvc mockMvc() {
        return MockMvcBuilders
                .webAppContextSetup(webApplicationContext)
                .apply(springSecurity())
                .build();
    }
}

并在此处添加:

@WebAppConfiguration
@ContextConfiguration(classes = {TestConfig.class})
@ActiveProfiles("mysql")
abstract public class AbstractControllerTest extends ApplicationAbstractTest {

但我收到了:

  

java.lang.IllegalStateException:springSecurityFilterChain不能   空值。确保名为springSecurityFilterChain的Bean   实现过滤器存在或注入要使用的过滤器。

2 个答案:

答案 0 :(得分:3)

WARN消息不会导致测试用例失败。它只是说实体经理工厂注册了两次。如果您使用相同的Entity Manager Factory对应用程序进行集群,则这只是一个问题。对于测试用例运行,它不是一个值得关注的原因。

测试用例失败的根本原因在于这两行

 .andExpect(view().name("details"))
 .andExpect(forwardedUrl("/WEB-INF/jsp/details.jsp"));

请检查项目是否有名为“details”的视图,并且前面的网址是“/WEB-INF/jsp/details.jsp”

<强>更新

请你试试这个

@Configuration
public class TestConfig {


@Autowired
private Filter springSecurityFilterChain;

@Autowired
private WebApplicationContext webApplicationContext;

@Bean
public MockMvc mockMvc() {
    return MockMvcBuilders
            .webAppContextSetup(webApplicationContext)
            .apply(springSecurityFilterChain)
            .build();
}
}

答案 1 :(得分:3)

创建一个配置文件,用于初始化测试用例的模拟对象。并放在所有测试用例类中。

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = {TestConfig.class})

它只会初始化所有模拟对象一次,然后缓存,并重用于所有测试用例。

  

或者如果您不想使用模拟配置,则可以直接使用   将实际的应用程序配置传递给ContextConfiguration as   以下

对于基于注释的应用程序配置(此处AppConfig和AppConfig2是您的配置类)

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = {AppConfig.class, AppConfig2.class})

对于基于xml的应用程序配置(此处appConfig.xml和appConfig2.xml是您的配置文件)

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:pathTo/appConfig.xml","classpath:pathTo/appConfig2.xml"})

参考:JUnit + Spring integration example