我正在使用Spring进行MVC测试
这是我的测试课
@RunWith(SpringRunner.class)
@WebMvcTest
public class ITIndexController {
@Autowired
WebApplicationContext context;
MockMvc mockMvc;
@MockBean
UserRegistrationApplicationService userRegistrationApplicationService;
@Before
public void setUp() {
this.mockMvc = MockMvcBuilders
.webAppContextSetup(context)
.apply(springSecurity())
.build();
}
@Test
public void should_render_index() throws Exception {
mockMvc.perform(get("/"))
.andExpect(status().isOk())
.andExpect(view().name("index"))
.andExpect(content().string(containsString("Login")));
}
}
下面是MVC配置
@Configuration
@EnableWebMvc
public class MvcConfig extends WebMvcConfigurerAdapter {
@Override
public void addViewControllers(ViewControllerRegistry registry) {
registry.addViewController("/").setViewName("index");
registry.addViewController("/login/form").setViewName("login");
}
}
这里是安全配置
@Configuration
@EnableWebSecurity
public class SecurityConfig extends WebSecurityConfigurerAdapter {
@Autowired
@Qualifier("customUserDetailsService")
UserDetailsService userDetailsService;
@Override
protected void configure(HttpSecurity http) throws Exception {
http.authorizeRequests()
.antMatchers("/resources/**", "/signup", "/signup/form", "/").permitAll()
.anyRequest().authenticated()
.and()
.formLogin()
.loginPage("/login/form").permitAll().loginProcessingUrl("/login").permitAll()
.and()
.logout().logoutSuccessUrl("/login/form?logout").permitAll()
.and()
.csrf().disable();
}
@Autowired
public void configureGlobalFromDatabase(AuthenticationManagerBuilder auth) throws Exception {
auth.userDetailsService(userDetailsService);
}
}
当我运行我的测试时,它失败并显示消息:
java.lang.AssertionError: Status expected:<200> but was:<401>
at org.springframework.test.util.AssertionErrors.fail(AssertionErrors.java:54)
at org.springframework.test.util.AssertionErrors.assertEquals(AssertionErrors.java:81)
at org.springframework.test.web.servlet.result.StatusResultMatchers$10.match(StatusResultMatchers.java:664)
at org.springframework.test.web.servlet.MockMvc$1.andExpect(MockMvc.java:171)
at com.marco.nutri.integration.web.controller.ITIndexController.should_render_index(ITIndexController.java:46)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:75)
at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:86)
at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:84)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:252)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:94)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:191)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:678)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)
我知道它失败是因为URL受到Spring Security 的保护,但是当我运行我的应用程序时,即使没有经过身份验证,我也可以访问该URL。
我做错什么了吗?
6条答案
按热度按时间3j86kqsm1#
我找到了答案
Spring Docs说:
@WebMvcTest将自动配置Spring MVC基础设施,并将扫描的bean限制为@Controller,@ControllerAdvice,@JsonComponent,Filter,WebMvcConfigurer和HandlerMethodArgumentResolver。使用此注解时,将不会扫描常规@Component bean。
根据Github上的这个问题:
https://github.com/spring-projects/spring-boot/issues/5476
如果spring-security-test存在于class path中(在我的例子中是),@WebMvcTest默认自动配置Spring Security 。
因此,由于WebSecurityConfigurer类没有被选中,默认的安全性是自动配置的,这就是我在URL中收到401的动机,它在我的安全配置中没有得到保护。Spring security default auto configuration通过基本身份验证保护所有url。
为了解决这个问题,我用@ContextConfiguration和@MockBean注解了这个类,就像文档中描述的那样:
通常,@WebMvcTest将被限制为单个控制器,并与@MockBean结合使用,为所需的协作者提供模拟实现。
这是测试类
Application、MvcConfig和SecurityConfig都是我的配置类
2lpgd9682#
不确定当最初的问题被问到时这是否可用,但如果真的不想测试Web请求的安全部分,(如果已知端点不安全,这似乎是合理的),那么我认为这可以简单地通过使用
@WebMvcTest
注解的secure
属性来完成(它默认为true
,因此将其设置为false
应该会禁用Spring Security的MockMvc支持的自动配置):javadocs中提供的更多信息
jrcvhitl3#
我也有同样的问题,并在这里的答案和@Sam Brannen评论的帮助下解决了这个问题。
您可能不需要使用@ContextConfiguration。简单地添加@Import(SecurityConfig.class)通常就足够了。
为了简化和更新答案,我想分享我如何在我的spring-boot 2项目中修复它。
我想在终点以下进行测试。
所有服务方法必须是接口,否则应用程序上下文无法初始化。你可以查看我的安全配置。
下面你可以看到我的SystemOptionControllerTest类。
所以我只需要将
@Import(SecurityConfig.class)
添加到mvc测试类中。vuktfyat4#
如果使用spring Boot 2.0+,请尝试此操作
@WebMvcTest(controllers = TestController.class,* excludeAutoConfiguration = {SecurityAutoConfiguration. class}*)
pwuypxnk5#
如果你使用SpringJUnit4ClassRunner而不是SpringRunner,你可以在安全层捕获你的请求。如果您使用的是基本身份验证,则必须在mockMvc中使用httpBasic方法。
ux6nzvsh6#
你可以添加这个.header(“apikey”,“apiValue”);
如果你在资源中声明了这个值,那么只有你可以添加它。