首页 文章

Spring Security OAuth2资源服务器始终返回无效令牌

提问于
浏览
18

我正在尝试使用Spring库运行基本的内存OAuth2服务器 . 我一直关注sparklr example .

我目前已配置服务器,几乎一切正常,但我无法从资源服务器访问我的受限资源 .

我的测试工作流程

{access_token:“f853bcc5-7801-42d3-9cb8-303fc67b0453”token_type:“bearer”refresh_token:“57100377-dea9-4df0-adab-62e33f2a1b49”expires_in:299范围:“read write”}

{error:“invalid_token”error_description:“访问令牌无效:f853bcc5-7801-42d3-9cb8-303fc67b0453”}

{access_token:“ed104994-899c-4cd9-8860-43d5689a9420”token_type:“bearer”refresh_token:“57100377-dea9-4df0-adab-62e33f2a1b49”expires_in:300范围:“read write”}

我真的不确定我做错了什么,但似乎除了访问受限制的uri之外的一切都在工作 . 这是我的配置:

@Configuration
public class Oauth2ServerConfiguration {

    private static final String SERVER_RESOURCE_ID = "oauth2-server";

    @Configuration
    @EnableResourceServer
    protected static class ResourceServerConfiguration extends ResourceServerConfigurerAdapter {

        @Override
        public void configure(ResourceServerSecurityConfigurer resources) {
            resources.resourceId(SERVER_RESOURCE_ID);
        }

        @Override
        public void configure(HttpSecurity http) throws Exception {
            http
                .sessionManagement()
                    .sessionCreationPolicy(SessionCreationPolicy.IF_REQUIRED)
                .and().requestMatchers()
                    .antMatchers("/me")
                .and().authorizeRequests()
                    .antMatchers("/me").access("#oauth2.clientHasRole('ROLE_CLIENT')")
            ;
        }
    }

    @Configuration
    @EnableAuthorizationServer
    protected static class AuthotizationServerConfiguration extends AuthorizationServerConfigurerAdapter {

        @Autowired
        private ClientDetailsService clientDetailsService;

        @Autowired
        @Qualifier("authenticationManagerBean")
        private AuthenticationManager authenticationManager;

        @Override
        public void configure(ClientDetailsServiceConfigurer clients) throws Exception {
            clients.inMemory()
                .withClient("client")
                    .resourceIds(SERVER_RESOURCE_ID)
                    .secret("secret")
                    .authorizedGrantTypes("authorization_code", "refresh_token")
                    .authorities("ROLE_CLIENT")
                    .scopes("read","write")
                    .redirectUris("http://localhost:8080/client")
                    .accessTokenValiditySeconds(300)
                    .autoApprove(true)
            ;
        }

        @Bean
        public TokenStore tokenStore() {
            return new InMemoryTokenStore();
        }

        @Override
        public void configure(AuthorizationServerEndpointsConfigurer endpoints) throws Exception {
            endpoints
                .tokenStore(tokenStore())
                .userApprovalHandler(userApprovalHandler())
                .authenticationManager(authenticationManager)
            ;
        }

        @Override
        public void configure(AuthorizationServerSecurityConfigurer oauthServer) throws Exception {
            oauthServer.realm("oauth");
        }

        @Bean
        public ApprovalStore approvalStore() throws Exception {
            TokenApprovalStore store = new TokenApprovalStore();
            store.setTokenStore(tokenStore());
            return store;
        }

        @Bean
        public UserApprovalHandler userApprovalHandler() throws Exception {
            TokenStoreUserApprovalHandler handler = new TokenStoreUserApprovalHandler();
            handler.setRequestFactory(new DefaultOAuth2RequestFactory(clientDetailsService));
            handler.setClientDetailsService(clientDetailsService);
            handler.setTokenStore(tokenStore());

            return handler;
        }
    }
}

有什么我想念的,或者我接近这个错误?任何帮助将不胜感激 .

