对于最新的稳定版本,请使用 Spring Security 6.4.5

OAuth 2.0 资源服务器不透明令牌

Introspection 的最小依赖项

JWT 的最小依赖项中所述,大多数 Resource Server 支持都收集在spring-security-oauth2-resource-server. 但是,除非自定义OpaqueTokenIntrospector,则资源服务器将回退到 NimbusOpaqueTokenIntrospector。 这意味着两者spring-security-oauth2-resource-serveroauth2-oidc-sdk是必需的,以便拥有支持不透明 Bearer Token 的工作最小 Resource Server。 请参考spring-security-oauth2-resource-server为了确定oauth2-oidc-sdk.

Introspection 的最小配置

通常,可以通过授权服务器托管的 OAuth 2.0 Introspection Endpoint 验证不透明令牌。 当需要吊销时,这可能很方便。

使用 Spring Boot 时,将应用程序配置为使用内省的资源服务器包括两个基本步骤。 首先,包括所需的依赖项,其次,指示自省终端节点详细信息。

指定 Authorization Server

要指定自省端点的位置,只需执行以下作:

spring:
  security:
    oauth2:
      resourceserver:
        opaque-token:
          introspection-uri: https://idp.example.com/introspect
          client-id: client
          client-secret: secret

哪里idp.example.com/introspect是由授权服务器托管的 Introspection 终端节点,client-idclient-secret是命中该终端节点所需的凭证。

Resource Server 将使用这些属性进一步自我配置并随后验证传入的 JWT。

使用内省时,授权服务器的话就是法律。 如果授权服务器响应令牌有效,则令牌有效。

就是这样!

创业期望

使用此属性和这些依赖项时,Resource Server 将自动配置自身以验证不透明不记名令牌。

此启动过程比 JWT 简单得多,因为不需要发现端点,也不需要添加额外的验证规则。

运行时预期

应用程序启动后,Resource Server 将尝试处理任何包含Authorization: Bearer页眉:

GET / HTTP/1.1
Authorization: Bearer some-token-value # Resource Server will process this

只要指定了此方案,Resource Server 就会尝试根据 Bearer Token 规范处理请求。

给定一个不透明的令牌,Resource Server 将

  1. 使用提供的凭证和令牌查询提供的自省终端节点

  2. 检查响应中是否有{ 'active' : true }属性

  3. 将每个范围映射到带有前缀的颁发机构SCOPE_

结果Authentication#getPrincipal默认情况下,是 Spring SecurityOAuth2AuthenticatedPrincipalobject 和Authentication#getName映射到令牌的sub属性(如果存在)。

从这里,您可能希望跳转到:

不透明令牌身份验证的工作原理

接下来,让我们看看 Spring Security 用于在基于 servlet 的应用程序中支持不透明令牌身份验证的架构组件,就像我们刚刚看到的一样。

让我们来看看OpaqueTokenAuthenticationProvider在 Spring Security 中工作。 该图详细介绍了AuthenticationManagerReading the Bearer Token works 中的数字中。

OpaqueTokenAuthenticationProvider
图 1.OpaqueTokenAuthenticationProvider用法

数字 1身份验证FilterReading the Bearer Token 传递一个BearerTokenAuthenticationTokenAuthenticationManager它由ProviderManager.

编号 2ProviderManager配置为使用 AuthenticationProvider 类型的OpaqueTokenAuthenticationProvider.

编号 3 OpaqueTokenAuthenticationProvider内省不透明令牌并使用OpaqueTokenIntrospector. 身份验证成功后,Authentication返回的 类型为BearerTokenAuthentication并且有一个主体,该主体是OAuth2AuthenticatedPrincipal由配置的OpaqueTokenIntrospector. 最终,返回的BearerTokenAuthentication将在SecurityContextHolder通过身份验证Filter.

身份验证后查找属性

令牌通过身份验证后,BearerTokenAuthenticationSecurityContext.

这意味着它可用于@Controller方法时使用@EnableWebMvc在您的配置中:

@GetMapping("/foo")
public String foo(BearerTokenAuthentication authentication) {
    return authentication.getTokenAttributes().get("sub") + " is the subject";
}

因为BearerTokenAuthentication持有OAuth2AuthenticatedPrincipal,这也意味着它也可用于控制器方法:

@GetMapping("/foo")
public String foo(@AuthenticationPrincipal OAuth2AuthenticatedPrincipal principal) {
    return principal.getAttribute("sub") + " is the subject";
}

