Package org.apache.tuscany.sca.implementation.java.invocation

Source Code of org.apache.tuscany.sca.implementation.java.invocation.JavaImplementationInvoker

/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements.  See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership.  The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License.  You may obtain a copy of the License at
*
*   http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied.  See the License for the
* specific language governing permissions and limitations
* under the License.   
*/
package org.apache.tuscany.sca.implementation.java.invocation;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;

import javax.xml.ws.Holder;

import org.apache.tuscany.sca.assembly.EndpointReference;
import org.apache.tuscany.sca.core.factory.ObjectCreationException;
import org.apache.tuscany.sca.core.scope.Scope;
import org.apache.tuscany.sca.core.scope.ScopeContainer;
import org.apache.tuscany.sca.core.scope.ScopedRuntimeComponent;
import org.apache.tuscany.sca.implementation.java.JavaImplementation;
import org.apache.tuscany.sca.implementation.java.context.ReflectiveInstanceWrapper;
import org.apache.tuscany.sca.implementation.java.injection.Injector;
import org.apache.tuscany.sca.implementation.java.introspect.JavaIntrospectionHelper;
import org.apache.tuscany.sca.interfacedef.DataType;
import org.apache.tuscany.sca.interfacedef.InterfaceContract;
import org.apache.tuscany.sca.interfacedef.Operation;
import org.apache.tuscany.sca.interfacedef.ParameterMode;
import org.apache.tuscany.sca.interfacedef.java.JavaInterface;
import org.apache.tuscany.sca.interfacedef.java.impl.JavaInterfaceUtil;
import org.apache.tuscany.sca.invocation.Invoker;
import org.apache.tuscany.sca.invocation.Message;
import org.apache.tuscany.sca.runtime.RuntimeComponent;
import org.oasisopen.sca.ServiceReference;
import org.oasisopen.sca.ServiceRuntimeException;

/**
* Responsible for synchronously dispatching an invocation to a Java component
* implementation instance
*
* @version $Rev: 998232 $ $Date: 2010-09-17 19:10:18 +0100 (Fri, 17 Sep 2010) $
*/
public class JavaImplementationInvoker implements Invoker {
    protected Operation operation;
    protected Method method;
    protected boolean allowsPBR;

    @SuppressWarnings("unchecked")
    protected final ScopeContainer scopeContainer;
  private final InterfaceContract interfaze;

    public JavaImplementationInvoker(Operation operation, Method method, RuntimeComponent component, InterfaceContract intf) {
        assert method != null : "Operation method cannot be null";
        this.method = method;
        this.operation = operation;
        this.scopeContainer = ((ScopedRuntimeComponent)component).getScopeContainer();
        this.allowsPBR = ((JavaImplementation)component.getImplementation()).isAllowsPassByReference(method);
        this.interfaze = intf;
    }

    public JavaImplementationInvoker(Operation operation, RuntimeComponent component, InterfaceContract intf) {
        // used if the method can't be computed statically in advance
        this.operation = operation;
        this.scopeContainer = ((ScopedRuntimeComponent)component).getScopeContainer();
        this.interfaze = intf;
    }

