@WebMvcTest 创建名称为 'securityConfiguration' 的 bean 时出错
@WebMvcTest Error creating bean with name 'securityConfiguration'
我有一个 spring boot rest 应用程序,我在其中使用 JWT 进行身份验证,当我刚启动该应用程序时它工作正常。
我只是想添加一些测试用例,但我无法解决这个问题(我 运行 测试用例 mvn test
:
Error states that Spring can not create userDetailsService bean but this seems incorrect, because my application can run without any issue.
不是:我尝试通过@MockBean 添加 userdetailservice mock。但是这次 spring 对 MyAuthenticationEntryPoint 显示相同的错误。我认为应该有一个解决方案来自动添加所有配置,或者为这个端点避免所有配置。
[ERROR] verifyRegisterUserHttpRequest Time elapsed: 0.008 s <<< ERROR!
java.lang.IllegalStateException: Failed to load ApplicationContext
Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException:
Error creating bean with name 'securityConfiguration': Unsatisfied dependency expressed
through field 'userDetailsService'; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException:
No qualifying bean of type 'com.myProjects.sample.security.MyUserDetailsService' available: expected at least 1 bean which qualifies as autowire candidate.
Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}
- 这里是测试class:
package com.myProjects.sample.controller;
@WebMvcTest(controllers = RegisterController.class)
public class RegisterControllerTest {
@Autowired
private MockMvc mockMvc;
@Autowired
private ObjectMapper objectMapper;
@MockBean
private RegisterService registerService;
@MockBean
private ValidationService validationService;
// 1. Verifying Http Request Matching
@Test
public void verifyRegisterUserHttpRequest() throws Exception {
NewCustomerRequest requestBody = new NewCustomerRequest();
requestBody.setEmailAddress("sample@sample.com");
requestBody.setPhoneNumber("01112223344");
requestBody.setPassword("password");
requestBody.setPasswordVerify("password");
mockMvc.perform(post(MyWebUrls.API.API + MyWebUrls.Register.REGISTER_USER)
.content(objectMapper.writeValueAsString(requestBody))
.contentType("application/json"))
.andExpect(status().isOk());
}
}
- 注册控制器和安全配置
package com.myProjects.sample.controller;
@RestController
@RequestMapping(MyWebUrls.API.API)
public class RegisterController {
@Autowired
private RegisterService registerService;
@Autowired
private ValidationService validationService;
@PostMapping(MyWebUrls.Register.REGISTER_USER)
public ResponseEntity<?> registerUser(@Valid @RequestBody NewCustomerRequest newCustomerRequest, BindingResult bindingResult){
String validationResultMessage = validationService.getFirstValidationErrorMessage(bindingResult);
// ...
}
}
}
package com.myProjects.sample.security;
@Configuration
@EnableWebSecurity
@EnableGlobalMethodSecurity(
// securedEnabled = true,
// jsr250Enabled = true,
prePostEnabled = true)
public class SecurityConfiguration extends WebSecurityConfigurerAdapter {
@Autowired
private MyUserDetailsService userDetailsService;
@Autowired
private MyAuthenticationEntryPoint unauthorizedHandler;
@Autowired
private CustomAccessDeniedHandler accessDeniedHandler;
@Bean
public AuthTokenFilter authJwtTokenFilter(){
return new AuthTokenFilter();
}
@Override
public void configure(AuthenticationManagerBuilder authenticationManagerBuilder) throws Exception {
authenticationManagerBuilder.userDetailsService(userDetailsService).passwordEncoder(passwordEncoder());
}
// ....
- 还可以查看我的自定义详细信息服务class
package com.myProjects.sample.security;
@Service
public class MyUserDetailsService implements UserDetailsService {
@Override
@Transactional
public UserDetails loadUserByUsername(String emailOrMobilePhone) throws UsernameNotFoundException {
// ...
}
}
你的项目可以添加如下依赖吗?
<dependency>
<groupId>org.springframework.security</groupId>
<artifactId>spring-security-test</artifactId>
<scope>test</scope>
</dependency>
这应该确保还为您的@WebMvcTest
填充安全配置。
然后,当您使用 MockMvc
:
访问您的应用程序时,您可以使用多种技术来模拟 credentials/user
@Test
@WithMockUser("duke")
或
@Test
public void testWithSecurityMockMvcRequestPostProcessors() throws Exception {
this.mockMvc
.perform(
post("/your/endpoint")
.with(jwt())
.with(csrf())
)
.andExpect(status().isOk());
}
有关 MockMvc
和 Spring 安全集成的更多信息,请参见 here。
我有一个 spring boot rest 应用程序,我在其中使用 JWT 进行身份验证,当我刚启动该应用程序时它工作正常。
我只是想添加一些测试用例,但我无法解决这个问题(我 运行 测试用例 mvn test
:
Error states that Spring can not create userDetailsService bean but this seems incorrect, because my application can run without any issue.
不是:我尝试通过@MockBean 添加 userdetailservice mock。但是这次 spring 对 MyAuthenticationEntryPoint 显示相同的错误。我认为应该有一个解决方案来自动添加所有配置,或者为这个端点避免所有配置。
[ERROR] verifyRegisterUserHttpRequest Time elapsed: 0.008 s <<< ERROR!
java.lang.IllegalStateException: Failed to load ApplicationContext
Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException:
Error creating bean with name 'securityConfiguration': Unsatisfied dependency expressed
through field 'userDetailsService'; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException:
No qualifying bean of type 'com.myProjects.sample.security.MyUserDetailsService' available: expected at least 1 bean which qualifies as autowire candidate.
Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}
- 这里是测试class:
package com.myProjects.sample.controller;
@WebMvcTest(controllers = RegisterController.class)
public class RegisterControllerTest {
@Autowired
private MockMvc mockMvc;
@Autowired
private ObjectMapper objectMapper;
@MockBean
private RegisterService registerService;
@MockBean
private ValidationService validationService;
// 1. Verifying Http Request Matching
@Test
public void verifyRegisterUserHttpRequest() throws Exception {
NewCustomerRequest requestBody = new NewCustomerRequest();
requestBody.setEmailAddress("sample@sample.com");
requestBody.setPhoneNumber("01112223344");
requestBody.setPassword("password");
requestBody.setPasswordVerify("password");
mockMvc.perform(post(MyWebUrls.API.API + MyWebUrls.Register.REGISTER_USER)
.content(objectMapper.writeValueAsString(requestBody))
.contentType("application/json"))
.andExpect(status().isOk());
}
}
- 注册控制器和安全配置
package com.myProjects.sample.controller;
@RestController
@RequestMapping(MyWebUrls.API.API)
public class RegisterController {
@Autowired
private RegisterService registerService;
@Autowired
private ValidationService validationService;
@PostMapping(MyWebUrls.Register.REGISTER_USER)
public ResponseEntity<?> registerUser(@Valid @RequestBody NewCustomerRequest newCustomerRequest, BindingResult bindingResult){
String validationResultMessage = validationService.getFirstValidationErrorMessage(bindingResult);
// ...
}
}
}
package com.myProjects.sample.security;
@Configuration
@EnableWebSecurity
@EnableGlobalMethodSecurity(
// securedEnabled = true,
// jsr250Enabled = true,
prePostEnabled = true)
public class SecurityConfiguration extends WebSecurityConfigurerAdapter {
@Autowired
private MyUserDetailsService userDetailsService;
@Autowired
private MyAuthenticationEntryPoint unauthorizedHandler;
@Autowired
private CustomAccessDeniedHandler accessDeniedHandler;
@Bean
public AuthTokenFilter authJwtTokenFilter(){
return new AuthTokenFilter();
}
@Override
public void configure(AuthenticationManagerBuilder authenticationManagerBuilder) throws Exception {
authenticationManagerBuilder.userDetailsService(userDetailsService).passwordEncoder(passwordEncoder());
}
// ....
- 还可以查看我的自定义详细信息服务class
package com.myProjects.sample.security;
@Service
public class MyUserDetailsService implements UserDetailsService {
@Override
@Transactional
public UserDetails loadUserByUsername(String emailOrMobilePhone) throws UsernameNotFoundException {
// ...
}
}
你的项目可以添加如下依赖吗?
<dependency>
<groupId>org.springframework.security</groupId>
<artifactId>spring-security-test</artifactId>
<scope>test</scope>
</dependency>
这应该确保还为您的@WebMvcTest
填充安全配置。
然后,当您使用 MockMvc
:
@Test
@WithMockUser("duke")
或
@Test
public void testWithSecurityMockMvcRequestPostProcessors() throws Exception {
this.mockMvc
.perform(
post("/your/endpoint")
.with(jwt())
.with(csrf())
)
.andExpect(status().isOk());
}
有关 MockMvc
和 Spring 安全集成的更多信息,请参见 here。