通过 SPEL 查找属性

当然,这也意味着可以通过 SPEL 访问属性。

例如,如果使用@EnableGlobalMethodSecurity这样您就可以使用@PreAuthorizeannotations 中,您可以执行以下作:

@PreAuthorize("principal?.attributes['sub'] == 'foo'")
public String forFoosEyesOnly() {
    return "foo";
}

覆盖或替换引导自动配置

有两个@Bean的 Spring Boot 代表 Resource Server 生成的。

第一个是SecurityFilterChain,将应用程序配置为资源服务器。 当使用 Opaque Token 时,这个SecurityFilterChain看来:

默认不透明令牌配置
@Bean
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
    http
        .authorizeHttpRequests(authorize -> authorize
            .anyRequest().authenticated()
        )
        .oauth2ResourceServer(OAuth2ResourceServerConfigurer::opaqueToken);
    return http.build();
}

如果应用程序没有公开SecurityFilterChainbean,则 Spring Boot 将公开上述默认的 bean。

替换它就像在应用程序中公开 bean 一样简单:

自定义不透明令牌配置
@EnableWebSecurity
public class MyCustomSecurityConfiguration {
    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http
            .authorizeHttpRequests(authorize -> authorize
                .requestMatchers("/messages/**").hasAuthority("SCOPE_message:read")
                .anyRequest().authenticated()
            )
            .oauth2ResourceServer(oauth2 -> oauth2
                .opaqueToken(opaqueToken -> opaqueToken
                    .introspector(myIntrospector())
                )
            );
        return http.build();
    }
}

以上要求message:read对于任何以/messages/.

方法oauth2ResourceServerDSL 还将覆盖或替换 auto 配置。

例如,第二个@BeanSpring Boot 创建的是一个OpaqueTokenIntrospector,解码String令牌转换为已验证的OAuth2AuthenticatedPrincipal:

@Bean
public OpaqueTokenIntrospector introspector() {
    return new NimbusOpaqueTokenIntrospector(introspectionUri, clientId, clientSecret);
}

如果应用程序没有公开OpaqueTokenIntrospectorbean,则 Spring Boot 将公开上述默认的 bean。

并且它的配置可以使用introspectionUri()introspectionClientCredentials()或使用introspector().

如果应用程序没有公开OpaqueTokenAuthenticationConverterbean,则 spring-security 将构建BearerTokenAuthentication.

或者,如果您根本不使用 Spring Boot,那么所有这些组件 - 过滤器链、OpaqueTokenIntrospector以及一个OpaqueTokenAuthenticationConverter可以在 XML 中指定。

过滤器链的指定方式如下:

默认不透明令牌配置
<http>
    <intercept-uri pattern="/**" access="authenticated"/>
    <oauth2-resource-server>
        <opaque-token introspector-ref="opaqueTokenIntrospector"
                authentication-converter-ref="opaqueTokenAuthenticationConverter"/>
    </oauth2-resource-server>
</http>
Opaque Token Introspector
<bean id="opaqueTokenIntrospector"
        class="org.springframework.security.oauth2.server.resource.introspection.NimbusOpaqueTokenIntrospector">
    <constructor-arg value="${spring.security.oauth2.resourceserver.opaquetoken.introspection_uri}"/>
    <constructor-arg value="${spring.security.oauth2.resourceserver.opaquetoken.client_id}"/>
    <constructor-arg value="${spring.security.oauth2.resourceserver.opaquetoken.client_secret}"/>
</bean>

OpaqueTokenAuthenticationConverter这样:

不透明令牌身份验证转换器
<bean id="opaqueTokenAuthenticationConverter"
        class="com.example.CustomOpaqueTokenAuthenticationConverter"/>

introspectionUri()

授权服务器的 Introspection Uri 可以配置为配置属性,也可以在 DSL 中提供:

Introspection URI 配置
@EnableWebSecurity
public class DirectlyConfiguredIntrospectionUri {
    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http
            .authorizeHttpRequests(authorize -> authorize
                .anyRequest().authenticated()
            )
            .oauth2ResourceServer(oauth2 -> oauth2
                .opaqueToken(opaqueToken -> opaqueToken
                    .introspectionUri("https://idp.example.com/introspect")
                    .introspectionClientCredentials("client", "secret")
                )
            );
        return http.build();
    }
}

introspectionUri()优先于任何配置属性。

introspector()

