Spring Security 与 OAuth2(资源服务器)

骑猪看日落 2021-09-12 05:14 502阅读 0赞

个人 OAuth2 全部文章

  • Spring Security 与 OAuth2(介绍):https://www.jianshu.com/p/68f22f9a00ee
  • Spring Security 与 OAuth2(授权服务器):https://www.jianshu.com/p/227f7e7503cb
  • Spring Security 与 OAuth2(资源服务器):https://www.jianshu.com/p/6dd03375224d
  • Spring Security 与 OAuth2(客户端):https://www.jianshu.com/p/03e515c2b43f
  • Spring Security 与 OAuth2(相关类参考):https://www.jianshu.com/p/c2395772bc86
  • Spring Security 与 OAuth2(完整案例):https://www.jianshu.com/p/d80061e6d900

resource-server(资源服务器)

资源服务器

  • 要访问资源服务器受保护的资源需要携带令牌(从授权服务器获得)
  • 客户端往往同时也是一个资源服务器,各个服务之间的通信(访问需要权限的资源)时需携带访问令牌
  • 资源服务器通过 @EnableResourceServer 注解来开启一个 OAuth2AuthenticationProcessingFilter 类型的过滤器
  • 通过继承 ResourceServerConfigurerAdapter 类来配置资源服务器

ResourceServerProperties

  • OAuth2 为资源服务器配置提供了 ResourceServerProperties 类,该类会读取配置文件中对资源服务器得配置信息(如授权服务器公钥访问地址)

ResourceServerSecurityConfigurer 可配置属性

  • tokenServices:ResourceServerTokenServices 类的实例,用来实现令牌业务逻辑服务
  • resourceId:这个资源服务的ID,这个属性是可选的,但是推荐设置并在授权服务中进行验证
  • tokenExtractor 令牌提取器用来提取请求中的令牌
  • 请求匹配器,用来设置需要进行保护的资源路径,默认的情况下是受保护资源服务的全部路径
  • 受保护资源的访问规则,默认的规则是简单的身份验证(plain authenticated)
  • 其他的自定义权限保护规则通过 HttpSecurity 来进行配置

解析令牌方法:

  • 使用 DefaultTokenServices 在资源服务器本地配置令牌存储、解码、解析方式
  • 使用 RemoteTokenServices 资源服务器通过 HTTP 请求来解码令牌,每次都请求授权服务器端点 /oauth/check_token
  • 若授权服务器是 JWT 非对称加密,则需要请求授权服务器的 /oauth/token_key 来获取公钥 key 进行解码

代码案例

令牌解析(JWT 对称加密)

资源服务器和授权服务器不在同一个应用,则需告诉资源服务器令牌如何存储与解析,并与授权服务器使用相同的密钥进行解密

  1. @Configuration
  2. @EnableResourceServer
  3. public class OAuth2ResourceConfig extends ResourceServerConfigurerAdapter{
  4. @Bean
  5. public TokenStore tokenStore() {
  6. return new JwtTokenStore(jwtAccessTokenConverter());
  7. }
  8. //与授权服务器使用共同的密钥进行解析
  9. @Bean
  10. public JwtAccessTokenConverter jwtAccessTokenConverter() {
  11. JwtAccessTokenConverter converter = new JwtAccessTokenConverter();
  12. converter.setSigningKey("123");
  13. return converter;
  14. }
  15. }

令牌解析(JWT 非对称加密)

  • 非对称加密需要公钥,可以从本地获取,也可以从授权服务器提供的公钥端点获取
  • 若本地获取不到公钥资源文件 pubkey.txt 则从授权服务器端点获取

    @Configuration
    @EnableResourceServer
    public class OAuth2ResourceConfig extends ResourceServerConfigurerAdapter {

    1. @Autowired
    2. private ResourceServerProperties resourceServerProperties;
    3. @Bean
    4. public TokenStore tokenStore() {
    5. return new JwtTokenStore(jwtAccessTokenConverter());
    6. }
    7. @Bean
    8. public JwtAccessTokenConverter jwtAccessTokenConverter() {
    9. JwtAccessTokenConverter converter = new JwtAccessTokenConverter();
    10. //设置用于解码的非对称加密的公钥
    11. converter.setVerifierKey(getPubKey());
    12. return converter;
    13. }
    14. private String getPubKey() {
    15. Resource resource = new ClassPathResource("pubkey.txt");
    16. try (BufferedReader br = new BufferedReader(new InputStreamReader(resource.getInputStream()))) {
    17. System.out.println("本地公钥");
    18. return br.lines().collect(Collectors.joining("\n"));
    19. } catch (IOException ioe) {
    20. return getKeyFromAuthorizationServer();
    21. }
    22. }
    23. private String getKeyFromAuthorizationServer() {
    24. ObjectMapper objectMapper = new ObjectMapper();
    25. String pubKey = new RestTemplate().getForObject(resourceServerProperties.getJwt().getKeyUri(), String.class);
    26. try {
    27. Map map = objectMapper.readValue(pubKey, Map.class);
    28. System.out.println("联网公钥");
    29. return map.get("value").toString();
    30. } catch (IOException e) {
    31. e.printStackTrace();
    32. }
    33. return null;
    34. }

    }

令牌解析(通过访问授权服务器解析令牌-适用 JDBC、内存存储)

  • 资源服务器通过访问授权服务器 /oauth/check_token 端点解析令牌需要使用 RemoteTokenServices
  • 并且使用 DefaultAccessTokenConverter 来实现令牌数据的存储

    @Autowired

    1. private OAuth2ClientProperties oAuth2ClientProperties;
    2. @Autowired
    3. private AuthorizationServerProperties authorizationServerProperties;
    4. @Bean
    5. public ResourceServerTokenServices tokenServices() {
    6. RemoteTokenServices remoteTokenServices = new RemoteTokenServices();
    7. remoteTokenServices.setCheckTokenEndpointUrl(authorizationServerProperties.getCheckTokenAccess());
    8. remoteTokenServices.setClientId(oAuth2ClientProperties.getClientId());
    9. remoteTokenServices.setClientSecret(oAuth2ClientProperties.getClientSecret());
    10. remoteTokenServices.setAccessTokenConverter(accessTokenConverter());
    11. return remoteTokenServices;
    12. }
    13. @Bean
    14. public AccessTokenConverter accessTokenConverter() {
    15. return new DefaultAccessTokenConverter();
    16. }
  • 修改配置文件

    security:
    oauth2:

    1. client:
    2. clientId: resource1
    3. clientSecret: secret
    4. userAuthorizationUri: http://localhost:9005/oauth/authorize
    5. grant-type: password
    6. scope: read
    7. access-token-uri: http://localhost:9005/oauth/token
    8. resource:
    9. userInfoUri: http://localhost:9005/user
    10. authorization:
    11. check-token-access: http://localhost:9005/oauth/check_token

    basic:

    1. enabled: false

发表评论

表情:
评论列表 (有 0 条评论,502人围观)

还没有评论,来说两句吧...

相关阅读