Coverage Report - org.kuali.rice.ksb.messaging.serviceproxies.AsynchronousServiceCallProxy
 
Classes in this File Line Coverage Branch Coverage Complexity
AsynchronousServiceCallProxy
0%
0/45
0%
0/12
2
 
 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  
 package org.kuali.rice.ksb.messaging.serviceproxies;
 17  
 
 18  
 import org.apache.log4j.Logger;
 19  
 import org.kuali.rice.core.api.exception.RiceRuntimeException;
 20  
 import org.kuali.rice.core.api.proxy.TargetedInvocationHandler;
 21  
 import org.kuali.rice.core.impl.proxy.BaseInvocationHandler;
 22  
 import org.kuali.rice.core.impl.resourceloader.ContextClassLoaderProxy;
 23  
 import org.kuali.rice.core.util.ClassLoaderUtils;
 24  
 import org.kuali.rice.ksb.messaging.*;
 25  
 import org.kuali.rice.ksb.service.KSBServiceLocator;
 26  
 import org.kuali.rice.ksb.util.KSBConstants;
 27  
 
 28  
 import java.io.Serializable;
 29  
 import java.lang.reflect.Method;
 30  
 import java.lang.reflect.Proxy;
 31  
 import java.util.List;
 32  
 
 33  
 
 34  
 /**
 35  
  * Standard default proxy used to call services asynchronously. Persists the method call to the db so call is never lost and
 36  
  * only sent when transaction is committed.
 37  
  * 
 38  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 39  
  * 
 40  
  */
 41  
 public class AsynchronousServiceCallProxy extends BaseInvocationHandler implements TargetedInvocationHandler {
 42  
     
 43  0
     private static final Logger LOG = Logger.getLogger(AsynchronousServiceCallProxy.class);
 44  
 
 45  
     private AsynchronousCallback callback;
 46  
 
 47  
     private List<RemotedServiceHolder> serviceDefs;
 48  
 
 49  
     private Serializable context;
 50  
 
 51  
     private String value1;
 52  
 
 53  
     private String value2;
 54  
 
 55  
     protected AsynchronousServiceCallProxy(List<RemotedServiceHolder> serviceDefs, AsynchronousCallback callback,
 56  0
             Serializable context, String value1, String value2) {
 57  0
         this.serviceDefs = serviceDefs;
 58  0
         this.callback = callback;
 59  0
         this.context = context;
 60  0
         this.value1 = value1;
 61  0
         this.value2 = value2;
 62  0
     }
 63  
 
 64  
     public static Object createInstance(List<RemotedServiceHolder> serviceDefs, AsynchronousCallback callback,
 65  
             Serializable context, String value1, String value2) {
 66  0
         if (serviceDefs == null || serviceDefs.isEmpty()) {
 67  0
             throw new RuntimeException("Cannot create service proxy, no service(s) passed in.");
 68  
         }
 69  
         try {
 70  0
             return Proxy.newProxyInstance(ClassLoaderUtils.getDefaultClassLoader(), ContextClassLoaderProxy
 71  
                     .getInterfacesToProxy(serviceDefs.get(0).getService()), new AsynchronousServiceCallProxy(
 72  
                     serviceDefs, callback, context, value1, value2));
 73  0
         } catch (Exception e) {
 74  0
             throw new RiceRuntimeException(e);
 75  
         }
 76  
     }
 77  
 
 78  
     @Override
 79  
     protected Object invokeInternal(Object proxy, Method method, Object[] arguments) throws Throwable {
 80  
 
 81  0
         if (LOG.isDebugEnabled()) {
 82  0
             LOG.debug("creating messages for method invocation: " + method.getName());
 83  
         }
 84  
         // there are multiple service calls to make in the case of topics.
 85  0
         AsynchronousCall methodCall = null;
 86  0
         PersistedMessageBO message = null;
 87  0
         synchronized (this) {
 88  
             // consider moving all this topic invocation stuff to the service
 89  
             // invoker for speed reasons
 90  0
             for (ServiceHolder remotedServiceHolder : this.serviceDefs) {
 91  0
                 ServiceInfo serviceInfo = remotedServiceHolder.getServiceInfo();
 92  0
                 methodCall = new AsynchronousCall(method.getParameterTypes(), arguments, serviceInfo, method.getName(),
 93  
                         this.callback, this.context);
 94  0
                 message = KSBServiceLocator.getRouteQueueService().getMessage(serviceInfo, methodCall);
 95  0
                 message.setValue1(this.value1);
 96  0
                 message.setValue2(this.value2);
 97  0
                 saveMessage(message);
 98  0
                 executeMessage(message);
 99  
                 // only do one iteration if this is a queue. The load balancing
 100  
                 // will be handled when the service is
 101  
                 // fetched by the MessageServiceInvoker through the GRL (and
 102  
                 // then through the RemoteResourceServiceLocatorImpl)
 103  0
                 if (serviceInfo.getServiceDefinition().getQueue()) {
 104  0
                     break;
 105  
                 }
 106  0
             }
 107  0
         }
 108  0
         if (LOG.isDebugEnabled()) {
 109  0
             LOG.debug("finished creating messages for method invocation: " + method.getName());
 110  
         }
 111  0
         return null;
 112  
     }
 113  
 
 114  
     protected void saveMessage(PersistedMessageBO message) {
 115  0
         message.setQueueStatus(KSBConstants.ROUTE_QUEUE_ROUTING);
 116  0
         KSBServiceLocator.getRouteQueueService().save(message);
 117  0
     }
 118  
 
 119  
     protected void executeMessage(PersistedMessageBO message) throws Exception {
 120  0
         MessageSender.sendMessage(message);
 121  0
     }
 122  
 
 123  
     /**
 124  
          * Returns the List<RemotedServiceHolder> of asynchronous services which will be invoked by calls to this proxy.
 125  
          * This is a List because, in the case of Topics, there can be more than one service invoked.
 126  
          */
 127  
     public Object getTarget() {
 128  0
         return this.serviceDefs;
 129  
     }
 130  
 
 131  
     public AsynchronousCallback getCallback() {
 132  0
         return this.callback;
 133  
     }
 134  
 
 135  
     public void setCallback(AsynchronousCallback callback) {
 136  0
         this.callback = callback;
 137  0
     }
 138  
 
 139  
     public List<RemotedServiceHolder> getServiceDefs() {
 140  0
         return this.serviceDefs;
 141  
     }
 142  
 
 143  
     public void setServiceDefs(List<RemotedServiceHolder> serviceDefs) {
 144  0
         this.serviceDefs = serviceDefs;
 145  0
     }
 146  
 }