java - Spring Security @PreAuthorize-ProviderNotFoundException

标签 java spring spring-mvc authentication spring-security

我有 Controller :

@RestController
public class CheckPermissionContr {

  @PreAuthorize("hasRole('ROLE_USER')")
  @RequestMapping(value = "/permission", method = RequestMethod.GET)
  public String checkPermission(@RequestParam("id") UUID id) {

当我从单元测试中调用它时,例如:

    @RunWith(SpringJUnit4ClassRunner.class)
    @WebAppConfiguration
    @ContextConfiguration("file:src/main/webapp/WEB-INF/mvc-dispatcher-servlet.xml")
    public class CheckPermissionContrTest {

    @SuppressWarnings("SpringJavaAutowiringInspection")
    @Autowired
    protected WebApplicationContext wac;

    private MockMvc mockMvc;


    @Before
    public void init() {

        MockitoAnnotations.initMocks(this);
        BasicConfigurator.configure();

        this.mockMvc = webAppContextSetup(wac).build();

        final GeneralAuthentication authen = mock(GeneralAuthentication.class);
        user = mock(User.class);

        when(user.getId()).thenReturn(UUID.randomUUID());

        final GrantedAuthority mockAuth = mock(GrantedAuthority.class);
        doReturn("ROLE_USER").when(mockAuth).getAuthority();
        Collection<GrantedAuthority> authority = Collections.singleton(mockAuth);

        doReturn(authority).when(authen).getAuthorities();
        when(authen.getPrincipal()).thenReturn(user);



        final SecurityContext secContext = mock(SecurityContext.class);
        when(secContext.getAuthentication()).thenReturn(authen);
        SecurityContextHolder.setContext(secContext);

        }

    @Test
    public void testCheckPermission() throws Exception {
        mockMvc.perform(get("/permission.json").param("id", id.toString())).andExpect(status().isOk()).andDo(print());

当我运行测试时:

org.springframework.web.util.NestedServletException: Request processing failed; nested exception is org.springframework.security.authentication.ProviderNotFoundException: No AuthenticationProvider found for com.XXXXX.GeneralAuthentication$$EnhancerByMockitoWithCGLIB$$6db1e4cf
    at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:973)
    at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:852)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:734)
    at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:837)
    at org.springframework.test.web.servlet.TestDispatcherServlet.service(TestDispatcherServlet.java:62)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:847)
    at org.springframework.mock.web.MockFilterChain$ServletFilterProxy.doFilter(MockFilterChain.java:170)
    at org.springframework.mock.web.MockFilterChain.doFilter(MockFilterChain.java:137)
    at org.springframework.test.web.servlet.MockMvc.perform(MockMvc.java:141)
    at com.XXXXX.CheckPermissionContrTest.testCheckPermission(CheckPermissionContrTest.java:121)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
    at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:74)
    at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:83)
    at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:72)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:232)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:89)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
    at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:71)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:175)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:157)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:74)
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:211)
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:67)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:120)
Caused by: org.springframework.security.authentication.ProviderNotFoundException: No AuthenticationProvider found for com.XXXXX.GeneralAuthentication$$EnhancerByMockitoWithCGLIB$$6db1e4cf
    at org.springframework.security.authentication.ProviderManager.authenticate(ProviderManager.java:199)
    at org.springframework.security.config.method.GlobalMethodSecurityBeanDefinitionParser$AuthenticationManagerDelegator.authenticate(GlobalMethodSecurityBeanDefinitionParser.java:433)
    at org.springframework.security.access.intercept.AbstractSecurityInterceptor.authenticateIfRequired(AbstractSecurityInterceptor.java:316)
    at org.springframework.security.access.intercept.AbstractSecurityInterceptor.beforeInvocation(AbstractSecurityInterceptor.java:202)
    at org.springframework.security.access.intercept.aopalliance.MethodSecurityInterceptor.invoke(MethodSecurityInterceptor.java:60)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)
    at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:644)
    at com.XXXXX.CheckPermissionContr$$EnhancerBySpringCGLIB$$f974fa1c.checkPermission(<generated>)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:215)
    at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:132)
    at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHand

但是当我用 @Secured("ROLE_USER") 替换 @PreAuthorize 时,一切都会顺利进行。

.xml包含:

WEB-INF/mvc-dispatcher-servlet.xml

      <import resource="spring-security.xml"/>

内部spring-security.xml

<security:global-method-security pre-post-annotations="enabled"/>

<security:http use-expressions="true" auto-config="false" entry-point-ref="restEntryPoint">
    <security:intercept-url pattern="/*"/>
</security:http>

<bean id="auhProvider" class="com.XXXXX.AuhProvider">
    <constructor-arg name="userRepo" ref="userRepo"/>

</bean>

<security:authentication-manager alias="manager">
    <security:authentication-provider ref="auhProvider">

    </security:authentication-provider>
</security:authentication-manager>

附加代码:

  @Override
    public boolean supports(Class<?> aClass) {

        return aClass.isAssignableFrom(GeneralAuthentication.class);
    }

最佳答案

com.XXXXX.AuhProvider 仅支持 GeneralAuthentication.class 类型的对象。由于您模拟 authen 它不再具有此类,而是 com.XXXXX.GeneralAuthentication$$EnhancerByMockitoWithCGLIB$$6db1e4cf

support方法中的测试必须是相反的:

GeneralAuthentication.class.isAssignableFrom(aClass)

这意味着“是一个Class GeneralAuthentication.class或其子类型”。

关于java - Spring Security @PreAuthorize-ProviderNotFoundException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23084029/

相关文章:

java - 读取Jar文件中包含的类文件

javascript - Css 样式根据值文本更改

java - 将maven项目导入到eclipse中

java - Windows Server 2012、Apache Tomcat、Spring MVC : Websocket connection blocked for external IP

java - CDI。如何检查bean是否实例化?

java - 如何在具有 I/O 多路复用的 Java 服务器中异步处理请求?

spring - HTTP 状态 500 - 处理程序处理失败;嵌套异常是 java.lang.NoSuchMethodError :

Spring Data查询执行优化: Parallel Execution of Hibernate @Query Method in JpaRepository

spring-boot - 通过 postman 发送二维字节数组(多个文件)作为多部分请求

java - 事件无限循环 SWT