我正在尝试使用Spring库运行基本的内存OAuth2服务器 . 我一直关注sparklr example .
我目前已配置服务器,几乎一切正常,但我无法从资源服务器访问我的受限资源 .
我的测试工作流程
-
访问oauth授权的URI以启动OAuth2流程:http://localhost:8080/server/oauth/authorize?response_type=code&client_id=client
-
重定向到登录页面:http://localhost:8080/server/login
-
处理批准并重定向到我配置的重定向页面,其中包含一个代码参数:http://localhost:8080/client?code=HMJO4K
-
使用基本身份验证使用客户端ID和密码以及授权类型和代码构造GET请求:http://localhost:8080/server/oauth/token?grant_type=authorization_code&code=HMJO4K
-
作为回报接收access_token和刷新令牌对象
{access_token:“f853bcc5-7801-42d3-9cb8-303fc67b0453”token_type:“bearer”refresh_token:“57100377-dea9-4df0-adab-62e33f2a1b49”expires_in:299范围:“read write”}
-
尝试使用access_token访问受限资源:http://localhost:8080/server/me?access_token=f853bcc5-7801-42d3-9cb8-303fc67b0453
-
收到无效的令牌回复
{error:“invalid_token”error_description:“访问令牌无效:f853bcc5-7801-42d3-9cb8-303fc67b0453”}
-
再次POST到令牌uri以刷新令牌:http://localhost:8080/server/oauth/token?grant_type=refresh_token&refresh_token=57100377-dea9-4df0-adab-62e33f2a1b49
-
收到新令牌
{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 回答
问题最终是资源服务器和授权服务器没有获得相同的令牌存储引用 . 不确定接线是如何工作不正确的,但在配置类中使用固定对象就像一个魅力 . 最终,我将转移到持久性支持的令牌存储,这可能不会有任何问题 .
感谢@OhadR的答案和帮助!
最终,我简化了配置,通过相同的工作流程,并且它成功了
任何偶然发现这篇文章的人,我建议更多地考虑单元测试而不是完整的sparklr / tonr示例,因为它有很多额外的配置,不一定需要开始 .
您的步骤#6是错误的 - 访问令牌不应该在URL中发送,因为它以这种方式易受攻击 . 比GET更好,使用POST .
此外,我不明白你的第一步 - 为什么你打电话/ oauth / authorize?当您尝试获取受保护资源时,应该隐式执行此操作 . 我的意思是,你的流程应该从以下开始:
然后谈判将在“幕后”开始:重定向到“/ oauth / authorize”等 .
此外,在步骤#8中,请注意您不是要求“另一个访问令牌”,而是要求“刷新令牌” . 好像您的访问令牌已过期 .
Note :身份提供者和资源服务器应该共享tokenStore!在这里阅读:Spring Security OAuth2 pure resource server
HTH
这对我有用: