[fixes #9225] - Get scopeIds from the AuthorizationRequestContext instead of session if DYNAMIC_SCOPES are enabled
Add a test to make sure ProtocolMappers run with Dynamic Scopes Change the way we create the DefaultClientSessionContext with respect to OAuth2 scopes, and standardize the way we obtain them from the parameter
This commit is contained in:
parent
8e6489459d
commit
76101e3591
8 changed files with 135 additions and 12 deletions
|
@ -29,6 +29,7 @@ import org.keycloak.broker.oidc.OIDCIdentityProvider;
|
|||
import org.keycloak.broker.provider.IdentityBrokerException;
|
||||
import org.keycloak.cluster.ClusterProvider;
|
||||
import org.keycloak.common.ClientConnection;
|
||||
import org.keycloak.common.Profile;
|
||||
import org.keycloak.common.VerificationException;
|
||||
import org.keycloak.common.util.Time;
|
||||
import org.keycloak.crypto.HashProvider;
|
||||
|
@ -76,6 +77,7 @@ import org.keycloak.services.managers.AuthenticationSessionManager;
|
|||
import org.keycloak.services.managers.UserSessionCrossDCManager;
|
||||
import org.keycloak.services.managers.UserSessionManager;
|
||||
import org.keycloak.services.resources.IdentityBrokerService;
|
||||
import org.keycloak.services.util.AuthorizationContextUtil;
|
||||
import org.keycloak.services.util.DefaultClientSessionContext;
|
||||
import org.keycloak.services.util.MtlsHoKTokenUtil;
|
||||
import org.keycloak.sessions.AuthenticationSessionModel;
|
||||
|
@ -544,7 +546,14 @@ public class TokenManager {
|
|||
clientSession.setRedirectUri(authSession.getRedirectUri());
|
||||
clientSession.setProtocol(authSession.getProtocol());
|
||||
|
||||
Set<String> clientScopeIds = authSession.getClientScopes();
|
||||
Set<String> clientScopeIds;
|
||||
if (Profile.isFeatureEnabled(Profile.Feature.DYNAMIC_SCOPES)) {
|
||||
clientScopeIds = AuthorizationContextUtil.getClientScopesStreamFromAuthorizationRequestContextWithClient(session, authSession.getClientNote(OAuth2Constants.SCOPE))
|
||||
.map(ClientScopeModel::getId)
|
||||
.collect(Collectors.toSet());
|
||||
} else {
|
||||
clientScopeIds = authSession.getClientScopes();
|
||||
}
|
||||
|
||||
Map<String, String> transferredNotes = authSession.getClientNotes();
|
||||
for (Map.Entry<String, String> entry : transferredNotes.entrySet()) {
|
||||
|
|
|
@ -424,7 +424,7 @@ public class TokenEndpoint {
|
|||
throw new CorsErrorResponseException(cors, OAuthErrorException.INVALID_SCOPE, "Client no longer has requested consent from user", Response.Status.BAD_REQUEST);
|
||||
}
|
||||
|
||||
ClientSessionContext clientSessionCtx = DefaultClientSessionContext.fromClientSessionAndClientScopes(clientSession, clientScopesSupplier.get(), session);
|
||||
ClientSessionContext clientSessionCtx = DefaultClientSessionContext.fromClientSessionAndScopeParameter(clientSession, scopeParam, session);
|
||||
|
||||
// Set nonce as an attribute in the ClientSessionContext. Will be used for the token generation
|
||||
clientSessionCtx.setAttribute(OIDCLoginProtocol.NONCE_PARAM, codeData.getNonce());
|
||||
|
|
|
@ -212,7 +212,7 @@ public class CibaGrantType {
|
|||
}
|
||||
|
||||
ClientSessionContext clientSessionCtx = DefaultClientSessionContext
|
||||
.fromClientSessionAndClientScopes(userSession.getAuthenticatedClientSessionByClient(client.getId()), TokenManager.getRequestedClientScopes(scopeParam, client), session);
|
||||
.fromClientSessionAndScopeParameter(userSession.getAuthenticatedClientSessionByClient(client.getId()), scopeParam, session);
|
||||
|
||||
int authTime = Time.currentTime();
|
||||
userSession.setNote(AuthenticationManager.AUTH_TIME, String.valueOf(authTime));
|
||||
|
|
|
@ -292,8 +292,8 @@ public class DeviceGrantType {
|
|||
"Client no longer has requested consent from user", Response.Status.BAD_REQUEST);
|
||||
}
|
||||
|
||||
ClientSessionContext clientSessionCtx = DefaultClientSessionContext.fromClientSessionAndClientScopes(clientSession,
|
||||
TokenManager.getRequestedClientScopes(scopeParam, client), session);
|
||||
ClientSessionContext clientSessionCtx = DefaultClientSessionContext.fromClientSessionAndScopeParameter(clientSession,
|
||||
scopeParam, session);
|
||||
|
||||
// Set nonce as an attribute in the ClientSessionContext. Will be used for the token generation
|
||||
clientSessionCtx.setAttribute(OIDCLoginProtocol.NONCE_PARAM, deviceCodeModel.getNonce());
|
||||
|
|
|
@ -1192,9 +1192,7 @@ public class AuthenticationManager {
|
|||
//if Dynamic Scopes are enabled, get the scopes from the AuthorizationRequestContext, passing the session and scopes as parameters
|
||||
// then concat a Stream with the ClientModel, as it's discarded in the getAuthorizationRequestContext method
|
||||
if (Profile.isFeatureEnabled(Profile.Feature.DYNAMIC_SCOPES)) {
|
||||
return Stream.concat(AuthorizationContextUtil.getAuthorizationRequestContextFromScopes(session, authSession.getClientNote(OAuth2Constants.SCOPE))
|
||||
.getAuthorizationDetailEntries().stream(),
|
||||
Collections.singletonList(new AuthorizationDetails(session.getContext().getClient())).stream());
|
||||
return AuthorizationContextUtil.getAuthorizationRequestsStreamFromScopesWithClient(session, authSession.getClientNote(OAuth2Constants.SCOPE));
|
||||
}
|
||||
// if dynamic scopes are not enabled, we retain the old behaviour, but the ClientScopes will be wrapped in
|
||||
// AuthorizationRequest objects to standardize the code handling these.
|
||||
|
|
|
@ -1,13 +1,47 @@
|
|||
/*
|
||||
* Copyright 2022 Red Hat, Inc. and/or its affiliates
|
||||
* and other contributors as indicated by the @author tags.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package org.keycloak.services.util;
|
||||
|
||||
import org.keycloak.common.Profile;
|
||||
import org.keycloak.models.ClientScopeModel;
|
||||
import org.keycloak.models.KeycloakSession;
|
||||
import org.keycloak.protocol.oidc.rar.AuthorizationRequestParserProvider;
|
||||
import org.keycloak.protocol.oidc.rar.parsers.ClientScopeAuthorizationRequestParserProviderFactory;
|
||||
import org.keycloak.rar.AuthorizationDetails;
|
||||
import org.keycloak.rar.AuthorizationRequestContext;
|
||||
import org.keycloak.rar.AuthorizationRequestSource;
|
||||
|
||||
import java.util.stream.Stream;
|
||||
|
||||
|
||||
/**
|
||||
* @author <a href="mailto:dgozalob@redhat.com">Daniel Gozalo</a>
|
||||
* Util class to unify a way to obtain the {@link AuthorizationRequestContext}.
|
||||
* <p>
|
||||
* As it can be obtained statically from just the OAuth2 scopes parameter, it can be easily referenced from almost anywhere.
|
||||
*/
|
||||
public class AuthorizationContextUtil {
|
||||
|
||||
/**
|
||||
* Base function to obtain a bare AuthorizationRequestContext with just OAuth2 Scopes
|
||||
* @param session
|
||||
* @param scope
|
||||
* @return an {@link AuthorizationRequestContext} with scope entries
|
||||
*/
|
||||
public static AuthorizationRequestContext getAuthorizationRequestContextFromScopes(KeycloakSession session, String scope) {
|
||||
if (!Profile.isFeatureEnabled(Profile.Feature.DYNAMIC_SCOPES)) {
|
||||
throw new RuntimeException("The Dynamic Scopes feature is not enabled and the AuthorizationRequestContext hasn't been generated");
|
||||
|
@ -23,4 +57,38 @@ public class AuthorizationContextUtil {
|
|||
return clientScopeParser.parseScopes(scope);
|
||||
}
|
||||
|
||||
/**
|
||||
* An extension of {@link AuthorizationContextUtil#getAuthorizationRequestContextFromScopes} that appends the current context's client
|
||||
* @param session
|
||||
* @param scope
|
||||
* @return an {@link AuthorizationRequestContext} with scope entries and a ClientModel
|
||||
*/
|
||||
public static AuthorizationRequestContext getAuthorizationRequestContextFromScopesWithClient(KeycloakSession session, String scope) {
|
||||
AuthorizationRequestContext authorizationRequestContext = getAuthorizationRequestContextFromScopes(session, scope);
|
||||
authorizationRequestContext.getAuthorizationDetailEntries().add(new AuthorizationDetails(session.getContext().getClient()));
|
||||
return authorizationRequestContext;
|
||||
}
|
||||
|
||||
/**
|
||||
* An extension of {@link AuthorizationContextUtil#getAuthorizationRequestContextFromScopesWithClient)} that returns the list as a Stream
|
||||
* @param session
|
||||
* @param scope
|
||||
* @return a Stream of {@link AuthorizationDetails} containing a ClientModel
|
||||
*/
|
||||
public static Stream<AuthorizationDetails> getAuthorizationRequestsStreamFromScopesWithClient(KeycloakSession session, String scope) {
|
||||
AuthorizationRequestContext authorizationRequestContext = getAuthorizationRequestContextFromScopesWithClient(session, scope);
|
||||
return authorizationRequestContext.getAuthorizationDetailEntries().stream();
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper method to return a Stream of all the {@link ClientScopeModel} in the current {@link AuthorizationRequestContext}
|
||||
* @param session
|
||||
* @param scope
|
||||
* @return see description
|
||||
*/
|
||||
public static Stream<ClientScopeModel> getClientScopesStreamFromAuthorizationRequestContextWithClient(KeycloakSession session, String scope) {
|
||||
return getAuthorizationRequestContextFromScopesWithClient(session, scope).getAuthorizationDetailEntries().stream()
|
||||
.filter(authorizationDetails -> authorizationDetails.getSource() == AuthorizationRequestSource.SCOPE)
|
||||
.map(AuthorizationDetails::getClientScope);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -71,8 +71,8 @@ public class DefaultClientSessionContext implements ClientSessionContext {
|
|||
private Map<String, Object> attributes = new HashMap<>();
|
||||
|
||||
private DefaultClientSessionContext(AuthenticatedClientSessionModel clientSession, Set<String> clientScopeIds, KeycloakSession session) {
|
||||
this.clientSession = clientSession;
|
||||
this.clientScopeIds = clientScopeIds;
|
||||
this.clientSession = clientSession;
|
||||
this.session = session;
|
||||
}
|
||||
|
||||
|
@ -86,7 +86,12 @@ public class DefaultClientSessionContext implements ClientSessionContext {
|
|||
|
||||
|
||||
public static DefaultClientSessionContext fromClientSessionAndScopeParameter(AuthenticatedClientSessionModel clientSession, String scopeParam, KeycloakSession session) {
|
||||
Stream<ClientScopeModel> requestedClientScopes = TokenManager.getRequestedClientScopes(scopeParam, clientSession.getClient());
|
||||
Stream<ClientScopeModel> requestedClientScopes;
|
||||
if (Profile.isFeatureEnabled(Profile.Feature.DYNAMIC_SCOPES)) {
|
||||
requestedClientScopes = AuthorizationContextUtil.getClientScopesStreamFromAuthorizationRequestContextWithClient(session, scopeParam);
|
||||
} else {
|
||||
requestedClientScopes = TokenManager.getRequestedClientScopes(scopeParam, clientSession.getClient());
|
||||
}
|
||||
return fromClientSessionAndClientScopes(clientSession, requestedClientScopes, session);
|
||||
}
|
||||
|
||||
|
@ -96,7 +101,10 @@ public class DefaultClientSessionContext implements ClientSessionContext {
|
|||
}
|
||||
|
||||
|
||||
public static DefaultClientSessionContext fromClientSessionAndClientScopes(AuthenticatedClientSessionModel clientSession,
|
||||
// in order to standardize the way we create this object and with that data, it's better to compute the client scopes internally instead of relying on external sources
|
||||
// i.e: the TokenManager.getRequestedClientScopes was being called in many places to obtain the ClientScopeModel stream.
|
||||
// by changing this method to private, we'll only call it in this class, while also having a single place to put the DYNAMIC_SCOPES feature flag condition
|
||||
private static DefaultClientSessionContext fromClientSessionAndClientScopes(AuthenticatedClientSessionModel clientSession,
|
||||
Stream<ClientScopeModel> clientScopes,
|
||||
KeycloakSession session) {
|
||||
Set<String> clientScopeIds = clientScopes.map(ClientScopeModel::getId).collect(Collectors.toSet());
|
||||
|
@ -157,7 +165,7 @@ public class DefaultClientSessionContext implements ClientSessionContext {
|
|||
|
||||
@Override
|
||||
public String getScopeString() {
|
||||
if(Profile.isFeatureEnabled(Profile.Feature.DYNAMIC_SCOPES)) {
|
||||
if (Profile.isFeatureEnabled(Profile.Feature.DYNAMIC_SCOPES)) {
|
||||
String scopeParam = buildScopesStringFromAuthorizationRequest();
|
||||
logger.tracef("Generated scope param with Dynamic Scopes enabled: %1s", scopeParam);
|
||||
String scopeSent = clientSession.getNote(OAuth2Constants.SCOPE);
|
||||
|
|
|
@ -30,6 +30,7 @@ import org.keycloak.common.Profile;
|
|||
import org.keycloak.common.util.UriUtils;
|
||||
import org.keycloak.jose.jws.JWSInput;
|
||||
import org.keycloak.models.AccountRoles;
|
||||
import org.keycloak.models.ClientScopeModel;
|
||||
import org.keycloak.protocol.oidc.OIDCLoginProtocol;
|
||||
import org.keycloak.protocol.oidc.OIDCLoginProtocolFactory;
|
||||
import org.keycloak.protocol.oidc.mappers.AddressMapper;
|
||||
|
@ -65,6 +66,7 @@ import javax.ws.rs.core.Response;
|
|||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.Arrays;
|
||||
import java.util.Collection;
|
||||
import java.util.Collections;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
@ -1268,6 +1270,44 @@ public class OIDCProtocolMappersTest extends AbstractKeycloakTest {
|
|||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
@EnableFeature(value = Profile.Feature.DYNAMIC_SCOPES, skipRestart = true)
|
||||
public void executeTokenMappersOnDynamicScopes() {
|
||||
ClientResource clientResource = findClientResourceByClientId(adminClient.realm("test"), "test-app");
|
||||
ClientScopeRepresentation scopeRep = new ClientScopeRepresentation();
|
||||
scopeRep.setName("dyn-scope-with-mapper");
|
||||
scopeRep.setProtocol("openid-connect");
|
||||
scopeRep.setAttributes(new HashMap<String, String>() {{
|
||||
put(ClientScopeModel.IS_DYNAMIC_SCOPE, "true");
|
||||
put(ClientScopeModel.DYNAMIC_SCOPE_REGEXP, "dyn-scope-with-mapper:*");
|
||||
}});
|
||||
// create the attribute mapper
|
||||
ProtocolMapperRepresentation protocolMapperRepresentation = createHardcodedClaim("dynamic-scope-hardcoded-mapper", "hardcoded-foo", "hardcoded-bar", "String", true, true);
|
||||
scopeRep.setProtocolMappers(Collections.singletonList(protocolMapperRepresentation));
|
||||
|
||||
try (Response resp = adminClient.realm("test").clientScopes().create(scopeRep)) {
|
||||
assertEquals(201, resp.getStatus());
|
||||
String clientScopeId = ApiUtil.getCreatedId(resp);
|
||||
getCleanup().addClientScopeId(clientScopeId);
|
||||
clientResource.addOptionalClientScope(clientScopeId);
|
||||
}
|
||||
|
||||
oauth.scope("openid dyn-scope-with-mapper:value");
|
||||
OAuthClient.AccessTokenResponse response = browserLogin("password", "test-user@localhost", "password");
|
||||
IDToken idToken = oauth.verifyIDToken(response.getIdToken());
|
||||
AccessToken accessToken = oauth.verifyToken(response.getAccessToken());
|
||||
|
||||
assertNotNull(idToken.getOtherClaims());
|
||||
assertNotNull(idToken.getOtherClaims().get("hardcoded-foo"));
|
||||
assertTrue(idToken.getOtherClaims().get("hardcoded-foo") instanceof String);
|
||||
assertEquals("hardcoded-bar", idToken.getOtherClaims().get("hardcoded-foo"));
|
||||
|
||||
assertNotNull(accessToken.getOtherClaims());
|
||||
assertNotNull(accessToken.getOtherClaims().get("hardcoded-foo"));
|
||||
assertTrue(accessToken.getOtherClaims().get("hardcoded-foo") instanceof String);
|
||||
assertEquals("hardcoded-bar", accessToken.getOtherClaims().get("hardcoded-foo"));
|
||||
}
|
||||
|
||||
private void assertRoles(List<String> actualRoleList, String ...expectedRoles){
|
||||
Assert.assertNames(actualRoleList, expectedRoles);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue