[请参阅下面的更新部分以获取最新版本]
我正在尝试使用 Spring Security 设置 Spring Boot REST 应用程序。我们的前端在 Okta 中使用 OIDC 对我们的用户进行身份验证,并在 Authorization Bearer 令牌中传递 JWT。Okta 正在使用 PKCE。
我正在引用以下示例,它似乎是一个非常简单的设置: https ://docs.spring.io/spring-security-oauth2-boot/docs/current/reference/html/boot-features-security-oauth2 -resource-server.html
我们的 Spring Boot 应用程序只是一个资源服务器,所以我想我只需要按照 2.1 到 2.2.3 节中的说明进行操作即可。我@EnableResourceServer
在我的应用程序中有以下内容application.yml
:
spring:
security:
oauth2:
resource:
jwt:
key-set-uri: https://dcsg-hub-dev.okta.com/oauth2/v1/keysZ?client_id=???
当我使用 Postman 调用我的应用程序时,我收到 401 错误:
{
"error": "invalid_token",
"error_description": "Invalid access token: eyJraWQ..."
}
它显示了有效的令牌,我可以通过调用 Oktauserinfo
端点来验证。不幸的是,它没有给我任何其他信息。
org.springframework.security.oauth2.provider.authentication.OAuth2AuthenticationProcessingFilter
我最终在课堂上设置了断点。进入从该位置调用的代码,错误是在代码中引起的org.springframework.security.oauth2.provider.authentication.OAuth2AuthenticationManager
:
public Authentication authenticate(Authentication authentication) throws AuthenticationException {
if (authentication == null) {
throw new InvalidTokenException("Invalid token (token not found)");
} else {
String token = (String)authentication.getPrincipal();
OAuth2Authentication auth = this.tokenServices.loadAuthentication(token);
if (auth == null) {
throw new InvalidTokenException("Invalid token: " + token);
在同一类中调用以下代码:
public OAuth2Authentication loadAuthentication(String accessTokenValue) throws AuthenticationException, InvalidTokenException {
OAuth2AccessToken accessToken = this.tokenStore.readAccessToken(accessTokenValue);
if (accessToken == null) {
throw new InvalidTokenException("Invalid access token: " + accessTokenValue);
似乎代码使用令牌本身作为从哈希映射中检索令牌的键。正如您在上面看到的,它getPrincipal()
用于获取“令牌”。在我看来,它认为令牌应该是一个用户 ID,它试图用它来查找用户的令牌。当值已经存在时,它使用令牌作为获取值的键似乎很奇怪。
有任何想法吗?
<<<<<<<<<<<<<<更新>>>>>>>>>>>>>>>>
我关注了此链接中的文档: https ://github.com/spring-projects/spring-security/tree/5.2.1.RELEASE/samples/boot/oauth2resourceserver-jwe
我application.yml
现在看起来像这样:
spring:
security:
oauth2:
resourceserver:
jwt:
issuer-uri: https://dcsg-hub-dev.okta.com
jwk-set-uri: https://dcsg-hub-dev.okta.com/oauth2/v1/keys?client_id=
这现在肯定是调用 JWK URI,而之前不是。但是,现在我收到以下错误:
Caused by: com.nimbusds.jose.proc.BadJOSEException: Signed JWT rejected: Another algorithm expected, or no matching key(s) found
我在该代码中设置了断点,无论出于何种原因,kid
从 Okta 中返回的令牌/JWT 与 Okta 中的任何键都不匹配jwk-set-uri
。