【问题标题】:Spring OAuth2 Redis Client DetailsSpring OAuth2 Redis 客户端详细信息
【发布时间】:2017-08-04 04:43:44
【问题描述】:

我有一个 oauth2 的工作示例,并为客户端和资源所有者提供内存身份验证和授权。

我想使用 Redis,但对如何设置它有点困惑。

如何调整下面的代码以便能够将数据持久保存在 Redis 中(例如令牌、refresh_token 和其他客户端详细信息)

@Configuration
@EnableAuthorizationServer
public class AuthorizationServerConfiguration extends AuthorizationServerConfigurerAdapter {

    @Value("${spring.oauth2.realm.id}")
    private String REALM;

    @Autowired
    private TokenStore tokenStore;

    @Autowired
    private UserApprovalHandler userApprovalHandler;

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

    @Override
    public void configure(ClientDetailsServiceConfigurer clients) throws Exception {
        clients.inMemory()
                .withClient("client001")
                .authorizedGrantTypes("client_credentials", "password")
                .authorities("ROLE_ADMIN", "ROLE_TRUSTED_CLIENT")
                .scopes("read", "write", "trust")
                .secret("secret")
                .accessTokenValiditySeconds(120) 
                .refreshTokenValiditySeconds(600);
    }

    @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(REALM + "/client");
    }

}

[更新]

我能够通过使用 RedisTokenStore 将令牌存储到 Redis。为此,请在您的 OAuth2SecurityConfiguration 中替换以下行:

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

与...

@Bean
public TokenStore tokenStore(final RedisConnectionFactory factory) {
    return new RedisTokenStore(factory);
}

现在遇到了一个新问题,在 AuthorizationServerConfiguration#configure(final ClientDetailsServiceConfigurer clients);。是否支持 redis

@Override
    public void configure(ClientDetailsServiceConfigurer clients) throws Exception {
      clients.inMemory()
        .withClient("client001")
        .secret("secret")
        .authorizedGrantTypes("client_credentials", "password")
        .authorities("ROLE_ADMIN", "ROLE_TRUSTED_CLIENT")
        .scopes("read", "write", "trust")
        .accessTokenValiditySeconds(60) 
        .refreshTokenValiditySeconds(120);
    }

当我查看 ClientDetailsS​​erviceConfigurer 时,它只支持 inMemory()jdbc()。我打算使用 withClientDetails() 来支持 redis。是否有 RedisClientDetailsS​​ervice 或类似的东西?

非常感谢任何关于其实施的 cmets 和建议。

【问题讨论】:

    标签: spring redis oauth-2.0 spring-oauth2


    【解决方案1】:

    我能够将客户详细信息存储到 Redis。

    这些是我做的步骤:

    1. 创建以下类:

      • RedisClientDetailsS​​erviceBuilder.java
      • RedisClientDetailsS​​ervice.java

    RedisClientDetailsS​​ervice

    public class RedisClientDetailsService implements ClientDetailsService {
    
        private PasswordEncoder passwordEncoder = NoOpPasswordEncoder.getInstance();
    
        private Map<String, ClientDetails> clientDetailsStore = new HashMap<String, ClientDetails>();
    
        public ClientDetails loadClientByClientId(final String clientId) throws ClientRegistrationException {
            final ClientDetails details = clientDetailsStore.get(clientId);
            if (details == null) {
                throw new NoSuchClientException("No client with requested id: " + clientId);
            }
            return details;
        }
    
        public void setClientDetailsStore(final Map<String, ? extends ClientDetails> clientDetailsStore) {
            this.clientDetailsStore = new HashMap<String, ClientDetails>(clientDetailsStore);
        }
    
        /**
         * @param passwordEncoder the password encoder to set
         */
        public void setPasswordEncoder(final PasswordEncoder passwordEncoder) {
            this.passwordEncoder = passwordEncoder;
        }
    }

    RedisClientDetailsS​​erviceBuilder

    public class RedisClientDetailsServiceBuilder extends ClientDetailsServiceBuilder<RedisClientDetailsServiceBuilder> {
    
        private Map<String, ClientDetails> clientDetails = new HashMap<String, ClientDetails>();
    
        private PasswordEncoder passwordEncoder; // for writing client secrets
    
        public RedisClientDetailsServiceBuilder passwordEncoder(PasswordEncoder passwordEncoder) {
            this.passwordEncoder = passwordEncoder;
            return this;
        }
    
        @Override
        protected void addClient(
                final String clientId,
                final ClientDetails build) {
    
            clientDetails.put(clientId, build);
        }
    
        @Override
        protected ClientDetailsService performBuild() {
            final RedisClientDetailsService redisClientDetailsService = new RedisClientDetailsService();
            if (passwordEncoder != null) {
                // This is used to encode secrets as they are added to the database (if it isn't set then the user has top
                // pass in pre-encoded secrets)
                redisClientDetailsService.setPasswordEncoder(passwordEncoder);
            }
    
            redisClientDetailsService.setClientDetailsStore(clientDetails);
            return redisClientDetailsService;
        }
    
    }
    1. 然后在 AuthorizationServerConfiguration#configure() 上,将客户端构建器设置为我们制作的构建器。

    @Override
    public void configure(final ClientDetailsServiceConfigurer clients) throws Exception {
        final RedisClientDetailsServiceBuilder builder = new RedisClientDetailsServiceBuilder();
        clients.setBuilder(builder);
    
        // @formatter:off
        builder.withClient("client001")
              .secret("secret")
              .authorizedGrantTypes("client_credentials", "password")
              .authorities("ROLE_ADMIN", "ROLE_TRUSTED_CLIENT")
              .scopes("read", "write", "trust")
              .accessTokenValiditySeconds(120)
              .refreshTokenValiditySeconds(400)
        // @formatter:on
    }

    【讨论】:

    • 你为什么不在这里简单地使用一个内存客户端呢?您的实现显示没有使用 Redis。
    猜你喜欢
    • 1970-01-01
    • 2017-10-02
    • 2017-10-11
    • 2017-03-15
    • 2010-11-26
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-12-27
    相关资源
    最近更新 更多