如果登录成功,Spring Boot @WebMvcTest 返回 404

seb*_*ira 1 java testing spring mocking spring-boot

我正在学习如何测试我的 SpringBoot 应用程序。

现在我正在尝试通过为现有工作项目创建测试来学​​习。

我从管理员登录时管理主页的AdminHomeController开始:

@Controller
@RequestMapping("/admin/home")
public class AdminHomeController {

private UsuarioService usuarioService;

@Autowired
public AdminHomeController(UsuarioService usuarioService) {
    this.usuarioService = usuarioService;
}

@RequestMapping(value={"", "/"}, method = RequestMethod.GET)
public ModelAndView admin_home(){
    ModelAndView modelAndView = new ModelAndView();

    Authentication auth = SecurityContextHolder.getContext().getAuthentication();
    Usuario loggedUser = usuarioService.findUsuarioByUsername(auth.getName());
    modelAndView.addObject("userFullName", loggedUser.getNombre() + " " + loggedUser.getApellido());
    modelAndView.addObject("userGravatar", Utils.getGravatarImageLink(loggedUser.getEmail()));

    modelAndView.addObject("totalUsuarios", usuarioService.getUsuariosCount());


    modelAndView.setViewName("admin/home");
    return modelAndView;
}
}
Run Code Online (Sandbox Code Playgroud)

这是我的测试:

@RunWith(SpringRunner.class)
@ContextConfiguration(classes = MyOwnProperties.class)
@WebMvcTest(AdminHomeController.class)
@Import(SecurityConfigurationGlobal.class)
public class AdminHomeControllerUnitTest {

@Autowired
private MockMvc mockMvc;

@MockBean
UsuarioService usuarioService;

@Autowired
MyOwnProperties myOwnProperties;

@MockBean
FacebookProfileService facebookProfileService;

@MockBean
MobileDeviceService mobileDeviceService;

@MockBean
PasswordEncoder passwordEncoder;

@MockBean
CustomAuthenticationProvider customAuthenticationProvider;


@Test
@WithMockUser(username = "user1", password = "pwd", authorities = "ADMIN")
public void shouldAllowAdminAccess() throws Exception{
    when(usuarioService.findUsuarioByUsername("user1")).thenReturn(new Usuario());


    mockMvc.perform(get("/admin/home"))
            .andDo(print())
            .andExpect(status().isOk())
            .andExpect(view().name("admin/home"));
}

}
Run Code Online (Sandbox Code Playgroud)

我认为我的 SecurityConfig 的相关部分是:

@Override
protected void configure(HttpSecurity http) throws Exception {
    http.
            authorizeRequests()
            .antMatchers("/", "/login", "/error/**", "/home").permitAll()
            .antMatchers(
                    myOwnProperties.getSecurity().getJwtLoginURL(),
                    myOwnProperties.getSecurity().getFacebookLoginURL()).permitAll()
            .antMatchers("/registration", "/registrationConfirm/**").permitAll()
            .antMatchers("/resetPass", "/resetPassConfirm/**", "/updatePass").permitAll()
            .antMatchers("/admin/**").hasAuthority(AUTHORITY_ADMIN)
            .antMatchers("/user/**").hasAuthority(AUTHORITY_USER)
            .anyRequest().authenticated()
            .and()
            .csrf().disable()
            .formLogin()
            .loginPage("/login")
            .failureUrl("/login?error=true")
            .successHandler(new CustomUrlAuthenticationSuccessHandler())
            .usernameParameter("username")
            .passwordParameter("password")
            .and()
            .logout()
            .logoutUrl("/logout")
            .logoutSuccessUrl("/")
            .and()
            .exceptionHandling().accessDeniedPage("/403");
}
Run Code Online (Sandbox Code Playgroud)

AUTHORITY_ADMIN是“ADMIN”的静态最终定义。

由于缺乏经验,我无法理解的是我的测试结果。

  • 如果我删除@WithMockUser,我会按预期得到 401
  • 如果我将@WithMockUser与“ADMIN”以外的任何其他权限一起使用,我会得到一个 403,这也是预期的响应
  • 最后,如果我使用具有“ADMIN”权限的@WithMockUser,则会得到 404

如上所述,我的应用程序正在运行,如果以管理员身份登录,我只能访问 /admin/home。

更新

运行另一个类似的测试工作正常,但这个需要加载完整的 SpringBoot 应用程序。我认为这将是一个集成测试,我只想“单独”测试控制器。仅使用 @WebMvcTest 的切片

@SpringBootTest
@AutoConfigureMockMvc
public class AdminHomeControllerTest {

@Autowired
private MockMvc mockMvc;


@MockBean
private UsuarioService usuarioService;

@Test
@WithMockUser(username = "user1", password = "pwd", authorities = "ADMIN")
public void shouldAllowAdminAccess() throws Exception{
    when(usuarioService.findUsuarioByUsername(anyString())).thenReturn(new Usuario());


    mockMvc.perform(get("/admin/home"))
            .andDo(print())
            .andExpect(status().isOk())
            .andExpect(view().name("admin/home"));
}
}
Run Code Online (Sandbox Code Playgroud)

更新 2

我使它通过改变传递@ContextConfiguration(类= MyOwnProperties.class)用于@Import

所以现在我的测试看起来像:

@RunWith(SpringRunner.class)
@WebMvcTest(AdminHomeController.class)
@Import({SecurityConfigurationGlobal.class, MyOwnProperties.class})
public class AdminHomeControllerUnitTest { 
....... Same as before
}
Run Code Online (Sandbox Code Playgroud)

我很高兴,因为测试通过了,但有人可以解释我为什么吗?我在其他 SO 帖子中阅读,要使用我自己的用 @ConfigurationProperties 注释的自定义属性文件,我需要使用 @ContextConfiguration 注释。

seb*_*ira 7

我的问题的解决方案是,以取代@ContextConfiguration(类= MyOwnProperties.class)@Import

所以它会变成:

@RunWith(SpringRunner.class)
@WebMvcTest(AdminHomeController.class)
@Import({SecurityConfigurationGlobal.class, MyOwnProperties.class})
public class AdminHomeControllerUnitTest { 
     ....... Same as before
}
Run Code Online (Sandbox Code Playgroud)

更新 SpringBoot 2.x

我现在已经将我的代码库迁移到了Spring Boot 2.4.1,这个测试又开始失败了。试错后,现在@Import需要被替换@ContextConfiguration