|  1 |     | 
     | 
  |  2 |     | 
     | 
  |  3 |     | 
     | 
  |  4 |     | 
     | 
  |  5 |     | 
     | 
  |  6 |     | 
     | 
  |  7 |     | 
     | 
  |  8 |     | 
     | 
  |  9 |     | 
     | 
  |  10 |     | 
     | 
  |  11 |     | 
     | 
  |  12 |     | 
     | 
  |  13 |     | 
     | 
  |  14 |     | 
     | 
  |  15 |     | 
     | 
  |  16 |     | 
   package org.kuali.rice.kim.service.impl;  | 
  |  17 |     | 
     | 
  |  18 |     | 
   import java.lang.ref.SoftReference;  | 
  |  19 |     | 
   import java.security.GeneralSecurityException;  | 
  |  20 |     | 
   import java.util.ArrayList;  | 
  |  21 |     | 
   import java.util.Collection;  | 
  |  22 |     | 
   import java.util.Collections;  | 
  |  23 |     | 
   import java.util.HashMap;  | 
  |  24 |     | 
   import java.util.Iterator;  | 
  |  25 |     | 
   import java.util.List;  | 
  |  26 |     | 
   import java.util.Map;  | 
  |  27 |     | 
     | 
  |  28 |     | 
   import org.apache.commons.beanutils.PropertyUtils;  | 
  |  29 |     | 
   import org.apache.commons.lang.StringUtils;  | 
  |  30 |     | 
   import org.apache.log4j.Logger;  | 
  |  31 |     | 
   import org.kuali.rice.core.api.services.CoreApiServiceLocator;  | 
  |  32 |     | 
   import org.kuali.rice.core.util.MaxAgeSoftReference;  | 
  |  33 |     | 
   import org.kuali.rice.kim.bo.Person;  | 
  |  34 |     | 
   import org.kuali.rice.kim.bo.entity.KimPrincipal;  | 
  |  35 |     | 
   import org.kuali.rice.kim.bo.entity.dto.KimEntityDefaultInfo;  | 
  |  36 |     | 
   import org.kuali.rice.kim.bo.entity.dto.KimEntityEntityTypeDefaultInfo;  | 
  |  37 |     | 
   import org.kuali.rice.kim.bo.impl.PersonImpl;  | 
  |  38 |     | 
   import org.kuali.rice.kim.bo.reference.dto.ExternalIdentifierTypeInfo;  | 
  |  39 |     | 
   import org.kuali.rice.kim.service.IdentityManagementService;  | 
  |  40 |     | 
   import org.kuali.rice.kim.service.KIMServiceLocator;  | 
  |  41 |     | 
   import org.kuali.rice.kim.service.PersonService;  | 
  |  42 |     | 
   import org.kuali.rice.kim.service.RoleManagementService;  | 
  |  43 |     | 
   import org.kuali.rice.kim.util.KIMPropertyConstants;  | 
  |  44 |     | 
   import org.kuali.rice.kns.bo.BusinessObject;  | 
  |  45 |     | 
   import org.kuali.rice.kns.bo.BusinessObjectRelationship;  | 
  |  46 |     | 
   import org.kuali.rice.kns.lookup.CollectionIncomplete;  | 
  |  47 |     | 
   import org.kuali.rice.kns.lookup.LookupUtils;  | 
  |  48 |     | 
   import org.kuali.rice.kns.service.BusinessObjectMetaDataService;  | 
  |  49 |     | 
   import org.kuali.rice.kns.service.KNSServiceLocator;  | 
  |  50 |     | 
   import org.kuali.rice.kns.service.KNSServiceLocatorWeb;  | 
  |  51 |     | 
   import org.kuali.rice.kns.service.MaintenanceDocumentDictionaryService;  | 
  |  52 |     | 
   import org.kuali.rice.kns.util.KNSConstants;  | 
  |  53 |     | 
   import org.kuali.rice.kns.util.KNSPropertyConstants;  | 
  |  54 |     | 
   import org.kuali.rice.kns.util.ObjectUtils;  | 
  |  55 |     | 
     | 
  |  56 |     | 
     | 
  |  57 |     | 
     | 
  |  58 |     | 
     | 
  |  59 |     | 
     | 
  |  60 |     | 
     | 
  |  61 |     | 
     | 
  |  62 |    0 |    public class PersonServiceImpl implements PersonService { | 
  |  63 |     | 
     | 
  |  64 |    0 |            private static Logger LOG = Logger.getLogger( PersonServiceImpl.class );  | 
  |  65 |     | 
           protected static final String ENTITY_EXT_ID_PROPERTY_PREFIX = "externalIdentifiers.";  | 
  |  66 |     | 
           protected static final String ENTITY_AFFILIATION_PROPERTY_PREFIX = "affiliations.";  | 
  |  67 |     | 
           protected static final String ENTITY_TYPE_PROPERTY_PREFIX = "entityTypes.";  | 
  |  68 |     | 
           protected static final String ENTITY_EMAIL_PROPERTY_PREFIX = "entityTypes.emailAddresses.";  | 
  |  69 |     | 
           protected static final String ENTITY_PHONE_PROPERTY_PREFIX = "entityTypes.phoneNumbers.";  | 
  |  70 |     | 
           protected static final String ENTITY_ADDRESS_PROPERTY_PREFIX = "entityTypes.addresses.";  | 
  |  71 |     | 
           protected static final String ENTITY_NAME_PROPERTY_PREFIX = "names.";  | 
  |  72 |     | 
           protected static final String PRINCIPAL_PROPERTY_PREFIX = "principals.";  | 
  |  73 |     | 
           protected static final String ENTITY_EMPLOYEE_ID_PROPERTY_PREFIX = "employmentInformation.";  | 
  |  74 |     | 
             | 
  |  75 |     | 
           protected static final String EXTENSION = "extension";  | 
  |  76 |     | 
             | 
  |  77 |     | 
           private IdentityManagementService identityManagementService;          | 
  |  78 |     | 
           private RoleManagementService roleManagementService;  | 
  |  79 |     | 
           private BusinessObjectMetaDataService businessObjectMetaDataService;  | 
  |  80 |     | 
           private MaintenanceDocumentDictionaryService maintenanceDocumentDictionaryService;  | 
  |  81 |     | 
     | 
  |  82 |     | 
             | 
  |  83 |    0 |            protected int personCacheMaxSize = 3000;  | 
  |  84 |    0 |            protected int personCacheMaxAgeSeconds = 3600;  | 
  |  85 |     | 
     | 
  |  86 |    0 |            protected Map<String,MaxAgeSoftReference<Person>> personCache = Collections.synchronizedMap( new HashMap<String,MaxAgeSoftReference<Person>>( personCacheMaxSize ) );  | 
  |  87 |     | 
             | 
  |  88 |     | 
     | 
  |  89 |    0 |            protected List<String> personEntityTypeCodes = new ArrayList<String>( 4 );  | 
  |  90 |     | 
             | 
  |  91 |    0 |            private String personEntityTypeLookupCriteria = null;  | 
  |  92 |     | 
         | 
  |  93 |    0 |            protected Map<String,String> baseLookupCriteria = new HashMap<String,String>();  | 
  |  94 |    0 |            protected Map<String,String> criteriaConversion = new HashMap<String,String>();  | 
  |  95 |    0 |            protected ArrayList<String> personCachePropertyNames = new ArrayList<String>();  | 
  |  96 |     | 
           { | 
  |  97 |     | 
                     | 
  |  98 |     | 
                     | 
  |  99 |    0 |                    baseLookupCriteria.put( KIMPropertyConstants.Person.ACTIVE, "Y" );  | 
  |  100 |    0 |                    baseLookupCriteria.put( ENTITY_TYPE_PROPERTY_PREFIX + KNSPropertyConstants.ACTIVE, "Y" );  | 
  |  101 |     | 
                     | 
  |  102 |     | 
                     | 
  |  103 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.ENTITY_ID, KIMPropertyConstants.Person.ENTITY_ID );  | 
  |  104 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.ACTIVE, PRINCIPAL_PROPERTY_PREFIX + KNSPropertyConstants.ACTIVE );  | 
  |  105 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.PRINCIPAL_ID, PRINCIPAL_PROPERTY_PREFIX + KIMPropertyConstants.Person.PRINCIPAL_ID );  | 
  |  106 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.PRINCIPAL_NAME, PRINCIPAL_PROPERTY_PREFIX + KIMPropertyConstants.Person.PRINCIPAL_NAME );  | 
  |  107 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.FIRST_NAME, "names.firstName" );  | 
  |  108 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.LAST_NAME, "names.lastName" );  | 
  |  109 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.MIDDLE_NAME, "names.middleName" );  | 
  |  110 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.EMAIL_ADDRESS, "entityTypes.emailAddresses.emailAddress" );  | 
  |  111 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.PHONE_NUMBER, "entityTypes.phoneNumbers.phoneNumber" );  | 
  |  112 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.ADDRESS_LINE_1, "entityTypes.addresses.line1" );  | 
  |  113 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.ADDRESS_LINE_2, "entityTypes.addresses.line2" );  | 
  |  114 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.ADDRESS_LINE_3, "entityTypes.addresses.line3" );  | 
  |  115 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.CITY_NAME, "entityTypes.addresses.cityName" );  | 
  |  116 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.STATE_CODE, "entityTypes.addresses.stateCode" );  | 
  |  117 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.POSTAL_CODE, "entityTypes.addresses.postalCode" );  | 
  |  118 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.COUNTRY_CODE, "entityTypes.addresses.countryCode" );  | 
  |  119 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.CAMPUS_CODE, "affiliations.campusCode" );  | 
  |  120 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.AFFILIATION_TYPE_CODE, "affiliations.affiliationTypeCode" );  | 
  |  121 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.EXTERNAL_IDENTIFIER_TYPE_CODE, "externalIdentifiers.externalIdentifierTypeCode" );  | 
  |  122 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.EXTERNAL_ID, "externalIdentifiers.externalId" );                  | 
  |  123 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.EMPLOYEE_TYPE_CODE, "employmentInformation.employeeTypeCode" );  | 
  |  124 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.EMPLOYEE_STATUS_CODE, "employmentInformation.employeeStatusCode" );  | 
  |  125 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.EMPLOYEE_ID, "employmentInformation.employeeId" );  | 
  |  126 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.BASE_SALARY_AMOUNT, "employmentInformation.baseSalaryAmount" );  | 
  |  127 |    0 |                    criteriaConversion.put( KIMPropertyConstants.Person.PRIMARY_DEPARTMENT_CODE, "employmentInformation.primaryDepartmentCode" );  | 
  |  128 |     | 
     | 
  |  129 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.PRINCIPAL_ID );  | 
  |  130 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.PRINCIPAL_NAME );  | 
  |  131 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.ENTITY_ID );  | 
  |  132 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.FIRST_NAME );  | 
  |  133 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.LAST_NAME );  | 
  |  134 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.MIDDLE_NAME );  | 
  |  135 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.CAMPUS_CODE );  | 
  |  136 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.EMPLOYEE_ID );  | 
  |  137 |    0 |                    personCachePropertyNames.add( KIMPropertyConstants.Person.PRIMARY_DEPARTMENT_CODE );  | 
  |  138 |    0 |            }  | 
  |  139 |     | 
     | 
  |  140 |     | 
             | 
  |  141 |     | 
             | 
  |  142 |     | 
     | 
  |  143 |     | 
     | 
  |  144 |     | 
           public Person getPerson(String principalId) { | 
  |  145 |    0 |                    if ( StringUtils.isBlank(principalId) ) { | 
  |  146 |    0 |                            return null;  | 
  |  147 |     | 
                   }  | 
  |  148 |     | 
                     | 
  |  149 |    0 |                    Person person = getPersonImplFromPrincipalIdCache( principalId );  | 
  |  150 |    0 |                    if ( person != null ) { | 
  |  151 |    0 |                            return person;  | 
  |  152 |     | 
                   }  | 
  |  153 |    0 |                    KimEntityDefaultInfo entity = null;  | 
  |  154 |     | 
                     | 
  |  155 |    0 |                    KimPrincipal principal = getIdentityManagementService().getPrincipal( principalId );  | 
  |  156 |     | 
                     | 
  |  157 |    0 |                    if ( principal != null ) { | 
  |  158 |    0 |                            entity = getIdentityManagementService().getEntityDefaultInfo( principal.getEntityId() );  | 
  |  159 |     | 
                   }  | 
  |  160 |     | 
                     | 
  |  161 |     | 
                     | 
  |  162 |    0 |                    if (entity != null ) { | 
  |  163 |    0 |                            person = convertEntityToPerson( entity, principal );  | 
  |  164 |    0 |                            addPersonToCache( person );  | 
  |  165 |     | 
                   }  | 
  |  166 |    0 |                    return person;  | 
  |  167 |     | 
           }  | 
  |  168 |     | 
     | 
  |  169 |     | 
           protected PersonImpl convertEntityToPerson( KimEntityDefaultInfo entity, KimPrincipal principal ) { | 
  |  170 |     | 
                   try { | 
  |  171 |     | 
                             | 
  |  172 |    0 |                            for ( String entityTypeCode : personEntityTypeCodes ) { | 
  |  173 |    0 |                                    KimEntityEntityTypeDefaultInfo entType = entity.getEntityType( entityTypeCode );  | 
  |  174 |     | 
                                     | 
  |  175 |    0 |                                    if ( entType == null ) { | 
  |  176 |    0 |                                            continue;  | 
  |  177 |     | 
                                   }  | 
  |  178 |     | 
                                     | 
  |  179 |     | 
                                     | 
  |  180 |    0 |                                    return new PersonImpl( principal, entity, entityTypeCode );  | 
  |  181 |     | 
                           }  | 
  |  182 |    0 |                            return null;  | 
  |  183 |    0 |                    } catch ( Exception ex ) { | 
  |  184 |     | 
                             | 
  |  185 |    0 |                            if ( ex instanceof RuntimeException ) { | 
  |  186 |    0 |                                    throw (RuntimeException)ex;  | 
  |  187 |     | 
                           }  | 
  |  188 |    0 |                            throw new RuntimeException( "Problem building person object", ex );  | 
  |  189 |     | 
                   }  | 
  |  190 |     | 
           }  | 
  |  191 |     | 
             | 
  |  192 |     | 
           protected Person getPersonImplFromPrincipalNameCache( String principalName ) { | 
  |  193 |    0 |                    SoftReference<Person> personRef = personCache.get( "principalName="+principalName );  | 
  |  194 |    0 |                    if ( personRef != null ) { | 
  |  195 |    0 |                            return personRef.get();  | 
  |  196 |     | 
                   }  | 
  |  197 |    0 |                    return null;  | 
  |  198 |     | 
           }  | 
  |  199 |     | 
     | 
  |  200 |     | 
           protected Person getPersonImplFromPrincipalIdCache( String principalId ) { | 
  |  201 |    0 |                    SoftReference<Person> personRef = personCache.get( "principalId="+principalId );  | 
  |  202 |    0 |                    if ( personRef != null ) { | 
  |  203 |    0 |                            return personRef.get();  | 
  |  204 |     | 
                   }  | 
  |  205 |    0 |                    return null;  | 
  |  206 |     | 
           }  | 
  |  207 |     | 
             | 
  |  208 |     | 
           protected Person getPersonImplFromEmployeeIdCache( String principalId ) { | 
  |  209 |    0 |                    SoftReference<Person> personRef = personCache.get( "employeeId="+principalId );  | 
  |  210 |    0 |                    if ( personRef != null ) { | 
  |  211 |    0 |                            return personRef.get();  | 
  |  212 |     | 
                   }  | 
  |  213 |    0 |                    return null;  | 
  |  214 |     | 
           }  | 
  |  215 |     | 
             | 
  |  216 |     | 
           protected void addPersonToCache( Person person ) { | 
  |  217 |    0 |                    if ( person != null ) { | 
  |  218 |    0 |                            synchronized (personCache) { | 
  |  219 |    0 |                                    personCache.put( "principalName="+person.getPrincipalName(), new MaxAgeSoftReference<Person>( personCacheMaxAgeSeconds, person ) );  | 
  |  220 |    0 |                                    personCache.put( "principalId="+person.getPrincipalId(), new MaxAgeSoftReference<Person>( personCacheMaxAgeSeconds, person ) );  | 
  |  221 |    0 |                                    personCache.put( "employeeId="+person.getEmployeeId(), new MaxAgeSoftReference<Person>( personCacheMaxAgeSeconds, person ) );  | 
  |  222 |    0 |                            }  | 
  |  223 |     | 
                   }  | 
  |  224 |    0 |            }  | 
  |  225 |     | 
             | 
  |  226 |     | 
           public void flushPersonCaches() { | 
  |  227 |    0 |                personCache.clear();  | 
  |  228 |    0 |            }  | 
  |  229 |     | 
             | 
  |  230 |     | 
             | 
  |  231 |     | 
             | 
  |  232 |     | 
     | 
  |  233 |     | 
     | 
  |  234 |     | 
           public Person getPersonByPrincipalName(String principalName) { | 
  |  235 |    0 |                    if ( StringUtils.isBlank(principalName) ) { | 
  |  236 |    0 |                            return null;  | 
  |  237 |     | 
                   }  | 
  |  238 |    0 |                    Person person = null;  | 
  |  239 |     | 
                     | 
  |  240 |    0 |                    person = getPersonImplFromPrincipalNameCache( principalName );  | 
  |  241 |    0 |                    if ( person != null ) { | 
  |  242 |    0 |                            return person;  | 
  |  243 |     | 
                   }  | 
  |  244 |    0 |                    KimEntityDefaultInfo entity = null;  | 
  |  245 |     | 
                     | 
  |  246 |    0 |                    KimPrincipal principal = getIdentityManagementService().getPrincipalByPrincipalName( principalName );  | 
  |  247 |     | 
                     | 
  |  248 |    0 |                    if ( principal != null ) { | 
  |  249 |    0 |                            entity = getIdentityManagementService().getEntityDefaultInfo( principal.getEntityId() );  | 
  |  250 |     | 
                   }  | 
  |  251 |     | 
                     | 
  |  252 |    0 |                    if ( entity != null ) { | 
  |  253 |    0 |                            person = convertEntityToPerson( entity, principal );  | 
  |  254 |     | 
                   }  | 
  |  255 |    0 |                    addPersonToCache( person );  | 
  |  256 |    0 |                    return person;  | 
  |  257 |     | 
           }  | 
  |  258 |     | 
     | 
  |  259 |     | 
           public Person getPersonByEmployeeId(String employeeId) { | 
  |  260 |    0 |                    if ( StringUtils.isBlank( employeeId  ) ) { | 
  |  261 |    0 |                            return null;  | 
  |  262 |     | 
                   }  | 
  |  263 |     | 
                     | 
  |  264 |    0 |                    Person person = getPersonImplFromEmployeeIdCache( employeeId );  | 
  |  265 |    0 |                    if ( person != null ) { | 
  |  266 |    0 |                            return person;  | 
  |  267 |     | 
                   }  | 
  |  268 |     | 
                     | 
  |  269 |    0 |                    Map<String,String> criteria = new HashMap<String,String>( 1 );  | 
  |  270 |    0 |                    criteria.put( KIMPropertyConstants.Person.EMPLOYEE_ID, employeeId );  | 
  |  271 |    0 |                    List<Person> people = findPeople( criteria );   | 
  |  272 |    0 |                    if ( !people.isEmpty() ) { | 
  |  273 |    0 |                            person = people.get(0);  | 
  |  274 |    0 |                            addPersonToCache( person );  | 
  |  275 |     | 
                   }  | 
  |  276 |    0 |                    return person;  | 
  |  277 |     | 
           }  | 
  |  278 |     | 
             | 
  |  279 |     | 
             | 
  |  280 |     | 
     | 
  |  281 |     | 
     | 
  |  282 |     | 
           public List<Person> findPeople(Map<String, String> criteria) { | 
  |  283 |    0 |                    return findPeople(criteria, true);  | 
  |  284 |     | 
           }  | 
  |  285 |     | 
             | 
  |  286 |     | 
             | 
  |  287 |     | 
     | 
  |  288 |     | 
     | 
  |  289 |     | 
           public List<Person> findPeople(Map<String, String> criteria, boolean unbounded) { | 
  |  290 |    0 |                    List<Person> people = null;  | 
  |  291 |     | 
                     | 
  |  292 |    0 |                    if ( criteria == null ) { | 
  |  293 |    0 |                            criteria = Collections.emptyMap();  | 
  |  294 |     | 
                   }  | 
  |  295 |     | 
                     | 
  |  296 |    0 |                    criteria = new HashMap<String, String>( criteria );  | 
  |  297 |     | 
                     | 
  |  298 |     | 
                     | 
  |  299 |    0 |                    String roleName = criteria.get( "lookupRoleName" );  | 
  |  300 |    0 |                    String namespaceCode = criteria.get( "lookupRoleNamespaceCode" );  | 
  |  301 |    0 |                    criteria.remove("lookupRoleName"); | 
  |  302 |    0 |                    criteria.remove("lookupRoleNamespaceCode"); | 
  |  303 |    0 |                    if ( StringUtils.isNotBlank(namespaceCode) && StringUtils.isNotBlank(roleName) ) { | 
  |  304 |    0 |                            Integer searchResultsLimit = LookupUtils.getSearchResultsLimit(PersonImpl.class);  | 
  |  305 |    0 |                            int searchResultsLimitInt = Integer.MAX_VALUE;  | 
  |  306 |    0 |                            if (searchResultsLimit != null) { | 
  |  307 |    0 |                                    searchResultsLimitInt = searchResultsLimit.intValue();  | 
  |  308 |     | 
                           }  | 
  |  309 |    0 |                            if ( LOG.isDebugEnabled() ) { | 
  |  310 |    0 |                                    LOG.debug("Performing Person search including role filter: " + namespaceCode + "/" + roleName ); | 
  |  311 |     | 
                           }  | 
  |  312 |    0 |                            if ( criteria.size() == 1 && criteria.containsKey(KIMPropertyConstants.Person.ACTIVE) ) {  | 
  |  313 |    0 |                                    if ( LOG.isDebugEnabled() ) { | 
  |  314 |    0 |                                            LOG.debug( "Only active criteria specified, running role search first" );  | 
  |  315 |     | 
                                   }  | 
  |  316 |     | 
                                     | 
  |  317 |    0 |                                    Collection<String> principalIds = getRoleManagementService().getRoleMemberPrincipalIds(namespaceCode, roleName, null);  | 
  |  318 |    0 |                                    StringBuffer sb = new StringBuffer(principalIds.size()*15);  | 
  |  319 |    0 |                                    Iterator<String> pi = principalIds.iterator();  | 
  |  320 |    0 |                                    while ( pi.hasNext() ) { | 
  |  321 |    0 |                                            sb.append( pi.next() );  | 
  |  322 |    0 |                                            if ( pi.hasNext() ) sb.append( '|' );  | 
  |  323 |     | 
                                   }  | 
  |  324 |     | 
                                     | 
  |  325 |    0 |                                    criteria.put( KIMPropertyConstants.Person.PRINCIPAL_ID, sb.toString() );  | 
  |  326 |    0 |                                    people = findPeopleInternal(criteria, false);   | 
  |  327 |    0 |                            } else if ( !criteria.isEmpty() ) {  | 
  |  328 |    0 |                                    if ( LOG.isDebugEnabled() ) { | 
  |  329 |    0 |                                            LOG.debug( "Person criteria also specified, running that search first" );  | 
  |  330 |     | 
                                   }  | 
  |  331 |     | 
                                     | 
  |  332 |    0 |                                    people = findPeopleInternal(criteria, true);   | 
  |  333 |     | 
                                     | 
  |  334 |     | 
                                     | 
  |  335 |    0 |                                    List<String> principalIds = peopleToPrincipalIds( people );  | 
  |  336 |     | 
                                     | 
  |  337 |    0 |                                    principalIds = getRoleManagementService().getPrincipalIdSubListWithRole(principalIds, namespaceCode, roleName, null);  | 
  |  338 |     | 
                                     | 
  |  339 |    0 |                                    if ( !unbounded && principalIds.size() > searchResultsLimitInt ) { | 
  |  340 |    0 |                                            int actualResultSize = principalIds.size();  | 
  |  341 |     | 
                                             | 
  |  342 |    0 |                                            principalIds = new ArrayList<String>(principalIds).subList(0, searchResultsLimitInt);   | 
  |  343 |    0 |                                            people = getPeople(principalIds);   | 
  |  344 |    0 |                                            people = new CollectionIncomplete<Person>( people.subList(0, searchResultsLimitInt), new Long(actualResultSize) );  | 
  |  345 |    0 |                                    } else { | 
  |  346 |    0 |                                            people = getPeople(principalIds);  | 
  |  347 |     | 
                                   }  | 
  |  348 |    0 |                            } else {  | 
  |  349 |    0 |                                    if ( LOG.isDebugEnabled() ) { | 
  |  350 |    0 |                                            LOG.debug( "No Person criteria specified - only using role service." );  | 
  |  351 |     | 
                                   }  | 
  |  352 |     | 
                                     | 
  |  353 |    0 |                                    Collection<String> principalIds = getRoleManagementService().getRoleMemberPrincipalIds(namespaceCode, roleName, null);  | 
  |  354 |    0 |                                    if ( !unbounded && principalIds.size() > searchResultsLimitInt ) { | 
  |  355 |    0 |                                            int actualResultSize = principalIds.size();  | 
  |  356 |     | 
                                             | 
  |  357 |    0 |                                            principalIds = new ArrayList<String>(principalIds).subList(0, searchResultsLimitInt);   | 
  |  358 |    0 |                                            people = getPeople(principalIds);   | 
  |  359 |    0 |                                            people = new CollectionIncomplete<Person>( people.subList(0, searchResultsLimitInt), new Long(actualResultSize) );  | 
  |  360 |    0 |                                    } else { | 
  |  361 |    0 |                                            people = getPeople(principalIds);   | 
  |  362 |     | 
                                   }  | 
  |  363 |     | 
                           }  | 
  |  364 |    0 |                    } else { | 
  |  365 |    0 |                            if ( LOG.isDebugEnabled() ) { | 
  |  366 |    0 |                                    LOG.debug( "No Role criteria specified, running person lookup as normal." );  | 
  |  367 |     | 
                           }  | 
  |  368 |    0 |                            people = findPeopleInternal(criteria, unbounded);  | 
  |  369 |     | 
                   }  | 
  |  370 |    0 |                    return people;  | 
  |  371 |     | 
           }  | 
  |  372 |     | 
             | 
  |  373 |     | 
           @SuppressWarnings("unchecked") | 
  |  374 |     | 
           protected List<Person> findPeopleInternal(Map<String,String> criteria, boolean unbounded ) { | 
  |  375 |     | 
                     | 
  |  376 |    0 |                    Map<String,String> entityCriteria = convertPersonPropertiesToEntityProperties( criteria );  | 
  |  377 |     | 
     | 
  |  378 |    0 |                    List<Person> people = new ArrayList<Person>();   | 
  |  379 |     | 
     | 
  |  380 |    0 |                    List<? extends KimEntityDefaultInfo> entities = getIdentityManagementService().lookupEntityDefaultInfo( entityCriteria, unbounded );  | 
  |  381 |     | 
     | 
  |  382 |    0 |                    for ( KimEntityDefaultInfo e : entities ) { | 
  |  383 |     | 
                             | 
  |  384 |    0 |                            for ( KimPrincipal p : e.getPrincipals() ) { | 
  |  385 |    0 |                                    people.add( convertEntityToPerson( e, p ) );  | 
  |  386 |     | 
                           }  | 
  |  387 |     | 
                   }  | 
  |  388 |     | 
                     | 
  |  389 |    0 |                    if ( entities instanceof CollectionIncomplete ) { | 
  |  390 |    0 |                            return new CollectionIncomplete( people, ((CollectionIncomplete)entities).getActualSizeIfTruncated() );  | 
  |  391 |     | 
                   }  | 
  |  392 |    0 |                    return people;  | 
  |  393 |     | 
           }  | 
  |  394 |     | 
     | 
  |  395 |     | 
           public Map<String,String> convertPersonPropertiesToEntityProperties( Map<String,String> criteria ) { | 
  |  396 |    0 |                    if ( LOG.isDebugEnabled() ) { | 
  |  397 |    0 |                            LOG.debug( "convertPersonPropertiesToEntityProperties: " + criteria );  | 
  |  398 |     | 
                   }  | 
  |  399 |    0 |                    boolean nameCriteria = false;  | 
  |  400 |    0 |                    boolean addressCriteria = false;  | 
  |  401 |    0 |                    boolean externalIdentifierCriteria = false;  | 
  |  402 |    0 |                    boolean affiliationCriteria = false;  | 
  |  403 |    0 |                    boolean affiliationDefaultOnlyCriteria = false;  | 
  |  404 |    0 |                    boolean phoneCriteria = false;  | 
  |  405 |    0 |                    boolean emailCriteria = false;  | 
  |  406 |    0 |                    boolean employeeIdCriteria = false;  | 
  |  407 |     | 
                     | 
  |  408 |    0 |                    HashMap<String,String> newCriteria = new HashMap<String,String>();  | 
  |  409 |    0 |                    newCriteria.putAll( baseLookupCriteria );  | 
  |  410 |    0 |                    newCriteria.put( "entityTypes.entityTypeCode", personEntityTypeLookupCriteria );  | 
  |  411 |    0 |                    if ( criteria != null ) { | 
  |  412 |    0 |                            for ( String key : criteria.keySet() ) { | 
  |  413 |     | 
                                                     | 
  |  414 |     | 
                                     | 
  |  415 |    0 |                                    if(key.equals(KIMPropertyConstants.Person.ACTIVE)) { | 
  |  416 |    0 |                                            newCriteria.put(KIMPropertyConstants.Person.ACTIVE, criteria.get(KIMPropertyConstants.Person.ACTIVE));  | 
  |  417 |     | 
                                   }  | 
  |  418 |     | 
                             | 
  |  419 |     | 
                                     | 
  |  420 |    0 |                                    if ( StringUtils.isEmpty( criteria.get(key) ) ) { | 
  |  421 |    0 |                                            continue;  | 
  |  422 |     | 
                                   }  | 
  |  423 |     | 
                                     | 
  |  424 |     | 
                                     | 
  |  425 |    0 |                                    if ( key.equals( KIMPropertyConstants.Person.EXTERNAL_ID ) && StringUtils.isNotBlank(criteria.get(key)) ) { | 
  |  426 |     | 
                                             | 
  |  427 |    0 |                                            if ( criteria.containsKey( KIMPropertyConstants.Person.EXTERNAL_IDENTIFIER_TYPE_CODE ) ) { | 
  |  428 |    0 |                                                    String extIdTypeCode = criteria.get(KIMPropertyConstants.Person.EXTERNAL_IDENTIFIER_TYPE_CODE);  | 
  |  429 |    0 |                                                    if ( StringUtils.isNotBlank(extIdTypeCode) ) { | 
  |  430 |     | 
                                                             | 
  |  431 |    0 |                                                            ExternalIdentifierTypeInfo extIdType = getIdentityManagementService().getExternalIdentifierType(extIdTypeCode);  | 
  |  432 |     | 
                                                             | 
  |  433 |    0 |                                                            if ( extIdType != null && extIdType.isEncryptionRequired() ) { | 
  |  434 |     | 
                                                                   try { | 
  |  435 |    0 |                                                                            criteria.put(key,   | 
  |  436 |     | 
                                                                                           CoreApiServiceLocator.getEncryptionService().encrypt(criteria.get(key))  | 
  |  437 |     | 
                                                                                           );  | 
  |  438 |    0 |                                                                    } catch (GeneralSecurityException ex) { | 
  |  439 |    0 |                                                                            LOG.error("Unable to encrypt value for external ID search of type " + extIdTypeCode, ex ); | 
  |  440 |    0 |                                                                    }                                                                  | 
  |  441 |     | 
                                                           }  | 
  |  442 |     | 
                                                   }  | 
  |  443 |     | 
                                           }  | 
  |  444 |     | 
                                   }  | 
  |  445 |     | 
                                     | 
  |  446 |     | 
                                     | 
  |  447 |    0 |                                    String entityProperty = criteriaConversion.get( key );  | 
  |  448 |    0 |                                    if ( entityProperty != null ) { | 
  |  449 |    0 |                                            newCriteria.put( entityProperty, criteria.get( key ) );  | 
  |  450 |     | 
                                   } else { | 
  |  451 |    0 |                                            entityProperty = key;  | 
  |  452 |     | 
                                             | 
  |  453 |    0 |                                            newCriteria.put( key, criteria.get( key ) );  | 
  |  454 |     | 
                                   }  | 
  |  455 |     | 
                                     | 
  |  456 |    0 |                                    if ( isNameEntityCriteria( entityProperty ) ) { | 
  |  457 |    0 |                                            nameCriteria = true;  | 
  |  458 |     | 
                                   }  | 
  |  459 |    0 |                                    if ( isExternalIdentifierEntityCriteria( entityProperty ) ) { | 
  |  460 |    0 |                                            externalIdentifierCriteria = true;  | 
  |  461 |     | 
                                   }  | 
  |  462 |    0 |                                    if ( isAffiliationEntityCriteria( entityProperty ) ) { | 
  |  463 |    0 |                                            affiliationCriteria = true;  | 
  |  464 |     | 
                                   }  | 
  |  465 |    0 |                                    if ( isAddressEntityCriteria( entityProperty ) ) { | 
  |  466 |    0 |                                            addressCriteria = true;  | 
  |  467 |     | 
                                   }  | 
  |  468 |    0 |                                    if ( isPhoneEntityCriteria( entityProperty ) ) { | 
  |  469 |    0 |                                            phoneCriteria = true;  | 
  |  470 |     | 
                                   }  | 
  |  471 |    0 |                                    if ( isEmailEntityCriteria( entityProperty ) ) { | 
  |  472 |    0 |                                            emailCriteria = true;  | 
  |  473 |     | 
                                   }  | 
  |  474 |    0 |                                    if ( isEmployeeIdEntityCriteria( entityProperty ) ) { | 
  |  475 |    0 |                                            employeeIdCriteria = true;  | 
  |  476 |     | 
                                   }                                  | 
  |  477 |     | 
                                     | 
  |  478 |     | 
                                     | 
  |  479 |    0 |                                    if ( key.equals( "campusCode" ) ) { | 
  |  480 |    0 |                                            affiliationDefaultOnlyCriteria = true;  | 
  |  481 |     | 
                                   }  | 
  |  482 |    0 |                            }                  | 
  |  483 |    0 |                            if ( nameCriteria ) { | 
  |  484 |    0 |                                    newCriteria.put( ENTITY_NAME_PROPERTY_PREFIX + "active", "Y" );  | 
  |  485 |    0 |                                    newCriteria.put( ENTITY_NAME_PROPERTY_PREFIX + "defaultValue", "Y" );  | 
  |  486 |     | 
                                     | 
  |  487 |     | 
                           }  | 
  |  488 |    0 |                            if ( addressCriteria ) { | 
  |  489 |    0 |                                    newCriteria.put( ENTITY_ADDRESS_PROPERTY_PREFIX + "active", "Y" );  | 
  |  490 |    0 |                                    newCriteria.put( ENTITY_ADDRESS_PROPERTY_PREFIX + "defaultValue", "Y" );  | 
  |  491 |     | 
                           }  | 
  |  492 |    0 |                            if ( phoneCriteria ) { | 
  |  493 |    0 |                                    newCriteria.put( ENTITY_PHONE_PROPERTY_PREFIX + "active", "Y" );  | 
  |  494 |    0 |                                    newCriteria.put( ENTITY_PHONE_PROPERTY_PREFIX + "defaultValue", "Y" );  | 
  |  495 |     | 
                           }  | 
  |  496 |    0 |                            if ( emailCriteria ) { | 
  |  497 |    0 |                                    newCriteria.put( ENTITY_EMAIL_PROPERTY_PREFIX + "active", "Y" );  | 
  |  498 |    0 |                                    newCriteria.put( ENTITY_EMAIL_PROPERTY_PREFIX + "defaultValue", "Y" );  | 
  |  499 |     | 
                           }  | 
  |  500 |    0 |                            if ( employeeIdCriteria ) { | 
  |  501 |    0 |                                    newCriteria.put( ENTITY_EMPLOYEE_ID_PROPERTY_PREFIX + "active", "Y" );  | 
  |  502 |    0 |                                    newCriteria.put( ENTITY_EMPLOYEE_ID_PROPERTY_PREFIX + "primary", "Y" );  | 
  |  503 |     | 
                           }  | 
  |  504 |    0 |                            if ( affiliationCriteria ) { | 
  |  505 |    0 |                                    newCriteria.put( ENTITY_AFFILIATION_PROPERTY_PREFIX + "active", "Y" );  | 
  |  506 |     | 
                           }  | 
  |  507 |    0 |                            if ( affiliationDefaultOnlyCriteria ) { | 
  |  508 |    0 |                                    newCriteria.put( ENTITY_AFFILIATION_PROPERTY_PREFIX + "defaultValue", "Y" );  | 
  |  509 |     | 
                           }  | 
  |  510 |    0 |                            if ( externalIdentifierCriteria ) { | 
  |  511 |    0 |                                    newCriteria.put( ENTITY_EXT_ID_PROPERTY_PREFIX + "active", "Y" );  | 
  |  512 |     | 
                           }  | 
  |  513 |     | 
                   }  | 
  |  514 |     | 
                     | 
  |  515 |    0 |                    if ( LOG.isDebugEnabled() ) { | 
  |  516 |    0 |                            LOG.debug( "Converted: " + newCriteria );  | 
  |  517 |     | 
                   }  | 
  |  518 |    0 |                    return newCriteria;                  | 
  |  519 |     | 
           }  | 
  |  520 |     | 
     | 
  |  521 |     | 
           protected boolean isNameEntityCriteria( String propertyName ) { | 
  |  522 |    0 |                    return propertyName.startsWith( ENTITY_NAME_PROPERTY_PREFIX );  | 
  |  523 |     | 
           }  | 
  |  524 |     | 
           protected boolean isAddressEntityCriteria( String propertyName ) { | 
  |  525 |    0 |                    return propertyName.startsWith( ENTITY_ADDRESS_PROPERTY_PREFIX );  | 
  |  526 |     | 
           }  | 
  |  527 |     | 
           protected boolean isPhoneEntityCriteria( String propertyName ) { | 
  |  528 |    0 |                    return propertyName.startsWith( ENTITY_PHONE_PROPERTY_PREFIX );  | 
  |  529 |     | 
           }  | 
  |  530 |     | 
           protected boolean isEmailEntityCriteria( String propertyName ) { | 
  |  531 |    0 |                    return propertyName.startsWith( ENTITY_EMAIL_PROPERTY_PREFIX );  | 
  |  532 |     | 
           }  | 
  |  533 |     | 
           protected boolean isEmployeeIdEntityCriteria( String propertyName ) { | 
  |  534 |    0 |                    return propertyName.startsWith( ENTITY_EMPLOYEE_ID_PROPERTY_PREFIX );  | 
  |  535 |     | 
           }  | 
  |  536 |     | 
           protected boolean isAffiliationEntityCriteria( String propertyName ) { | 
  |  537 |    0 |                    return propertyName.startsWith( ENTITY_AFFILIATION_PROPERTY_PREFIX );  | 
  |  538 |     | 
           }  | 
  |  539 |     | 
           protected boolean isExternalIdentifierEntityCriteria( String propertyName ) { | 
  |  540 |    0 |                    return propertyName.startsWith( ENTITY_EXT_ID_PROPERTY_PREFIX );  | 
  |  541 |     | 
           }  | 
  |  542 |     | 
             | 
  |  543 |     | 
             | 
  |  544 |     | 
     | 
  |  545 |     | 
     | 
  |  546 |     | 
     | 
  |  547 |     | 
     | 
  |  548 |     | 
           public List<String> getPersonEntityTypeCodes() { | 
  |  549 |    0 |                    return this.personEntityTypeCodes;  | 
  |  550 |     | 
           }  | 
  |  551 |     | 
     | 
  |  552 |     | 
           public void setPersonEntityTypeCodes(List<String> personEntityTypeCodes) { | 
  |  553 |    0 |                    this.personEntityTypeCodes = personEntityTypeCodes;  | 
  |  554 |    0 |                    personEntityTypeLookupCriteria = null;  | 
  |  555 |    0 |                    for ( String entityTypeCode : personEntityTypeCodes ) { | 
  |  556 |    0 |                            if ( personEntityTypeLookupCriteria == null ) { | 
  |  557 |    0 |                                    personEntityTypeLookupCriteria = entityTypeCode;  | 
  |  558 |     | 
                           } else { | 
  |  559 |    0 |                                    personEntityTypeLookupCriteria = personEntityTypeLookupCriteria + "|" + entityTypeCode;  | 
  |  560 |     | 
                           }  | 
  |  561 |     | 
                   }  | 
  |  562 |    0 |            }  | 
  |  563 |     | 
     | 
  |  564 |     | 
             | 
  |  565 |     | 
           protected List<Person> getPeople( Collection<String> principalIds ) { | 
  |  566 |    0 |                    List<Person> people = new ArrayList<Person>( principalIds.size() );  | 
  |  567 |    0 |                    for ( String principalId : principalIds ) { | 
  |  568 |    0 |                            people.add( getPerson(principalId) );  | 
  |  569 |     | 
                   }  | 
  |  570 |    0 |                    return people;  | 
  |  571 |     | 
           }  | 
  |  572 |     | 
             | 
  |  573 |     | 
           protected List<String> peopleToPrincipalIds( List<Person> people ) { | 
  |  574 |    0 |                    List<String> principalIds = new ArrayList<String>();  | 
  |  575 |     | 
                     | 
  |  576 |    0 |                    for ( Person person : people ) { | 
  |  577 |    0 |                            principalIds.add( person.getPrincipalId() );  | 
  |  578 |     | 
                   }  | 
  |  579 |     | 
                     | 
  |  580 |    0 |                    return principalIds;  | 
  |  581 |     | 
           }  | 
  |  582 |     | 
             | 
  |  583 |     | 
             | 
  |  584 |     | 
     | 
  |  585 |     | 
     | 
  |  586 |     | 
           public List<Person> getPersonByExternalIdentifier(String externalIdentifierTypeCode, String externalId) { | 
  |  587 |    0 |                    if (StringUtils.isBlank( externalIdentifierTypeCode ) || StringUtils.isBlank( externalId ) ) { | 
  |  588 |    0 |                            return null;  | 
  |  589 |     | 
                   }  | 
  |  590 |    0 |                    Map<String,String> criteria = new HashMap<String,String>( 2 );  | 
  |  591 |    0 |                    criteria.put( KIMPropertyConstants.Person.EXTERNAL_IDENTIFIER_TYPE_CODE, externalIdentifierTypeCode );  | 
  |  592 |    0 |                    criteria.put( KIMPropertyConstants.Person.EXTERNAL_ID, externalId );  | 
  |  593 |    0 |                    return findPeople( criteria );  | 
  |  594 |     | 
           }  | 
  |  595 |     | 
             | 
  |  596 |     | 
             | 
  |  597 |     | 
     | 
  |  598 |     | 
     | 
  |  599 |     | 
       public Person updatePersonIfNecessary(String sourcePrincipalId, Person currentPerson ) { | 
  |  600 |    0 |            if (currentPerson  == null   | 
  |  601 |     | 
                   || !StringUtils.equals(sourcePrincipalId, currentPerson.getPrincipalId() )   | 
  |  602 |     | 
                   || currentPerson.getEntityId() == null ) {  | 
  |  603 |    0 |                Person person = getPerson( sourcePrincipalId );  | 
  |  604 |     | 
                 | 
  |  605 |     | 
                 | 
  |  606 |    0 |                if ( person == null ) { | 
  |  607 |    0 |                    if ( currentPerson != null && currentPerson.getEntityId() == null ) { | 
  |  608 |    0 |                        return currentPerson;  | 
  |  609 |     | 
                   }  | 
  |  610 |     | 
               }  | 
  |  611 |     | 
                 | 
  |  612 |    0 |                if ( person == null && currentPerson == null ) { | 
  |  613 |     | 
                       try { | 
  |  614 |    0 |                                return new PersonImpl();  | 
  |  615 |    0 |                        } catch ( Exception ex ) { | 
  |  616 |    0 |                                LOG.error( "unable to instantiate an object of type: " + getPersonImplementationClass() + " - returning null", ex );  | 
  |  617 |    0 |                                return null;  | 
  |  618 |     | 
                       }  | 
  |  619 |     | 
               }  | 
  |  620 |    0 |                return person;  | 
  |  621 |     | 
           }  | 
  |  622 |     | 
             | 
  |  623 |    0 |            return currentPerson;  | 
  |  624 |     | 
       }  | 
  |  625 |     | 
     | 
  |  626 |     | 
         | 
  |  627 |     | 
     | 
  |  628 |     | 
     | 
  |  629 |     | 
     | 
  |  630 |     | 
       private Map<String,String> getNonPersonSearchCriteria( BusinessObject bo, Map<String,String> fieldValues) { | 
  |  631 |    0 |            Map<String,String> nonUniversalUserSearchCriteria = new HashMap<String,String>();  | 
  |  632 |    0 |            for ( String propertyName : fieldValues.keySet() ) { | 
  |  633 |    0 |                if (!isPersonProperty(bo, propertyName)) { | 
  |  634 |    0 |                    nonUniversalUserSearchCriteria.put(propertyName, fieldValues.get(propertyName));  | 
  |  635 |     | 
               }  | 
  |  636 |     | 
           }  | 
  |  637 |    0 |            return nonUniversalUserSearchCriteria;  | 
  |  638 |     | 
       }  | 
  |  639 |     | 
     | 
  |  640 |     | 
     | 
  |  641 |     | 
       private boolean isPersonProperty(BusinessObject bo, String propertyName) { | 
  |  642 |     | 
           try { | 
  |  643 |    0 |                    if ( ObjectUtils.isNestedAttribute( propertyName )   | 
  |  644 |     | 
                               && !StringUtils.contains(propertyName, "add.") ) { | 
  |  645 |    0 |                            Class<?> type = PropertyUtils.getPropertyType(bo, ObjectUtils.getNestedAttributePrefix( propertyName ));  | 
  |  646 |     | 
                             | 
  |  647 |    0 |                            if ( type != null ) { | 
  |  648 |    0 |                                    return Person.class.isAssignableFrom(type);  | 
  |  649 |     | 
                           }  | 
  |  650 |    0 |                            LOG.warn( "Unable to determine type of nested property: " + bo.getClass().getName() + " / " + propertyName );  | 
  |  651 |     | 
                   }  | 
  |  652 |    0 |            } catch (Exception ex) { | 
  |  653 |    0 |                    if ( LOG.isDebugEnabled() ) { | 
  |  654 |    0 |                            LOG.debug("Unable to determine if property on " + bo.getClass().getName() + " to a person object: " + propertyName, ex ); | 
  |  655 |     | 
                   }  | 
  |  656 |    0 |            }  | 
  |  657 |    0 |            return false;  | 
  |  658 |     | 
       }  | 
  |  659 |     | 
         | 
  |  660 |     | 
         | 
  |  661 |     | 
     | 
  |  662 |     | 
     | 
  |  663 |     | 
       public boolean hasPersonProperty(Class<? extends BusinessObject> businessObjectClass, Map<String,String> fieldValues) { | 
  |  664 |    0 |                if ( businessObjectClass == null || fieldValues == null ) { | 
  |  665 |    0 |                        return false;  | 
  |  666 |     | 
               }  | 
  |  667 |     | 
               try { | 
  |  668 |    0 |                        BusinessObject bo = businessObjectClass.newInstance();  | 
  |  669 |    0 |                    for ( String propertyName : fieldValues.keySet() ) { | 
  |  670 |    0 |                        if (isPersonProperty(bo, propertyName)) { | 
  |  671 |    0 |                                return true;  | 
  |  672 |     | 
                       }  | 
  |  673 |     | 
                   }  | 
  |  674 |    0 |                } catch (Exception ex) { | 
  |  675 |    0 |                        if ( LOG.isDebugEnabled() ) { | 
  |  676 |    0 |                                LOG.debug( "Error instantiating business object class passed into hasPersonProperty", ex );  | 
  |  677 |     | 
                       }  | 
  |  678 |     | 
                             | 
  |  679 |    0 |                    }  | 
  |  680 |    0 |            return false;  | 
  |  681 |     | 
       }      | 
  |  682 |     | 
     | 
  |  683 |     | 
         | 
  |  684 |     | 
     | 
  |  685 |     | 
     | 
  |  686 |     | 
       @SuppressWarnings("unchecked") | 
  |  687 |     | 
           public Map<String,String> resolvePrincipalNamesToPrincipalIds(BusinessObject businessObject, Map<String,String> fieldValues) { | 
  |  688 |    0 |                if ( fieldValues == null ) { | 
  |  689 |    0 |                        return null;  | 
  |  690 |     | 
               }  | 
  |  691 |    0 |                if ( businessObject == null ) { | 
  |  692 |    0 |                        return fieldValues;  | 
  |  693 |     | 
               }  | 
  |  694 |    0 |                StringBuffer resolvedPrincipalIdPropertyName = new StringBuffer();  | 
  |  695 |     | 
                 | 
  |  696 |     | 
                 | 
  |  697 |    0 |            Map<String,String> processedFieldValues = getNonPersonSearchCriteria(businessObject, fieldValues);  | 
  |  698 |    0 |            for ( String propertyName : fieldValues.keySet() ) {                 | 
  |  699 |    0 |                if (        !StringUtils.isBlank(fieldValues.get(propertyName))    | 
  |  700 |     | 
                               && isPersonProperty(businessObject, propertyName)   | 
  |  701 |     | 
                               ) { | 
  |  702 |     | 
                         | 
  |  703 |    0 |                    String personPropertyName = ObjectUtils.getNestedAttributePrimitive( propertyName );  | 
  |  704 |     | 
                     | 
  |  705 |    0 |                    if ( StringUtils.equals( KIMPropertyConstants.Person.PRINCIPAL_NAME, personPropertyName) ) { | 
  |  706 |    0 |                        Class targetBusinessObjectClass = null;  | 
  |  707 |    0 |                        BusinessObject targetBusinessObject = null;  | 
  |  708 |    0 |                        resolvedPrincipalIdPropertyName.setLength( 0 );   | 
  |  709 |     | 
                             | 
  |  710 |     | 
                             | 
  |  711 |    0 |                            String personReferenceObjectPropertyName = ObjectUtils.getNestedAttributePrefix( propertyName );  | 
  |  712 |     | 
                             | 
  |  713 |     | 
                             | 
  |  714 |    0 |                        if ( ObjectUtils.isNestedAttribute( personReferenceObjectPropertyName ) ) { | 
  |  715 |    0 |                            String targetBusinessObjectPropertyName = ObjectUtils.getNestedAttributePrefix( personReferenceObjectPropertyName );  | 
  |  716 |    0 |                            targetBusinessObject = (BusinessObject)ObjectUtils.getPropertyValue( businessObject, targetBusinessObjectPropertyName );  | 
  |  717 |    0 |                            if (targetBusinessObject != null) { | 
  |  718 |    0 |                                targetBusinessObjectClass = targetBusinessObject.getClass();  | 
  |  719 |    0 |                                resolvedPrincipalIdPropertyName.append(targetBusinessObjectPropertyName).append("."); | 
  |  720 |     | 
                           } else { | 
  |  721 |    0 |                                LOG.error("Could not find target property '"+propertyName+"' in class "+businessObject.getClass().getName()+". Property value was null."); | 
  |  722 |     | 
                           }  | 
  |  723 |    0 |                        } else {  | 
  |  724 |    0 |                            targetBusinessObjectClass = businessObject.getClass();  | 
  |  725 |    0 |                            targetBusinessObject = businessObject;  | 
  |  726 |     | 
                       }  | 
  |  727 |     | 
                         | 
  |  728 |    0 |                        if (targetBusinessObjectClass != null) { | 
  |  729 |     | 
                                 | 
  |  730 |     | 
                                 | 
  |  731 |     | 
                                 | 
  |  732 |    0 |                            String propName = ObjectUtils.getNestedAttributePrimitive( personReferenceObjectPropertyName );  | 
  |  733 |    0 |                            BusinessObjectRelationship rel = getBusinessObjectMetaDataService().getBusinessObjectRelationship( targetBusinessObject, propName );  | 
  |  734 |    0 |                            if ( rel != null ) { | 
  |  735 |    0 |                                String sourcePrimitivePropertyName = rel.getParentAttributeForChildAttribute(KIMPropertyConstants.Person.PRINCIPAL_ID);  | 
  |  736 |    0 |                                resolvedPrincipalIdPropertyName.append(sourcePrimitivePropertyName);  | 
  |  737 |     | 
                                     | 
  |  738 |    0 |                                String principalName = fieldValues.get( propertyName );  | 
  |  739 |    0 |                                    KimPrincipal principal = getIdentityManagementService().getPrincipalByPrincipalName( principalName );  | 
  |  740 |    0 |                                if (principal != null ) { | 
  |  741 |    0 |                                    processedFieldValues.put(resolvedPrincipalIdPropertyName.toString(), principal.getPrincipalId());  | 
  |  742 |     | 
                               } else { | 
  |  743 |    0 |                                    processedFieldValues.put(resolvedPrincipalIdPropertyName.toString(), null);  | 
  |  744 |     | 
                                   try { | 
  |  745 |     | 
                                         | 
  |  746 |     | 
                                         | 
  |  747 |     | 
                                         | 
  |  748 |     | 
                                         | 
  |  749 |    0 |                                        ObjectUtils.setObjectProperty(targetBusinessObject, resolvedPrincipalIdPropertyName.toString(), null );  | 
  |  750 |    0 |                                        ObjectUtils.setObjectProperty(targetBusinessObject, propName, null );  | 
  |  751 |    0 |                                        ObjectUtils.setObjectProperty(targetBusinessObject, propName + ".principalName", principalName );  | 
  |  752 |    0 |                                    } catch ( Exception ex ) { | 
  |  753 |    0 |                                        LOG.error( "Unable to blank out the person object after finding that the person with the given principalName does not exist.", ex );  | 
  |  754 |    0 |                                    }  | 
  |  755 |     | 
                               }  | 
  |  756 |    0 |                            } else { | 
  |  757 |    0 |                                    LOG.error( "Missing relationship for " + propName + " on " + targetBusinessObjectClass.getName() );  | 
  |  758 |     | 
                           }  | 
  |  759 |    0 |                        } else {  | 
  |  760 |    0 |                            processedFieldValues.put(resolvedPrincipalIdPropertyName.toString(), null);  | 
  |  761 |     | 
                       }  | 
  |  762 |     | 
                   }  | 
  |  763 |     | 
                 | 
  |  764 |     | 
                 | 
  |  765 |     | 
                 | 
  |  766 |    0 |                } else if (propertyName.endsWith("." + KIMPropertyConstants.Person.PRINCIPAL_NAME)){ | 
  |  767 |     | 
                     | 
  |  768 |    0 |                    String principalName = fieldValues.get(propertyName);  | 
  |  769 |    0 |                    if ( StringUtils.isNotEmpty( principalName ) ) { | 
  |  770 |    0 |                        String containerPropertyName = propertyName;  | 
  |  771 |    0 |                        if (containerPropertyName.startsWith(KNSConstants.MAINTENANCE_ADD_PREFIX)) { | 
  |  772 |    0 |                            containerPropertyName = StringUtils.substringAfter( propertyName, KNSConstants.MAINTENANCE_ADD_PREFIX );  | 
  |  773 |     | 
                       }  | 
  |  774 |     | 
                         | 
  |  775 |     | 
                         | 
  |  776 |     | 
                         | 
  |  777 |    0 |                        if ( ObjectUtils.isNestedAttribute( containerPropertyName ) ) { | 
  |  778 |     | 
                                 | 
  |  779 |    0 |                            String collectionName = StringUtils.substringBefore( containerPropertyName, "." );  | 
  |  780 |     | 
                             | 
  |  781 |     | 
                             | 
  |  782 |     | 
                             | 
  |  783 |    0 |                            Class<? extends BusinessObject> collectionBusinessObjectClass = getMaintenanceDocumentDictionaryService()  | 
  |  784 |     | 
                                           .getCollectionBusinessObjectClass(  | 
  |  785 |     | 
                                                           getMaintenanceDocumentDictionaryService()  | 
  |  786 |     | 
                                                                           .getDocumentTypeName(businessObject.getClass()), collectionName);  | 
  |  787 |    0 |                            if (collectionBusinessObjectClass != null) { | 
  |  788 |     | 
                                 | 
  |  789 |     | 
                                     | 
  |  790 |    0 |                                List<BusinessObjectRelationship> relationships =   | 
  |  791 |     | 
                                               getBusinessObjectMetaDataService().getBusinessObjectRelationships( collectionBusinessObjectClass );  | 
  |  792 |     | 
                                 | 
  |  793 |     | 
                                 | 
  |  794 |    0 |                                for ( BusinessObjectRelationship rel : relationships ) { | 
  |  795 |    0 |                                        String parentAttribute = rel.getParentAttributeForChildAttribute( KIMPropertyConstants.Person.PRINCIPAL_ID );  | 
  |  796 |    0 |                                        if ( parentAttribute == null ) { | 
  |  797 |    0 |                                                continue;  | 
  |  798 |     | 
                                       }  | 
  |  799 |     | 
                                     | 
  |  800 |    0 |                                        processedFieldValues.remove( propertyName );  | 
  |  801 |    0 |                                            String fieldPrefix = StringUtils.substringBeforeLast( StringUtils.substringBeforeLast( propertyName, "." + KIMPropertyConstants.Person.PRINCIPAL_NAME ), "." );  | 
  |  802 |    0 |                                    String relatedPrincipalIdPropertyName = fieldPrefix + "." + parentAttribute;  | 
  |  803 |     | 
                                     | 
  |  804 |    0 |                                              if(EXTENSION.equals(StringUtils.substringAfterLast(fieldPrefix, ".")) && EXTENSION.equals(StringUtils.substringBefore(parentAttribute, ".")))  | 
  |  805 |     | 
                                             { | 
  |  806 |    0 |                                                      relatedPrincipalIdPropertyName = fieldPrefix + "." + StringUtils.substringAfter(parentAttribute, ".");  | 
  |  807 |     | 
                                             }  | 
  |  808 |    0 |                                    String currRelatedPersonPrincipalId = processedFieldValues.get(relatedPrincipalIdPropertyName);  | 
  |  809 |    0 |                                    if ( StringUtils.isBlank( currRelatedPersonPrincipalId ) ) { | 
  |  810 |    0 |                                            KimPrincipal principal = getIdentityManagementService().getPrincipalByPrincipalName( principalName );  | 
  |  811 |    0 |                                            if ( principal != null ) { | 
  |  812 |    0 |                                                    processedFieldValues.put(relatedPrincipalIdPropertyName, principal.getPrincipalId());  | 
  |  813 |     | 
                                           } else { | 
  |  814 |    0 |                                                    processedFieldValues.put(relatedPrincipalIdPropertyName, null);  | 
  |  815 |     | 
                                           }  | 
  |  816 |     | 
                                   }  | 
  |  817 |    0 |                                }   | 
  |  818 |    0 |                            } else { | 
  |  819 |    0 |                                    if ( LOG.isDebugEnabled() ) { | 
  |  820 |    0 |                                            LOG.debug( "Unable to determine class for collection referenced as part of property: " + containerPropertyName + " on " + businessObject.getClass().getName() );  | 
  |  821 |     | 
                                   }  | 
  |  822 |     | 
                           }  | 
  |  823 |    0 |                        } else { | 
  |  824 |    0 |                                if ( LOG.isDebugEnabled() ) { | 
  |  825 |    0 |                                        LOG.debug( "Non-nested property ending with 'principalName': " + containerPropertyName + " on " + businessObject.getClass().getName() );  | 
  |  826 |     | 
                               }  | 
  |  827 |     | 
                       }  | 
  |  828 |     | 
                   }  | 
  |  829 |    0 |                }  | 
  |  830 |     | 
           }  | 
  |  831 |    0 |            return processedFieldValues;  | 
  |  832 |     | 
       }  | 
  |  833 |     | 
             | 
  |  834 |     | 
             | 
  |  835 |     | 
     | 
  |  836 |     | 
           protected IdentityManagementService getIdentityManagementService() { | 
  |  837 |    0 |                    if ( identityManagementService == null ) { | 
  |  838 |    0 |                            identityManagementService = KIMServiceLocator.getIdentityManagementService();  | 
  |  839 |     | 
                   }  | 
  |  840 |    0 |                    return identityManagementService;  | 
  |  841 |     | 
           }  | 
  |  842 |     | 
     | 
  |  843 |     | 
           protected RoleManagementService getRoleManagementService() { | 
  |  844 |    0 |                    if ( roleManagementService == null ) { | 
  |  845 |    0 |                            roleManagementService = KIMServiceLocator.getRoleManagementService();  | 
  |  846 |     | 
                   }  | 
  |  847 |    0 |                    return roleManagementService;  | 
  |  848 |     | 
           }  | 
  |  849 |     | 
     | 
  |  850 |     | 
     | 
  |  851 |     | 
           public Class<? extends Person> getPersonImplementationClass() { | 
  |  852 |    0 |                    return PersonImpl.class;  | 
  |  853 |     | 
           }  | 
  |  854 |     | 
             | 
  |  855 |     | 
           protected BusinessObjectMetaDataService getBusinessObjectMetaDataService() { | 
  |  856 |    0 |                    if ( businessObjectMetaDataService == null ) { | 
  |  857 |    0 |                            businessObjectMetaDataService = KNSServiceLocatorWeb.getBusinessObjectMetaDataService();  | 
  |  858 |     | 
                   }  | 
  |  859 |    0 |                    return businessObjectMetaDataService;  | 
  |  860 |     | 
           }  | 
  |  861 |     | 
     | 
  |  862 |     | 
           protected MaintenanceDocumentDictionaryService getMaintenanceDocumentDictionaryService() { | 
  |  863 |    0 |                    if ( maintenanceDocumentDictionaryService == null ) { | 
  |  864 |    0 |                            maintenanceDocumentDictionaryService = KNSServiceLocatorWeb.getMaintenanceDocumentDictionaryService();  | 
  |  865 |     | 
                   }  | 
  |  866 |    0 |                    return maintenanceDocumentDictionaryService;  | 
  |  867 |     | 
           }  | 
  |  868 |     | 
     | 
  |  869 |     | 
           public void setPersonCacheMaxSize(int personCacheMaxSize) { | 
  |  870 |    0 |                    this.personCacheMaxSize = personCacheMaxSize;  | 
  |  871 |    0 |            }  | 
  |  872 |     | 
     | 
  |  873 |     | 
           public void setPersonCacheMaxAgeSeconds(int personCacheMaxAgeSeconds) { | 
  |  874 |    0 |                    this.personCacheMaxAgeSeconds = personCacheMaxAgeSeconds;  | 
  |  875 |    0 |            }  | 
  |  876 |     | 
             | 
  |  877 |     | 
   }  |