Coverage Report - org.kuali.rice.kim.service.impl.RoleServiceBase
 
Classes in this File Line Coverage Branch Coverage Complexity
RoleServiceBase
0%
0/534
0%
0/436
4.067
RoleServiceBase$1
0%
0/1
N/A
4.067
RoleServiceBase$RoleDaoAction
0%
0/10
N/A
4.067
RoleServiceBase$RoleMemberCacheKeyHelper
0%
0/11
0%
0/2
4.067
 
 1  
 /*
 2  
  * Copyright 2007-2010 The Kuali Foundation
 3  
  *
 4  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 5  
  * you may not use this file except in compliance with the License.
 6  
  * You may obtain a copy of the License at
 7  
  *
 8  
  * http://www.opensource.org/licenses/ecl2.php
 9  
  *
 10  
  * Unless required by applicable law or agreed to in writing, software
 11  
  * distributed under the License is distributed on an "AS IS" BASIS,
 12  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13  
  * See the License for the specific language governing permissions and
 14  
  * limitations under the License.
 15  
  */
 16  
 package org.kuali.rice.kim.service.impl;
 17  
 
 18  
 import java.util.ArrayList;
 19  
 import java.util.Collection;
 20  
 import java.util.Collections;
 21  
 import java.util.HashMap;
 22  
 import java.util.HashSet;
 23  
 import java.util.List;
 24  
 import java.util.Map;
 25  
 import java.util.Set;
 26  
 
 27  
 import javax.xml.namespace.QName;
 28  
 
 29  
 import org.apache.commons.collections.CollectionUtils;
 30  
 import org.apache.commons.lang.StringUtils;
 31  
 import org.kuali.rice.kew.service.KEWServiceLocator;
 32  
 import org.kuali.rice.kew.util.KEWConstants;
 33  
 import org.kuali.rice.kim.bo.Group;
 34  
 import org.kuali.rice.kim.bo.Role;
 35  
 import org.kuali.rice.kim.bo.entity.KimPrincipal;
 36  
 import org.kuali.rice.kim.bo.impl.RoleImpl;
 37  
 import org.kuali.rice.kim.bo.role.dto.DelegateMemberCompleteInfo;
 38  
 import org.kuali.rice.kim.bo.role.dto.RoleMemberCompleteInfo;
 39  
 import org.kuali.rice.kim.bo.role.dto.RoleResponsibilityActionInfo;
 40  
 import org.kuali.rice.kim.bo.role.impl.KimDelegationImpl;
 41  
 import org.kuali.rice.kim.bo.role.impl.KimDelegationMemberImpl;
 42  
 import org.kuali.rice.kim.bo.role.impl.RoleMemberAttributeDataImpl;
 43  
 import org.kuali.rice.kim.bo.role.impl.RoleMemberImpl;
 44  
 import org.kuali.rice.kim.bo.role.impl.RoleResponsibilityActionImpl;
 45  
 import org.kuali.rice.kim.bo.types.dto.AttributeSet;
 46  
 import org.kuali.rice.kim.bo.types.impl.KimAttributeImpl;
 47  
 import org.kuali.rice.kim.dao.KimRoleDao;
 48  
 import org.kuali.rice.kim.service.IdentityManagementNotificationService;
 49  
 import org.kuali.rice.kim.service.IdentityManagementService;
 50  
 import org.kuali.rice.kim.service.KIMServiceLocator;
 51  
 import org.kuali.rice.kim.service.ResponsibilityInternalService;
 52  
 import org.kuali.rice.kim.service.support.KimDelegationTypeService;
 53  
 import org.kuali.rice.kim.service.support.KimRoleTypeService;
 54  
 import org.kuali.rice.kim.util.KIMPropertyConstants;
 55  
 import org.kuali.rice.kim.util.KimCommonUtils;
 56  
 import org.kuali.rice.kim.util.KimConstants;
 57  
 import org.kuali.rice.kns.service.BusinessObjectService;
 58  
 import org.kuali.rice.kns.service.KNSServiceLocator;
 59  
 import org.kuali.rice.kns.service.LookupService;
 60  
 import org.kuali.rice.kns.service.SequenceAccessorService;
 61  
 import org.kuali.rice.kns.util.KNSPropertyConstants;
 62  
 import org.kuali.rice.ksb.cache.RiceCacheAdministrator;
 63  
 import org.kuali.rice.ksb.service.KSBServiceLocator;
 64  
 
 65  
 /**
 66  
  * This is a description of what this class does - jjhanso don't forget to fill this in. 
 67  
  * 
 68  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 69  
  *
 70  
  */
 71  0
 public class RoleServiceBase {
 72  
         protected static final String ROLE_IMPL_CACHE_PREFIX = "RoleImpl-ID-";
 73  
         protected static final String ROLE_IMPL_BY_NAME_CACHE_PREFIX = "RoleImpl-Name-";
 74  
         protected static final String ROLE_IMPL_CACHE_GROUP = "RoleImpl";
 75  
         
 76  
         protected static final String ROLE_MEMBER_IMPL_CACHE_PREFIX = "RoleMemberImpl-ID-";
 77  
         protected static final String ROLE_MEMBER_IMPL_LIST_CACHE_PREFIX = "RoleMemberImpl-List-";
 78  
         protected static final String ROLE_MEMBER_IMPL_CACHE_GROUP = "RoleMemberImpl";
 79  
         
 80  
         protected static final String DELEGATION_IMPL_CACHE_PREFIX = "KimDelegationImpl-ID-";
 81  
         protected static final String DELEGATION_IMPL_LIST_CACHE_PREFIX = "KimDelegationImpl-List-";
 82  
         protected static final String DELEGATION_IMPL_CACHE_GROUP = "KimDelegationImpl";
 83  
         
 84  
         protected static final String DELEGATION_MEMBER_IMPL_CACHE_PREFIX = "KimDelegationMemberImpl-ID-";
 85  
         protected static final String DELEGATION_MEMBER_IMPL_BY_DLGN_AND_ID_CACHE_PREFIX = "KimDelegationMemberImpl-DelegationAndId-";
 86  
         protected static final String DELEGATION_MEMBER_IMPL_LIST_CACHE_PREFIX = "KimDelegationMemberImpl-List-";
 87  
         protected static final String DELEGATION_MEMBER_IMPL_LIST_BY_MBR_DLGN_CACHE_PREFIX = "KimDelegationMemberImpl-List-MemberAndDelegationId-";
 88  
         protected static final String DELEGATION_MEMBER_IMPL_CACHE_GROUP = "KimDelegationMemberImpl";
 89  
         
 90  
         private BusinessObjectService businessObjectService;
 91  
         private LookupService lookupService;
 92  
         private RiceCacheAdministrator cacheAdministrator;
 93  
         private SequenceAccessorService sequenceAccessorService;
 94  
         private IdentityManagementService identityManagementService;
 95  
         private ResponsibilityInternalService responsibilityInternalService;
 96  
 
 97  0
     private Map<String,KimRoleTypeService> roleTypeServiceCache = Collections.synchronizedMap( new HashMap<String,KimRoleTypeService>() );
 98  0
         private Map<String,KimDelegationTypeService> delegationTypeServiceCache = Collections.synchronizedMap( new HashMap<String,KimDelegationTypeService>() );
 99  
 
 100  0
         private Map<String,Boolean> applicationRoleTypeCache = Collections.synchronizedMap( new HashMap<String,Boolean>() );
 101  
         private KimRoleDao roleDao;
 102  
         
 103  
         /**
 104  
          * A helper enumeration for indicating which KimRoleDao method to use when attempting to get role/delegation-related lists that are not in the cache.
 105  
          * 
 106  
          * @author Kuali Rice Team (rice.collab@kuali.org)
 107  
          */
 108  0
         protected static enum RoleDaoAction {
 109  0
                 ROLE_PRINCIPALS_FOR_PRINCIPAL_ID_AND_ROLE_IDS("principalMembers-"),
 110  0
                 ROLE_GROUPS_FOR_GROUP_IDS_AND_ROLE_IDS("groupMembers-"),
 111  0
                 ROLE_MEMBERS_FOR_ROLE_IDS("membersOfRole-"),
 112  0
                 ROLE_MEMBERSHIPS_FOR_ROLE_IDS_AS_MEMBERS("rolesAsMembers-"),
 113  0
                 ROLE_MEMBERS_FOR_ROLE_IDS_WITH_FILTERS("roleIdsWithFilters-"),
 114  0
                 DELEGATION_PRINCIPALS_FOR_PRINCIPAL_ID_AND_DELEGATION_IDS("delegationPrincipals-"),
 115  0
                 DELEGATION_GROUPS_FOR_GROUP_IDS_AND_DELEGATION_IDS("delegationGroups-"),
 116  0
                 DELEGATION_MEMBERS_FOR_DELEGATION_IDS("delegationMembers-");
 117  
                 
 118  
                 public final String DAO_ACTION_CACHE_PREFIX;
 119  0
                 private RoleDaoAction(String daoActionCachePrefix) { this.DAO_ACTION_CACHE_PREFIX = daoActionCachePrefix;  }
 120  
         }
 121  
         
 122  
     // -----------------------------------------------------------------------------------------------------------------
 123  
     // Role Membership Caching Methods
 124  
     // -----------------------------------------------------------------------------------------------------------------
 125  
 
 126  
         /**
 127  
          * Generates a String key to use for storing or retrieving a RoleMemberImpl to/from the cache.
 128  
          * 
 129  
          * @param roleMemberId The ID of the RoleMemberImpl to generate a key for.
 130  
          * @return A cache key for the RoleMemberImpl with the given ID.
 131  
          */
 132  
         protected String getRoleMemberCacheKey(String roleMemberId) {
 133  0
                 return ROLE_MEMBER_IMPL_CACHE_PREFIX + roleMemberId;
 134  
         }
 135  
         
 136  
         /**
 137  
          * Generates a String key to use for storing or retrieving a list of RoleMemberImpls to/from the cache. The key is generated by specifying
 138  
          * certain properties that are common to all the RoleMemberImpl instances in the list to be cached. Note that at least one common
 139  
          * property will always be null; for instance, a role member cannot have a member ID that refers to both a principal and a group, so at least
 140  
          * the principalId or the groupId parameter will have a null value passed in by the calling code in this RoleService implementation. Also,
 141  
          * the value of the RoleDaoAction parameter passed in will affect which subsequent parameters will be used in generating the cache key.
 142  
          * 
 143  
          * @param roleDaoAction The RoleDaoAction signifying which KimRoleDao call found this list; will determine how and which parameters are used.
 144  
          * @param roleId The role ID (possibly as a member ID) shared among the members of the list; will be interpreted as an empty String if this is blank.
 145  
          * @param principalId The (principal) member ID shared among the members of the list; will be interpreted as an empty String if this is blank.
 146  
          * @param groupId The (group) member ID shared among the members of the list; will be interpreted as an empty String if this is blank.
 147  
          * @param memberTypeCode The member type code shared among the members of the list; will be interpreted as an empty String if this is blank.
 148  
          * @return A cache key for the RoleMemberImpl list whose members share the given roleId, principalId, groupId, and memberTypeCode.
 149  
          * @throws IllegalArgumentException if the RoleDaoAction parameter does not refer to a role-member-related enumeration value.
 150  
          */
 151  
         protected String getRoleMemberListCacheKey(RoleDaoAction roleDaoAction, String roleId, String principalId, String groupId, String memberTypeCode) {
 152  0
                 switch (roleDaoAction) {
 153  
                         case ROLE_PRINCIPALS_FOR_PRINCIPAL_ID_AND_ROLE_IDS : // Search for principal role members only.
 154  0
                                 return new StringBuilder(ROLE_MEMBER_IMPL_LIST_CACHE_PREFIX).append(roleDaoAction.DAO_ACTION_CACHE_PREFIX).append(
 155  
                                                 StringUtils.isBlank(roleId) ? "" : roleId).append('-').append(StringUtils.isBlank(principalId) ? "" : principalId).toString();
 156  
                         case ROLE_GROUPS_FOR_GROUP_IDS_AND_ROLE_IDS : // Search for group role members only.
 157  0
                                 return new StringBuilder(ROLE_MEMBER_IMPL_LIST_CACHE_PREFIX).append(roleDaoAction.DAO_ACTION_CACHE_PREFIX).append(
 158  
                                                 StringUtils.isBlank(roleId) ? "" : roleId).append('-').append(StringUtils.isBlank(groupId) ? "" : groupId).toString();
 159  
                         case ROLE_MEMBERS_FOR_ROLE_IDS : // Search for role members with the given member type code.
 160  0
                                 return new StringBuilder(ROLE_MEMBER_IMPL_LIST_CACHE_PREFIX).append(roleDaoAction.DAO_ACTION_CACHE_PREFIX).append(
 161  
                                                 StringUtils.isBlank(roleId) ? "" : roleId).append('-').append(StringUtils.isBlank(memberTypeCode) ? "" : memberTypeCode).toString();
 162  
                         case ROLE_MEMBERSHIPS_FOR_ROLE_IDS_AS_MEMBERS : // Search for role members who are also roles.
 163  0
                                 return new StringBuilder(ROLE_MEMBER_IMPL_LIST_CACHE_PREFIX).append(roleDaoAction.DAO_ACTION_CACHE_PREFIX).append(
 164  
                                                 StringUtils.isBlank(roleId) ? "" : roleId).toString();
 165  
                         case ROLE_MEMBERS_FOR_ROLE_IDS_WITH_FILTERS : // Search for role members that might be roles, principals, or groups.
 166  0
                                 return new StringBuilder(ROLE_MEMBER_IMPL_LIST_CACHE_PREFIX).append(roleDaoAction.DAO_ACTION_CACHE_PREFIX).append(
 167  
                                                 StringUtils.isBlank(roleId) ? "" : roleId).append('-').append(StringUtils.isBlank(principalId) ? "" : principalId).append(
 168  
                                                                 '-').append(StringUtils.isBlank(groupId) ? "" : groupId).append('-').append(
 169  
                                                                                 StringUtils.isBlank(memberTypeCode) ? "" : memberTypeCode).toString();
 170  
                         default : // The daoActionToTake parameter is invalid; throw an exception.
 171  0
                                 throw new IllegalArgumentException("The 'roleDaoAction' parameter cannot refer to a non-role-member-related value!");
 172  
                 }
 173  
         }
 174  
         
 175  
         /**
 176  
          * Converts the Qualifier Name/Value Role qualification set into Qualifier AttributeID/Value set
 177  
          * 
 178  
          * @param qualification The original role qualification attribute set
 179  
          * @return Converted attributeSet containing ID/value pairs
 180  
          */        
 181  
         private AttributeSet convertQualifierKeys(AttributeSet qualification) {
 182  0
                 AttributeSet convertedQualification = new AttributeSet();
 183  0
                 if(qualification != null && CollectionUtils.isNotEmpty(qualification.keySet())) { 
 184  0
                         for(String attributeName : qualification.keySet()) {
 185  0
                                 if(StringUtils.isNotEmpty(getKimAttributeId(attributeName))) {
 186  0
                                         convertedQualification.put(getKimAttributeId(attributeName), qualification.get(attributeName));
 187  
                                 }
 188  
                         }
 189  
                 }
 190  0
                 return convertedQualification;
 191  
         }
 192  
         
 193  
     public Set<String> getRoleTypeRoleMemberIds( String roleId ){
 194  0
             Set<String> results = new HashSet();
 195  0
             getNestedRoleTypeMemberIds( roleId, results);
 196  0
             return results;
 197  
     }
 198  
 
 199  
     protected void getNestedRoleTypeMemberIds( String roleId, Set members) throws RuntimeException
 200  
     {
 201  0
             ArrayList<String> roleList = new ArrayList<String>( 1 );
 202  0
                 roleList.add( roleId );
 203  0
                 List<RoleMemberImpl> firstLevelMembers = getStoredRoleMembersForRoleIds(roleList, KimConstants.KimUIConstants.MEMBER_TYPE_ROLE_CODE, null);
 204  0
                 for (RoleMemberImpl member : firstLevelMembers) {
 205  0
                         if (KimConstants.KimUIConstants.MEMBER_TYPE_ROLE_CODE.equals(member.getMemberTypeCode())){
 206  0
                                 if (!members.contains(member.getMemberId())){
 207  0
                                         members.add(member.getMemberId());
 208  0
                                         getNestedRoleTypeMemberIds( member.getMemberId(), members);
 209  
                                 }
 210  
                         }
 211  
             }            
 212  0
     }
 213  
     
 214  
     public List<String> getMemberParentRoleIds(String memberType, String memberId) {        
 215  0
                 List<RoleMemberImpl> parentRoleMembers = roleDao.getRoleMembershipsForMemberId(memberType, memberId, null);
 216  
 
 217  0
                 List<String> parentRoleIds = new ArrayList<String>(parentRoleMembers.size());
 218  0
                 for (RoleMemberImpl parentRoleMember: parentRoleMembers) {
 219  0
                         parentRoleIds.add(parentRoleMember.getRoleId());
 220  
                 }
 221  
                 
 222  0
             return parentRoleIds;
 223  
     }
 224  
     
 225  
         /**
 226  
          * Retrieves a list of RoleMemberImpl instances from the cache and/or the KimRoleDao as appropriate.
 227  
          * 
 228  
          * @param daoActionToTake An indicator for which KimRoleDao method should be used to get the results if the desired RoleMemberImpls are not cached.
 229  
          * @param roleIds The role IDs to filter by; may get used as the IDs for members that are also roles, depending on the daoActionToTake value.
 230  
          * @param principalId The principal ID to filter by; may get ignored depending on the daoActionToTake value.
 231  
          * @param groupIds The group IDs to filter by; may get ignored depending on the daoActionToTake value.
 232  
          * @param memberTypeCode The member type code to filter by; may get overridden depending on the daoActionToTake value.
 233  
          * @param qualification The original role qualification attribute set 
 234  
          * @return A list of RoleMemberImpl instances based on the provided parameters.
 235  
          * @throws IllegalArgumentException if daoActionToTake refers to an enumeration constant that is not role-member-related.
 236  
          */
 237  
         protected List<RoleMemberImpl> getRoleMemberImplList(RoleDaoAction daoActionToTake, Collection<String> roleIds, String principalId,
 238  
                         Collection<String> groupIds, String memberTypeCode, AttributeSet qualification) {
 239  0
                 List<RoleMemberImpl> finalResults = new ArrayList<RoleMemberImpl>();
 240  0
                 List<RoleMemberCacheKeyHelper> searchKeys = new ArrayList<RoleMemberCacheKeyHelper>();
 241  0
                 List<RoleMemberCacheKeyHelper> uncachedKeys = new ArrayList<RoleMemberCacheKeyHelper>();
 242  0
                 Set<String> usedKeys = new HashSet<String>();
 243  0
                 AttributeSet convertedQualification = convertQualifierKeys(qualification);
 244  
                 
 245  0
                 if (roleIds == null || roleIds.isEmpty()) { roleIds = Collections.singletonList(null); }
 246  0
                 if (groupIds == null || groupIds.isEmpty()) { groupIds = Collections.singletonList(null); }
 247  
                 
 248  
                 // Attempt to find any pre-cached role members based on what KimRoleDao method call is desired.
 249  0
                 switch (daoActionToTake) {
 250  
                         case ROLE_PRINCIPALS_FOR_PRINCIPAL_ID_AND_ROLE_IDS : // Search for principal role members only.
 251  0
                                 for (String roleId : roleIds) {
 252  0
                                         searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, principalId, null, Role.PRINCIPAL_MEMBER_TYPE));
 253  
                                 }
 254  0
                                 break;
 255  
                         case ROLE_GROUPS_FOR_GROUP_IDS_AND_ROLE_IDS : // Search for group role members only.
 256  0
                                 for (String roleId : roleIds) {
 257  0
                                         for (String groupId : groupIds) {
 258  0
                                                 searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, null, groupId, Role.GROUP_MEMBER_TYPE));
 259  
                                         }
 260  
                                 }
 261  0
                                 break;
 262  
                         case ROLE_MEMBERS_FOR_ROLE_IDS : // Search for role members with the given member type code.
 263  0
                                 for (String roleId : roleIds) {
 264  0
                                         searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, null, null, memberTypeCode));
 265  
                                 }
 266  0
                                 break;
 267  
                         case ROLE_MEMBERSHIPS_FOR_ROLE_IDS_AS_MEMBERS : // Search for role members who are also roles.
 268  0
                                 for (String roleId : roleIds) {
 269  0
                                         searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, null, null, Role.ROLE_MEMBER_TYPE));
 270  
                                 }
 271  0
                                 break;
 272  
                         case ROLE_MEMBERS_FOR_ROLE_IDS_WITH_FILTERS : // Search for role members that might be roles, principals, or groups.
 273  0
                                 for (String roleId : roleIds) {
 274  0
                                         searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, null, null, Role.ROLE_MEMBER_TYPE));
 275  0
                                         searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, principalId, null, Role.PRINCIPAL_MEMBER_TYPE));
 276  0
                                         for (String groupId : groupIds) {
 277  0
                                                 searchKeys.add(new RoleMemberCacheKeyHelper(daoActionToTake, roleId, null, groupId, Role.GROUP_MEMBER_TYPE));
 278  
                                         }
 279  
                                 }
 280  0
                                 break;
 281  
                         default : // The daoActionToTake parameter is invalid; throw an exception.
 282  0
                                 throw new IllegalArgumentException("The 'daoActionToTake' parameter cannot refer to a non-role-member-related value!");
 283  
                 }
 284  
                 
 285  
                 // Attempt to find any pre-cached role members.
 286  0
                 for (RoleMemberCacheKeyHelper searchKey : searchKeys) {
 287  0
                         if (!usedKeys.contains(searchKey.getCacheKey())) {
 288  0
                                 List<RoleMemberImpl> tempMembers = (List<RoleMemberImpl>) getCacheAdministrator().getFromCache(searchKey.getCacheKey());
 289  0
                                 if (tempMembers != null) {
 290  0
                                         finalResults.addAll(tempMembers);
 291  
                                 } else {
 292  0
                                         uncachedKeys.add(searchKey);
 293  
                                 }
 294  0
                                 usedKeys.add(searchKey.getCacheKey());
 295  0
                         }
 296  
                 }
 297  
                 
 298  
                 // If any portion of the result set was not in the cache, then retrieve and cache the missing sections.
 299  0
                 if (!uncachedKeys.isEmpty()) {
 300  0
                         Set<String> uncachedRoleSet = new HashSet<String>();
 301  0
                         Set<String> uncachedGroupSet = new HashSet<String>();
 302  0
                         for (RoleMemberCacheKeyHelper uncachedKey : uncachedKeys) {
 303  0
                                 if (uncachedKey.ROLE_ID != null) { uncachedRoleSet.add(uncachedKey.ROLE_ID); }
 304  0
                                 if (uncachedKey.GROUP_ID != null) { uncachedGroupSet.add(uncachedKey.GROUP_ID); }
 305  
                         }
 306  
                         
 307  0
                         List<String> uncachedRoles = (uncachedRoleSet.isEmpty()) ? null : new ArrayList<String>(uncachedRoleSet);
 308  0
                         List<String> uncachedGroups = (uncachedGroupSet.isEmpty()) ? null : new ArrayList<String>(uncachedGroupSet);
 309  0
                         List<RoleMemberImpl> uncachedResults = null;
 310  
                         
 311  
                         // Retrieve the uncached RoleMemberImpls via the appropriate RoleDao method.
 312  0
                         switch (daoActionToTake) {
 313  
                                 case ROLE_PRINCIPALS_FOR_PRINCIPAL_ID_AND_ROLE_IDS : // Search for principal role members only.
 314  0
                                         uncachedResults = roleDao.getRolePrincipalsForPrincipalIdAndRoleIds(uncachedRoles, principalId, convertedQualification);
 315  0
                                         break;
 316  
                                 case ROLE_GROUPS_FOR_GROUP_IDS_AND_ROLE_IDS : // Search for group role members only.
 317  0
                                         uncachedResults = roleDao.getRoleGroupsForGroupIdsAndRoleIds(uncachedRoles, uncachedGroups, convertedQualification);
 318  0
                                         break;
 319  
                                 case ROLE_MEMBERS_FOR_ROLE_IDS : // Search for role members with the given member type code.
 320  0
                                         uncachedResults = roleDao.getRoleMembersForRoleIds(uncachedRoles, memberTypeCode, convertedQualification);
 321  0
                                         break;
 322  
                                 case ROLE_MEMBERSHIPS_FOR_ROLE_IDS_AS_MEMBERS : // Search for role members who are also roles.
 323  0
                                         uncachedResults = roleDao.getRoleMembershipsForRoleIdsAsMembers(uncachedRoles, convertedQualification);
 324  0
                                         break;
 325  
                                 case ROLE_MEMBERS_FOR_ROLE_IDS_WITH_FILTERS : // Search for role members that might be roles, principals, or groups.
 326  0
                                         uncachedResults = roleDao.getRoleMembersForRoleIdsWithFilters(uncachedRoles, principalId, uncachedGroups, convertedQualification);
 327  0
                                         break;
 328  
                                 default : // This should never happen, since the previous switch block should handle this case appropriately.
 329  0
                                         throw new IllegalArgumentException("The 'daoActionToTake' parameter cannot refer to a non-role-member-related value!");
 330  
                         }
 331  
                         
 332  0
                         cacheRoleMemberLists(uncachedKeys, uncachedResults);
 333  0
                         for (RoleMemberImpl uncachedMember : uncachedResults) {
 334  0
                                 addRoleMemberImplToCache(uncachedMember);
 335  
                         }
 336  0
                         finalResults.addAll(uncachedResults);
 337  
                 }
 338  0
                 return finalResults;
 339  
         }
 340  
         
 341  
         /**
 342  
          * Caches several Lists of role members that are constructed based on the given search keys. Note that a given List will not be cached if
 343  
          * it contains any role members that belong to a role that disallows caching of its members.
 344  
          * 
 345  
          * @param keysToCache The keys of the role member Lists that will be used to store the uncached results.
 346  
          * @param membersToCache The uncached role members.
 347  
          */
 348  
         private void cacheRoleMemberLists(List<RoleMemberCacheKeyHelper> keysToCache, List<RoleMemberImpl> membersToCache) {
 349  0
                 if (membersToCache == null) { membersToCache = new ArrayList<RoleMemberImpl>(); }
 350  0
                 for (RoleMemberCacheKeyHelper keyToCache : keysToCache) {
 351  0
                         List<RoleMemberImpl> roleMembers = new ArrayList<RoleMemberImpl>();
 352  
                         
 353  
                         // Cache the Lists that do not contain role members belonging to roles that forbid caching of their members.
 354  0
                         if (RoleDaoAction.ROLE_MEMBERSHIPS_FOR_ROLE_IDS_AS_MEMBERS.equals(keyToCache.ROLE_DAO_ACTION)) {
 355  0
                                 boolean safeToCacheList = true;
 356  0
                                 for (RoleMemberImpl memberToCache : membersToCache) {
 357  0
                                         if ( (keyToCache.ROLE_ID == null || keyToCache.ROLE_ID.equals(memberToCache.getMemberId())) &&
 358  
                                                         (keyToCache.MEMBER_TYPE_CODE == null || keyToCache.MEMBER_TYPE_CODE.equals(memberToCache.getMemberTypeCode())) ) {
 359  0
                                                 if (shouldCacheMembersOfRole(memberToCache.getRoleId())) {
 360  0
                                                         roleMembers.add(memberToCache);
 361  
                                                 } else {
 362  0
                                                         safeToCacheList = false;
 363  0
                                                         break;
 364  
                                                 }
 365  
                                         }
 366  
                                 }
 367  0
                                 if (safeToCacheList) { getCacheAdministrator().putInCache(keyToCache.getCacheKey(), roleMembers, ROLE_MEMBER_IMPL_CACHE_GROUP); }
 368  0
                         } else if (keyToCache.ROLE_ID == null || shouldCacheMembersOfRole(keyToCache.ROLE_ID)) {
 369  0
                                 for (RoleMemberImpl memberToCache : membersToCache) {
 370  0
                                         if ( (keyToCache.ROLE_ID == null || keyToCache.ROLE_ID.equals(memberToCache.getRoleId())) &&
 371  
                                                         (keyToCache.PRINCIPAL_ID == null || keyToCache.PRINCIPAL_ID.equals(memberToCache.getMemberId())) &&
 372  
                                                         (keyToCache.GROUP_ID == null || keyToCache.GROUP_ID.equals(memberToCache.getMemberId())) &&
 373  
                                                         (keyToCache.MEMBER_TYPE_CODE == null || keyToCache.MEMBER_TYPE_CODE.equals(memberToCache.getMemberTypeCode())) ) {
 374  0
                                                 roleMembers.add(memberToCache);
 375  
                                         }
 376  
                                 }
 377  0
                                 getCacheAdministrator().putInCache(keyToCache.getCacheKey(), roleMembers, ROLE_MEMBER_IMPL_CACHE_GROUP);
 378  
                         }
 379  0
                 }
 380  0
         }
 381  
         
 382  
         /** Calls the KimRoleDao's "getRolePrincipalsForPrincipalIdAndRoleIds" method and/or retrieves any corresponding members from the cache. */
 383  
         protected List<RoleMemberImpl> getStoredRolePrincipalsForPrincipalIdAndRoleIds(Collection<String> roleIds, String principalId, AttributeSet qualification) {
 384  0
                 return getRoleMemberImplList(RoleDaoAction.ROLE_PRINCIPALS_FOR_PRINCIPAL_ID_AND_ROLE_IDS, roleIds, principalId, null, null, qualification);
 385  
         }
 386  
         
 387  
         /** Calls the KimRoleDao's "getRoleGroupsForGroupIdsAndRoleIds" method and/or retrieves any corresponding members from the cache. */
 388  
         protected List<RoleMemberImpl> getStoredRoleGroupsForGroupIdsAndRoleIds(Collection<String> roleIds, Collection<String> groupIds, AttributeSet qualification) {
 389  0
                 return getRoleMemberImplList(RoleDaoAction.ROLE_GROUPS_FOR_GROUP_IDS_AND_ROLE_IDS, roleIds, null, groupIds, null, qualification);
 390  
         }
 391  
         
 392  
         /** Calls the KimRoleDao's "getRoleMembersForRoleIds" method and/or retrieves any corresponding members from the cache. */
 393  
         protected List<RoleMemberImpl> getStoredRoleMembersForRoleIds(Collection<String> roleIds, String memberTypeCode, AttributeSet qualification) {
 394  0
                 return getRoleMemberImplList(RoleDaoAction.ROLE_MEMBERS_FOR_ROLE_IDS, roleIds, null, null, memberTypeCode, qualification);
 395  
         }
 396  
         
 397  
         /** Calls the KimRoleDao's "getRoleMembershipsForRoleIdsAsMembers" method and/or retrieves any corresponding members from the cache. */
 398  
         protected List<RoleMemberImpl> getStoredRoleMembershipsForRoleIdsAsMembers(Collection<String> roleIds, AttributeSet qualification) {
 399  0
                 return getRoleMemberImplList(RoleDaoAction.ROLE_MEMBERSHIPS_FOR_ROLE_IDS_AS_MEMBERS, roleIds, null, null, null, qualification);
 400  
         }
 401  
         
 402  
         /** Calls the KimRoleDao's "getRoleMembersForRoleIdsWithFilters" method and/or retrieves any corresponding members from the cache. */
 403  
         protected List<RoleMemberImpl> getStoredRoleMembersForRoleIdsWithFilters(Collection<String> roleIds, String principalId, List<String> groupIds, AttributeSet qualification) {
 404  0
                 return getRoleMemberImplList(RoleDaoAction.ROLE_MEMBERS_FOR_ROLE_IDS_WITH_FILTERS, roleIds, principalId, groupIds, null, qualification);
 405  
         }
 406  
         
 407  
         /**
 408  
          * Determines whether or not the given role should allow its members to be cached. The default implementation always returns true, but
 409  
          * subclasses can override this method if other non-default Role implementations forbid their members from being cached.
 410  
          * 
 411  
          * @param roleId The ID of the role to check for determining whether or not to allow caching of its members.
 412  
          * @return True if the given role allows its members to be cached; false otherwise.
 413  
          */
 414  
         protected boolean shouldCacheMembersOfRole(String roleId) {
 415  0
                 return true;
 416  
         }
 417  
         
 418  
         protected void addRoleMemberImplToCache(RoleMemberImpl roleMember) {
 419  0
                 if (roleMember != null && shouldCacheMembersOfRole(roleMember.getRoleId())) {
 420  0
                         getCacheAdministrator().putInCache(getRoleMemberCacheKey(roleMember.getRoleMemberId()), roleMember, ROLE_MEMBER_IMPL_CACHE_GROUP);
 421  
                 }
 422  0
         }
 423  
         
 424  
         protected RoleMemberImpl getRoleMemberFromCache(String roleMemberId) {
 425  0
                 return (RoleMemberImpl)getCacheAdministrator().getFromCache(getRoleMemberCacheKey(roleMemberId));
 426  
         }
 427  
         
 428  
         public void flushInternalRoleMemberCache() {
 429  0
                 getCacheAdministrator().flushGroup(ROLE_MEMBER_IMPL_CACHE_GROUP);
 430  0
         }
 431  
         
 432  
         /**
 433  
          * Retrieves a RoleMemberImpl object by its ID. If the role member already exists in the cache, this method will return the cached
 434  
          * version; otherwise, it will retrieve the uncached version from the database and then cache it (if it belongs to a role that allows
 435  
          * its members to be cached) before returning it.
 436  
          */
 437  
     protected RoleMemberImpl getRoleMemberImpl( String roleMemberId ) {
 438  0
             if (StringUtils.isBlank(roleMemberId)) {
 439  0
                     return null;
 440  
             }
 441  
             
 442  
             // If the RoleMemberImpl exists in the cache, return the cached one.
 443  0
             RoleMemberImpl tempRoleMemberImpl = getRoleMemberFromCache(roleMemberId);
 444  0
             if (tempRoleMemberImpl != null) {
 445  0
                     return tempRoleMemberImpl;
 446  
             }
 447  
             // Otherwise, retrieve it normally.
 448  0
             tempRoleMemberImpl = (RoleMemberImpl)getBusinessObjectService().findByPrimaryKey( RoleMemberImpl.class,
 449  
                             Collections.singletonMap(KIMPropertyConstants.RoleMember.ROLE_MEMBER_ID, roleMemberId) );
 450  0
             addRoleMemberImplToCache(tempRoleMemberImpl);
 451  0
             return tempRoleMemberImpl;
 452  
     }
 453  
         
 454  
     // -----------------------------------------------------------------------------------------------------------------
 455  
     // Delegation Caching Methods
 456  
     // -----------------------------------------------------------------------------------------------------------------
 457  
     
 458  
         /**
 459  
          * Generates a String key to use for storing or retrieving a KimDelegationImpl to/from the cache.
 460  
          * 
 461  
          * @param delegationId The ID of the KimDelegationImpl to generate a key for.
 462  
          * @return A cache key for the KimDelegationImpl with the given ID.
 463  
          */
 464  
         protected String getDelegationCacheKey(String delegationId) {
 465  0
                 return DELEGATION_IMPL_CACHE_PREFIX + delegationId;
 466  
         }
 467  
 
 468  
     /**
 469  
      * Generates a String key to use for storing or retrieving a List of KimDelegationImpls to/from the cache based on a role's ID.
 470  
      * @param roleId The ID of the role that the KIM delegations belong to.
 471  
      * @return A cache key for the KimDelegationImpls with the given role ID.
 472  
      */
 473  
     protected String getDelegationListCacheKey(String roleId) {
 474  0
             return DELEGATION_IMPL_LIST_CACHE_PREFIX + (StringUtils.isBlank(roleId) ? "" : roleId);
 475  
     }
 476  
         
 477  
     /** Calls the KimRoleDao's "getDelegationImplMapFromRoleIds" method and/or retrieves any corresponding delegations from the cache. */
 478  
         protected Map<String,KimDelegationImpl> getStoredDelegationImplMapFromRoleIds(Collection<String> roleIds) {
 479  0
                 Map<String,KimDelegationImpl> finalResults = Collections.emptyMap();
 480  
                 
 481  0
                 if (roleIds != null && !roleIds.isEmpty()) {
 482  0
                         Map<String,List<KimDelegationImpl>> uncachedLists = new HashMap<String,List<KimDelegationImpl>>();
 483  
                         // Retrieve any existing results from the cache.
 484  0
                         finalResults = getKimDelegationImplMap(uncachedLists, roleIds);
 485  
                         
 486  
                         // Retrieve any uncached results from the database and then cache them.
 487  0
                         if (!uncachedLists.isEmpty()) {
 488  0
                                 Map<String,KimDelegationImpl> uncachedResults = roleDao.getDelegationImplMapFromRoleIds(uncachedLists.keySet());
 489  
                                 
 490  0
                                 for (Map.Entry<String,KimDelegationImpl> uncachedResult : uncachedResults.entrySet()) {
 491  0
                                         finalResults.put(uncachedResult.getKey(), uncachedResult.getValue());
 492  0
                                         addKimDelegationImplToCache(uncachedResult.getValue());
 493  0
                                         uncachedLists.get(uncachedResult.getValue().getRoleId()).add(uncachedResult.getValue());
 494  
                                 }
 495  0
                                 for (Map.Entry<String,List<KimDelegationImpl>> uncachedList : uncachedLists.entrySet()) {
 496  0
                                         getCacheAdministrator().putInCache(getDelegationListCacheKey(uncachedList.getKey()),
 497  
                                                         uncachedList.getValue(), DELEGATION_IMPL_CACHE_GROUP);
 498  
                                 }
 499  
                         }
 500  
                 }
 501  
                 
 502  0
                 return finalResults;
 503  
         }
 504  
         
 505  
         /** Calls the KimRoleDao's "getDelegationImplsForRoleIds" method and/or retrieves any corresponding delegations from the cache. */
 506  
         protected List<KimDelegationImpl> getStoredDelegationImplsForRoleIds(Collection<String> roleIds) {
 507  0
                 List<KimDelegationImpl> finalResults = new ArrayList<KimDelegationImpl>();
 508  
                 
 509  0
                 if (roleIds != null && !roleIds.isEmpty()) {
 510  0
                         Map<String,List<KimDelegationImpl>> uncachedLists = new HashMap<String,List<KimDelegationImpl>>();
 511  
                         // Retrieve any existing results from the cache.
 512  0
                         Map<String,KimDelegationImpl> tempDelegations = getKimDelegationImplMap(uncachedLists, roleIds);
 513  0
                         finalResults.addAll(tempDelegations.values());
 514  
                         
 515  
                         // Retrieve any uncached results from the database and then cache them.
 516  0
                         if (!uncachedLists.isEmpty()) {
 517  0
                                 List<KimDelegationImpl> uncachedResults = roleDao.getDelegationImplsForRoleIds(uncachedLists.keySet());
 518  
                                 
 519  0
                                 for (KimDelegationImpl uncachedResult : uncachedResults) {
 520  0
                                         addKimDelegationImplToCache(uncachedResult);
 521  0
                                         uncachedLists.get(uncachedResult.getRoleId()).add(uncachedResult);
 522  
                                 }
 523  0
                                 for (Map.Entry<String,List<KimDelegationImpl>> uncachedList : uncachedLists.entrySet()) {
 524  0
                                         getCacheAdministrator().putInCache(getDelegationListCacheKey(uncachedList.getKey()),
 525  
                                                         uncachedList.getValue(), DELEGATION_IMPL_CACHE_GROUP);
 526  
                                 }
 527  0
                                 finalResults.addAll(uncachedResults);
 528  
                         }
 529  
                 }
 530  
                 
 531  0
                 return finalResults;
 532  
         }
 533  
         
 534  
         /**
 535  
          * Retrieves any existing delegation lists from the cache for the given role IDs. If the delegations for a given role have not been cached,
 536  
          * then a new entry containing the uncached role ID and an empty List will be added to the given Map.
 537  
          * 
 538  
          * @param uncachedLists The Map in which to place any uncached lists; cannot be null.
 539  
          * @param roleIds The IDs of the roles containing the delegations.
 540  
          * @return A mutable Map containing any existing results from the cache and which maps the delegations' IDs to the delegation objects.
 541  
          * @throws IllegalArgumentException if the provided Map is null.
 542  
          */
 543  
         protected Map<String,KimDelegationImpl> getKimDelegationImplMap(Map<String,List<KimDelegationImpl>> uncachedLists, Collection<String> roleIds) {
 544  0
                 if (uncachedLists == null) {
 545  0
                         throw new IllegalArgumentException("'uncachedLists' parameter cannot be null!");
 546  
                 }
 547  0
                 Map<String,KimDelegationImpl> delegationMap = new HashMap<String,KimDelegationImpl>();
 548  
                 
 549  
                 // Retrieve any existing results from the cache.
 550  0
                 if (roleIds != null && !roleIds.isEmpty()) {
 551  0
                         for (String roleId : roleIds) {
 552  0
                                 List<KimDelegationImpl> tempDelegations = (List<KimDelegationImpl>) getCacheAdministrator().getFromCache(getDelegationListCacheKey(roleId));
 553  0
                                 if (tempDelegations != null) {
 554  0
                                         for (KimDelegationImpl tempDelegation : tempDelegations) {
 555  0
                                                 delegationMap.put(tempDelegation.getDelegationId(), tempDelegation);
 556  
                                         }
 557  
                                 } else {
 558  0
                                         uncachedLists.put(roleId, new ArrayList<KimDelegationImpl>());
 559  
                                 }
 560  0
                         }
 561  
                 }
 562  
                 
 563  0
                 return delegationMap;
 564  
         }
 565  
         
 566  
         protected void addKimDelegationImplToCache(KimDelegationImpl delegation) {
 567  0
                 if (delegation != null) {
 568  0
                         getCacheAdministrator().putInCache(getDelegationCacheKey(delegation.getDelegationId()),
 569  
                                         delegation, DELEGATION_IMPL_CACHE_GROUP);
 570  
                 }
 571  0
         }
 572  
         
 573  
         protected KimDelegationImpl getDelegationFromCache(String delegationId) {
 574  0
                 return (KimDelegationImpl)getCacheAdministrator().getFromCache(getDelegationCacheKey(delegationId));
 575  
         }
 576  
         
 577  
         public void flushInternalDelegationCache() {
 578  0
                 getCacheAdministrator().flushGroup(DELEGATION_IMPL_CACHE_GROUP);
 579  0
         }
 580  
         
 581  
         /**
 582  
          * Retrieves a KimDelegationImpl object by its ID. If the delegation already exists in the cache, this method will return the cached
 583  
          * version; otherwise, it will retrieve the uncached version from the database and then cache it before returning it.
 584  
          */
 585  
     protected KimDelegationImpl getKimDelegationImpl( String delegationId ) {
 586  0
             if (StringUtils.isBlank(delegationId)) {
 587  0
                     return null;
 588  
             }
 589  
             
 590  
             // If the KimDelegationImpl exists in the cache, return the cached one.
 591  0
             KimDelegationImpl tempDelegation = getDelegationFromCache(delegationId);
 592  0
             if (tempDelegation != null) {
 593  0
                     return tempDelegation;
 594  
             }
 595  
             // Otherwise, retrieve it normally.
 596  0
             tempDelegation = (KimDelegationImpl)getBusinessObjectService().findByPrimaryKey( KimDelegationImpl.class,
 597  
                             Collections.singletonMap(KimConstants.PrimaryKeyConstants.DELEGATION_ID, delegationId) );
 598  0
             addKimDelegationImplToCache(tempDelegation);
 599  0
             return tempDelegation;
 600  
     }
 601  
     
 602  
     // -----------------------------------------------------------------------------------------------------------------
 603  
     // Delegation Membership Caching Methods
 604  
     // -----------------------------------------------------------------------------------------------------------------
 605  
     
 606  
         /**
 607  
          * Generates a String key to use for storing or retrieving a KimDelegationMemberImpl to/from the cache.
 608  
          * 
 609  
          * @param delegationMemberId The ID of the KimDelegationMemberImpl to generate a key for.
 610  
          * @return A cache key for the KimDelegationMemberImpl with the given ID.
 611  
          */
 612  
         protected String getDelegationMemberCacheKey(String delegationMemberId) {
 613  0
                 return DELEGATION_MEMBER_IMPL_CACHE_PREFIX + delegationMemberId;
 614  
         }
 615  
     
 616  
         /**
 617  
          * Generates a String key to use for storing or retrieving a KimDelegationMemberImpl to/from the cache by both delegation ID and delegation member ID.
 618  
          * 
 619  
          * @param delegationId The ID of the delegation that the KimDelegationMemberImpl belongs to.
 620  
          * @param delegationMemberId The ID of the KimDelegationMemberImpl to generate a key for.
 621  
          * @return A cache key for the KimDelegationMemberImpl with the given delegation ID and delegation member ID.
 622  
          */
 623  
         protected String getDelegationMemberByDelegationAndIdCacheKey(String delegationId, String delegationMemberId) {
 624  0
                 return new StringBuilder(DELEGATION_MEMBER_IMPL_BY_DLGN_AND_ID_CACHE_PREFIX).append(delegationId).append('-').append(delegationMemberId).toString();
 625  
         }
 626  
         
 627  
         /**
 628  
          * Generates a String key to use for storing or retrieving a KimDelegationMemberImpl List to/from the cache by both delegation ID and member ID.
 629  
          * 
 630  
          * @param memberId The principal/group/role ID of the KimDelegationMemberImpl(s) in the List.
 631  
          * @param delegationId The ID of the delegation that the KimDelegationMemberImpl(s) in the List belong to.
 632  
          * @return A cache key for the KimDelegationMemberImpl List with the given delegation ID and member ID.
 633  
          */
 634  
         protected String getDelegationMemberListByMemberAndDelegationIdCacheKey(String memberId, String delegationId) {
 635  0
                 return new StringBuilder(DELEGATION_MEMBER_IMPL_LIST_BY_MBR_DLGN_CACHE_PREFIX).append(StringUtils.isBlank(memberId) ? "" : memberId).append(
 636  
                                 '-').append(StringUtils.isBlank(delegationId) ? "" : delegationId).toString();
 637  
         }
 638  
         
 639  
         /**
 640  
          * Generates a String key to use for storing or retrieving a List of KimDelegationMemberImpl lists to/from the cache. Some parameters may get
 641  
          * ignored depending on which KimRoleDao call is desired.
 642  
          * 
 643  
          * @param daoAction The RoleDaoAction signifying which KimRoleDao call found this list; will determine how and which parameters are used.
 644  
          * @param delegationId The ID of the KimDelegationImpl that the indicated delegation member belongs to; will be interpreted as an empty String if blank.
 645  
          * @param principalId The (principal) member ID of the delegation members; will be interpreted as an empty String if blank.
 646  
          * @param groupId The (group) member ID of the delegation members; will be interpreted as an empty String if blank.
 647  
          * @return A cache key for the KimDelegationMemberImpl List with the given criteria.
 648  
          * @throws IllegalArgumentException if daoAction does not represent a delegation-member-related enumeration value.
 649  
          */
 650  
         protected String getDelegationMemberListCacheKey(RoleDaoAction daoAction, String delegationId, String principalId, String groupId) {
 651  0
                 switch (daoAction) {
 652  
                         case DELEGATION_PRINCIPALS_FOR_PRINCIPAL_ID_AND_DELEGATION_IDS : // Search for principal delegation members.
 653  0
                                 return new StringBuilder(DELEGATION_MEMBER_IMPL_LIST_CACHE_PREFIX).append(daoAction.DAO_ACTION_CACHE_PREFIX).append(
 654  
                                                 StringUtils.isBlank(delegationId) ? "" : delegationId).append('-').append(
 655  
                                                                 StringUtils.isBlank(principalId) ? "" : principalId).toString();
 656  
                         case DELEGATION_GROUPS_FOR_GROUP_IDS_AND_DELEGATION_IDS : // Search for group delegation members.
 657  0
                                 return new StringBuilder(DELEGATION_MEMBER_IMPL_LIST_CACHE_PREFIX).append(daoAction.DAO_ACTION_CACHE_PREFIX).append(
 658  
                                                 StringUtils.isBlank(delegationId) ? "" : delegationId).append('-').append(StringUtils.isBlank(groupId) ? "" : groupId).toString();
 659  
                         case DELEGATION_MEMBERS_FOR_DELEGATION_IDS : // Search for delegation members regardless of their member type.
 660  0
                                 return new StringBuilder(DELEGATION_MEMBER_IMPL_LIST_CACHE_PREFIX).append(daoAction.DAO_ACTION_CACHE_PREFIX).append(
 661  
                                                 StringUtils.isBlank(delegationId) ? "" : delegationId).toString();
 662  
                         default : // daoActionToTake is invalid; throw an exception.
 663  0
                                 throw new IllegalArgumentException("The 'daoActionToTake' parameter cannot refer to a non-delegation-member-related value!");
 664  
                 }
 665  
         }
 666  
         
 667  
         /**
 668  
          * Retrieves a List of delegation members from the cache and/or the KimRoleDao as appropriate.
 669  
          * 
 670  
          * @param daoActionToTake An indicator for which KimRoleDao method to use for retrieving uncached results.
 671  
          * @param delegationIds The IDs of the delegations that the members belong to.
 672  
          * @param principalId The principal ID of the principal delegation members; may get ignored depending on the RoleDaoAction value.
 673  
          * @param groupIds The group IDs of the group delegation members; may get ignored depending on the RoleDaoAction value.
 674  
          * @return A List of KimDelegationMemberImpl objects based on the provided parameters.
 675  
          * @throws IllegalArgumentException if daoActionToTake does not represent a delegation-member-list-related enumeration value.
 676  
          */
 677  
         protected List<KimDelegationMemberImpl> getKimDelegationMemberImplList(RoleDaoAction daoActionToTake, Collection<String> delegationIds,
 678  
                         String principalId, List<String> groupIds) {
 679  0
                 List<KimDelegationMemberImpl> finalResults = new ArrayList<KimDelegationMemberImpl>();
 680  0
                 List<String[]> uncachedKeys = new ArrayList<String[]>();
 681  0
                 Set<String> usedKeys = new HashSet<String>();
 682  0
                 if (delegationIds == null || delegationIds.isEmpty()) { delegationIds = Collections.singletonList(null); }
 683  0
                 if (groupIds == null || groupIds.isEmpty()) { groupIds = Collections.singletonList(null); }
 684  
                 
 685  
                 // Search for cached values based on the intended search action.
 686  0
                 switch (daoActionToTake) {
 687  
                         case DELEGATION_PRINCIPALS_FOR_PRINCIPAL_ID_AND_DELEGATION_IDS : // Search for principal delegation members.
 688  0
                                 for (String delegationId : delegationIds) {
 689  0
                                         String tempKey = getDelegationMemberListCacheKey(daoActionToTake, delegationId, principalId, null);
 690  0
                                         if (!usedKeys.contains(tempKey)) {
 691  0
                                                 List<KimDelegationMemberImpl> tempMembers = (List<KimDelegationMemberImpl>) getCacheAdministrator().getFromCache(tempKey);
 692  0
                                                 if (tempMembers != null) {
 693  0
                                                         finalResults.addAll(tempMembers);
 694  
                                                 } else {
 695  0
                                                         uncachedKeys.add(new String[] {delegationId, principalId, null});
 696  
                                                 }
 697  0
                                                 usedKeys.add(tempKey);
 698  
                                         }
 699  0
                                 }
 700  0
                                 break;
 701  
                         case DELEGATION_GROUPS_FOR_GROUP_IDS_AND_DELEGATION_IDS : // Search for group delegation members.
 702  0
                                 for (String delegationId : delegationIds) {
 703  0
                                         for (String groupId : groupIds) {
 704  0
                                                 String tempKey = getDelegationMemberListCacheKey(daoActionToTake, delegationId, null, groupId);
 705  0
                                                 if (!usedKeys.contains(tempKey)) {
 706  0
                                                         List<KimDelegationMemberImpl> tempMembers = (List<KimDelegationMemberImpl>) getCacheAdministrator().getFromCache(tempKey);
 707  0
                                                         if (tempMembers != null) {
 708  0
                                                                 finalResults.addAll(tempMembers);
 709  
                                                         } else {
 710  0
                                                                 uncachedKeys.add(new String[] {delegationId, null, groupId});
 711  
                                                         }
 712  
                                                 }
 713  0
                                         }
 714  
                                 }
 715  0
                                 break;
 716  
                         default : // daoActionToTake is invalid; throw an exception.
 717  0
                                 throw new IllegalArgumentException("The 'daoActionToTake' parameter cannot refer to a non-delegation-member-list-related value!");
 718  
                 }
 719  
                 
 720  
                 // Retrieve any uncached values based on the intended search action.
 721  0
                 if (!uncachedKeys.isEmpty()) {
 722  0
                         List<KimDelegationMemberImpl> uncachedResults = new ArrayList<KimDelegationMemberImpl>();
 723  0
                         Set<String> uncachedDelegationSet = new HashSet<String>();
 724  0
                         Set<String> uncachedGroupSet = new HashSet<String>();
 725  0
                         for (String[] uncachedKey : uncachedKeys) {
 726  0
                                 if (uncachedKey[0] != null) { uncachedDelegationSet.add(uncachedKey[0]); }
 727  0
                                 if (uncachedKey[2] != null) { uncachedGroupSet.add(uncachedKey[2]); }
 728  
                         }
 729  0
                         List<String> uncachedDelegations = (uncachedDelegationSet.isEmpty()) ? null : new ArrayList<String>(uncachedDelegationSet);
 730  0
                         List<String> uncachedGroups = (uncachedGroupSet.isEmpty()) ? null : new ArrayList<String>(uncachedGroupSet);
 731  0
                         String memberTypeCode = null;
 732  
                         
 733  
                         // Search for uncached values based on the intended search action.
 734  0
                         switch (daoActionToTake) {
 735  
                                 case DELEGATION_PRINCIPALS_FOR_PRINCIPAL_ID_AND_DELEGATION_IDS : // Search for principal delegation members.
 736  0
                                         uncachedResults = roleDao.getDelegationPrincipalsForPrincipalIdAndDelegationIds(uncachedDelegations, principalId);
 737  0
                                         memberTypeCode = Role.PRINCIPAL_MEMBER_TYPE;
 738  0
                                         break;
 739  
                                 case DELEGATION_GROUPS_FOR_GROUP_IDS_AND_DELEGATION_IDS : // Search for group delegation members.
 740  0
                                         uncachedResults = roleDao.getDelegationGroupsForGroupIdsAndDelegationIds(uncachedDelegations, uncachedGroups);
 741  0
                                         memberTypeCode = Role.GROUP_MEMBER_TYPE;
 742  0
                                         break;
 743  
                                 default : // This should never happen since the previous switch block should handle this case appropriately.
 744  0
                                         throw new IllegalArgumentException("The 'daoActionToTake' parameter cannot refer to a non-delegation-member-list-related value!");
 745  
                         }
 746  
                         
 747  
                         // Cache the delegation members and add them to the final results.
 748  0
                         cacheDelegationMemberLists(daoActionToTake, uncachedKeys, memberTypeCode, uncachedResults);
 749  0
                         for (KimDelegationMemberImpl uncachedResult : uncachedResults) {
 750  0
                                 addKimDelegationMemberImplToCache(uncachedResult);
 751  
                         }
 752  0
                         finalResults.addAll(uncachedResults);
 753  
                 }
 754  
                 
 755  0
                 return finalResults;
 756  
         }
 757  
         
 758  
         /**
 759  
          * Caches several Lists of delegation members that are constructed based on the given search parameters.
 760  
          * 
 761  
          * @param daoActionToTake The enumeration constant representing the KimRoleDao call that returned the results.
 762  
          * @param uncachedKeys The keys of the delegation member Lists that will be used to store the uncached results.
 763  
          * @param memberTypeCode The member type code of all the delegation members in the uncached results.
 764  
          * @param uncachedMembers The uncached delegation members.
 765  
          */
 766  
         private void cacheDelegationMemberLists(RoleDaoAction daoActionToTake, List<String[]> uncachedKeys,
 767  
                         String memberTypeCode, List<KimDelegationMemberImpl> uncachedMembers) {
 768  
                 // Place the uncached delegation members into the list cache appropriately.
 769  0
                 for (String[] uncachedKey : uncachedKeys) {
 770  0
                         List<KimDelegationMemberImpl> tempMembers = new ArrayList<KimDelegationMemberImpl>();
 771  0
                         for (KimDelegationMemberImpl uncachedMember : uncachedMembers) {
 772  0
                                 if ( (memberTypeCode == null || memberTypeCode.equals(uncachedMember.getMemberTypeCode())) &&
 773  
                                                 (uncachedKey[0] == null || uncachedKey[0].equals(uncachedMember.getDelegationId())) &&
 774  
                                                                 (uncachedKey[1] == null || uncachedKey[1].equals(uncachedMember.getMemberId())) &&
 775  
                                                                                 (uncachedKey[2] == null || uncachedKey[2].equals(uncachedMember.getMemberId())) ) {
 776  0
                                         tempMembers.add(uncachedMember);
 777  
                                 }
 778  
                         }
 779  0
                         getCacheAdministrator().putInCache(getDelegationMemberListCacheKey(daoActionToTake, uncachedKey[0], uncachedKey[1], uncachedKey[2]),
 780  
                                         tempMembers, DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 781  0
                 }
 782  0
         }
 783  
         
 784  
         /** Calls the KimRoleDao's "getDelegationPrincipalsForPrincipalIdAndDelegationIds" method and/or retrieves any corresponding members from the cache. */
 785  
         protected List<KimDelegationMemberImpl> getStoredDelegationPrincipalsForPrincipalIdAndDelegationIds(Collection<String> delegationIds,String principalId){
 786  0
                 return getKimDelegationMemberImplList(RoleDaoAction.DELEGATION_PRINCIPALS_FOR_PRINCIPAL_ID_AND_DELEGATION_IDS, delegationIds, principalId, null);
 787  
         }
 788  
         
 789  
         /** Calls the KimRoleDao's "getDelegationGroupsForGroupIdAndDelegationIds" method and/or retrieves any corresponding members from the cache. */
 790  
         protected List<KimDelegationMemberImpl> getStoredDelegationGroupsForGroupIdsAndDelegationIds(Collection<String> delegationIds, List<String> groupIds) {
 791  0
                 return getKimDelegationMemberImplList(RoleDaoAction.DELEGATION_GROUPS_FOR_GROUP_IDS_AND_DELEGATION_IDS, delegationIds, null, groupIds);
 792  
         }
 793  
         
 794  
         /** Calls the KimRoleDao's "getDelegationMembersForDelegationIds" method and/or retrieves any corresponding members from the cache. */
 795  
         protected Map<String,List<KimDelegationMemberImpl>> getStoredDelegationMembersForDelegationIds(List<String> delegationIds) {
 796  0
                 Map<String,List<KimDelegationMemberImpl>> finalResults = new HashMap<String,List<KimDelegationMemberImpl>>();
 797  0
                 Set<String> uncachedDelegationIds = new HashSet<String>();
 798  0
                 boolean idListWasNullOrEmpty = (delegationIds == null || delegationIds.isEmpty());
 799  0
                 if (idListWasNullOrEmpty) { delegationIds = Collections.singletonList(null); }
 800  
                 
 801  
                 // Retrieve any existing Lists from the cache.
 802  0
                 for (String delegationId : delegationIds) {
 803  0
                         List<KimDelegationMemberImpl> tempMembers = (List<KimDelegationMemberImpl>) getCacheAdministrator().getFromCache(
 804  
                                         getDelegationMemberListCacheKey(RoleDaoAction.DELEGATION_MEMBERS_FOR_DELEGATION_IDS, delegationId, null, null));
 805  0
                         if (tempMembers != null) {
 806  0
                                 finalResults.put(delegationId, tempMembers);
 807  
                         } else {
 808  0
                                 uncachedDelegationIds.add(delegationId);
 809  
                         }
 810  0
                 }
 811  
                 
 812  
                 // Retrieve and cache any uncached results. If the initial delegation ID List was null or empty, then also cache a List holding all the results.
 813  0
                 if (!uncachedDelegationIds.isEmpty()) {
 814  0
                         List<String> uncachedIdsList = (idListWasNullOrEmpty) ? new ArrayList<String>() : new ArrayList<String>(uncachedDelegationIds);
 815  
                         
 816  0
                         Map<String,List<KimDelegationMemberImpl>> tempMemberMap = roleDao.getDelegationMembersForDelegationIds(uncachedIdsList);
 817  0
                         List<KimDelegationMemberImpl> allMembers = new ArrayList<KimDelegationMemberImpl>();
 818  
                         
 819  0
                         for (Map.Entry<String,List<KimDelegationMemberImpl>> tempMemberEntry : tempMemberMap.entrySet()) {
 820  0
                                 getCacheAdministrator().putInCache(getDelegationMemberListCacheKey(RoleDaoAction.DELEGATION_MEMBERS_FOR_DELEGATION_IDS,
 821  
                                                 tempMemberEntry.getKey(), null, null), tempMemberEntry.getValue(), DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 822  0
                                 for (KimDelegationMemberImpl tempMember : tempMemberEntry.getValue()) {
 823  0
                                         addKimDelegationMemberImplToCache(tempMember);
 824  
                                 }
 825  0
                                 if (idListWasNullOrEmpty) {
 826  0
                                         allMembers.addAll(tempMemberEntry.getValue());
 827  
                                 }
 828  0
                                 finalResults.put(tempMemberEntry.getKey(), tempMemberEntry.getValue());
 829  
                         }
 830  
                         
 831  0
                         if (idListWasNullOrEmpty) {
 832  0
                                 getCacheAdministrator().putInCache(getDelegationMemberListCacheKey(RoleDaoAction.DELEGATION_MEMBERS_FOR_DELEGATION_IDS,
 833  
                                                 null, null, null), allMembers, DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 834  
                         }
 835  
                 }
 836  
                 
 837  0
                 return finalResults;
 838  
         }
 839  
         
 840  
         protected void addKimDelegationMemberImplToCache(KimDelegationMemberImpl delegationMember) {
 841  0
                 if (delegationMember != null) {
 842  0
                         getCacheAdministrator().putInCache(getDelegationMemberCacheKey(delegationMember.getDelegationMemberId()),
 843  
                                         delegationMember, DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 844  0
                         getCacheAdministrator().putInCache(getDelegationMemberByDelegationAndIdCacheKey(delegationMember.getDelegationId(),
 845  
                                         delegationMember.getDelegationMemberId()), delegationMember, DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 846  
                 }
 847  0
         }
 848  
         
 849  
         protected void addKimDelegationMemberImplListByMemberAndDelegationIdToCache(
 850  
                         List<KimDelegationMemberImpl> memberList, String memberId, String delegationId) {
 851  0
                 if (memberList != null) {
 852  0
                         getCacheAdministrator().putInCache(getDelegationMemberListByMemberAndDelegationIdCacheKey(memberId, delegationId),
 853  
                                         memberList, DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 854  
                 }
 855  0
         }
 856  
         
 857  
         protected KimDelegationMemberImpl getDelegationMemberFromCache(String delegationMemberId) {
 858  0
                 return (KimDelegationMemberImpl)getCacheAdministrator().getFromCache(getDelegationMemberCacheKey(delegationMemberId));
 859  
         }
 860  
         
 861  
         protected KimDelegationMemberImpl getDelegationMemberByDelegationAndIdFromCache(String delegationId, String delegationMemberId) {
 862  0
                 return (KimDelegationMemberImpl)getCacheAdministrator().getFromCache(getDelegationMemberByDelegationAndIdCacheKey(delegationId,delegationMemberId));
 863  
         }
 864  
         
 865  
         protected List<KimDelegationMemberImpl> getDelegationMemberListByMemberAndDelegationIdFromCache(String memberId, String delegationId) {
 866  0
                 return (List<KimDelegationMemberImpl>)
 867  
                                 getCacheAdministrator().getFromCache(getDelegationMemberListByMemberAndDelegationIdCacheKey(memberId, delegationId));
 868  
         }
 869  
         
 870  
         public void flushInternalDelegationMemberCache() {
 871  0
                 getCacheAdministrator().flushGroup(DELEGATION_MEMBER_IMPL_CACHE_GROUP);
 872  0
         }
 873  
         
 874  
         /**
 875  
          * Retrieves a KimDelegationMemberImpl object by its ID. If the delegation member already exists in the cache, this method will return the cached
 876  
          * version; otherwise, it will retrieve the uncached version from the database and then cache it before returning it.
 877  
          */
 878  
     protected KimDelegationMemberImpl getKimDelegationMemberImpl( String delegationMemberId ) {
 879  0
             if (StringUtils.isBlank(delegationMemberId)) {
 880  0
                     return null;
 881  
             }
 882  
             
 883  
             // If the KimDelegationMemberImpl exists in the cache, return the cached one.
 884  0
             KimDelegationMemberImpl tempDelegationMember = getDelegationMemberFromCache(delegationMemberId);
 885  0
             if (tempDelegationMember != null) {
 886  0
                     return tempDelegationMember;
 887  
             }
 888  
             // Otherwise, retrieve it normally.
 889  0
             tempDelegationMember = (KimDelegationMemberImpl)getBusinessObjectService().findByPrimaryKey( KimDelegationMemberImpl.class,
 890  
                             Collections.singletonMap(KimConstants.PrimaryKeyConstants.DELEGATION_MEMBER_ID, delegationMemberId) );
 891  0
             addKimDelegationMemberImplToCache(tempDelegationMember);
 892  0
             return tempDelegationMember;
 893  
     }
 894  
 
 895  
     /**
 896  
          * Retrieves a KimDelegationMemberImpl object by its ID and the ID of the delegation it belongs to. If the delegation member exists in the cache,
 897  
          * this method will return the cached one; otherwise, it will retrieve the uncached version from the database and then cache it before returning it.
 898  
          */
 899  
         protected KimDelegationMemberImpl getKimDelegationMemberImplByDelegationAndId(String delegationId, String delegationMemberId) {
 900  0
                 if (StringUtils.isBlank(delegationId) || StringUtils.isBlank(delegationMemberId)) {
 901  0
                     return null;
 902  
             }
 903  
             
 904  
             // If the KimDelegationMemberImpl exists in the cache, return the cached one.
 905  0
             KimDelegationMemberImpl tempDelegationMember = getDelegationMemberByDelegationAndIdFromCache(delegationId, delegationMemberId);
 906  0
             if (tempDelegationMember != null) {
 907  0
                     return tempDelegationMember;
 908  
             }
 909  
             // Otherwise, retrieve it normally.
 910  0
             Map<String,String> searchCriteria = new HashMap<String,String>();
 911  0
             searchCriteria.put(KimConstants.PrimaryKeyConstants.DELEGATION_ID, delegationId);
 912  0
             searchCriteria.put(KimConstants.PrimaryKeyConstants.DELEGATION_MEMBER_ID, delegationMemberId);
 913  0
             List<KimDelegationMemberImpl> memberList =
 914  
                     (List<KimDelegationMemberImpl>) getBusinessObjectService().findMatching(KimDelegationMemberImpl.class, searchCriteria);
 915  0
             if (memberList != null && !memberList.isEmpty()) {
 916  0
                     tempDelegationMember = memberList.get(0);
 917  0
                     addKimDelegationMemberImplToCache(tempDelegationMember);
 918  
             }
 919  0
             return tempDelegationMember;
 920  
         }
 921  
     
 922  
         /**
 923  
          * Retrieves a KimDelegationMemberImpl List by (principal/group/role) member ID and delegation ID. If the List already exists in the cache,
 924  
          * this method will return the cached one; otherwise, it will retrieve the uncached version from the database and then cache it before returning it.
 925  
          */
 926  
         protected List<KimDelegationMemberImpl> getKimDelegationMemberImplListByMemberAndDelegationId(String memberId, String delegationId) {
 927  
                 // If the KimDelegationMemberImpl List exists in the cache, return the cached one.
 928  0
             List<KimDelegationMemberImpl> memberList = getDelegationMemberListByMemberAndDelegationIdFromCache(memberId, delegationId);
 929  0
             if (memberList != null) {
 930  0
                     return memberList;
 931  
             }
 932  
                 
 933  
             // Otherwise, retrieve it normally.
 934  0
             Map<String,String> searchCriteria = new HashMap<String,String>();
 935  0
                 searchCriteria.put(KimConstants.PrimaryKeyConstants.MEMBER_ID, memberId);
 936  0
             searchCriteria.put(KimConstants.PrimaryKeyConstants.DELEGATION_ID, delegationId);
 937  0
             List<KimDelegationMemberImpl> tempList =
 938  
                     (List<KimDelegationMemberImpl>)getBusinessObjectService().findMatching(KimDelegationMemberImpl.class, searchCriteria);
 939  0
             if (tempList != null && !tempList.isEmpty()) {
 940  0
                     memberList = new ArrayList<KimDelegationMemberImpl>();
 941  0
                     memberList.addAll(tempList);
 942  0
                     addKimDelegationMemberImplListByMemberAndDelegationIdToCache(memberList, memberId, delegationId);
 943  
             }
 944  0
                 return memberList;
 945  
         }
 946  
         
 947  
         public void flushInternalRoleCache() {
 948  0
             getCacheAdministrator().flushGroup(ROLE_IMPL_CACHE_GROUP);
 949  0
     }
 950  
         
 951  
         public RoleMemberCompleteInfo findRoleMemberCompleteInfo(String roleMemberId){
 952  0
             Map<String, String> fieldValues = new HashMap<String, String>();
 953  0
             fieldValues.put(KimConstants.PrimaryKeyConstants.ROLE_MEMBER_ID, roleMemberId);
 954  0
             List<RoleMemberCompleteInfo> roleMemberInfos = findRoleMembersCompleteInfo(fieldValues);
 955  0
             if(roleMemberInfos!=null && roleMemberInfos.size()>0)
 956  0
                     return roleMemberInfos.get(0);
 957  0
             return null;
 958  
     }
 959  
         
 960  
         public List<RoleMemberCompleteInfo> findRoleMembersCompleteInfo(Map<String, String> fieldValues){
 961  0
             List<RoleMemberCompleteInfo> roleMembersCompleteInfos = new ArrayList<RoleMemberCompleteInfo>();
 962  
             RoleMemberCompleteInfo roleMembersCompleteInfo;
 963  0
             List<RoleMemberImpl> roleMembers = (List<RoleMemberImpl>)getLookupService().findCollectionBySearchHelper(
 964  
                                 RoleMemberImpl.class, fieldValues, true);
 965  0
             for(RoleMemberImpl roleMember: roleMembers){
 966  0
                     roleMembersCompleteInfo = roleMember.toSimpleInfo();
 967  0
                         Object member = getMember(roleMembersCompleteInfo.getMemberTypeCode(), roleMembersCompleteInfo.getMemberId());
 968  0
                         roleMembersCompleteInfo.setMemberName(getMemberName(member));
 969  0
                         roleMembersCompleteInfo.setMemberNamespaceCode(getMemberNamespaceCode(member));
 970  0
                         roleMembersCompleteInfo.setRoleRspActions(getRoleMemberResponsibilityActionInfo(roleMember.getRoleMemberId()));
 971  0
                         roleMembersCompleteInfos.add(roleMembersCompleteInfo);
 972  0
             }
 973  0
             return roleMembersCompleteInfos;
 974  
     }
 975  
         
 976  
         public List<RoleResponsibilityActionInfo> getRoleMemberResponsibilityActionInfo(String roleMemberId){
 977  0
                 Map<String, String> criteria = new HashMap<String, String>(1);                
 978  0
                 criteria.put(KimConstants.PrimaryKeyConstants.ROLE_MEMBER_ID, roleMemberId);
 979  0
                 List<RoleResponsibilityActionImpl> responsibilityImpls = (List<RoleResponsibilityActionImpl>)
 980  
                         getBusinessObjectService().findMatching(RoleResponsibilityActionImpl.class, criteria);
 981  0
                 List<RoleResponsibilityActionInfo> roleResponsibilityActionInfos = new ArrayList<RoleResponsibilityActionInfo>();
 982  
                 RoleResponsibilityActionInfo roleResponsibilityActionInfo;
 983  0
                 for(RoleResponsibilityActionImpl responsibilityActionImpl: responsibilityImpls){
 984  0
                         roleResponsibilityActionInfo = new RoleResponsibilityActionInfo();
 985  0
                         KimCommonUtils.copyProperties(roleResponsibilityActionInfo, responsibilityActionImpl);
 986  0
                         roleResponsibilityActionInfos.add(roleResponsibilityActionInfo);
 987  
                 }
 988  0
                 return roleResponsibilityActionInfos;
 989  
         }
 990  
         
 991  
         public List<DelegateMemberCompleteInfo> findDelegateMembersCompleteInfo(final Map<String, String> fieldValues){
 992  0
             List<DelegateMemberCompleteInfo> delegateMembersCompleteInfo = new ArrayList<DelegateMemberCompleteInfo>();
 993  
             DelegateMemberCompleteInfo delegateMemberCompleteInfo;
 994  0
             List<KimDelegationImpl> delegations = (List<KimDelegationImpl>)getLookupService().findCollectionBySearchHelper(
 995  
                                 KimDelegationImpl.class, fieldValues, true);
 996  0
             if(delegations!=null && !delegations.isEmpty()){
 997  0
                     Map<String, String> delegationMemberFieldValues = new HashMap<String, String>();
 998  0
                     for(String key: fieldValues.keySet()){
 999  0
                             if(key.startsWith(KimConstants.KimUIConstants.MEMBER_ID_PREFIX)){
 1000  0
                                     delegationMemberFieldValues.put(
 1001  
                                                     key.substring(key.indexOf(
 1002  
                                                     KimConstants.KimUIConstants.MEMBER_ID_PREFIX)+KimConstants.KimUIConstants.MEMBER_ID_PREFIX.length()), 
 1003  
                                                     fieldValues.get(key));
 1004  
                             }
 1005  
                     }
 1006  0
                         StringBuffer memberQueryString = new StringBuffer();
 1007  0
                     for(KimDelegationImpl delegation: delegations)
 1008  0
                             memberQueryString.append(delegation.getDelegationId()+KimConstants.KimUIConstants.OR_OPERATOR);
 1009  0
                     delegationMemberFieldValues.put(KimConstants.PrimaryKeyConstants.DELEGATION_ID, 
 1010  
                                     KimCommonUtils.stripEnd(memberQueryString.toString(), KimConstants.KimUIConstants.OR_OPERATOR));
 1011  0
                     List<KimDelegationMemberImpl> delegateMembers = (List<KimDelegationMemberImpl>)getLookupService().findCollectionBySearchHelper(
 1012  
                                         KimDelegationMemberImpl.class, delegationMemberFieldValues, true);
 1013  
                     KimDelegationImpl delegationTemp;
 1014  0
                     for(KimDelegationMemberImpl delegateMember: delegateMembers){
 1015  0
                             delegateMemberCompleteInfo = delegateMember.toSimpleInfo();
 1016  0
                             delegationTemp = getDelegationImpl(delegations, delegateMember.getDelegationId());
 1017  0
                             delegateMemberCompleteInfo.setRoleId(delegationTemp.getRoleId());
 1018  0
                             delegateMemberCompleteInfo.setDelegationTypeCode(delegationTemp.getDelegationTypeCode());
 1019  0
                                 Object member = getMember(delegateMemberCompleteInfo.getMemberTypeCode(), delegateMemberCompleteInfo.getMemberId());
 1020  0
                                 delegateMemberCompleteInfo.setMemberName(getMemberName(member));
 1021  0
                                 delegateMemberCompleteInfo.setMemberNamespaceCode(getMemberNamespaceCode(member));
 1022  0
                                 delegateMembersCompleteInfo.add(delegateMemberCompleteInfo);
 1023  0
                     }
 1024  
 
 1025  
             }
 1026  0
             return delegateMembersCompleteInfo;
 1027  
     }
 1028  
         
 1029  
         protected KimDelegationImpl getDelegationImpl(List<KimDelegationImpl> delegations, String delegationId){
 1030  0
             if(StringUtils.isEmpty(delegationId) || delegations==null)
 1031  0
                     return null;
 1032  0
             for(KimDelegationImpl delegation: delegations){
 1033  0
                     if(StringUtils.equals(delegation.getDelegationId(), delegationId))
 1034  0
                             return delegation;
 1035  
             }
 1036  0
             return null;
 1037  
     }
 1038  
         
 1039  
         protected Object getMember(String memberTypeCode, String memberId){
 1040  0
                 if ( StringUtils.isBlank(memberId) ) {
 1041  0
                         return null;
 1042  
                 }
 1043  0
         if(KimConstants.KimUIConstants.MEMBER_TYPE_PRINCIPAL_CODE.equals(memberTypeCode)){
 1044  0
                 return getIdentityManagementService().getPrincipal(memberId);
 1045  0
         } else if(KimConstants.KimUIConstants.MEMBER_TYPE_GROUP_CODE.equals(memberTypeCode)){
 1046  0
                 return getIdentityManagementService().getGroup(memberId);
 1047  0
         } else if(KimConstants.KimUIConstants.MEMBER_TYPE_ROLE_CODE.equals(memberTypeCode)){
 1048  0
                 return getRoleImpl(memberId);
 1049  
         }
 1050  0
         return null;
 1051  
     }
 1052  
         
 1053  
         protected String getMemberName(Object member){
 1054  0
         if( member == null ) {
 1055  0
                 return "";
 1056  
         }
 1057  0
         if ( member instanceof KimPrincipal ) {
 1058  0
                 return ((KimPrincipal)member).getPrincipalName();
 1059  
         }
 1060  0
         if ( member instanceof Group ) {
 1061  0
                 return ((Group)member).getGroupName();
 1062  
         }
 1063  0
         if ( member instanceof Role ) {
 1064  0
                 return ((Role)member).getRoleName();
 1065  
         }
 1066  0
         return member.toString();
 1067  
     }
 1068  
         
 1069  
         protected String getMemberNamespaceCode(Object member){
 1070  0
         if( member == null ) {
 1071  0
                 return "";
 1072  
         }
 1073  0
         if ( member instanceof KimPrincipal ) {
 1074  0
                 return "";
 1075  
         }
 1076  0
         if ( member instanceof Group ) {
 1077  0
                 return ((Group)member).getNamespaceCode();
 1078  
         }
 1079  0
         if ( member instanceof Role ) {
 1080  0
                 return ((Role)member).getNamespaceCode();
 1081  
         }
 1082  0
         return "";
 1083  
     }
 1084  
         
 1085  
         protected RoleImpl getRoleImpl(String roleId) {
 1086  0
                 if ( StringUtils.isBlank( roleId ) ) {
 1087  0
                         return null;
 1088  
                 }
 1089  
                 // check for a non-null result in the cache, return it if found
 1090  0
                 RoleImpl cachedResult = getRoleFromCache( roleId );
 1091  0
                 if ( cachedResult != null ) {
 1092  0
                         return cachedResult;
 1093  
                 }
 1094  
                 // otherwise, run the query
 1095  0
                 RoleImpl result = (RoleImpl)getBusinessObjectService().findBySinglePrimaryKey(RoleImpl.class, roleId);
 1096  0
                 addRoleImplToCache( result );
 1097  0
                 return result;
 1098  
         }
 1099  
         
 1100  
         protected KimDelegationImpl getDelegationOfType(String roleId, String delegationTypeCode){
 1101  0
             List<KimDelegationImpl> roleDelegations = getRoleDelegations(roleId);
 1102  0
         if(isDelegationPrimary(delegationTypeCode))
 1103  0
             return getPrimaryDelegation(roleId, roleDelegations);
 1104  
         else
 1105  0
             return getSecondaryDelegation(roleId, roleDelegations);
 1106  
     }
 1107  
         
 1108  
         private KimDelegationImpl getSecondaryDelegation(String roleId, List<KimDelegationImpl> roleDelegations){
 1109  0
         KimDelegationImpl secondaryDelegation = null;
 1110  0
         RoleImpl roleImpl = getRoleImpl(roleId);
 1111  0
         for(KimDelegationImpl delegation: roleDelegations){
 1112  0
             if(isDelegationSecondary(delegation.getDelegationTypeCode()))
 1113  0
                 secondaryDelegation = delegation;
 1114  
         }
 1115  0
         if(secondaryDelegation==null){
 1116  0
             secondaryDelegation = new KimDelegationImpl();
 1117  0
             secondaryDelegation.setRoleId(roleId);
 1118  0
             secondaryDelegation.setDelegationId(getNewDelegationId());
 1119  0
             secondaryDelegation.setDelegationTypeCode(KEWConstants.DELEGATION_SECONDARY);
 1120  0
             secondaryDelegation.setKimTypeId(roleImpl.getKimTypeId());
 1121  
         }
 1122  0
         return secondaryDelegation;
 1123  
     }
 1124  
         
 1125  
         protected KimDelegationImpl getPrimaryDelegation(String roleId, List<KimDelegationImpl> roleDelegations){
 1126  0
         KimDelegationImpl primaryDelegation = null;
 1127  0
         RoleImpl roleImpl = getRoleImpl(roleId);
 1128  0
         for(KimDelegationImpl delegation: roleDelegations){
 1129  0
             if(isDelegationPrimary(delegation.getDelegationTypeCode()))
 1130  0
                 primaryDelegation = delegation;
 1131  
         }
 1132  0
         if(primaryDelegation==null){
 1133  0
             primaryDelegation = new KimDelegationImpl();
 1134  0
             primaryDelegation.setRoleId(roleId);
 1135  0
             primaryDelegation.setDelegationId(getNewDelegationId());
 1136  0
             primaryDelegation.setDelegationTypeCode(KEWConstants.DELEGATION_PRIMARY);
 1137  0
             primaryDelegation.setKimTypeId(roleImpl.getKimTypeId());
 1138  
         }
 1139  0
         return primaryDelegation;
 1140  
     }
 1141  
         
 1142  
         protected RoleMemberImpl matchingMemberRecord( List<RoleMemberImpl> roleMembers, String memberId, String memberTypeCode, AttributeSet qualifier ) {
 1143  0
                 for ( RoleMemberImpl rm : roleMembers ) {
 1144  0
                         if ( doesMemberMatch( rm, memberId, memberTypeCode, qualifier ) ) {
 1145  0
                                 return rm;
 1146  
                         }
 1147  
                 }
 1148  0
                 return null;
 1149  
         }
 1150  
         
 1151  
         protected boolean isDelegationPrimary(String delegationTypeCode){
 1152  0
         return KEWConstants.DELEGATION_PRIMARY.equals(delegationTypeCode);
 1153  
     }
 1154  
         
 1155  
         protected boolean isDelegationSecondary(String delegationTypeCode){
 1156  0
         return KEWConstants.DELEGATION_SECONDARY.equals(delegationTypeCode);
 1157  
     }
 1158  
         
 1159  
         
 1160  
         private List<KimDelegationImpl> getRoleDelegations(String roleId){
 1161  0
                 if(roleId==null)
 1162  0
                         return new ArrayList<KimDelegationImpl>();
 1163  0
                 return getStoredDelegationImplsForRoleIds(Collections.singletonList(roleId));
 1164  
 
 1165  
         }
 1166  
         
 1167  
         protected RoleImpl getRoleImplByName( String namespaceCode, String roleName ) {
 1168  0
                 if ( StringUtils.isBlank( namespaceCode )
 1169  
                                 || StringUtils.isBlank( roleName ) ) {
 1170  0
                         return null;
 1171  
                 }
 1172  
                 // check for a non-null result in the cache, return it if found
 1173  0
                 RoleImpl cachedResult = getRoleFromCache( namespaceCode, roleName );
 1174  0
                 if ( cachedResult != null ) {
 1175  0
                         return cachedResult;
 1176  
                 }
 1177  0
                 AttributeSet criteria = new AttributeSet();
 1178  0
                 criteria.put(KimConstants.UniqueKeyConstants.NAMESPACE_CODE, namespaceCode);
 1179  0
                 criteria.put(KimConstants.UniqueKeyConstants.ROLE_NAME, roleName);
 1180  0
                 criteria.put(KNSPropertyConstants.ACTIVE, "Y");
 1181  
                 // while this is not actually the primary key - there will be at most one row with these criteria
 1182  0
                 RoleImpl result = (RoleImpl)getBusinessObjectService().findByPrimaryKey(RoleImpl.class, criteria);
 1183  0
                 addRoleImplToCache( result );
 1184  0
                 return result;
 1185  
         }
 1186  
         
 1187  
         protected boolean doAnyMemberRecordsMatch( List<RoleMemberImpl> roleMembers, String memberId, String memberTypeCode, AttributeSet qualifier ) {
 1188  0
                 for ( RoleMemberImpl rm : roleMembers ) {
 1189  0
                         if ( doesMemberMatch( rm, memberId, memberTypeCode, qualifier ) ) {
 1190  0
                                 return true;
 1191  
                         }
 1192  
                 }
 1193  0
                 return false;
 1194  
         }
 1195  
         
 1196  
         protected boolean doesMemberMatch( RoleMemberImpl roleMember, String memberId, String memberTypeCode, AttributeSet qualifier ) {
 1197  0
                 if ( roleMember.getMemberId().equals( memberId ) && roleMember.getMemberTypeCode().equals( memberTypeCode ) ) {
 1198  
                         // member ID/type match
 1199  0
                     AttributeSet roleQualifier = roleMember.getQualifier();
 1200  0
                     if ( (qualifier == null || qualifier.isEmpty())
 1201  
                                     && (roleQualifier == null || roleQualifier.isEmpty()) ) {
 1202  0
                             return true; // blank qualifier match
 1203  
                     } else {
 1204  0
                             if ( qualifier != null && roleQualifier != null && qualifier.equals( roleQualifier ) ) {
 1205  0
                                     return true; // qualifier match
 1206  
                             }
 1207  
                     }
 1208  
                 }
 1209  0
                 return false;
 1210  
         }
 1211  
 
 1212  
         /**
 1213  
          * 
 1214  
          * This method tests to see if assigning a role to another role will create a circular reference.
 1215  
          * The Role is checked to see if it is a member (direct or nested) of the role to be assigned as a member.
 1216  
          * 
 1217  
          * @param newMemberId
 1218  
          * @param role
 1219  
          * @return true  - assignment is allowed, no circular reference will be created.
 1220  
          *         false - illegal assignment, it will create a circular membership
 1221  
          */
 1222  
         protected boolean checkForCircularRoleMembership( String newMemberId, RoleImpl role ) {
 1223  
                 // get all nested role members that are of type role
 1224  0
                 Set<String> newRoleMemberIds = getRoleTypeRoleMemberIds(newMemberId);
 1225  0
                 if (newRoleMemberIds.contains(role.getRoleId())){
 1226  0
                                         return false;
 1227  
                 }
 1228  0
                 return true;
 1229  
         }
 1230  
         
 1231  
         // TODO: pulling attribute IDs repeatedly is inefficient - consider caching the entire list as a map
 1232  
         @SuppressWarnings("unchecked")
 1233  
         protected String getKimAttributeId( String attributeName ) {
 1234  0
                 String result = null;
 1235  0
                 Map<String,Object> critieria = new HashMap<String,Object>( 1 );
 1236  0
                 critieria.put( "attributeName", attributeName );
 1237  0
                 Collection<KimAttributeImpl> defs = getBusinessObjectService().findMatching( KimAttributeImpl.class, critieria );
 1238  0
                 if(CollectionUtils.isNotEmpty(defs)) {
 1239  0
                         result = defs.iterator().next().getKimAttributeId();
 1240  
                 }
 1241  0
                 return result;
 1242  
         }
 1243  
         
 1244  
     /**
 1245  
          * @return the applicationRoleTypeCache
 1246  
          */
 1247  
         protected Map<String, Boolean> getApplicationRoleTypeCache() {
 1248  0
                 return this.applicationRoleTypeCache;
 1249  
         }
 1250  
         /**
 1251  
          * @return the roleTypeServiceCache
 1252  
          */
 1253  
         protected Map<String, KimRoleTypeService> getRoleTypeServiceCache() {
 1254  0
                 return this.roleTypeServiceCache;
 1255  
         }
 1256  
 
 1257  
         /**
 1258  
          * @return the delegationTypeServiceCache
 1259  
          */
 1260  
         protected Map<String, KimDelegationTypeService> getDelegationTypeServiceCache() {
 1261  0
                 return this.delegationTypeServiceCache;
 1262  
         }
 1263  
         protected String getRoleCacheKey( String roleId ) {
 1264  0
             return ROLE_IMPL_CACHE_PREFIX + roleId;
 1265  
     }
 1266  
 
 1267  
     protected String getRoleByNameCacheKey( String namespaceCode, String roleName ) {
 1268  0
             return ROLE_IMPL_BY_NAME_CACHE_PREFIX + namespaceCode + "-" + roleName;
 1269  
             }
 1270  
 
 1271  
     protected void addRoleImplToCache( RoleImpl role ) {
 1272  0
             if (role != null) {
 1273  0
                     getCacheAdministrator().putInCache(getRoleCacheKey(role.getRoleId()), role, ROLE_IMPL_CACHE_GROUP);
 1274  0
                     getCacheAdministrator().putInCache(getRoleByNameCacheKey(role.getNamespaceCode(),role.getRoleName()), role, ROLE_IMPL_CACHE_GROUP);
 1275  
                     }
 1276  0
             }
 1277  
 
 1278  
     protected RoleImpl getRoleFromCache( String roleId ) {
 1279  0
             return (RoleImpl)getCacheAdministrator().getFromCache(getRoleCacheKey(roleId));
 1280  
     }
 1281  
 
 1282  
     protected RoleImpl getRoleFromCache( String namespaceCode, String roleName ) {
 1283  0
             return (RoleImpl)getCacheAdministrator().getFromCache(getRoleByNameCacheKey(namespaceCode,roleName));
 1284  
     }
 1285  
     
 1286  
     protected String getNewDelegationId(){
 1287  0
             SequenceAccessorService sas = getSequenceAccessorService();
 1288  0
             Long nextSeq = sas.getNextAvailableSequenceNumber(
 1289  
                 KimConstants.SequenceNames.KRIM_DLGN_ID_S,
 1290  
                 KimDelegationImpl.class);
 1291  0
         return nextSeq.toString();
 1292  
     }
 1293  
 
 1294  
     protected String getNewAttributeDataId(){
 1295  0
                 SequenceAccessorService sas = getSequenceAccessorService();                
 1296  0
                 Long nextSeq = sas.getNextAvailableSequenceNumber(
 1297  
                                 KimConstants.SequenceNames.KRIM_ATTR_DATA_ID_S, 
 1298  
                                 RoleMemberAttributeDataImpl.class );
 1299  0
                 return nextSeq.toString();
 1300  
     }
 1301  
     
 1302  
     protected String getNewDelegationMemberId(){
 1303  0
             SequenceAccessorService sas = getSequenceAccessorService();
 1304  0
             Long nextSeq = sas.getNextAvailableSequenceNumber(
 1305  
                 KimConstants.SequenceNames.KRIM_DLGN_MBR_ID_S,
 1306  
                 KimDelegationImpl.class);
 1307  0
         return nextSeq.toString();
 1308  
     }
 1309  
     
 1310  
     protected BusinessObjectService getBusinessObjectService() {
 1311  0
                 if ( businessObjectService == null ) {
 1312  0
                         businessObjectService = KNSServiceLocator.getBusinessObjectService();
 1313  
                 }
 1314  0
                 return businessObjectService;
 1315  
         }
 1316  
     
 1317  
     /**
 1318  
          * @return the lookupService
 1319  
          */
 1320  
     protected LookupService getLookupService() {
 1321  0
                 if(lookupService == null) {
 1322  0
                         lookupService = KNSServiceLocator.getLookupService();
 1323  
                 }
 1324  0
                 return lookupService;
 1325  
         }
 1326  
     
 1327  
     protected RiceCacheAdministrator getCacheAdministrator() {
 1328  0
                 if ( cacheAdministrator == null ) {
 1329  0
                         cacheAdministrator = KEWServiceLocator.getCacheAdministrator();
 1330  
                 }
 1331  0
                 return cacheAdministrator;
 1332  
     }
 1333  
     
 1334  
     protected IdentityManagementService getIdentityManagementService() {
 1335  0
                 if ( identityManagementService == null ) {
 1336  0
                         identityManagementService = KIMServiceLocator.getIdentityManagementService();
 1337  
                 }
 1338  
 
 1339  0
                 return identityManagementService;
 1340  
         }
 1341  
 
 1342  
         protected SequenceAccessorService getSequenceAccessorService() {
 1343  0
                 if ( sequenceAccessorService == null ) {
 1344  0
                         sequenceAccessorService = KNSServiceLocator.getSequenceAccessorService();
 1345  
                 }
 1346  0
                 return sequenceAccessorService;
 1347  
         }
 1348  
         
 1349  
         protected ResponsibilityInternalService getResponsibilityInternalService() {
 1350  0
                 if ( responsibilityInternalService == null ) {
 1351  0
                         responsibilityInternalService = KIMServiceLocator.getResponsibilityInternalService();
 1352  
                 }
 1353  0
                 return responsibilityInternalService;
 1354  
         }
 1355  
         
 1356  
         protected IdentityManagementNotificationService getIdentityManagementNotificationService() {
 1357  0
         return (IdentityManagementNotificationService)KSBServiceLocator.getMessageHelper().getServiceAsynchronously(new QName("KIM", "kimIdentityManagementNotificationService"));
 1358  
     }
 1359  
         
 1360  
         /**
 1361  
          * An internal helper class for encapsulating the information related to generating a key for a RoleMemberImpl list. 
 1362  
          * 
 1363  
          * @author Kuali Rice Team (rice.collab@kuali.org)
 1364  
          */
 1365  0
         private class RoleMemberCacheKeyHelper {
 1366  
                 private final RoleDaoAction ROLE_DAO_ACTION;
 1367  
                 private final String ROLE_ID;
 1368  
                 private final String PRINCIPAL_ID;
 1369  
                 private final String GROUP_ID;
 1370  
                 private final String MEMBER_TYPE_CODE;
 1371  
                 private String cacheKey;
 1372  
                 
 1373  0
                 private RoleMemberCacheKeyHelper(RoleDaoAction roleDaoAction, String roleId, String principalId, String groupId, String memberTypeCode) {
 1374  0
                         this.ROLE_DAO_ACTION = roleDaoAction;
 1375  0
                         this.ROLE_ID = roleId;
 1376  0
                         this.PRINCIPAL_ID = principalId;
 1377  0
                         this.GROUP_ID = groupId;
 1378  0
                         this.MEMBER_TYPE_CODE = memberTypeCode;
 1379  0
                 }
 1380  
                 
 1381  
                 private String getCacheKey() {
 1382  0
                         if (this.cacheKey == null) {
 1383  0
                                 this.cacheKey = getRoleMemberListCacheKey(ROLE_DAO_ACTION, ROLE_ID, PRINCIPAL_ID, GROUP_ID, MEMBER_TYPE_CODE);
 1384  
                         }
 1385  0
                         return this.cacheKey;
 1386  
                 }
 1387  
         }
 1388  
         /**
 1389  
          * @return the roleDao
 1390  
          */
 1391  
         public KimRoleDao getRoleDao() {
 1392  0
                 return this.roleDao;
 1393  
         }
 1394  
 
 1395  
         /**
 1396  
          * @param roleDao the roleDao to set
 1397  
          */
 1398  
         public void setRoleDao(KimRoleDao roleDao) {
 1399  0
                 this.roleDao = roleDao;
 1400  0
         }
 1401  
 }