    @SuppressWarnings("unchecked")
    public Message invoke(Message msg) {
     
        Operation op = msg.getOperation();
        if (op == null) {
            op = this.operation;
        }
        Object payload = msg.getBody();

        Object contextId = null;

        EndpointReference from = msg.getFrom();

        // store the current thread context classloader
        // as we need to replace it with the class loader
        // used to load the java class as per SCA Spec
        ClassLoader tccl = Thread.currentThread().getContextClassLoader();
       
        try {
            // The following call might create a new conversation, as a result, the msg.getConversationID() might
            // return a new value
            ReflectiveInstanceWrapper wrapper = (ReflectiveInstanceWrapper) scopeContainer.getWrapper(contextId);
           
            // If there is a callback interface and the implementation is stateless, we need to
            // inject callbacks at invocation time. For Composite scope, this has already been done.
            if (( interfaze.getCallbackInterface() != null && (scopeContainer.getScope().equals(Scope.STATELESS))){
              injectCallbacks(wrapper, (JavaInterface)interfaze.getCallbackInterface());
            }
           
            Object instance = wrapper.getInstance();

            // If the method couldn't be computed statically, or the instance being
            // invoked is a user-specified callback object that doesn't implement
            // the service interface from which the reflective method was obtained,
            // compute the method object dynamically for this invocation.
            Method imethod = method;
            if (imethod == null || !imethod.getDeclaringClass().isInstance(instance)) {
                try {
                    imethod = JavaInterfaceUtil.findMethod(instance.getClass(), operation);
                } catch (NoSuchMethodException e) {
                    throw new IllegalArgumentException("Callback object does not provide method " + e.getMessage());
                }
            }
           
            // Set the thread context classloader of the thread used to invoke an operation
            // of a Java POJO component implementation is the class loader of the contribution
            // that contains the POJO implementation class.
           
            Thread.currentThread().setContextClassLoader(instance.getClass().getClassLoader());
           
            int argumentHolderCount = 0;

            // Holder pattern. Any payload parameters <T> which are should be in holders are placed in Holder<T>.
            // Only check Holder for remotable interfaces
            if (imethod != null && op.getInterface().isRemotable()) {
                List<DataType> inputTypes = op.getInputType().getLogical();
                for (int i = 0, size = inputTypes.size(); i < size; i++) {
                    if (ParameterMode.IN != op.getParameterModes().get(i)) {
                        // Promote array params from [<T>] to [Holder<T>]
                        Object[] payloadArray = (Object[])payload;
                 
                        if ( ParameterMode.INOUT == op.getParameterModes().get(i)) {
                          Object item = payloadArray[i];                          
                          payloadArray[i] = new Holder(item);
                        } else {
                          // Create an empty Holder since we should not pass values for OUT parameters
                          payloadArray[i] = new Holder();
                        }
                       
                        argumentHolderCount++;
                    }
                }
            }

            Object ret;
            if (payload != null && !payload.getClass().isArray()) {
                ret = imethod.invoke(instance, payload);
            } else {
                ret = imethod.invoke(instance, (Object[])payload);
            }

            scopeContainer.returnWrapper(wrapper, contextId);
           
                       
            if (argumentHolderCount > 0) {
                // Holder pattern. Any payload Holder<T> types are returned as the message body.
                List<Object> returnArgs = new ArrayList<Object>();
                returnArgs.add(ret);
                if (imethod != null) {
                  Object[] payloadArray = (Object[])payload;
                    for (int i = 0, size = op.getParameterModes().size(); i < size; i++) {
                        // System.out.println( "JavaImplementationInvoker.invoke return parameter " + i + " type=" + parameter.getClass().getName() );
                        if (ParameterMode.IN != op.getParameterModes().get(i)) {                         
                            // Demote array params from Holder<T> to <T>.                                                  
                                Holder<Object> item = (Holder<Object>)payloadArray[i];
                                payloadArray[i] = item.value;
                                returnArgs.add(payloadArray[i]);
                        }
                    }
                }

                msg.setBody(returnArgs.toArray());

            } else {
                msg.setBody(ret);
            }
        } catch (InvocationTargetException e) {
            Throwable cause = e.getTargetException();
            boolean isChecked = false;
            for (DataType<?> d : operation.getFaultTypes()) {
                if (d.getPhysical().isInstance(cause)) {
                    isChecked = true;
                    msg.setFaultBody(cause);
                    break;
                }
            }
            if (!isChecked) {
                if (cause instanceof RuntimeException) {
                    throw (RuntimeException)cause;
                }
                if (cause instanceof Error) {
                    throw (Error)cause;
                } else {
                    throw new ServiceRuntimeException(cause.getMessage(), cause);
                }
            }           
               
        } catch (ObjectCreationException e) {
            throw new ServiceRuntimeException(e.getMessage(), e);
        } catch (Exception e) {
            msg.setFaultBody(e);          
        } finally {
            // set the tccl
            Thread.currentThread().setContextClassLoader(tccl);
        }
        return msg;
    }

  private void injectCallbacks(ReflectiveInstanceWrapper wrapper,
      JavaInterface callbackInterface) {
 
    for (Injector injector : wrapper.getCallbackInjectors()) {
      if (injector != null) {
        try {      
          if (ServiceReference.class.isAssignableFrom(injector.getType())) {
            Class<?> intf = JavaIntrospectionHelper.getBusinessInterface(injector.getType(), injector.getGenericType());
            if ( intf.isAssignableFrom(callbackInterface.getJavaClass())) {                                                         
              injector.inject(wrapper.getInstance());        
            }
          } else if (injector.getType().isAssignableFrom(callbackInterface.getJavaClass())) {
            injector.inject(wrapper.getInstance());
          } else {
            injector.injectNull(wrapper.getInstance());
          }
        } catch (Exception e) {                    
          throw new ObjectCreationException("Exception invoking injector - " + e.getMessage(), e);
        }
      }
    }
   
  }   

}
TOP

Related Classes of org.apache.tuscany.sca.implementation.java.invocation.JavaImplementationInvoker

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.