3 回答

  • 5

    问题最终是资源服务器和授权服务器没有获得相同的令牌存储引用 . 不确定接线是如何工作不正确的,但在配置类中使用固定对象就像一个魅力 . 最终,我将转移到持久性支持的令牌存储,这可能不会有任何问题 .

    感谢@OhadR的答案和帮助!

    最终,我简化了配置,通过相同的工作流程,并且它成功了

    @Configuration
    public class Oauth2ServerConfiguration {
    
        private static final String SERVER_RESOURCE_ID = "oauth2-server";
    
        private static InMemoryTokenStore tokenStore = new InMemoryTokenStore();
    
    
        @Configuration
        @EnableResourceServer
        protected static class ResourceServer extends ResourceServerConfigurerAdapter {
    
            @Override
            public void configure(ResourceServerSecurityConfigurer resources) throws Exception {
                resources.tokenStore(tokenStore).resourceId(SERVER_RESOURCE_ID);
            }
    
            @Override
            public void configure(HttpSecurity http) throws Exception {
                http.requestMatchers().antMatchers("/me").and().authorizeRequests().antMatchers("/me").access("#oauth2.hasScope('read')");
            }
        }
    
        @Configuration
        @EnableAuthorizationServer
        protected static class AuthConfig extends AuthorizationServerConfigurerAdapter {
    
            @Autowired
            private AuthenticationManager authenticationManager;
    
    
            @Override
            public void configure(AuthorizationServerEndpointsConfigurer endpoints) throws Exception {
                endpoints.authenticationManager(authenticationManager).tokenStore(tokenStore).approvalStoreDisabled();
            }
    
            @Override
            public void configure(ClientDetailsServiceConfigurer clients) throws Exception {
                clients.inMemory()
                    .withClient("client")
                        .authorizedGrantTypes("authorization_code","refresh_token")
                        .authorities("ROLE_CLIENT")
                        .scopes("read")
                        .resourceIds(SERVER_RESOURCE_ID)
                        .secret("secret")
                ;
            }
        }
    }
    

    任何偶然发现这篇文章的人,我建议更多地考虑单元测试而不是完整的sparklr / tonr示例,因为它有很多额外的配置,不一定需要开始 .

  • 0

    您的步骤#6是错误的 - 访问令牌不应该在URL中发送,因为它以这种方式易受攻击 . 比GET更好,使用POST .

    此外,我不明白你的第一步 - 为什么你打电话/ oauth / authorize?当您尝试获取受保护资源时,应该隐式执行此操作 . 我的意思是,你的流程应该从以下开始:

    尝试使用access_token访问受限资源:http:// localhost:8080 / server / me

    然后谈判将在“幕后”开始:重定向到“/ oauth / authorize”等 .

    此外,在步骤#8中,请注意您不是要求“另一个访问令牌”,而是要求“刷新令牌” . 好像您的访问令牌已过期 .

    Note :身份提供者和资源服务器应该共享tokenStore!在这里阅读:Spring Security OAuth2 pure resource server

    HTH

  • 5

    这对我有用:

    @Configuration
    public class Oauth2ServerConfiguration {
    
        private static final String SERVER_RESOURCE_ID = "oauth2-server";
    
        @Autowired
        private TokenStore tokenStore;
    
        @Bean
        public TokenStore tokenStore() {
            return new InMemoryTokenStore();
        }
    
        @Configuration
        @EnableResourceServer
        protected static class ResourceServer extends ResourceServerConfigurerAdapter {
    
            @Override
            public void configure(ResourceServerSecurityConfigurer resources) throws Exception {
                resources.tokenStore(tokenStore).resourceId(SERVER_RESOURCE_ID);
            }
    
            @Override
            public void configure(HttpSecurity http) throws Exception {
                // ... Not important at this stage
            }
        }
    
        @Configuration
        @EnableAuthorizationServer
        protected static class AuthConfig extends AuthorizationServerConfigurerAdapter {
    
            @Autowired
            private AuthenticationManager authenticationManager;
    
    
            @Override
            public void configure(AuthorizationServerEndpointsConfigurer endpoints) throws Exception {
                endpoints.authenticationManager(authenticationManager).tokenStore(tokenStore).approvalStoreDisabled();
            }
    
            @Override
            public void configure(ClientDetailsServiceConfigurer clients) throws Exception {
                //... Not important at this stage
            }
        }
    }
    

相关问题