OAuth 2.0 资源服务器不透明令牌
自检的最小依赖项
如JWT 的最小依赖项中所述,大部分资源服务器支持都包含在spring-security-oauth2-resource-server
中。但是,除非提供自定义OpaqueTokenIntrospector
,否则资源服务器将回退到NimbusOpaqueTokenIntrospector。这意味着为了拥有一个支持不透明 Bearer 令牌的工作最小资源服务器,spring-security-oauth2-resource-server
和oauth2-oidc-sdk
都是必要的。请参考spring-security-oauth2-resource-server
以确定oauth2-oidc-sdk
的正确版本。
自检的最小配置
通常,可以通过授权服务器托管的OAuth 2.0 自检端点验证不透明令牌。当需要撤销时,这非常方便。
使用Spring Boot时,将应用程序配置为使用自检的资源服务器包括两个基本步骤。首先,包含所需的依赖项;其次,指示自检端点详细信息。
指定授权服务器
要指定自检端点的位置,只需执行以下操作:
spring:
security:
oauth2:
resourceserver:
opaque-token:
introspection-uri: https://idp.example.com/introspect
client-id: client
client-secret: secret
其中idp.example.com/introspect
是授权服务器托管的自检端点,client-id
和client-secret
是访问该端点所需的凭据。
资源服务器将使用这些属性进行进一步的自我配置,然后验证传入的 JWT。
使用自检时,授权服务器的判断为最终判断。如果授权服务器响应令牌有效,则令牌有效。 |
就是这样!
运行时预期
应用程序启动后,资源服务器将尝试处理任何包含Authorization: Bearer
头的请求。
GET / HTTP/1.1
Authorization: Bearer some-token-value # Resource Server will process this
只要指示了此方案,资源服务器将尝试根据 Bearer 令牌规范处理请求。
对于不透明令牌,资源服务器将:
-
使用提供的凭据和令牌查询提供的自检端点。
-
检查响应中是否存在
{ 'active' : true }
属性。 -
将每个范围映射到具有前缀
SCOPE_
的权限。
生成的Authentication#getPrincipal
默认为 Spring Security OAuth2AuthenticatedPrincipal
对象,Authentication#getName
映射到令牌的sub
属性(如果存在)。
接下来,您可能想跳转到:
不透明令牌身份验证的工作原理
接下来,让我们看看 Spring Security 用于在基于 servlet 的应用程序(就像我们刚才看到的那个一样)中支持不透明令牌身份验证的架构组件。
OpaqueTokenAuthenticationProvider
是一个AuthenticationProvider
实现,它利用OpaqueTokenIntrospector
来验证不透明令牌。
让我们来看看OpaqueTokenAuthenticationProvider
在 Spring Security 中是如何工作的。该图解释了AuthenticationManager
(来自读取 Bearer 令牌中的图)的详细工作原理。
OpaqueTokenAuthenticationProvider
的使用 来自读取 Bearer 令牌的认证Filter
将BearerTokenAuthenticationToken
传递给由ProviderManager
实现的AuthenticationManager
。
ProviderManager
配置为使用类型为OpaqueTokenAuthenticationProvider
的AuthenticationProvider。
OpaqueTokenAuthenticationProvider
对不透明令牌进行内省,并使用OpaqueTokenIntrospector
添加授予的权限。身份验证成功后,返回的Authentication
类型为BearerTokenAuthentication
,其主体是配置的OpaqueTokenIntrospector
返回的OAuth2AuthenticatedPrincipal
。最终,返回的BearerTokenAuthentication
将由身份验证Filter
设置到SecurityContextHolder
上。
身份验证后查找属性
令牌经过身份验证后,BearerTokenAuthentication
实例将设置在SecurityContext
中。
这意味着在配置中使用@EnableWebMvc
时,它在@Controller
方法中可用。
-
Java
-
Kotlin
@GetMapping("/foo")
public String foo(BearerTokenAuthentication authentication) {
return authentication.getTokenAttributes().get("sub") + " is the subject";
}
@GetMapping("/foo")
fun foo(authentication: BearerTokenAuthentication): String {
return authentication.tokenAttributes["sub"].toString() + " is the subject"
}
由于BearerTokenAuthentication
包含一个OAuth2AuthenticatedPrincipal
,这意味着它也对控制器方法可用。
-
Java
-
Kotlin
@GetMapping("/foo")
public String foo(@AuthenticationPrincipal OAuth2AuthenticatedPrincipal principal) {
return principal.getAttribute("sub") + " is the subject";
}
@GetMapping("/foo")
fun foo(@AuthenticationPrincipal principal: OAuth2AuthenticatedPrincipal): String {
return principal.getAttribute<Any>("sub").toString() + " is the subject"
}
通过 SpEL 查找属性
当然,这也意味着可以通过 SpEL 访问属性。
例如,如果使用@EnableGlobalMethodSecurity
以便可以使用@PreAuthorize
注解,则可以执行以下操作:
-
Java
-
Kotlin
@PreAuthorize("principal?.attributes['sub'] == 'foo'")
public String forFoosEyesOnly() {
return "foo";
}
@PreAuthorize("principal?.attributes['sub'] == 'foo'")
fun forFoosEyesOnly(): String {
return "foo"
}
覆盖或替换 Boot 自动配置
Spring Boot 代表资源服务器生成两个@Bean
。
第一个是SecurityFilterChain
,它将应用程序配置为资源服务器。使用不透明令牌时,此SecurityFilterChain
看起来像这样:
-
Java
-
Kotlin
@Bean
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
http
.authorizeHttpRequests(authorize -> authorize
.anyRequest().authenticated()
)
.oauth2ResourceServer(OAuth2ResourceServerConfigurer::opaqueToken);
return http.build();
}
@Bean
open fun filterChain(http: HttpSecurity): SecurityFilterChain {
http {
authorizeRequests {
authorize(anyRequest, authenticated)
}
oauth2ResourceServer {
opaqueToken { }
}
}
return http.build()
}
如果应用程序没有公开SecurityFilterChain
bean,则 Spring Boot 将公开上述默认 bean。
替换它就像在应用程序中公开该 bean 一样简单:
-
Java
-
Kotlin
import static org.springframework.security.oauth2.core.authorization.OAuth2AuthorizationManagers.hasScope;
@Configuration
@EnableWebSecurity
public class MyCustomSecurityConfiguration {
@Bean
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
http
.authorizeHttpRequests(authorize -> authorize
.requestMatchers("/messages/**").access(hasScope("message:read"))
.anyRequest().authenticated()
)
.oauth2ResourceServer(oauth2 -> oauth2
.opaqueToken(opaqueToken -> opaqueToken
.introspector(myIntrospector())
)
);
return http.build();
}
}
import org.springframework.security.oauth2.core.authorization.OAuth2AuthorizationManagers.hasScope;
@Configuration
@EnableWebSecurity
class MyCustomSecurityConfiguration {
@Bean
open fun filterChain(http: HttpSecurity): SecurityFilterChain {
http {
authorizeRequests {
authorize("/messages/**", hasScope("SCOPE_message:read"))
authorize(anyRequest, authenticated)
}
oauth2ResourceServer {
opaqueToken {
introspector = myIntrospector()
}
}
}
return http.build()
}
}
上述操作要求任何以/messages/
开头的 URL 都具有message:read
范围。
oauth2ResourceServer
DSL 上的方法也将覆盖或替换自动配置。
例如,Spring Boot 创建的第二个@Bean
是OpaqueTokenIntrospector
,它将String
令牌解码为经过验证的OAuth2AuthenticatedPrincipal
实例
-
Java
-
Kotlin
@Bean
public OpaqueTokenIntrospector introspector() {
return new NimbusOpaqueTokenIntrospector(introspectionUri, clientId, clientSecret);
}
@Bean
fun introspector(): OpaqueTokenIntrospector {
return NimbusOpaqueTokenIntrospector(introspectionUri, clientId, clientSecret)
}
如果应用程序没有公开OpaqueTokenIntrospector
bean,则 Spring Boot 将公开上述默认 bean。
可以使用introspectionUri()
和introspectionClientCredentials()
覆盖其配置,或者使用introspector()
替换。
如果应用程序没有公开OpaqueTokenAuthenticationConverter
bean,则 spring-security 将构建BearerTokenAuthentication
。
或者,如果您根本不使用 Spring Boot,则所有这些组件(过滤器链、OpaqueTokenIntrospector
和OpaqueTokenAuthenticationConverter
)都可以在 XML 中指定。
过滤器链的指定方式如下:
-
Xml
<http>
<intercept-uri pattern="/**" access="authenticated"/>
<oauth2-resource-server>
<opaque-token introspector-ref="opaqueTokenIntrospector"
authentication-converter-ref="opaqueTokenAuthenticationConverter"/>
</oauth2-resource-server>
</http>
-
Xml
<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
:
-
Xml
<bean id="opaqueTokenAuthenticationConverter"
class="com.example.CustomOpaqueTokenAuthenticationConverter"/>
使用introspectionUri()
授权服务器的内省 URI 可以作为配置属性进行配置,也可以在 DSL 中提供。
-
Java
-
Kotlin
-
Xml
@Configuration
@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();
}
}
@Configuration
@EnableWebSecurity
class DirectlyConfiguredIntrospectionUri {
@Bean
open fun filterChain(http: HttpSecurity): SecurityFilterChain {
http {
authorizeRequests {
authorize(anyRequest, authenticated)
}
oauth2ResourceServer {
opaqueToken {
introspectionUri = "https://idp.example.com/introspect"
introspectionClientCredentials("client", "secret")
}
}
}
return http.build()
}
}
<bean id="opaqueTokenIntrospector"
class="org.springframework.security.oauth2.server.resource.introspection.NimbusOpaqueTokenIntrospector">
<constructor-arg value="https://idp.example.com/introspect"/>
<constructor-arg value="client"/>
<constructor-arg value="secret"/>
</bean>
使用introspectionUri()
优先于任何配置属性。
使用introspector()
比introspectionUri()
更强大的方法是introspector()
,它将完全替换OpaqueTokenIntrospector
的任何 Boot 自动配置。
-
Java
-
Kotlin
-
Xml
@Configuration
@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();
}
}
@Configuration
@EnableWebSecurity
class DirectlyConfiguredIntrospector {
@Bean
open fun filterChain(http: HttpSecurity): SecurityFilterChain {
http {
authorizeRequests {
authorize(anyRequest, authenticated)
}
oauth2ResourceServer {
opaqueToken {
introspector = myCustomIntrospector()
}
}
}
return http.build()
}
}
<http>
<intercept-uri pattern="/**" access="authenticated"/>
<oauth2-resource-server>
<opaque-token introspector-ref="myCustomIntrospector"/>
</oauth2-resource-server>
</http>
公开OpaqueTokenIntrospector
@Bean
或者,公开OpaqueTokenIntrospector
@Bean
与introspector()
具有相同的效果。
@Bean
public OpaqueTokenIntrospector introspector() {
return new NimbusOpaqueTokenIntrospector(introspectionUri, clientId, clientSecret);
}
配置授权
OAuth 2.0 内省端点通常会返回一个scope
属性,指示它已被授予的范围(或权限),例如:
{ …, "scope" : "messages contacts"}
在这种情况下,资源服务器将尝试将这些范围强制转换为授予的权限列表,并在每个范围前面添加字符串“SCOPE_”。
这意味着要使用从不透明令牌派生的范围来保护端点或方法,相应的表达式应包含此前缀:
-
Java
-
Kotlin
-
Xml
import static org.springframework.security.oauth2.core.authorization.OAuth2AuthorizationManagers.hasScope;
@Configuration
@EnableWebSecurity
public class MappedAuthorities {
@Bean
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
http
.authorizeHttpRequests(authorizeRequests -> authorizeRequests
.requestMatchers("/contacts/**").access(hasScope("contacts"))
.requestMatchers("/messages/**").access(hasScope("messages"))
.anyRequest().authenticated()
)
.oauth2ResourceServer(OAuth2ResourceServerConfigurer::opaqueToken);
return http.build();
}
}
import org.springframework.security.oauth2.core.authorization.OAuth2AuthorizationManagers.hasScope
@Configuration
@EnableWebSecurity
class MappedAuthorities {
@Bean
open fun filterChain(http: HttpSecurity): SecurityFilterChain {
http {
authorizeRequests {
authorize("/contacts/**", hasScope("contacts"))
authorize("/messages/**", hasScope("messages"))
authorize(anyRequest, authenticated)
}
oauth2ResourceServer {
opaqueToken { }
}
}
return http.build()
}
}
<http>
<intercept-uri pattern="/contacts/**" access="hasAuthority('SCOPE_contacts')"/>
<intercept-uri pattern="/messages/**" access="hasAuthority('SCOPE_messages')"/>
<oauth2-resource-server>
<opaque-token introspector-ref="opaqueTokenIntrospector"/>
</oauth2-resource-server>
</http>
或者类似的方法安全:
-
Java
-
Kotlin
@PreAuthorize("hasAuthority('SCOPE_messages')")
public List<Message> getMessages(...) {}
@PreAuthorize("hasAuthority('SCOPE_messages')")
fun getMessages(): List<Message?> {}
手动提取权限
默认情况下,不透明令牌支持将从内省响应中提取范围声明,并将其解析为单独的GrantedAuthority
实例。
例如,如果内省响应为:
{
"active" : true,
"scope" : "message:read message:write"
}
则资源服务器将生成一个具有两个权限的Authentication
,一个用于message:read
,另一个用于message:write
。
当然,可以使用自定义OpaqueTokenIntrospector
进行自定义,该自定义程序查看属性集并以其自身的方式进行转换。
-
Java
-
Kotlin
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());
}
}
class CustomAuthoritiesOpaqueTokenIntrospector : OpaqueTokenIntrospector {
private val delegate: OpaqueTokenIntrospector = NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret")
override fun introspect(token: String): OAuth2AuthenticatedPrincipal {
val principal: OAuth2AuthenticatedPrincipal = delegate.introspect(token)
return DefaultOAuth2AuthenticatedPrincipal(
principal.name, principal.attributes, extractAuthorities(principal))
}
private fun extractAuthorities(principal: OAuth2AuthenticatedPrincipal): Collection<GrantedAuthority> {
val scopes: List<String> = principal.getAttribute(OAuth2IntrospectionClaimNames.SCOPE)
return scopes
.map { SimpleGrantedAuthority(it) }
}
}
此后,只需将其作为@Bean
公开即可配置此自定义内省器。
-
Java
-
Kotlin
@Bean
public OpaqueTokenIntrospector introspector() {
return new CustomAuthoritiesOpaqueTokenIntrospector();
}
@Bean
fun introspector(): OpaqueTokenIntrospector {
return CustomAuthoritiesOpaqueTokenIntrospector()
}
配置超时
默认情况下,资源服务器使用 30 秒的连接和套接字超时来与授权服务器协调。
在某些情况下,这可能太短了。此外,它没有考虑更复杂的模式,例如回退和发现。
要调整资源服务器连接到授权服务器的方式,NimbusOpaqueTokenIntrospector
接受RestOperations
实例。
-
Java
-
Kotlin
@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);
}
@Bean
fun introspector(builder: RestTemplateBuilder, properties: OAuth2ResourceServerProperties): OpaqueTokenIntrospector? {
val rest: RestOperations = builder
.basicAuthentication(properties.opaquetoken.clientId, properties.opaquetoken.clientSecret)
.setConnectTimeout(Duration.ofSeconds(60))
.setReadTimeout(Duration.ofSeconds(60))
.build()
return NimbusOpaqueTokenIntrospector(introspectionUri, rest)
}
使用 JWT 进行内省
一个常见的问题是内省是否与 JWT 兼容。Spring Security 的不透明令牌支持旨在不关心令牌的格式——它将乐于将任何令牌传递给提供的内省端点。
因此,假设您有一个要求,要求您在每次请求时都与授权服务器进行检查,以防 JWT 被吊销。
即使您使用 JWT 格式作为令牌,您的验证方法也是内省,这意味着您需要执行以下操作:
spring:
security:
oauth2:
resourceserver:
opaque-token:
introspection-uri: https://idp.example.org/introspection
client-id: client
client-secret: secret
在这种情况下,生成的Authentication
将是BearerTokenAuthentication
。相应的OAuth2AuthenticatedPrincipal
中的任何属性都将是内省端点返回的任何内容。
但是,假设奇怪的是,内省端点仅返回令牌是否有效。现在怎么办?
在这种情况下,您可以创建一个自定义OpaqueTokenIntrospector
,它仍然会访问端点,然后更新返回的主体以使 JWT 声明成为属性。
-
Java
-
Kotlin
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();
}
}
}
class JwtOpaqueTokenIntrospector : OpaqueTokenIntrospector {
private val delegate: OpaqueTokenIntrospector = NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret")
private val jwtDecoder: JwtDecoder = NimbusJwtDecoder(ParseOnlyJWTProcessor())
override fun introspect(token: String): OAuth2AuthenticatedPrincipal {
val principal = delegate.introspect(token)
return try {
val jwt: Jwt = jwtDecoder.decode(token)
DefaultOAuth2AuthenticatedPrincipal(jwt.claims, NO_AUTHORITIES)
} catch (ex: JwtException) {
throw OAuth2IntrospectionException(ex.message)
}
}
private class ParseOnlyJWTProcessor : DefaultJWTProcessor<SecurityContext>() {
override fun process(jwt: SignedJWT, context: SecurityContext): JWTClaimsSet {
return jwt.jwtClaimsSet
}
}
}
此后,只需将其作为@Bean
公开即可配置此自定义内省器。
-
Java
-
Kotlin
@Bean
public OpaqueTokenIntrospector introspector() {
return new JwtOpaqueTokenIntrospector();
}
@Bean
fun introspector(): OpaqueTokenIntrospector {
return JwtOpaqueTokenIntrospector()
}
调用/userinfo
端点
一般来说,资源服务器并不关心底层用户,而是关心已授予的权限。
也就是说,有时将授权语句与用户关联起来可能很有价值。
如果应用程序也使用spring-security-oauth2-client
并设置了相应的ClientRegistrationRepository
,则使用自定义OpaqueTokenIntrospector
非常简单。以下实现执行三件事:
-
委托给内省端点,以确认令牌的有效性。
-
查找与
/userinfo
端点关联的相应客户端注册。 -
调用并返回
/userinfo
端点的响应。
-
Java
-
Kotlin
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);
}
}
class UserInfoOpaqueTokenIntrospector : OpaqueTokenIntrospector {
private val delegate: OpaqueTokenIntrospector = NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret")
private val oauth2UserService = DefaultOAuth2UserService()
private val repository: ClientRegistrationRepository? = null
// ... constructor
override fun introspect(token: String): OAuth2AuthenticatedPrincipal {
val authorized = delegate.introspect(token)
val issuedAt: Instant? = authorized.getAttribute(ISSUED_AT)
val expiresAt: Instant? = authorized.getAttribute(EXPIRES_AT)
val clientRegistration: ClientRegistration = repository!!.findByRegistrationId("registration-id")
val accessToken = OAuth2AccessToken(BEARER, token, issuedAt, expiresAt)
val oauth2UserRequest = OAuth2UserRequest(clientRegistration, accessToken)
return oauth2UserService.loadUser(oauth2UserRequest)
}
}
如果您不使用spring-security-oauth2-client
,它仍然非常简单。您只需要使用您自己的WebClient
实例调用/userinfo
。
-
Java
-
Kotlin
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);
}
}
class UserInfoOpaqueTokenIntrospector : OpaqueTokenIntrospector {
private val delegate: OpaqueTokenIntrospector = NimbusOpaqueTokenIntrospector("https://idp.example.org/introspect", "client", "secret")
private val rest: WebClient = WebClient.create()
override fun introspect(token: String): OAuth2AuthenticatedPrincipal {
val authorized = delegate.introspect(token)
return makeUserInfoRequest(authorized)
}
}
无论哪种方式,创建了OpaqueTokenIntrospector
后,都应将其作为@Bean
发布以覆盖默认值。
-
Java
-
Kotlin
@Bean
OpaqueTokenIntrospector introspector() {
return new UserInfoOpaqueTokenIntrospector(...);
}
@Bean
fun introspector(): OpaqueTokenIntrospector {
return UserInfoOpaqueTokenIntrospector(...)
}