introspectionUri()introspector(),它将完全替换OpaqueTokenIntrospector:

Introspector 配置
@EnableWebSecurity
public class DirectlyConfiguredIntrospector {
    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http
            .authorizeHttpRequests(authorize -> authorize
                .anyRequest().authenticated()
            )
            .oauth2ResourceServer(oauth2 -> oauth2
                .opaqueToken(opaqueToken -> opaqueToken
                    .introspector(myCustomIntrospector())
                )
            );
        return http.build();
    }
}

当需要更深入的配置(如权限映射JWT 撤销请求超时)时,这非常方便。

公开OpaqueTokenIntrospector @Bean

或者,公开OpaqueTokenIntrospector @Bean具有相同的效果introspector():

@Bean
public OpaqueTokenIntrospector introspector() {
    return new NimbusOpaqueTokenIntrospector(introspectionUri, clientId, clientSecret);
}

配置授权

OAuth 2.0 Introspection 终端节点通常会返回scope属性,指示已授予的范围(或权限),例如:

{ …​, "scope" : "messages contacts"}

在这种情况下,Resource Server 将尝试将这些范围强制转换为已授予的权限列表,并在每个范围前加上字符串“SCOPE_”。

这意味着,要保护具有从 Opaque Token 派生的范围的终端节点或方法,相应的表达式应包含以下前缀:

授权不透明令牌配置
@EnableWebSecurity
public class MappedAuthorities {
    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http
            .authorizeHttpRequests(authorizeRequests -> authorizeRequests
                .requestMatchers("/contacts/**").hasAuthority("SCOPE_contacts")
                .requestMatchers("/messages/**").hasAuthority("SCOPE_messages")
                .anyRequest().authenticated()
            )
            .oauth2ResourceServer(OAuth2ResourceServerConfigurer::opaqueToken);
        return http.build();
    }
}

或者与方法安全性类似:

@PreAuthorize("hasAuthority('SCOPE_messages')")
public List<Message> getMessages(...) {}

手动提取权限

默认情况下,Opaque Token 支持将从自省响应中提取 scope 声明,并将其解析为单个GrantedAuthority实例。

例如,如果内省响应为:

{
    "active" : true,
    "scope" : "message:read message:write"
}

然后 Resource Server 将生成一个Authentication有两个权限,一个用于message:read另一个用于message:write.

当然,这可以使用自定义OpaqueTokenIntrospector,它查看了 attribute set 并以自己的方式进行转换:

public class CustomAuthoritiesOpaqueTokenIntrospector implements OpaqueTokenIntrospector {
    private OpaqueTokenIntrospector delegate =
            new NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret");

    public OAuth2AuthenticatedPrincipal introspect(String token) {
        OAuth2AuthenticatedPrincipal principal = this.delegate.introspect(token);
        return new DefaultOAuth2AuthenticatedPrincipal(
                principal.getName(), principal.getAttributes(), extractAuthorities(principal));
    }

    private Collection<GrantedAuthority> extractAuthorities(OAuth2AuthenticatedPrincipal principal) {
        List<String> scopes = principal.getAttribute(OAuth2IntrospectionClaimNames.SCOPE);
        return scopes.stream()
                .map(SimpleGrantedAuthority::new)
                .collect(Collectors.toList());
    }
}

此后,只需将这个自定义 introspector 公开为@Bean:

@Bean
public OpaqueTokenIntrospector introspector() {
    return new CustomAuthoritiesOpaqueTokenIntrospector();
}

配置超时

默认情况下,Resource Server 使用各 30 秒的连接和套接字超时来与授权服务器进行协调。

在某些情况下,这可能太短了。 此外,它没有考虑更复杂的模式,如 back-off 和 discovery。

要调整 Resource Server 与授权服务器的连接方式,NimbusOpaqueTokenIntrospector接受RestOperations:

@Bean
public OpaqueTokenIntrospector introspector(RestTemplateBuilder builder, OAuth2ResourceServerProperties properties) {
    RestOperations rest = builder
            .basicAuthentication(properties.getOpaquetoken().getClientId(), properties.getOpaquetoken().getClientSecret())
            .setConnectTimeout(Duration.ofSeconds(60))
            .setReadTimeout(Duration.ofSeconds(60))
            .build();

    return new NimbusOpaqueTokenIntrospector(introspectionUri, rest);
}

将 Introspection 与 JWT 结合使用

