Coverage Report - org.kuali.rice.kim.service.impl.IdentityArchiveServiceImpl
 
Classes in this File Line Coverage Branch Coverage Complexity
IdentityArchiveServiceImpl
0%
0/40
0%
0/12
2.409
IdentityArchiveServiceImpl$1
N/A
N/A
2.409
IdentityArchiveServiceImpl$CallableAdapter
0%
0/8
N/A
2.409
IdentityArchiveServiceImpl$EntityArchiveWriter
0%
0/11
N/A
2.409
IdentityArchiveServiceImpl$EntityArchiveWriter$1
0%
0/8
0%
0/8
2.409
IdentityArchiveServiceImpl$EntityArchiveWriter$2
0%
0/20
0%
0/14
2.409
IdentityArchiveServiceImpl$EntityArchiveWriter$3
0%
0/11
0%
0/8
2.409
IdentityArchiveServiceImpl$PreLogCallableWrapper
0%
0/7
N/A
2.409
IdentityArchiveServiceImpl$WriteQueue
0%
0/8
0%
0/2
2.409
 
 1  
 /*
 2  
  * Copyright 2006-2011 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  
 
 17  
 package org.kuali.rice.kim.service.impl;
 18  
 
 19  
 import org.apache.commons.lang.StringUtils;
 20  
 import org.apache.log4j.Level;
 21  
 import org.apache.log4j.Logger;
 22  
 import org.kuali.rice.core.api.config.property.ConfigurationService;
 23  
 import org.kuali.rice.kim.api.identity.IdentityArchiveService;
 24  
 import org.kuali.rice.kim.api.identity.entity.EntityDefault;
 25  
 import org.kuali.rice.kim.api.identity.principal.Principal;
 26  
 import org.kuali.rice.kim.bo.entity.impl.KimEntityDefaultInfoCacheImpl;
 27  
 import org.kuali.rice.kim.util.KimConstants;
 28  
 import org.kuali.rice.krad.service.BusinessObjectService;
 29  
 import org.kuali.rice.krad.service.KRADServiceLocatorInternal;
 30  
 import org.kuali.rice.ksb.service.KSBServiceLocator;
 31  
 import org.springframework.beans.factory.DisposableBean;
 32  
 import org.springframework.beans.factory.InitializingBean;
 33  
 import org.springframework.transaction.PlatformTransactionManager;
 34  
 import org.springframework.transaction.TransactionStatus;
 35  
 import org.springframework.transaction.support.TransactionCallback;
 36  
 import org.springframework.transaction.support.TransactionTemplate;
 37  
 
 38  
 import java.util.ArrayList;
 39  
 import java.util.Arrays;
 40  
 import java.util.Collection;
 41  
 import java.util.Collections;
 42  
 import java.util.Comparator;
 43  
 import java.util.HashMap;
 44  
 import java.util.HashSet;
 45  
 import java.util.List;
 46  
 import java.util.Map;
 47  
 import java.util.Set;
 48  
 import java.util.concurrent.Callable;
 49  
 import java.util.concurrent.ConcurrentLinkedQueue;
 50  
 import java.util.concurrent.TimeUnit;
 51  
 import java.util.concurrent.atomic.AtomicBoolean;
 52  
 import java.util.concurrent.atomic.AtomicInteger;
 53  
 
 54  
 /**
 55  
  * This is the default implementation for the IdentityArchiveService.
 56  
  * @see IdentityArchiveService
 57  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 58  
  *
 59  
  */
 60  0
 public class IdentityArchiveServiceImpl implements IdentityArchiveService, InitializingBean, DisposableBean {
 61  0
         private static final Logger LOG = Logger.getLogger( IdentityArchiveServiceImpl.class );
 62  
 
 63  
         private BusinessObjectService businessObjectService;
 64  
         private ConfigurationService kualiConfigurationService;
 65  
 
 66  
         private static final String EXEC_INTERVAL_SECS = "kim.identityArchiveServiceImpl.executionIntervalSeconds";
 67  
         private static final String MAX_WRITE_QUEUE_SIZE = "kim.identityArchiveServiceImpl.maxWriteQueueSize";
 68  
         private static final int EXECUTION_INTERVAL_SECONDS_DEFAULT = 600; // by default, flush the write queue this often
 69  
         private static final int MAX_WRITE_QUEUE_SIZE_DEFAULT = 300; // cache this many KEDI's before forcing write
 70  
 
 71  0
         private final WriteQueue writeQueue = new WriteQueue();
 72  0
         private final EntityArchiveWriter writer = new EntityArchiveWriter();
 73  
 
 74  
         // all this ceremony just decorates the writer so it logs a message first, and converts the Callable to Runnable
 75  0
         private final Runnable maxQueueSizeExceededWriter =
 76  
                 new CallableAdapter(new PreLogCallableWrapper<Boolean>(writer, Level.DEBUG, "max size exceeded, flushing write queue"));
 77  
 
 78  
         // ditto
 79  0
         private final Runnable scheduledWriter =
 80  
                 new CallableAdapter(new PreLogCallableWrapper<Boolean>(writer, Level.DEBUG, "scheduled write out, flushing write queue"));
 81  
 
 82  
         // ditto
 83  0
         private final Runnable shutdownWriter =
 84  
                 new CallableAdapter(new PreLogCallableWrapper<Boolean>(writer, Level.DEBUG, "rice is shutting down, flushing write queue"));
 85  
         
 86  
         private int getExecutionIntervalSeconds() {
 87  0
                 final String prop = kualiConfigurationService.getPropertyValueAsString(EXEC_INTERVAL_SECS);
 88  
                 try {
 89  0
                         return Integer.valueOf(prop).intValue();
 90  0
                 } catch (NumberFormatException e) {
 91  0
                         return EXECUTION_INTERVAL_SECONDS_DEFAULT;
 92  
                 }
 93  
         }
 94  
         
 95  
         private int getMaxWriteQueueSize() {
 96  0
                 final String prop = kualiConfigurationService.getPropertyValueAsString(MAX_WRITE_QUEUE_SIZE);
 97  
                 try {
 98  0
                         return Integer.valueOf(prop).intValue();
 99  0
                 } catch (NumberFormatException e) {
 100  0
                         return MAX_WRITE_QUEUE_SIZE_DEFAULT;
 101  
                 }
 102  
         }
 103  
 
 104  
         @Override
 105  
         public EntityDefault getEntityDefaultFromArchive( String entityId ) {
 106  0
             Map<String,String> criteria = new HashMap<String, String>(1);
 107  0
             criteria.put(KimConstants.PrimaryKeyConstants.ENTITY_ID, entityId);
 108  0
             KimEntityDefaultInfoCacheImpl cachedValue = businessObjectService.findByPrimaryKey(KimEntityDefaultInfoCacheImpl.class, criteria);
 109  0
             return (cachedValue == null) ? null : cachedValue.convertCacheToEntityDefaultInfo();
 110  
     }
 111  
 
 112  
     @Override
 113  
         public EntityDefault getEntityDefaultFromArchiveByPrincipalId(String principalId) {
 114  0
             Map<String,String> criteria = new HashMap<String, String>(1);
 115  0
             criteria.put("principalId", principalId);
 116  0
             KimEntityDefaultInfoCacheImpl cachedValue = businessObjectService.findByPrimaryKey(KimEntityDefaultInfoCacheImpl.class, criteria);
 117  0
             return (cachedValue == null) ? null : cachedValue.convertCacheToEntityDefaultInfo();
 118  
     }
 119  
 
 120  
     @Override
 121  
         public EntityDefault getEntityDefaultFromArchiveByPrincipalName(String principalName) {
 122  0
             Map<String,String> criteria = new HashMap<String, String>(1);
 123  0
             criteria.put("principalName", principalName);
 124  0
             Collection<KimEntityDefaultInfoCacheImpl> entities = businessObjectService.findMatching(KimEntityDefaultInfoCacheImpl.class, criteria);
 125  0
             return (entities == null || entities.isEmpty()) ? null : entities.iterator().next().convertCacheToEntityDefaultInfo();
 126  
     }
 127  
 
 128  
     @Override
 129  
         public void saveEntityDefaultToArchive(EntityDefault entity) {
 130  
             // if the max size has been reached, schedule now
 131  0
             if (getMaxWriteQueueSize() <= writeQueue.offerAndGetSize(entity) /* <- this enqueues the KEDI */ &&
 132  
                             writer.requestSubmit()) {
 133  0
                     KSBServiceLocator.getThreadPool().execute(maxQueueSizeExceededWriter);
 134  
             }
 135  0
     }
 136  
 
 137  
         public void setBusinessObjectService(BusinessObjectService businessObjectService) {
 138  0
                 this.businessObjectService = businessObjectService;
 139  0
         }
 140  
 
 141  
         public void setKualiConfigurationService(
 142  
                         ConfigurationService kualiConfigurationService) {
 143  0
                 this.kualiConfigurationService = kualiConfigurationService;
 144  0
         }
 145  
     
 146  
     /** schedule the writer on the KSB scheduled pool. */
 147  
         @Override
 148  
         public void afterPropertiesSet() throws Exception {
 149  0
                 LOG.info("scheduling writer...");
 150  0
                 KSBServiceLocator.getScheduledPool().scheduleAtFixedRate(scheduledWriter,
 151  
                                 getExecutionIntervalSeconds(), getExecutionIntervalSeconds(), TimeUnit.SECONDS);
 152  0
         }
 153  
 
 154  
         /** flush the write queue immediately. */
 155  
         @Override
 156  
         public void destroy() throws Exception {
 157  0
                 KSBServiceLocator.getThreadPool().execute(shutdownWriter);
 158  0
         }
 159  
 
 160  
         /**
 161  
          * store the person to the database, but do this an alternate thread to
 162  
          * prevent transaction issues since this service is non-transactional
 163  
          *
 164  
          * @author Kuali Rice Team (rice.collab@kuali.org)
 165  
          *
 166  
          */
 167  0
         private class EntityArchiveWriter implements Callable {
 168  
 
 169  
                 // flag used to prevent multiple processes from being submitted at once
 170  0
                 AtomicBoolean currentlySubmitted = new AtomicBoolean(false);
 171  
 
 172  0
                 private final Comparator<Comparable> nullSafeComparator = new Comparator<Comparable>() {
 173  
                         @Override
 174  
                         public int compare(Comparable i1, Comparable i2) {
 175  0
                                 if (i1 != null && i2 != null) {
 176  0
                                         return i1.compareTo(i2);
 177  0
                                 } else if (i1 == null) {
 178  0
                                         if (i2 == null) {
 179  0
                                                 return 0;
 180  
                                         } else {
 181  0
                                                 return -1;
 182  
                                         }
 183  
                                 } else { // if (entityId2 == null) {
 184  0
                                         return 1;
 185  
                                 }
 186  
                         };
 187  
                 };
 188  
 
 189  
                 /**
 190  
                  * Comparator that attempts to impose a total ordering on EntityDefault instances
 191  
                  */
 192  0
                 private final Comparator<EntityDefault> kediComparator = new Comparator<EntityDefault>() {
 193  
                         /**
 194  
                          * compares by entityId value
 195  
                          * @see java.util.Comparator#compare(java.lang.Object, java.lang.Object)
 196  
                          */
 197  
                         @Override
 198  
                         public int compare(EntityDefault o1, EntityDefault o2) {
 199  0
                                 String entityId1 = (o1 == null) ? null : o1.getEntityId();
 200  0
                                 String entityId2 = (o2 == null) ? null : o2.getEntityId();
 201  
 
 202  0
                                 int result = nullSafeComparator.compare(entityId1, entityId2);
 203  
 
 204  0
                                 if (result == 0) {
 205  0
                                         result = getPrincipalIdsString(o1).compareTo(getPrincipalIdsString(o2));
 206  
                                 }
 207  
 
 208  0
                                 return result;
 209  
                         }
 210  
 
 211  
                         /**
 212  
                          * This method builds a newline delimited String containing the identity's principal IDs in sorted order
 213  
                          *
 214  
                          * @param entity
 215  
                          * @return
 216  
                          */
 217  
                         private String getPrincipalIdsString(EntityDefault entity) {
 218  0
                                 String result = "";
 219  0
                                 if (entity != null) {
 220  0
                                         List<Principal> principals = entity.getPrincipals();
 221  0
                                         if (principals != null) {
 222  0
                                                 if (principals.size() == 1) { // one
 223  0
                                                         result = principals.get(0).getPrincipalId();
 224  
                                                 } else { // multiple
 225  0
                                                         String [] ids = new String [principals.size()];
 226  0
                                                         int insertIndex = 0;
 227  0
                                                         for (Principal principal : principals) {
 228  0
                                                                 ids[insertIndex++] = principal.getPrincipalId();
 229  
                                                         }
 230  0
                                                         Arrays.sort(ids);
 231  0
                                                         result = StringUtils.join(ids, "\n");
 232  
                                                 }
 233  
                                         }
 234  
                                 }
 235  0
                                 return result;
 236  
                         }
 237  
                 };
 238  
 
 239  
                 public boolean requestSubmit() {
 240  0
                         return currentlySubmitted.compareAndSet(false, true);
 241  
                 }
 242  
 
 243  
                 /**
 244  
                  * Call that tries to flush the write queue.
 245  
                  * @see Callable#call()
 246  
                  */
 247  
                 @Override
 248  
                 public Object call() {
 249  
                         try {
 250  
                                 // the strategy is to grab chunks of entities, dedupe & sort them, and insert them in a big
 251  
                                 // batch to reduce transaction overhead.  Sorting is done so insertion order is guaranteed, which
 252  
                                 // prevents deadlocks between concurrent writers to the database.
 253  0
                                 PlatformTransactionManager transactionManager = KRADServiceLocatorInternal.getTransactionManager();
 254  0
                                 TransactionTemplate template = new TransactionTemplate(transactionManager);
 255  0
                                 template.execute(new TransactionCallback() {
 256  
                                         @Override
 257  
                                         public Object doInTransaction(TransactionStatus status) {
 258  0
                                                 EntityDefault entity = null;
 259  0
                                                 ArrayList<EntityDefault> entitiesToInsert = new ArrayList<EntityDefault>(getMaxWriteQueueSize());
 260  0
                                                 Set<String> deduper = new HashSet<String>(getMaxWriteQueueSize());
 261  
 
 262  
                                                 // order is important in this conditional so that elements aren't dequeued and then ignored
 263  0
                                                 while (entitiesToInsert.size() < getMaxWriteQueueSize() && null != (entity = writeQueue.poll())) {
 264  0
                                                         if (deduper.add(entity.getEntityId())) {
 265  0
                                                                 entitiesToInsert.add(entity);
 266  
                                                         }
 267  
                                                 }
 268  
 
 269  0
                                                 Collections.sort(entitiesToInsert, kediComparator);
 270  
 
 271  0
                                                 for (EntityDefault entityToInsert : entitiesToInsert) {
 272  0
                                                         businessObjectService.save( new KimEntityDefaultInfoCacheImpl( entityToInsert ) );
 273  
                                                 }
 274  0
                                                 return null;
 275  
                                         }
 276  
                                 });
 277  
                         } finally { // make sure our running flag is unset, otherwise we'll never run again
 278  0
                                 currentlySubmitted.compareAndSet(true, false);
 279  0
                         }
 280  
 
 281  0
                         return Boolean.TRUE;
 282  
                 }
 283  
         }
 284  
 
 285  
         /**
 286  
          * A class encapsulating a {@link ConcurrentLinkedQueue} and an {@link AtomicInteger} to
 287  
          * provide fast offer(enqueue)/poll(dequeue) and size checking.  Size may be approximate due to concurrent
 288  
          * activity, but for our purposes that is fine.
 289  
          *
 290  
          * @author Kuali Rice Team (rice.collab@kuali.org)
 291  
          *
 292  
          */
 293  0
         private static class WriteQueue {
 294  0
                 AtomicInteger writeQueueSize = new AtomicInteger(0);
 295  0
                 ConcurrentLinkedQueue<EntityDefault> queue = new ConcurrentLinkedQueue<EntityDefault>();
 296  
 
 297  
                 public int offerAndGetSize(EntityDefault entity) {
 298  0
                         queue.add(entity);
 299  0
                         return writeQueueSize.incrementAndGet();
 300  
                 }
 301  
 
 302  
                 private EntityDefault poll() {
 303  0
                         EntityDefault result = queue.poll();
 304  0
                         if (result != null) { writeQueueSize.decrementAndGet(); }
 305  0
                         return result;
 306  
                 }
 307  
         }
 308  
 
 309  
         /**
 310  
          * decorator for a callable to log a message before it is executed
 311  
          *
 312  
          * @author Kuali Rice Team (rice.collab@kuali.org)
 313  
          *
 314  
          */
 315  
         private static class PreLogCallableWrapper<A> implements Callable<A> {
 316  
                 
 317  
                 private final Callable inner;
 318  
                 private final Level level;
 319  
                 private final String message;
 320  
 
 321  0
                 public PreLogCallableWrapper(Callable inner, Level level, String message) {
 322  0
                         this.inner = inner;
 323  0
                         this.level = level;
 324  0
                         this.message = message;
 325  0
                 }
 326  
 
 327  
                 /**
 328  
                  * logs the message then calls the inner Callable
 329  
                  * 
 330  
                  * @see java.util.concurrent.Callable#call()
 331  
                  */
 332  
                 @Override
 333  
                 @SuppressWarnings("unchecked")
 334  
                 public A call() throws Exception {
 335  0
                         LOG.log(level, message);
 336  0
                         return (A)inner.call();
 337  
                 }
 338  
         }
 339  
 
 340  
         /**
 341  
          * Adapts a Callable to be Runnable
 342  
          *
 343  
          * @author Kuali Rice Team (rice.collab@kuali.org)
 344  
          *
 345  
          */
 346  0
         private static class CallableAdapter implements Runnable {
 347  
 
 348  
                 private final Callable callable;
 349  
 
 350  0
                 public CallableAdapter(Callable callable) {
 351  0
                         this.callable = callable;
 352  0
                 }
 353  
 
 354  
                 @Override
 355  
                 public void run() {
 356  
                         try {
 357  0
                                 callable.call();
 358  0
                         } catch (Exception e) {
 359  0
                                 throw new RuntimeException(e);
 360  0
                         }
 361  0
                 }
 362  
         }
 363  
 }