KEYCLOAK-2630 Add caching for user consents

This commit is contained in:
mposolda 2016-03-21 13:04:30 +01:00
parent 4193856fbb
commit e2e96f7183
3 changed files with 127 additions and 10 deletions

View file

@ -17,12 +17,28 @@
package org.keycloak.models.cache.infinispan; package org.keycloak.models.cache.infinispan;
import org.keycloak.models.*; import org.keycloak.models.ClientModel;
import org.keycloak.models.cache.CacheUserProvider; import org.keycloak.models.GroupModel;
import org.keycloak.models.KeycloakSession;
import org.keycloak.models.ProtocolMapperModel;
import org.keycloak.models.RealmModel;
import org.keycloak.models.RoleContainerModel;
import org.keycloak.models.RoleModel;
import org.keycloak.models.UserConsentModel;
import org.keycloak.models.UserCredentialModel;
import org.keycloak.models.UserCredentialValueModel;
import org.keycloak.models.UserModel;
import org.keycloak.models.cache.infinispan.entities.CachedUser; import org.keycloak.models.cache.infinispan.entities.CachedUser;
import org.keycloak.models.cache.infinispan.entities.CachedUserConsent;
import org.keycloak.models.utils.KeycloakModelUtils; import org.keycloak.models.utils.KeycloakModelUtils;
import java.util.*; import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
/** /**
* @author <a href="mailto:bill@burkecentral.com">Bill Burke</a> * @author <a href="mailto:bill@burkecentral.com">Bill Burke</a>
@ -380,16 +396,28 @@ public class UserAdapter implements UserModel {
@Override @Override
public UserConsentModel getConsentByClient(String clientId) { public UserConsentModel getConsentByClient(String clientId) {
// TODO: caching? if (updated != null) return updated.getConsentByClient(clientId);
getDelegateForUpdate(); CachedUserConsent cachedConsent = cached.getConsents().get(clientId);
return updated.getConsentByClient(clientId); if (cachedConsent == null) {
return null;
}
return toConsentModel(cachedConsent);
} }
@Override @Override
public List<UserConsentModel> getConsents() { public List<UserConsentModel> getConsents() {
// TODO: caching? if (updated != null) return updated.getConsents();
getDelegateForUpdate(); Collection<CachedUserConsent> cachedConsents = cached.getConsents().values();
return updated.getConsents();
List<UserConsentModel> result = new LinkedList<>();
for (CachedUserConsent cachedConsent : cachedConsents) {
UserConsentModel consent = toConsentModel(cachedConsent);
if (consent != null) {
result.add(consent);
}
}
return result;
} }
@Override @Override
@ -404,6 +432,26 @@ public class UserAdapter implements UserModel {
return updated.revokeConsentForClient(clientId); return updated.revokeConsentForClient(clientId);
} }
private UserConsentModel toConsentModel(CachedUserConsent cachedConsent) {
ClientModel client = keycloakSession.realms().getClientById(cachedConsent.getClientDbId(), realm);
if (client == null) {
return null;
}
UserConsentModel consentModel = new UserConsentModel(client);
for (String roleId : cachedConsent.getRoleIds()) {
RoleModel role = keycloakSession.realms().getRoleById(roleId, realm);
if (role != null) {
consentModel.addGrantedRole(role);
}
}
for (ProtocolMapperModel protocolMapper : cachedConsent.getProtocolMappers()) {
consentModel.addGrantedProtocolMapper(protocolMapper);
}
return consentModel;
}
@Override @Override
public boolean equals(Object o) { public boolean equals(Object o) {
if (this == o) return true; if (this == o) return true;

View file

@ -20,14 +20,16 @@ package org.keycloak.models.cache.infinispan.entities;
import org.keycloak.models.GroupModel; import org.keycloak.models.GroupModel;
import org.keycloak.models.RealmModel; import org.keycloak.models.RealmModel;
import org.keycloak.models.RoleModel; import org.keycloak.models.RoleModel;
import org.keycloak.models.UserConsentModel;
import org.keycloak.models.UserCredentialValueModel; import org.keycloak.models.UserCredentialValueModel;
import org.keycloak.models.UserModel; import org.keycloak.models.UserModel;
import org.keycloak.common.util.MultivaluedHashMap; import org.keycloak.common.util.MultivaluedHashMap;
import java.io.Serializable; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map;
import java.util.Set; import java.util.Set;
/** /**
@ -51,6 +53,7 @@ public class CachedUser extends AbstractRevisioned implements InRealm {
private Set<String> requiredActions = new HashSet<>(); private Set<String> requiredActions = new HashSet<>();
private Set<String> roleMappings = new HashSet<>(); private Set<String> roleMappings = new HashSet<>();
private Set<String> groups = new HashSet<>(); private Set<String> groups = new HashSet<>();
private Map<String, CachedUserConsent> consents = new HashMap<>(); // Key is client DB Id
@ -79,6 +82,13 @@ public class CachedUser extends AbstractRevisioned implements InRealm {
groups.add(group.getId()); groups.add(group.getId());
} }
} }
List<UserConsentModel> consents = user.getConsents();
if (consents != null) {
for (UserConsentModel consent : consents) {
this.consents.put(consent.getClient().getId(), new CachedUserConsent(consent));
}
}
} }
public String getRealm() { public String getRealm() {
@ -144,4 +154,8 @@ public class CachedUser extends AbstractRevisioned implements InRealm {
public Set<String> getGroups() { public Set<String> getGroups() {
return groups; return groups;
} }
public Map<String, CachedUserConsent> getConsents() {
return consents;
}
} }

View file

@ -0,0 +1,55 @@
/*
* Copyright 2016 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.models.cache.infinispan.entities;
import java.util.HashSet;
import java.util.Set;
import org.keycloak.models.ProtocolMapperModel;
import org.keycloak.models.RoleModel;
import org.keycloak.models.UserConsentModel;
/**
* @author <a href="mailto:mposolda@redhat.com">Marek Posolda</a>
*/
public class CachedUserConsent {
private final String clientDbId;
private final Set<ProtocolMapperModel> protocolMappers = new HashSet<>();
private final Set<String> roleIds = new HashSet<>();
public CachedUserConsent(UserConsentModel consentModel) {
this.clientDbId = consentModel.getClient().getId();
this.protocolMappers.addAll(consentModel.getGrantedProtocolMappers());
for (RoleModel role : consentModel.getGrantedRoles()) {
this.roleIds.add(role.getId());
}
}
public String getClientDbId() {
return clientDbId;
}
public Set<ProtocolMapperModel> getProtocolMappers() {
return protocolMappers;
}
public Set<String> getRoleIds() {
return roleIds;
}
}