一个常见的问题是 introspection 是否与 JWT 兼容。 Spring Security 的 Opaque Token 支持被设计为不关心令牌的格式 - 它会很乐意将任何令牌传递给提供的内省端点。

因此,假设您有一个要求,要求您在每个请求上与授权服务器进行检查,以防 JWT 被撤销。

即使您对令牌使用 JWT 格式,您的验证方法也是 introspection,这意味着您需要执行以下作:

spring:
  security:
    oauth2:
      resourceserver:
        opaque-token:
          introspection-uri: https://idp.example.org/introspection
          client-id: client
          client-secret: secret

在这种情况下,生成的AuthenticationBearerTokenAuthentication. 相应OAuth2AuthenticatedPrincipal将是 Introspection 终端节点返回的任何内容。

但是,奇怪的是,内省端点只返回令牌是否处于活动状态。 现在怎么办?

在这种情况下,您可以创建自定义OpaqueTokenIntrospector,它仍然命中终端节点,但随后更新返回的主体以将 JWTs 声明作为属性:

public class JwtOpaqueTokenIntrospector implements OpaqueTokenIntrospector {
    private OpaqueTokenIntrospector delegate =
            new NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret");
    private JwtDecoder jwtDecoder = new NimbusJwtDecoder(new ParseOnlyJWTProcessor());

    public OAuth2AuthenticatedPrincipal introspect(String token) {
        OAuth2AuthenticatedPrincipal principal = this.delegate.introspect(token);
        try {
            Jwt jwt = this.jwtDecoder.decode(token);
            return new DefaultOAuth2AuthenticatedPrincipal(jwt.getClaims(), NO_AUTHORITIES);
        } catch (JwtException ex) {
            throw new OAuth2IntrospectionException(ex);
        }
    }

    private static class ParseOnlyJWTProcessor extends DefaultJWTProcessor<SecurityContext> {
    	JWTClaimsSet process(SignedJWT jwt, SecurityContext context)
                throws JOSEException {
            return jwt.getJWTClaimsSet();
        }
    }
}

此后,只需将这个自定义 introspector 公开为@Bean:

@Bean
public OpaqueTokenIntrospector introspector() {
    return new JwtOpaqueTokenIntrospector();
}

调用/userinfo端点

一般来说,Resource Server 不关心底层用户,而是关心已授予的权限。

也就是说,有时将授权声明绑定回用户可能很有价值。

如果应用程序还使用spring-security-oauth2-client,在设置适当的ClientRegistrationRepository,那么使用自定义OpaqueTokenIntrospector. 下面的实现执行三项作:

public class UserInfoOpaqueTokenIntrospector implements OpaqueTokenIntrospector {
    private final OpaqueTokenIntrospector delegate =
            new NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret");
    private final OAuth2UserService oauth2UserService = new DefaultOAuth2UserService();

    private final ClientRegistrationRepository repository;

    // ... constructor

    @Override
    public OAuth2AuthenticatedPrincipal introspect(String token) {
        OAuth2AuthenticatedPrincipal authorized = this.delegate.introspect(token);
        Instant issuedAt = authorized.getAttribute(ISSUED_AT);
        Instant expiresAt = authorized.getAttribute(EXPIRES_AT);
        ClientRegistration clientRegistration = this.repository.findByRegistrationId("registration-id");
        OAuth2AccessToken token = new OAuth2AccessToken(BEARER, token, issuedAt, expiresAt);
        OAuth2UserRequest oauth2UserRequest = new OAuth2UserRequest(clientRegistration, token);
        return this.oauth2UserService.loadUser(oauth2UserRequest);
    }
}

如果您未使用spring-security-oauth2-client,还是挺简单的。 您只需调用/userinfo替换为您自己的WebClient:

public class UserInfoOpaqueTokenIntrospector implements OpaqueTokenIntrospector {
    private final OpaqueTokenIntrospector delegate =
            new NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret");
    private final WebClient rest = WebClient.create();

    @Override
    public OAuth2AuthenticatedPrincipal introspect(String token) {
        OAuth2AuthenticatedPrincipal authorized = this.delegate.introspect(token);
        return makeUserInfoRequest(authorized);
    }
}

无论哪种方式,创建OpaqueTokenIntrospector,您应该将其发布为@Bean要覆盖默认值,请执行以下作:

@Bean
OpaqueTokenIntrospector introspector() {
    return new UserInfoOpaqueTokenIntrospector(...);
}

APP信息