HttpSecurity 配置不适用于 Spring 上的 Keycloak Adapter

Posted

技术标签:

【中文标题】HttpSecurity 配置不适用于 Spring 上的 Keycloak Adapter【英文标题】:HttpSecurity configuration does not work with Keycloak Adapter on Spring 【发布时间】:2019-05-25 04:23:54 【问题描述】:

完整代码:https://github.com/czetsuya/Spring-Keycloak-with-REST-API

我正在尝试在 Spring 中实现一个 REST API,该 API 由 Keycloak (4.8.1) 保护,带有仅承载客户端。

问题:configure(HttpSecurity http) 不受尊重,只要用户通过身份验证,REST 端点就可以访问。

例如使用 .antMatchers("/admin*").hasRole("ADMIN"),/admin 应该只能由具有 ADMIN 角色的用户访问,但我可以使用 USER 角色访问。

我还尝试在 application.yml 中设置安全约束(但没有帮助):

  security-constraints:
  - auth-roles:
    - ADMIN
  - security-collections:
    - name: admin
    - patterns:
      - /admin*

结合使用@EnableGlobalMethodSecurity 和@PreAuthorize("hasRole('ADMIN')") 可以解决问题,但真的没有其他办法了吗?

这是 application.xml。

keycloak:
  enabled: true
  realm: dev
  auth-server-url: http://localhost:8083/auth
  ssl-required: external
  resource: dev-api
  bearer-only: true
  confidential-port: 0
  use-resource-role-mappings: false
  principal-attribute: preferred_username

以下是对pom的依赖:

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-security</artifactId>
    </dependency>
    <dependency>
        <groupId>org.keycloak</groupId>
        <artifactId>keycloak-spring-boot-starter</artifactId>
    </dependency>
    .....

以及 SecurityConfig 类的一部分:

@KeycloakConfiguration
@EnableGlobalMethodSecurity(prePostEnabled = true)
public class SecurityConfig extends KeycloakWebSecurityConfigurerAdapter 

    @Autowired
    public void configureGlobal(AuthenticationManagerBuilder auth) 
        KeycloakAuthenticationProvider keycloakAuthenticationProvider = keycloakAuthenticationProvider();
        SimpleAuthorityMapper simpleAuthorityMapper = new SimpleAuthorityMapper();
        simpleAuthorityMapper.setConvertToUpperCase(true);
        keycloakAuthenticationProvider.setGrantedAuthoritiesMapper(simpleAuthorityMapper);
        auth.authenticationProvider(keycloakAuthenticationProvider);
    

    @Bean
    @Override
    protected SessionAuthenticationStrategy sessionAuthenticationStrategy() 
        return new NullAuthenticatedSessionStrategy();
    

    @Bean
    public KeycloakConfigResolver keycloakConfigResolver() 
        return new KeycloakSpringBootConfigResolver();
    

    @Autowired
    public KeycloakClientRequestFactory keycloakClientRequestFactory;

    @Bean
    @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    public KeycloakRestTemplate keycloakRestTemplate() 
        return new KeycloakRestTemplate(keycloakClientRequestFactory);
    

    /**
     * Secure appropriate endpoints
     */
    @Override
    protected void configure(HttpSecurity http) throws Exception 
        super.configure(http);
        http.authorizeRequests() //
                .antMatchers("/users*").hasRole("USER") //
                .antMatchers("/admin*").hasRole("ADMIN") //
                .anyRequest().authenticated() //
                .and().csrf().disable() //
        ;
    

启用详细日志后。我发现正在应用 application.yml 中定义的安全约束,而不是 java 类中定义的约束。

现在的问题是如何使用java约束而不是定义的application.yml。

【问题讨论】:

【参考方案1】:

问题是 yml 配置错误。这是正确的版本:

security-constraints:
    - auth-roles:
      - User
      security-collections:
      - name: unsecured
        patterns:
        - /users
    - auth-roles: 
      - Admin
      security-collections:
      - name: secured
        patterns:
        - /admin

请注意,当定义 KeycloakSpringBootConfigResolver 时,还会调用 configure(httpSecurity)。

在 application.yml 中启用下面的日志将向我们展示安全约束检查:

logging:
  level:
    org.apache.catalina: DEBUG

这里有更详细的解释代码:http://czetsuya-tech.blogspot.com/2018/12/secure-spring-boot-rest-project-with.html

【讨论】:

以上是关于HttpSecurity 配置不适用于 Spring 上的 Keycloak Adapter的主要内容,如果未能解决你的问题,请参考以下文章

spring security 继承 WebSecurityConfigurerAdapter 的重写方法configure() 参数 HttpSecurity 常用方法及说明

spring security 继承 WebSecurityConfigurerAdapter 的重写方法configure() 参数 HttpSecurity 常用方法及说明

Spring Security之注销登录

Spring Security Logout 不适用于 Spring 4 CORS

Spring Security:为 HttpSecurity 全局配置时忽略 hasAuthority

第五篇深入理解HttpSecurity的设计