/*
* The Apache Software License, Version 1.1
*
*
* Copyright (c) 2001-2003 The Apache Software Foundation.  All rights
* reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
*    notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
*    notice, this list of conditions and the following disclaimer in
*    the documentation and/or other materials provided with the
*    distribution.
*
* 3. The end-user documentation included with the redistribution,
*    if any, must include the following acknowledgment:
*       "This product includes software developed by the
*        Apache Software Foundation (http://www.apache.org/)."
*    Alternately, this acknowledgment may appear in the software itself,
*    if and wherever such third-party acknowledgments normally appear.
*
* 4. The names "Axis" and "Apache Software Foundation" must
*    not be used to endorse or promote products derived from this
*    software without prior written permission. For written
*    permission, please contact apache@apache.org.
*
* 5. Products derived from this software may not be called "Apache",
*    nor may "Apache" appear in their name, without prior written
*    permission of the Apache Software Foundation.
*
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
* ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation.  For more
* information on the Apache Software Foundation, please see
* <http://www.apache.org/>.
*/

package org.jboss.axis.client;

import org.jboss.axis.AxisEngine;
import org.jboss.axis.AxisFault;
import org.jboss.axis.AxisProperties;
import org.jboss.axis.Constants;
import org.jboss.axis.Handler;
import org.jboss.axis.InternalException;
import org.jboss.axis.Message;
import org.jboss.axis.MessageContext;
import org.jboss.axis.attachments.Attachments;
import org.jboss.axis.description.FaultDesc;
import org.jboss.axis.description.OperationDesc;
import org.jboss.axis.description.ParameterDesc;
import org.jboss.axis.encoding.DeserializerFactory;
import org.jboss.axis.encoding.SerializerFactory;
import org.jboss.axis.encoding.TypeMapping;
import org.jboss.axis.encoding.TypeMappingRegistry;
import org.jboss.axis.encoding.XMLType;
import org.jboss.axis.encoding.ser.BaseDeserializerFactory;
import org.jboss.axis.encoding.ser.BaseSerializerFactory;
import org.jboss.axis.enums.Style;
import org.jboss.axis.enums.Use;
import org.jboss.axis.handlers.soap.SOAPService;
import org.jboss.axis.message.RPCElement;
import org.jboss.axis.message.RPCHeaderParam;
import org.jboss.axis.message.RPCParam;
import org.jboss.axis.message.SOAPBodyElementAxisImpl;
import org.jboss.axis.message.SOAPEnvelopeAxisImpl;
import org.jboss.axis.message.SOAPFaultImpl;
import org.jboss.axis.message.SOAPHeaderElementAxisImpl;
import org.jboss.axis.soap.SOAPConstants;
import org.jboss.axis.transport.http.HTTPTransport;
import org.jboss.axis.transport.java.JavaTransport;
import org.jboss.axis.transport.local.LocalTransport;
import org.jboss.axis.utils.ClassUtils;
import org.jboss.axis.utils.JavaUtils;
import org.jboss.axis.utils.Messages;
import org.jboss.axis.wsdl.symbolTable.BindingEntry;
import org.jboss.axis.wsdl.symbolTable.FaultInfo;
import org.jboss.axis.wsdl.symbolTable.Parameter;
import org.jboss.axis.wsdl.symbolTable.Parameters;
import org.jboss.axis.wsdl.symbolTable.SymbolTable;
import org.jboss.axis.wsdl.toJava.Utils;
import org.jboss.logging.Logger;

import javax.wsdl.Binding;
import javax.wsdl.BindingInput;
import javax.wsdl.BindingOperation;
import javax.wsdl.Operation;
import javax.wsdl.Part;
import javax.wsdl.Port;
import javax.wsdl.PortType;
import javax.wsdl.extensions.soap.SOAPAddress;
import javax.wsdl.extensions.soap.SOAPBody;
import javax.wsdl.extensions.soap.SOAPOperation;
import javax.xml.namespace.QName;
import javax.xml.rpc.JAXRPCException;
import javax.xml.rpc.ParameterMode;
import javax.xml.rpc.holders.Holder;
import java.net.MalformedURLException;
import java.net.URL;
import java.rmi.RemoteException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.StringTokenizer;
import java.util.Vector;

/**
 * Axis' JAXRPC Dynamic Invocation Interface implementation of the Call
 * interface.  This class should be used to actually invoke the Web Service.
 * It can be prefilled by a WSDL document (on the constructor to the Service
 * object) or you can fill in the data yourself.
 * <pre>
 * Standard properties defined by in JAX-RPC's javax..xml.rpc.Call interface:
 *     USERNAME_PROPERTY        - User name for authentication
 *     PASSWORD_PROPERTY        - Password for authentication
 *     SESSION_PROPERTY         - Participate in a session with the endpoint?
 *     OPERATION_STYLE_PROPERTY - "rpc" or "document"
 *     SOAPACTION_USE_PROPERTY  - Should SOAPAction be used?
 *     SOAPACTION_URI_PROPERTY  - If SOAPAction is used, this is that action
 *     ENCODING_STYLE_PROPERTY  - Default is SOAP 1.1:  "http://schemas.xmlsoap.org/soap/encoding/"
 * <p/>
 * AXIS properties:
 *     SEND_TYPE_ATTR - Should we send the XSI type attributes (true/false)
 *     TIMEOUT        - Timeout used by transport sender in milliseconds
 *     TRANSPORT_NAME - Name of transport handler to use
 *     ATTACHMENT_ENCAPSULATION_FORMAT- Send attachments as MIME the default, or DIME.
 * </pre>
 *
 * @author Doug Davis (dug@us.ibm.com)
 */

public class Call implements javax.xml.rpc.Call
{

   private static Logger log = Logger.getLogger(Call.class.getName());

   private static Logger tlog = Logger.getLogger(Constants.TIME_LOG_CATEGORY);

   private boolean parmAndRetReq = true;
   private Service service;
   private QName portName;
   private QName operationName;

   protected MessageContext msgContext;

   // Collection of properties to store and put in MessageContext at
   // invoke() time.  Known ones are stored in actual variables for
   // efficiency/type-consistency.  Unknown ones are in myProperties.
   private Hashtable myProperties = new Hashtable();
   private String username;
   private String password;
   private boolean maintainSession;
   private boolean useSOAPAction;
   private String SOAPActionURI;
   private Integer timeout;

   /**
    * Metadata for the operation associated with this Call
    */
   private OperationDesc operation;
   /**
    * This will be true if an OperationDesc is handed to us whole
    */
   private boolean operationSetManually;

   // Is this a one-way call?
   private boolean invokeOneWay;
   private boolean isMsg;

   // Our Transport, if any
   private Transport transport;
   private String transportName;

   // A couple places to store output parameters.
   // As a HashMap, retrievable via QName (for getOutputParams).
   private HashMap outParams;
   // As a list, retrievable by index (for getOutputValues).
   private ArrayList outParamsList;

   // A place to store any client-specified headers
   private Vector headers;

   protected Vector attachmentParts = new Vector();

   public static final String SEND_TYPE_ATTR = "send_type_attr";
   public static final String TRANSPORT_NAME = "transport_name";
   public static final String TRANSPORT_PROPERTY = "java.protocol.handler.pkgs";

   public static final String WSDL_SERVICE = "wsdl.service";

   public static final String WSDL_PORT_NAME = "wsdl.portName";

   // @deprecated use WSDL_SERVICE instead.
   public static final String JAXRPC_SERVICE = WSDL_SERVICE;

   // @deprected use WSDL_PORT_NAME instead.
   public static final String JAXRPC_PORTTYPE_NAME = WSDL_PORT_NAME;

   // If true, the code will throw a fault if there is no
   // response message from the server.  Otherwise, the
   // invoke method will return a null.
   public static final boolean FAULT_ON_NO_RESPONSE = false;

   /**
    * Property for setting attachment format.
    */
   public static final String ATTACHMENT_ENCAPSULATION_FORMAT =
           "attachment_encapsulation_format";
   /**
    * Property value for setting attachment format as MIME.
    */
   public static final String ATTACHMENT_ENCAPSULATION_FORMAT_MIME =
           "axis.attachment.style.mime";
   /**
    * Property value for setting attachment format as DIME.
    */
   public static final String ATTACHMENT_ENCAPSULATION_FORMAT_DIME =
           "axis.attachment.style.dime";

   /**
    * A Hashtable mapping protocols (Strings) to Transports (classes)
    */
   private static Hashtable transports = new Hashtable();

   static ParameterMode[] modes = new ParameterMode[]{null,
                                                      ParameterMode.IN,
                                                      ParameterMode.OUT,
                                                      ParameterMode.INOUT};

   /**
    * This is true when someone has called setEncodingStyle()
    */
   private boolean encodingStyleExplicitlySet;
   /**
    * This is true when someone has called setOperationUse()
    */
   private boolean useExplicitlySet;

   /************************************************************************/
   /* Start of core JAX-RPC stuff                                          */
   /************************************************************************/

   /**
    * Default constructor - not much else to say.
    */
   public Call(Service service)
   {
      this.service = service;
      maintainSession = service.getMaintainSession();
      initialize();
   }

   /**
    * Build a call from a URL string
    *
    * @param url the target endpoint URL
    * @throws MalformedURLException
    */
   public Call(String url) throws MalformedURLException
   {
      this(new Service());
      setTargetEndpointAddress(new URL(url));
   }

   /**
    * Build a call from a URL
    *
    * @param url the target endpoint URL
    */
   public Call(URL url)
   {
      this(new Service());
      setTargetEndpointAddress(url);
   }

   ////////////////////////////
   //
   // Properties and the shortcuts for common ones.
   //

   /**
    * Allows you to set a named property to the passed in value.
    * There are a few known properties (like username, password, etc)
    * that are variables in Call.  The rest of the properties are
    * stored in a Hashtable.  These common properties should be
    * accessed via the accessors for speed/type safety, but they may
    * still be obtained via this method.  It's up to one of the
    * Handlers (or the Axis engine itself) to go looking for
    * one of them.
    *
    * @param name  Name of the property
    * @param value Value of the property
    */
   public void setProperty(String name, Object value)
   {
      if (name == null || value == null)
      {
         throw new JAXRPCException(Messages.getMessage(name == null ?
                 "badProp03" : "badProp04"));
      }
      else if (name.equals(USERNAME_PROPERTY))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]{
               name, "java.lang.String", value.getClass().getName()}));
         }
         setUsername((String)value);
      }
      else if (name.equals(PASSWORD_PROPERTY))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]{
               name, "java.lang.String", value.getClass().getName()}));
         }
         setPassword((String)value);
      }
      else if (name.equals(SESSION_MAINTAIN_PROPERTY))
      {
         if (!(value instanceof Boolean))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]
            {name,
             "java.lang.Boolean",
             value.getClass().getName()}));
         }
         setMaintainSession(((Boolean)value).booleanValue());
      }
      else if (name.equals(OPERATION_STYLE_PROPERTY))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]{
               name, "java.lang.String", value.getClass().getName()}));
         }
         setOperationStyle((String)value);
         if (getOperationStyle() == Style.DOCUMENT ||
                 getOperationStyle() == Style.WRAPPED)
         {
            setOperationUse(Use.LITERAL_STR);
         }
         else if (getOperationStyle() == Style.RPC)
         {
            setOperationUse(Use.ENCODED_STR);
         }
      }
      else if (name.equals(SOAPACTION_USE_PROPERTY))
      {
         if (!(value instanceof Boolean))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]
            {name,
             "java.lang.Boolean",
             value.getClass().getName()}));
         }
         setUseSOAPAction(((Boolean)value).booleanValue());
      }
      else if (name.equals(SOAPACTION_URI_PROPERTY))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]
            {name,
             "java.lang.String",
             value.getClass().getName()}));
         }
         setSOAPActionURI((String)value);
      }
      else if (name.equals(ENCODINGSTYLE_URI_PROPERTY))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]
            {name,
             "java.lang.String",
             value.getClass().getName()}));
         }
         setEncodingStyle((String)value);
      }
      else if (name.equals(Stub.ENDPOINT_ADDRESS_PROPERTY))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]
            {name,
             "java.lang.String",
             value.getClass().getName()}));
         }
         setTargetEndpointAddress((String)value);
      }
      else if (name.equals(TRANSPORT_NAME))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]{
               name, "java.lang.String", value.getClass().getName()}));
         }
         transportName = (String)value;
         if (transport != null)
            transport.setTransportName((String)value);
      }
      else if (name.equals(ATTACHMENT_ENCAPSULATION_FORMAT))
      {
         if (!(value instanceof String))
         {
            throw new JAXRPCException(Messages.getMessage("badProp00", new String[]{
               name, "java.lang.String", value.getClass().getName()}));
         }
         if (!value.equals(ATTACHMENT_ENCAPSULATION_FORMAT_MIME) &&
                 !value.equals(ATTACHMENT_ENCAPSULATION_FORMAT_DIME))
            throw new JAXRPCException(Messages.getMessage("badattachmenttypeerr", new String[]{
               (String)value, ATTACHMENT_ENCAPSULATION_FORMAT_MIME + " "
                    + ATTACHMENT_ENCAPSULATION_FORMAT_DIME}));
      }
      else if (name.startsWith("java.") || name.startsWith("javax."))
      {
         throw new JAXRPCException(Messages.getMessage("badProp05", name));
      }
      myProperties.put(name, value);
   } // setProperty

   /**
    * Returns the value associated with the named property
    *
    * @return Object value of the property or null if the property is not set
    * @throws JAXRPCException if the requested property is not a supported property
    */
   public Object getProperty(String name)
   {
      if (Stub.ENDPOINT_ADDRESS_PROPERTY.equals(name))
      {
         return getTargetEndpointAddress();
      }
      else if (!isPropertySupported(name))
      {
         throw new JAXRPCException(name == null ?
                 Messages.getMessage("badProp03") :
                 Messages.getMessage("badProp05", name));
      }
      return myProperties.get(name);
   } // getProperty

   /**
    * Removes (if set) the named property.
    *
    * @param name name of the property to remove
    */
   public void removeProperty(String name)
   {
      if (name == null || !isPropertySupported(name))
      {
         throw new JAXRPCException(name == null ?
                 Messages.getMessage("badProp03") :
                 Messages.getMessage("badProp05", name));
      }
      myProperties.remove(name);
   } // removeProperty

   /**
    * Set a scoped property on the call (i.e. one that propagates down into
    * the runtime).
    * <p/>
    * Deprecated, since setProperty() now does the right thing here.  Expect
    * this to disappear in 1.1.
    *
    * @param name
    * @param value
    * @deprecated
    */
   public void setScopedProperty(String name, Object value)
   {
      if (name == null || value == null)
      {
         throw new JAXRPCException(Messages.getMessage(name == null ?
                 "badProp03" : "badProp04"));
      }
      myProperties.put(name, value);
   } // setScopedProperty

   /**
    * Get a scoped property (i.e. one that propagates down into the runtime).
    * <p/>
    * Deprecated, since there's only one property bag now.  Expect this to
    * disappear in 1.1.
    *
    * @param name
    * @return
    * @deprecated
    */
   public Object getScopedProperty(String name)
   {
      if (name != null)
      {
         return myProperties.get(name);
      }
      return null;
   } // getScopedProperty

   /**
    * Remove a scoped property (i.e. one that propagates down into the
    * runtime).
    * <p/>
    * Deprecated, since there's only one property bag now.  Expect this to
    * disappear in 1.1.
    *
    * @param name
    * @deprecated
    */
   public void removeScopedProperty(String name)
   {
      if (name == null || myProperties == null) return;
      myProperties.remove(name);
   } // removeScopedProperty

   /**
    * Configurable properties supported by this Call object.
    */
   private static ArrayList propertyNames = new ArrayList();

   static
   {
      propertyNames.add(USERNAME_PROPERTY);
      propertyNames.add(PASSWORD_PROPERTY);
      propertyNames.add(SESSION_MAINTAIN_PROPERTY);
      propertyNames.add(ATTACHMENT_ENCAPSULATION_FORMAT);
      propertyNames.add(OPERATION_STYLE_PROPERTY);
      propertyNames.add(SOAPACTION_USE_PROPERTY);
      propertyNames.add(SOAPACTION_URI_PROPERTY);
      propertyNames.add(ENCODINGSTYLE_URI_PROPERTY);
      propertyNames.add(TRANSPORT_NAME);
      propertyNames.add(ATTACHMENT_ENCAPSULATION_FORMAT);
   }

   public Iterator getPropertyNames()
   {
      return propertyNames.iterator();
   }

   public boolean isPropertySupported(String name)
   {
      boolean isSupported = false;
      if (name != null)
      {
         isSupported = propertyNames.contains(name) || (!name.startsWith("java.") && !name.startsWith("javax."));
      }
      return isSupported;
   }

   /**
    * Set the username.
    */
   public void setUsername(String username)
   {
      this.username = username;
   } // setUsername

   /**
    * Get the user name
    */
   public String getUsername()
   {
      return username;
   } // getUsername

   /**
    * Set the password.
    */
   public void setPassword(String password)
   {
      this.password = password;
   } // setPassword

   /**
    * Get the password
    */
   public String getPassword()
   {
      return password;
   } // getPassword

   /**
    * Determine whether we'd like to track sessions or not.  This
    * overrides the default setting from the service.
    * This just passes through the value into the MessageContext.
    * Note: Not part of JAX-RPC specification.
    *
    * @param yesno true if session state is desired, false if not.
    */
   public void setMaintainSession(boolean yesno)
   {
      maintainSession = yesno;
   }

   /**
    * Get the value of maintainSession flag.
    */
   public boolean getMaintainSession()
   {
      return maintainSession;
   }

   /**
    * Set the operation style: "document", "rpc"
    *
    * @param operationStyle string designating style
    */
   public void setOperationStyle(String operationStyle)
   {
      Style style = Style.getStyle(operationStyle, Style.DEFAULT);
      setOperationStyle(style);
   } // setOperationStyle

   /**
    * Set the operation style
    *
    * @param operationStyle
    */
   public void setOperationStyle(Style operationStyle)
   {
      if (operation == null)
      {
         operation = new OperationDesc();
      }

      operation.setStyle(operationStyle);

      // If no one has explicitly set the use, we should track
      // the style.  If it's non-RPC, default to LITERAL.
      if (!useExplicitlySet)
      {
         if (operationStyle != Style.RPC)
         {
            operation.setUse(Use.LITERAL);
         }
      }

      // If no one has explicitly set the encodingStyle, we should
      // track the style.  If it's RPC, default to SOAP-ENC, otherwise
      // default to "".
      if (!encodingStyleExplicitlySet)
      {
         String encStyle = "";
         if (operationStyle == Style.RPC)
         {
            // RPC style defaults to encoded, otherwise default to literal
            encStyle = getMessageContext().getSOAPConstants().getEncodingURI();
         }
         getMessageContext().setEncodingStyle(encStyle);
      }
   }

   /**
    * Get the operation style.
    */
   public Style getOperationStyle()
   {
      if (operation != null)
      {
         return operation.getStyle();
      }
      return Style.DEFAULT;
   } // getOperationStyle

   /**
    * Set the operation use: "literal", "encoded"
    *
    * @param operationUse string designating use
    */
   public void setOperationUse(String operationUse)
   {
      Use use = Use.getUse(operationUse, Use.DEFAULT);
      setOperationUse(use);
   } // setOperationUse

   /**
    * Set the operation use
    *
    * @param operationUse
    */
   public void setOperationUse(Use operationUse)
   {
      useExplicitlySet = true;

      if (operation == null)
      {
         operation = new OperationDesc();
      }

      operation.setUse(operationUse);
      if (!encodingStyleExplicitlySet)
      {
         String encStyle = "";
         if (operationUse == Use.ENCODED)
         {
            // RPC style defaults to encoded, otherwise default to literal
            encStyle = getMessageContext().getSOAPConstants().getEncodingURI();
         }
         getMessageContext().setEncodingStyle(encStyle);
      }
   }

   /**
    * Get the operation use.
    */
   public Use getOperationUse()
   {
      if (operation != null)
      {
         return operation.getUse();
      }
      return Use.DEFAULT;
   } // getOperationStyle

   /**
    * Should soapAction be used?
    */
   public void setUseSOAPAction(boolean useSOAPAction)
   {
      this.useSOAPAction = useSOAPAction;
   } // setUseSOAPAction

   /**
    * Are we using soapAction?
    */
   public boolean useSOAPAction()
   {
      return useSOAPAction;
   } // useSOAPAction

   /**
    * Set the soapAction URI.
    */
   public void setSOAPActionURI(String SOAPActionURI)
           throws IllegalArgumentException
   {
      useSOAPAction = true;
      this.SOAPActionURI = SOAPActionURI;
   } // setSOAPActionURI

   /**
    * Get the soapAction URI.
    */
   public String getSOAPActionURI()
   {
      return SOAPActionURI;
   } // getSOAPActionURI

   /**
    * Sets the encoding style to the URL passed in.
    *
    * @param namespaceURI URI of the encoding to use.
    */
   public void setEncodingStyle(String namespaceURI)
   {
      encodingStyleExplicitlySet = true;
      getMessageContext().setEncodingStyle(namespaceURI);
   }

   /**
    * Returns the encoding style as a URI that should be used for the SOAP
    * message.
    *
    * @return String URI of the encoding style to use
    */
   public String getEncodingStyle()
   {
      return getMessageContext().getEncodingStyle();
   }

   /**
    * Sets the endpoint address of the target service port. This address must
    * correspond to the transport specified in the binding for this Call
    * instance.
    *
    * @param address - Endpoint address of the target service port; specified
    *                as URI
    */
   public void setTargetEndpointAddress(String address)
   {
      URL urlAddress;
      try
      {
         urlAddress = new URL(address);
      }
      catch (MalformedURLException mue)
      {
         throw new JAXRPCException(mue);
      }
      setTargetEndpointAddress(urlAddress);
   }

   /**
    * Sets the URL of the target Web Service.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param address URL of the target Web Service
    */
   public void setTargetEndpointAddress(java.net.URL address)
   {
      try
      {
         if (address == null)
         {
            setTransport(null);
            return;
         }

         String protocol = address.getProtocol();

         // Handle the case where the protocol is the same but we
         // just want to change the URL - if so just set the URL,
         // creating a new Transport object will drop all session
         // data - and we want that stuff to persist between invoke()s.
         // Technically the session data should be in the message
         // context so that it can be persistent across transports
         // as well, but for now the data is in the Transport object.
         ////////////////////////////////////////////////////////////////
         if (this.transport != null)
         {
            String oldAddr = this.transport.getUrl();
            if (oldAddr != null && !oldAddr.equals(""))
            {
               URL tmpURL = new URL(oldAddr);
               String oldProto = tmpURL.getProtocol();
               if (protocol.equals(oldProto))
               {
                  this.transport.setUrl(address.toString());
                  return;
               }
            }
         }

         // Do we already have a transport for this address?
         Transport transport = service.getTransportForURL(address);
         if (transport != null)
         {
            setTransport(transport);
         }
         else
         {
            // We don't already have a transport for this address.  Create one.
            transport = getTransportForProtocol(protocol);
            if (transport == null)
               throw new AxisFault("Call.setTargetEndpointAddress",
                       Messages.getMessage("noTransport01",
                               protocol), null, null);
            transport.setUrl(address.toString());
            setTransport(transport);
            service.registerTransportForURL(address, transport);
         }
      }
      catch (Exception exp)
      {
         log.error(Messages.getMessage("exception00"), exp);
         // do what?
         // throw new AxisFault("Call.setTargetEndpointAddress",
         //"Malformed URL Exception: " + e.getMessage(), null, null);
      }
   }

   /**
    * Returns the URL of the target Web Service.
    *
    * @return URL URL of the target Web Service
    */
   public String getTargetEndpointAddress()
   {
      try
      {
         if (transport == null) return (null);
         return (transport.getUrl());
      }
      catch (Exception exp)
      {
         return (null);
      }
   }

   public Integer getTimeout()
   {
      return timeout;
   }

   public void setTimeout(Integer timeout)
   {
      this.timeout = timeout;
   }
   //
   // end properties code.
   //
   ////////////////////////////

   /**
    * Is the caller required to provide the parameter and return type
    * specification?
    * If true, then
    * addParameter and setReturnType MUST be called to provide the meta data.
    * If false, then
    * addParameter and setReturnType SHOULD NOT be called because the
    * Call object already has the meta data describing the
    * parameters and return type. If addParameter is called, the specified
    * parameter is added to the end of the list of parameters.
    */
   public boolean isParameterAndReturnSpecRequired(QName operationName)
   {
      return parmAndRetReq;
   } // isParameterAndReturnSpecRequired

   /**
    * Adds the specified parameter to the list of parameters for the
    * operation associated with this Call object.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param paramName     Name that will be used for the parameter in the XML
    * @param xmlType       XMLType of the parameter
    * @param parameterMode one of IN, OUT or INOUT
    */
   public void addParameter(QName paramName, QName xmlType,
                            ParameterMode parameterMode)
   {
      Class javaType = null;
      TypeMapping tm = getTypeMapping();
      if (tm != null)
      {
         javaType = tm.getClassForQName(xmlType);
      }
      addParameter(paramName, xmlType, javaType, parameterMode);
   }

   /**
    * Adds the specified parameter to the list of parameters for the
    * operation associated with this Call object.
    * <p/>
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param paramName     Name that will be used for the parameter in the XML
    * @param xmlType       XMLType of the parameter
    * @param javaType      The Java class of the parameter
    * @param parameterMode one of IN, OUT or INOUT
    */
   public void addParameter(QName paramName, QName xmlType,
                            Class javaType, ParameterMode parameterMode,
                            boolean inHeader, boolean outHeader)
   {

      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      // In order to allow any Call to be re-used, Axis
      // chooses to allow parameters to be added when
      // parmAndRetReq==false.  This does not conflict with
      // JSR 101 which indicates an exception MAY be thrown.

      //if (parmAndRetReq) {
      ParameterDesc param = new ParameterDesc();
      param.setQName(paramName);
      param.setTypeQName(xmlType);

      byte mode = ParameterDesc.IN;
      if (parameterMode == ParameterMode.INOUT)
      {
         mode = ParameterDesc.INOUT;
      }
      else if (parameterMode == ParameterMode.OUT)
      {
         mode = ParameterDesc.OUT;
      }
      param.setMode(mode);

      param.setJavaType(javaType);

      param.setInHeader(inHeader);
      param.setOutHeader(outHeader);

      operation.addParameter(param);
      parmAndRetReq = true;
      //}
      //else {
      //throw new JAXRPCException(Messages.getMessage("noParmAndRetReq"));
      //}
   }

   /**
    * Adds the specified parameter to the list of parameters for the
    * operation associated with this Call object.
    * <p/>
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param paramName     Name that will be used for the parameter in the XML
    * @param xmlType       XMLType of the parameter
    * @param javaType      The Java class of the parameter
    * @param parameterMode one of IN, OUT or INOUT
    */
   public void addParameter(QName paramName, QName xmlType,
                            Class javaType, ParameterMode parameterMode)
   {

      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      // In order to allow any Call to be re-used, Axis
      // chooses to allow parameters to be added when
      // parmAndRetReq==false.  This does not conflict with
      // JSR 101 which indicates an exception MAY be thrown.

      //if (parmAndRetReq) {
      ParameterDesc param = new ParameterDesc();
      param.setQName(paramName);
      param.setTypeQName(xmlType);
      param.setJavaType(javaType);
      byte mode = ParameterDesc.IN;
      if (parameterMode == ParameterMode.INOUT)
      {
         mode = ParameterDesc.INOUT;
      }
      else if (parameterMode == ParameterMode.OUT)
      {
         mode = ParameterDesc.OUT;
      }
      param.setMode(mode);

      operation.addParameter(param);
      parmAndRetReq = true;
      //}
      //else {
      //throw new JAXRPCException(Messages.getMessage("noParmAndRetReq"));
      //}
   }

   /**
    * Adds the specified parameter to the list of parameters for the
    * operation associated with this Call object.
    *
    * @param paramName     Name that will be used for the parameter in the XML
    * @param xmlType       XMLType of the parameter
    * @param parameterMode one of IN, OUT or INOUT
    */
   public void addParameter(String paramName, QName xmlType,
                            ParameterMode parameterMode)
   {
      Class javaType = null;
      TypeMapping tm = getTypeMapping();
      if (tm != null)
      {
         javaType = tm.getClassForQName(xmlType);
      }
      addParameter(new QName("", paramName), xmlType,
              javaType, parameterMode);
   }

   /**
    * Adds a parameter type and mode for a specific operation. Note that the
    * client code is not required to call any addParameter and setReturnType
    * methods before calling the invoke method. A Call implementation class
    * can determine the parameter types by using the Java reflection and
    * configured type mapping registry.
    *
    * @param paramName     - Name of the parameter
    * @param xmlType       - XML datatype of the parameter
    * @param javaType      - The Java class of the parameter
    * @param parameterMode - Mode of the parameter-whether IN, OUT or INOUT
    * @throws JAXRPCException - if isParameterAndReturnSpecRequired returns
    *                         false, then addParameter MAY throw
    *                         JAXRPCException....actually Axis allows
    *                         modification in such cases
    */
   public void addParameter(String paramName, QName xmlType,
                            Class javaType, ParameterMode parameterMode)
   {
      addParameter(new QName("", paramName), xmlType,
              javaType, parameterMode);
   }

   /**
    * Adds a parameter type as a soap:header.
    *
    * @param paramName     - Name of the parameter
    * @param xmlType       - XML datatype of the parameter
    * @param javaType      - The Java class of the parameter
    * @param parameterMode - Mode of the parameter-whether IN, OUT or INOUT
    * @param headerMode    - Mode of the header.  Even if this is an INOUT
    *                      parameter, it need not be in the header in both
    *                      directions.
    * @throws JAXRPCException - if isParameterAndReturnSpecRequired returns
    *                         false, then addParameter MAY throw
    *                         JAXRPCException....actually Axis allows
    *                         modification in such cases
    */
   public void addParameterAsHeader(QName paramName, QName xmlType,
                                    Class javaType, ParameterMode parameterMode,
                                    ParameterMode headerMode)
   {
      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      ParameterDesc param = new ParameterDesc();
      param.setQName(paramName);
      param.setTypeQName(xmlType);
      param.setJavaType(javaType);
      if (parameterMode == ParameterMode.IN)
      {
         param.setMode(ParameterDesc.IN);
      }
      else if (parameterMode == ParameterMode.INOUT)
      {
         param.setMode(ParameterDesc.INOUT);
      }
      else if (parameterMode == ParameterMode.OUT)
      {
         param.setMode(ParameterDesc.OUT);
      }
      if (headerMode == ParameterMode.IN)
      {
         param.setInHeader(true);
      }
      else if (headerMode == ParameterMode.INOUT)
      {
         param.setInHeader(true);
         param.setOutHeader(true);
      }
      else if (headerMode == ParameterMode.OUT)
      {
         param.setOutHeader(true);
      }
      operation.addParameter(param);
      parmAndRetReq = true;
   } // addParameterAsHeader

   /**
    * Return the QName of the type of the parameters with the given name.
    *
    * @param paramName name of the parameter to return
    * @return XMLType    XMLType of paramName, or null if not found.
    */
   public QName getParameterTypeByName(String paramName)
   {
      QName paramQName = new QName("", paramName);

      return getParameterTypeByQName(paramQName);
   }

   /**
    * Return the QName of the type of the parameters with the given name.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param paramQName QName of the parameter to return
    * @return XMLType    XMLType of paramQName, or null if not found.
    */
   public QName getParameterTypeByQName(QName paramQName)
   {
      ParameterDesc param = operation.getParamByQName(paramQName);
      if (param != null)
      {
         return param.getTypeQName();
      }
      return (null);
   }

   /**
    * Sets the return type of the operation associated with this Call object.
    *
    * @param returnType QName of the return value type.
    */
   public void setReturnType(QName returnType)
   {
      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      // [TDI 09-Aug-2004] How do we treat anonymous names ???
      while (returnType != null && returnType.getLocalPart().startsWith(">"))
      {
         log.debug("Removing leading '>' from anonymous type" + returnType);
         returnType = new QName(returnType.getNamespaceURI(), returnType.getLocalPart().substring(1));
      }

      // In order to allow any Call to be re-used, Axis
      // chooses to allow setReturnType to be changed when
      // parmAndRetReq==false.  This does not conflict with
      // JSR 101 which indicates an exception MAY be thrown.

      //if (parmAndRetReq) {
      operation.setReturnType(returnType);
      TypeMapping tm = getTypeMapping();
      operation.setReturnClass(tm.getClassForQName(returnType));
      parmAndRetReq = true;
      //}
      //else {
      //throw new JAXRPCException(Messages.getMessage("noParmAndRetReq"));
      //}
   }

   /**
    * Sets the return type for a specific operation.
    *
    * @param xmlType  - QName of the data type of the return value
    * @param javaType - Java class of the return value
    * @throws JAXRPCException - if isParameterAndReturnSpecRequired returns
    *                         false, then setReturnType MAY throw JAXRPCException...Axis allows
    *                         modification without throwing the exception.
    */
   public void setReturnType(QName xmlType, Class javaType)
   {
      setReturnType(xmlType);
      // Use specified type as the operation return
      operation.setReturnClass(javaType);
   }

   /**
    * Set the return type as a header
    */
   public void setReturnTypeAsHeader(QName xmlType)
   {
      setReturnType(xmlType);
      operation.setReturnHeader(true);
   } // setReturnTypeAsHeader

   /**
    * Set the return type as a header
    */
   public void setReturnTypeAsHeader(QName xmlType, Class javaType)
   {
      setReturnType(xmlType, javaType);
      operation.setReturnHeader(true);
   } // setReturnTypeAsHeader

   /**
    * Returns the QName of the type of the return value of this Call - or null
    * if not set.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @return the XMLType specified for this Call (or null).
    */
   public QName getReturnType()
   {
      if (operation != null)
         return operation.getReturnType();

      return null;
   }

   /**
    * Set the QName of the return element
    * <p/>
    * NOT part of JAX-RPC
    */
   public void setReturnQName(QName qname)
   {
      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      operation.setReturnQName(qname);
   }

   /**
    * Sets the desired return Java Class.  This is a convenience method
    * which will cause the Call to automatically convert return values
    * into a desired class if possible.  For instance, we return object
    * arrays by default now for SOAP arrays - you could specify:
    * <p/>
    * setReturnClass(Vector.class)
    * <p/>
    * and you'd get a Vector back from invoke() instead of having to do
    * the conversion yourself.
    * <p/>
    * Note: Not part of JAX-RPC specification.  To be JAX-RPC compliant,
    * use setReturnType(QName, Class).
    *
    * @param cls the desired return class.
    */
   public void setReturnClass(Class cls)
   {
      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      operation.setReturnClass(cls);
      TypeMapping tm = getTypeMapping();
      operation.setReturnType(tm.getTypeQName(cls));
      parmAndRetReq = true;
   }

   /**
    * Clears the list of parameters.
    *
    * @throws JAXRPCException - if isParameterAndReturnSpecRequired returns
    *                         false, then removeAllParameters MAY throw JAXRPCException...Axis allows
    *                         modification to the Call object without throwing an exception.
    */
   public void removeAllParameters()
   {
      //if (parmAndRetReq) {
      operation = new OperationDesc();
      operationSetManually = false;
      parmAndRetReq = true;
      //}
      //else {
      //throw new JAXRPCException(Messages.getMessage("noParmAndRetReq"));
      //}
   }

   /**
    * Returns the operation name associated with this Call object.
    *
    * @return String Name of the operation or null if not set.
    */
   public QName getOperationName()
   {
      return (operationName);
   }

   /**
    * Sets the operation name associated with this Call object.  This will
    * not check the WSDL (if there is WSDL) to make sure that it's a valid
    * operation name.
    *
    * @param opName Name of the operation.
    */
   public void setOperationName(QName opName)
   {
      operationName = opName;
   }

   /**
    * This is a convenience method.  If the user doesn't care about the QName
    * of the operation, the user can call this method, which converts a String
    * operation name to a QName.
    */
   public void setOperationName(String opName)
   {
      operationName = new QName(opName);
   }

   /**
    * Prefill as much info from the WSDL as it can.
    * Right now it's SOAPAction, operation qname, parameter types
    * and return type of the Web Service.
    * <p/>
    * This methods considers that port name and target endpoint address have
    * already been set. This is useful when you want to use the same Call
    * instance for several calls on the same Port
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param opName Operation(method) that's going to be invoked
    */
   public void setOperation(String opName)
   {
      if (service == null)
         throw new JAXRPCException(Messages.getMessage("noService04"));

      // remove all settings concerning an operation
      // leave portName and targetEndPoint as they are
      this.setOperationName(opName);
      this.setEncodingStyle(null);
      this.setReturnType(null);
      this.removeAllParameters();

      javax.wsdl.Service wsdlService = service.getWSDLService();
      // Nothing to do is the WSDL is not already set.
      if (wsdlService == null)
         return;

      Port port = wsdlService.getPort(portName.getLocalPart());
      if (port == null)
         throw new JAXRPCException(Messages.getMessage("noPort00", "" +
                 portName));

      Binding binding = port.getBinding();
      PortType portType = binding.getPortType();
      if (portType == null)
         throw new JAXRPCException(Messages.getMessage("noPortType00", "" +
                 portName));

      List operations = portType.getOperations();
      if (operations == null)
         throw new JAXRPCException(Messages.getMessage("noOperation01",
                 opName));

      Operation wsdlOp = null;
      for (int i = 0; i < operations.size(); i++, wsdlOp = null)
      {
         wsdlOp = (Operation)operations.get(i);
         String wsdlOpName = getWsdlOpName(opName);
         if (wsdlOp.getName().equals(wsdlOpName)) break;
      }
      if (wsdlOp == null)
         throw new UnsupportedOperationException(Messages.getMessage("noOperation01", opName));

      // form now on, the opName is the wsdl op name
      opName = wsdlOp.getName();

      // Get the SOAPAction
      ////////////////////////////////////////////////////////////////////
      List list = port.getExtensibilityElements();
      String opStyle = null;
      BindingOperation bop = binding.getBindingOperation(opName,
              null, null);
      if (bop == null)
         throw new JAXRPCException(Messages.getMessage("noOperation02",
                 opName));
      list = bop.getExtensibilityElements();
      for (int i = 0; list != null && i < list.size(); i++)
      {
         Object obj = list.get(i);
         if (obj instanceof SOAPOperation)
         {
            SOAPOperation sop = (SOAPOperation)obj;
            opStyle = ((SOAPOperation)obj).getStyle();
            String action = sop.getSoapActionURI();
            if (action != null)
            {
               setUseSOAPAction(true);
               setSOAPActionURI(action);
            }
            else
            {
               setUseSOAPAction(false);
               setSOAPActionURI(null);
            }
            break;
         }
      }

      // Get the body's namespace URI and encoding style
      ////////////////////////////////////////////////////////////////////
      BindingInput bIn = bop.getBindingInput();
      if (bIn != null)
      {
         list = bIn.getExtensibilityElements();
         for (int i = 0; list != null && i < list.size(); i++)
         {
            Object obj = list.get(i);
            if (obj instanceof
                    javax.wsdl.extensions.mime.MIMEMultipartRelated)
            {
               javax.wsdl.extensions.mime.MIMEMultipartRelated mpr =
                       (javax.wsdl.extensions.mime.MIMEMultipartRelated)obj;
               Object part = null;
               List l = mpr.getMIMEParts();
               for (int j = 0; l != null && j < l.size() && part == null; j++)
               {
                  javax.wsdl.extensions.mime.MIMEPart mp
                          = (javax.wsdl.extensions.mime.MIMEPart)l.get(j);
                  List ll = mp.getExtensibilityElements();
                  for (int k = 0; ll != null && k < ll.size() && part == null;
                       k++)
                  {
                     part = ll.get(k);
                     if (!(part instanceof SOAPBody)) part = null;
                  }
               }
               if (null != part) obj = part;
            }

            if (obj instanceof SOAPBody)
            {
               SOAPBody sBody = (SOAPBody)obj;
               list = sBody.getEncodingStyles();
               if (list != null && list.size() > 0)
                  this.setEncodingStyle((String)list.get(0));
               String ns = sBody.getNamespaceURI();
               if (ns != null && !ns.equals(""))
                  setOperationName(new QName(ns, opName));
               break;
            }
         }
      }

      Service service = this.getService();
      SymbolTable symbolTable = service.getWSDLParser().getSymbolTable();
      BindingEntry bEntry = symbolTable.getBindingEntry(binding.getQName());
      Parameters parameters = bEntry.getParameters(bop.getOperation());

      // loop over paramters and set up in/out params
      for (int j = 0; j < parameters.list.size(); ++j)
      {
         Parameter p = (Parameter)parameters.list.get(j);
         // Get the QName representing the parameter type
         QName paramType = Utils.getXSIType(p);

         boolean isInHeader = p.isInHeader();
         boolean isOutHeader = p.isOutHeader();
         this.addParameter(p.getQName(), paramType, modes[p.getMode()], isInHeader, isOutHeader);
      }

      Map faultMap = bEntry.getFaults();
      // Get the list of faults for this operation
      ArrayList faults = (ArrayList)faultMap.get(bop);

      // check for no faults
      if (faults == null)
      {
         return;
      }
      // For each fault, register its information
      for (Iterator faultIt = faults.iterator(); faultIt.hasNext();)
      {
         FaultInfo info = (FaultInfo)faultIt.next();
         QName qname = info.getQName();

         // if no parts in fault, skip it!
         if (qname == null)
         {
            continue;
         }
         try
         {
            Class clazz = getTypeMapping().getClassForQName(info.getXMLType());
            addFault(qname, clazz, info.getXMLType(), true);
         }
         catch (Exception e)
         {
            //TODO: ???
         }
      }

      // set output type
      if (parameters.returnParam != null)
      {
         // Get the QName for the return Type
         QName returnType = Utils.getXSIType(parameters.returnParam);
         QName returnQName = parameters.returnParam.getQName();

         // Get the javaType
         String javaType = null;
         if (parameters.returnParam.getMIMEInfo() != null)
         {
            javaType = "javax.activation.DataHandler";
         }
         else
         {
            javaType = parameters.returnParam.getType().getName();
         }
         if (javaType == null)
         {
            javaType = "";
         }
         else
         {
            javaType = javaType + ".class";
         }
         this.setReturnType(returnType);
         try
         {
            this.setReturnClass(ClassUtils.forName(javaType));
         }
         catch (Exception e)
         {
            //TODO: ???
         }
         this.setReturnQName(returnQName);
      }
      else
      {
         this.setReturnType(XMLType.AXIS_VOID);
      }

      boolean hasMIME = Utils.hasMIME(bEntry, bop);
      Use use = bEntry.getInputBodyType(bop.getOperation());
      Style style = Style.getStyle(opStyle, bEntry.getBindingStyle());
      if (use == Use.LITERAL)
      {
         // Turn off encoding
         setEncodingStyle(null);
         // turn off XSI types
         setProperty(Call.SEND_TYPE_ATTR, Boolean.FALSE);
      }
      if (hasMIME || use == Use.LITERAL)
      {
         // If it is literal, turn off multirefs.
         //
         // If there are any MIME types, turn off multirefs.
         // I don't know enough about the guts to know why
         // attachments don't work with multirefs, but they don't.
         setProperty(AxisEngine.PROP_DOMULTIREFS, Boolean.FALSE);
      }

      if (style == Style.DOCUMENT && symbolTable.isWrapped())
      {
         style = Style.WRAPPED;
      }

      // Operation name
      if (style == Style.WRAPPED)
      {
         // We need to make sure the operation name, which is what we
         // wrap the elements in, matches the Qname of the parameter
         // element.
         Map partsMap = bop.getOperation().getInput().getMessage().getParts();
         Part p = (Part)partsMap.values().iterator().next();
         QName q = p.getElementName();
         setOperationName(q);
      }
      else
      {
         QName elementQName =
                 Utils.getOperationQName(bop, bEntry, symbolTable);
         if (elementQName != null)
         {
            setOperationName(elementQName);
         }
      }

      // Indicate that the parameters and return no longer
      // need to be specified with addParameter calls.
      parmAndRetReq = false;
      return;

   }

   private void addParameter(QName paramName, QName xmlType,
                             ParameterMode parameterMode, boolean inHeader, boolean outHeader)
   {
      Class javaType = null;
      TypeMapping tm = getTypeMapping();
      if (tm != null)
      {
         javaType = tm.getClassForQName(xmlType);
      }

      if (parameterMode != ParameterMode.IN && javaType != null && !Holder.class.isAssignableFrom(javaType))
         javaType = JavaUtils.getHolderType(javaType);

      addParameter(paramName, xmlType, javaType, parameterMode, inHeader, outHeader);
   }

   /**
    * The default implementation simply returns the java operation name.
    * A ws4ee implementation would take the jaxrpc-mapping file into consideration
    * and return the corresponding wsdl operation name
    */
   protected String getWsdlOpName(String javaOpName)
   {
      return javaOpName;
   }

   /**
    * prefill as much info from the WSDL as it can.
    * Right now it's target URL, SOAPAction, Parameter types,
    * and return type of the Web Service.
    * <p/>
    * If wsdl is not present, this function set port name and operation name
    * and does not modify target endpoint address.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param portName PortName in the WSDL doc to search for
    * @param opName   Operation(method) that's going to be invoked
    */
   public void setOperation(QName portName, String opName)
   {
      if (service == null)
         throw new JAXRPCException(Messages.getMessage("noService04"));

      // Make sure we're making a fresh start.
      this.setPortName(portName);
      this.setOperationName(opName);
      this.setEncodingStyle(null);
      this.setReturnType(null);
      this.removeAllParameters();

      javax.wsdl.Service wsdlService = service.getWSDLService();
      // Nothing to do is the WSDL is not already set.
      if (wsdlService == null)
         return;

      // we reinitialize target endpoint only if we have wsdl
      this.setTargetEndpointAddress((URL)null);

      Port port = wsdlService.getPort(portName.getLocalPart());
      if (port == null)
         throw new JAXRPCException(Messages.getMessage("noPort00", "" +
                 portName));

      Binding binding = port.getBinding();
      PortType portType = binding.getPortType();
      if (portType == null)
         throw new JAXRPCException(Messages.getMessage("noPortType00", "" +
                 portName));

      // Get the URL
      ////////////////////////////////////////////////////////////////////
      List list = port.getExtensibilityElements();
      for (int i = 0; list != null && i < list.size(); i++)
      {
         Object obj = list.get(i);
         if (obj instanceof SOAPAddress)
         {
            try
            {
               SOAPAddress addr = (SOAPAddress)obj;
               URL url = new URL(addr.getLocationURI());
               this.setTargetEndpointAddress(url);
            }
            catch (Exception exp)
            {
               throw new JAXRPCException(Messages.getMessage("cantSetURI00", "" + exp));
            }
         }
      }

      // Get the SOAPAction
      ////////////////////////////////////////////////////////////////////
      BindingOperation bop = binding.getBindingOperation(opName,
              null, null);
      if (bop == null)
         throw new JAXRPCException(Messages.getMessage("noOperation02",
                 opName));
      list = bop.getExtensibilityElements();
      for (int i = 0; list != null && i < list.size(); i++)
      {
         Object obj = list.get(i);
         if (obj instanceof SOAPOperation)
         {
            SOAPOperation sop = (SOAPOperation)obj;
            String action = sop.getSoapActionURI();
            if (action != null)
            {
               setUseSOAPAction(true);
               setSOAPActionURI(action);
            }
            else
            {
               setUseSOAPAction(false);
               setSOAPActionURI(null);
            }
            break;
         }
      }
      setOperation(opName);
   }

   /**
    * Returns the fully qualified name of the port for this Call object
    * (if there is one).
    *
    * @return QName Fully qualified name of the port (or null if not set)
    */
   public QName getPortName()
   {
      return (portName);
   } // getPortName

   /**
    * Sets the port name of this Call object.  This call will not set
    * any additional fields, nor will it do any checking to verify that
    * this port name is actually defined in the WSDL - for now anyway.
    *
    * @param portName Fully qualified name of the port
    */
   public void setPortName(QName portName)
   {
      this.portName = portName;
   } // setPortName

   /**
    * Returns the fully qualified name of the port for this Call object
    * (if there is one).
    *
    * @return QName Fully qualified name of the port
    * @deprecated This is really the service's port name, not portType name.
    *             Use getPortName instead.
    */
   public QName getPortTypeName()
   {
      return portName == null ? new QName("") : portName;
   }

   /**
    * Sets the port name of this Call object.  This call will not set
    * any additional fields, nor will it do any checking to verify that
    * this port type is actually defined in the WSDL - for now anyway.
    *
    * @param portType Fully qualified name of the portType
    * @deprecated This is really the service's port name, not portType name.
    *             Use setPortName instead.
    */
   public void setPortTypeName(QName portType)
   {
      setPortName(portType);
   }

   /**
    * Allow the user to set the default SOAP version.  For SOAP 1.2, pass
    * SOAPConstants.SOAP12_CONSTANTS.
    *
    * @param soapConstants the SOAPConstants object representing the correct
    *                      version
    */
   public void setSOAPVersion(SOAPConstants soapConstants)
   {
      getMessageContext().setSOAPConstants(soapConstants);
   }

   /**
    * Invokes a specific operation using a synchronous request-response interaction mode. The invoke method takes
    * as parameters the object values corresponding to these defined parameter types. Implementation of the invoke
    * method must check whether the passed parameter values correspond to the number, order and types of parameters
    * specified in the corresponding operation specification.
    *
    * @param operationName - Name of the operation to invoke
    * @param params        - Parameters for this invocation
    * @return the value returned from the other end.
    * @throws java.rmi.RemoteException - if there is any error in the remote method invocation or if the Call
    *                                  object is not configured properly.
    */
   public Object invoke(QName operationName, Object[] params)
           throws java.rmi.RemoteException
   {
      QName origOpName = this.operationName;
      this.operationName = operationName;
      try
      {
         return this.invoke(params);
      }
      catch (AxisFault af)
      {
         this.operationName = origOpName;
         if (af.detail != null && af.detail instanceof RemoteException)
         {
            throw ((RemoteException)af.detail);
         }
         throw af;
      }
      catch (java.rmi.RemoteException re)
      {
         this.operationName = origOpName;
         throw re;
      }
      catch (RuntimeException re)
      {
         this.operationName = origOpName;
         throw re;
      }
      catch (Error e)
      {
         this.operationName = origOpName;
         throw e;
      }
   } // invoke

   /**
    * Invokes the operation associated with this Call object using the
    * passed in parameters as the arguments to the method.
    * <p/>
    * For Messaging (ie. non-RPC) the params argument should be an array
    * of SOAPBodyElements.  <b>All</b> of them need to be SOAPBodyElements,
    * if any of them are not this method will default back to RPC.  In the
    * Messaging case the return value will be a vector of SOAPBodyElements.
    *
    * @param params Array of parameters to invoke the Web Service with
    * @return Object Return value of the operation/method - or null
    * @throws java.rmi.RemoteException if there's an error
    */
   public Object invoke(Object[] params) throws java.rmi.RemoteException
   {
      return invokeInternal(params);
   }

   private Object invokeInternal(Object[] params) throws java.rmi.RemoteException
   {
      long t0 = 0, t1 = 0;
      if (tlog.isDebugEnabled())
      {
         t0 = System.currentTimeMillis();
      }
      /* First see if we're dealing with Messaging instead of RPC.        */
      /* If ALL of the params are SOAPBodyElements then we're doing       */
      /* Messaging, otherwise just fall through to normal RPC processing. */
      /********************************************************************/
      SOAPEnvelopeAxisImpl env = null;

      boolean isSOAPBodyElement = (params != null && params.length == 1 && params[0] instanceof SOAPBodyElementAxisImpl);
      if (isSOAPBodyElement)
      {
         /* ok, we're doing Messaging, so build up the message */
         /******************************************************/
         isMsg = true;

         SOAPBodyElementAxisImpl sbElement = (SOAPBodyElementAxisImpl)params[0];
         if (sbElement.getParentElement() != null && sbElement.getParentElement().getParentElement() != null)
         {
            env = (SOAPEnvelopeAxisImpl)sbElement.getParentElement().getParentElement();
         }
         else
         {
            env = new SOAPEnvelopeAxisImpl(getMessageContext().getSOAPConstants(), getMessageContext().getSchemaVersion());
            env.addBodyElement(sbElement);
         }

         Message msg = new Message(env);
         setRequestMessage(msg);

         invoke();

         msg = getMessageContext().getResponseMessage();
         if (msg == null)
         {
            if (FAULT_ON_NO_RESPONSE)
            {
               throw new AxisFault(Messages.getMessage("nullResponse00"));
            }
            else
            {
               return null;
            }
         }

         env = msg.getSOAPEnvelope();
         Vector bodyElements = env.getBodyElements();
         return (bodyElements.size() > 0 ? bodyElements.get(0) : null);
      }


      if (operationName == null)
         throw new AxisFault(Messages.getMessage("noOperation00"));
      try
      {
         Object res = this.invoke(operationName.getNamespaceURI(),
                 operationName.getLocalPart(), params);
         if (tlog.isDebugEnabled())
         {
            t1 = System.currentTimeMillis();
            tlog.debug("axis.Call.invoke: " + (t1 - t0) + " " + operationName);
         }
         return res;
      }
      catch (AxisFault af)
      {
         if (af.detail != null && af.detail instanceof RemoteException)
         {
            throw ((RemoteException)af.detail);
         }
         throw af;
      }
      catch (RuntimeException ex)
      {
         throw ex;
      }
      catch (Exception exp)
      {
         log.debug(Messages.getMessage("toAxisFault00"), exp);
         throw new AxisFault(Messages.getMessage("errorInvoking00", "\n" + exp));
      }
   }

   /**
    * Invokes the operation associated with this Call object using the passed
    * in parameters as the arguments to the method.  This will return
    * immediately rather than waiting for the server to complete its
    * processing.
    * <p/>
    * NOTE: the return immediately part isn't implemented yet
    *
    * @param params Array of parameters to invoke the Web Service with
    * @throws JAXRPCException is there's an error
    */
   public void invokeOneWay(Object[] params)
   {
      try
      {
         invokeOneWay = true;
         invokeInternal(params);
      }
      catch (Exception exp)
      {
         throw new JAXRPCException(exp.toString());
      }
      finally
      {
         invokeOneWay = false;
      }
   }

   /************************************************************************/
   /* End of core JAX-RPC stuff                                            */
   /************************************************************************/

   /**
    * Invoke the service with a custom SOAPEnvelope.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param env a SOAPEnvelope to send.
    * @throws AxisFault
    */
   public SOAPEnvelopeAxisImpl invoke(SOAPEnvelopeAxisImpl env)
           throws java.rmi.RemoteException
   {
      try
      {
         Message msg = null;

         msg = new Message(env);
         setRequestMessage(msg);
         invoke();
         msg = getMessageContext().getResponseMessage();
         if (msg == null)
         {
            if (FAULT_ON_NO_RESPONSE)
            {
               throw new AxisFault(Messages.getMessage("nullResponse00"));
            }
            else
            {
               return null;
            }
         }
         return (msg.getSOAPEnvelope());
      }
      catch (AxisFault fault)
      {
         throw (AxisFault)fault;
      }
      catch (Exception ex)
      {
         log.error(Messages.getMessage("toAxisFault00"), ex);
         throw AxisFault.makeFault(ex);
      }
   }


   /**
    * Register a Transport that should be used for URLs of the specified
    * protocol.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param protocol       the URL protocol (i.e. "tcp" for "tcp://" urls)
    * @param transportClass the class of a Transport type which will be used
    *                       for matching URLs.
    */
   public static void setTransportForProtocol(String protocol,
                                              Class transportClass)
   {
      if (Transport.class.isAssignableFrom(transportClass))
         transports.put(protocol, transportClass);
      else
         throw new InternalException(transportClass.toString());
   }

   /**
    * Set up the default transport URL mappings.
    * <p/>
    * This must be called BEFORE doing non-standard URL parsing (i.e. if you
    * want the system to accept a "local:" URL).  This is why the Options class
    * calls it before parsing the command-line URL argument.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    */
   public static synchronized void initialize()
   {
      addTransportPackage("transport");

      setTransportForProtocol("java", JavaTransport.class);
      setTransportForProtocol("local", LocalTransport.class);
      setTransportForProtocol("http", HTTPTransport.class);
      setTransportForProtocol("https", HTTPTransport.class);
   }

   /**
    * Cache of transport packages we've already added to the system
    * property.
    */
   private static ArrayList transportPackages = null;

   /**
    * Add a package to the system protocol handler search path.  This
    * enables users to create their own URLStreamHandler classes, and thus
    * allow custom protocols to be used in Axis (typically on the client
    * command line).
    * <p/>
    * For instance, if you add "samples.transport" to the packages property,
    * and have a class samples.transport.tcp.Handler, the system will be able
    * to parse URLs of the form "tcp://host:port..."
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param packageName the package in which to search for protocol names.
    */
   public static synchronized void addTransportPackage(String packageName)
   {
      if (transportPackages == null)
      {
         transportPackages = new ArrayList();
         String currentPackages =
                 AxisProperties.getProperty(TRANSPORT_PROPERTY);
         if (currentPackages != null)
         {
            StringTokenizer tok = new StringTokenizer(currentPackages,
                    "|");
            while (tok.hasMoreTokens())
            {
               transportPackages.add(tok.nextToken());
            }
         }
      }

      if (transportPackages.contains(packageName))
         return;

      transportPackages.add(packageName);

      StringBuffer currentPackages = new StringBuffer();
      for (Iterator i = transportPackages.iterator(); i.hasNext();)
      {
         String thisPackage = (String)i.next();
         currentPackages.append(thisPackage);
         currentPackages.append('|');
      }

      System.setProperty(TRANSPORT_PROPERTY, currentPackages.toString());
   }

   /**
    * Convert the list of objects into RPCParam's based on the paramNames,
    * paramXMLTypes and paramModes variables.  If those aren't set then just
    * return what was passed in.
    *
    * @param params Array of parameters to pass into the operation/method
    * @return Object[] Array of parameters to pass to invoke()
    */
   private Object[] getParamList(Object[] params)
   {
      int numParams = 0;

      // If we never set-up any names... then just return what was passed in
      //////////////////////////////////////////////////////////////////////
      if (log.isDebugEnabled())
      {
         log.debug("operation=" + operation);
         if (operation != null)
            log.debug("operation.getNumParams()=" +
                    operation.getNumParams());
      }
      if (operation == null || operation.getNumParams() == 0)
         return (params);

      // Count the number of IN and INOUT params, this needs to match the
      // number of params passed in - if not throw an error
      /////////////////////////////////////////////////////////////////////
      numParams = operation.getNumInParams();

      // [TDI] Workaround for document style operations that don't have a parameter
      // http://jira.jboss.com/jira/browse/JBWS-70
      if (operation.getStyle() == Style.DOCUMENT && params == null && numParams == 1)
      {
         ParameterDesc param = operation.getParameter(0);
         Class javaType = param.getJavaType();
         if (javaType != null)
         {
            try
            {
               Object obj = javaType.newInstance();
               params = new Object[]{obj};
            }
            catch (Exception e)
            {
               log.warn("Cannot instanciate: " + javaType);
            }
         }
      }

      if (params == null || numParams != params.length)
         throw new JAXRPCException("Parameter count mismatch: " + numParams);

      log.debug("getParamList number of params: " + params.length);

      // All ok - so now produce an array of RPCParams
      //////////////////////////////////////////////////
      Vector result = new Vector();
      int j = 0;
      ArrayList parameters = operation.getParameters();

      for (int i = 0; i < parameters.size(); i++)
      {
         ParameterDesc param = (ParameterDesc)parameters.get(i);
         if (param.getMode() != ParameterDesc.OUT)
         {
            QName paramQName = param.getQName();

            // Create an RPCParam if param isn't already an RPCParam.
            RPCParam rpcParam = null;
            Object p = params[j++];
            if (p instanceof RPCParam)
            {
               rpcParam = (RPCParam)p;
            }
            else
            {
               rpcParam = new RPCParam(paramQName.getNamespaceURI(),
                       paramQName.getLocalPart(),
                       p);
            }
            // Attach the ParameterDescription to the RPCParam
            // so that the serializer can use the (javaType, xmlType)
            // information.
            rpcParam.setParamDesc(param);

            // Add the param to the header or vector depending
            // on whether it belongs in the header or body.
            if (param.isInHeader())
            {
               addHeader(new RPCHeaderParam(rpcParam));
            }
            else
            {
               result.add(rpcParam);
            }
         }
      }
      return (result.toArray());
   }

   /**
    * Set the Transport
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param trans the Transport object we'll use to set up
    *              MessageContext properties.
    */
   public void setTransport(Transport trans)
   {
      transport = trans;
      if (log.isDebugEnabled())
         log.debug(Messages.getMessage("transport00", "" + transport));
   }

   /**
    * Get the Transport registered for the given protocol.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param protocol a protocol such as "http" or "local" which may
    *                 have a Transport object associated with it.
    * @return the Transport registered for this protocol, or null if none.
    */
   public Transport getTransportForProtocol(String protocol)
   {
      Class transportClass = (Class)transports.get(protocol);
      Transport ret = null;
      if (transportClass != null)
      {
         try
         {
            ret = (Transport)transportClass.newInstance();
         }
         catch (InstantiationException e)
         {
         }
         catch (IllegalAccessException e)
         {
         }
      }
      return ret;
   }

   /**
    * Directly set the request message in our MessageContext.
    * <p/>
    * This allows custom message creation.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param msg the new request message.
    */
   public void setRequestMessage(Message msg)
   {
      addAttachmentParts(msg);
      getMessageContext().setRequestMessage(msg);
   }

   /** Stub out to allow ws4ee layer to add attachments to the message
    */
   protected void addAttachmentParts(Message msg)
   {
      String attachformat = (String)getProperty(ATTACHMENT_ENCAPSULATION_FORMAT);
      if (attachformat != null)
      {
         Attachments attachments = msg.getAttachmentsImpl();
         if (attachments != null)
         {
            if (attachformat.equals(ATTACHMENT_ENCAPSULATION_FORMAT_MIME))
               attachments.setSendType(Attachments.SEND_TYPE_MIME);
            else if (attachformat.equals(ATTACHMENT_ENCAPSULATION_FORMAT_DIME))
               attachments.setSendType(Attachments.SEND_TYPE_DIME);
         }
      }

      if (attachmentParts.isEmpty() == false)
      {
         try
         {
            Attachments attachments = msg.getAttachmentsImpl();
            if (attachments != null)
               attachments.setAttachmentParts(attachmentParts);
         }
         catch (AxisFault ex)
         {
            log.info(Messages.getMessage("axisFault00"), ex);
            throw  new RuntimeException(ex.getMessage());
         }
      }
      attachmentParts.clear();
   }

   /**
    * Directly get the response message in our MessageContext.
    * <p/>
    * Shortcut for having to go thru the msgContext
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @return the response Message object in the msgContext
    */
   public Message getResponseMessage()
   {
      return getMessageContext().getResponseMessage();
   }

   /**
    * Obtain a reference to our MessageContext.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @return the MessageContext.
    */
   public MessageContext getMessageContext()
   {
      if (msgContext == null)
         msgContext = new MessageContext(service.getEngine());

      return msgContext;
   }

   /**
    * Add a header which should be inserted into each outgoing message
    * we generate.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param header a SOAPHeaderElement to be inserted into messages
    */
   public void addHeader(SOAPHeaderElementAxisImpl header)
   {
      if (headers == null)
      {
         headers = new Vector();
      }
      headers.add(header);
   }

   /**
    * Clear the list of headers which we insert into each message
    * <p/>
    * Note: Not part of JAX-RPC specification.
    */
   public void clearHeaders()
   {
      headers = null;
   }

   public TypeMapping getTypeMapping()
   {
      // Get the TypeMappingRegistry
      TypeMappingRegistry tmr = getMessageContext().getTypeMappingRegistry();

      // If a TypeMapping is not available, add one.
      return tmr.getOrMakeTypeMapping(getEncodingStyle());
   }

   /**
    * Register type mapping information for serialization/deserialization
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param javaType is  the Java class of the data type.
    * @param xmlType  the xsi:type QName of the associated XML type.
    * @param sf/df    are the factories (or the Class objects of the factory).
    */
   public void registerTypeMapping(Class javaType, QName xmlType,
                                   SerializerFactory sf,
                                   DeserializerFactory df)
   {
      registerTypeMapping(javaType, xmlType, sf, df, true);
   }

   /**
    * Register type mapping information for serialization/deserialization
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param javaType is  the Java class of the data type.
    * @param xmlType  the xsi:type QName of the associated XML type.
    * @param sf/df    are the factories (or the Class objects of the factory).
    * @param force    Indicates whether to add the information if already registered.
    */
   public void registerTypeMapping(Class javaType, QName xmlType,
                                   SerializerFactory sf,
                                   DeserializerFactory df,
                                   boolean force)
   {
      TypeMapping tm = getTypeMapping();
      if (!force && tm.isRegistered(javaType, xmlType))
         return;

      // Register the information
      tm.register(javaType, xmlType, sf, df);
   }

   public void registerTypeMapping(Class javaType, QName xmlType,
                                   Class sfClass, Class dfClass)
   {
      registerTypeMapping(javaType, xmlType, sfClass, dfClass, true);
   }

   public void registerTypeMapping(Class javaType,
                                   QName xmlType,
                                   Class sfClass,
                                   Class dfClass,
                                   boolean force)
   {
      // Instantiate the factory using introspection.
      SerializerFactory sf =
              BaseSerializerFactory.createFactory(sfClass, javaType, xmlType);
      DeserializerFactory df =
              BaseDeserializerFactory.createFactory(dfClass,
                      javaType,
                      xmlType);
      if (sf != null || df != null)
      {
         registerTypeMapping(javaType, xmlType, sf, df, force);
      }
   }

   /************************************************
    * Invocation
    */

   /**
    * Invoke an RPC service with a method name and arguments.
    * <p/>
    * This will call the service, serializing all the arguments, and
    * then deserialize the return value.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param namespace the desired namespace URI of the method element
    * @param method    the method name
    * @param args      an array of Objects representing the arguments to the
    *                  invoked method.  If any of these objects are RPCParams,
    *                  Axis will use the embedded name of the RPCParam as the
    *                  name of the parameter.  Otherwise, we will serialize
    *                  each argument as an XML element called "arg<n>".
    * @return a deserialized Java Object containing the return value
    * @throws AxisFault
    */
   public Object invoke(String namespace, String method, Object[] args)
           throws AxisFault
   {

      if (log.isDebugEnabled())
      {
         log.debug("Enter: Call::invoke(ns, meth, args)");
      }

      /**
       * Since JAX-RPC requires us to specify all or nothing, if setReturnType
       * was called (returnType != null) and we have args but addParameter
       * wasn't called (paramXMLTypes == null), then toss a fault.
       */
      if (getReturnType() != null && args != null && args.length != 0
              && operation.getNumParams() == 0)
      {
         throw new AxisFault(Messages.getMessage("mustSpecifyParms"));
      }

      RPCElement body = new RPCElement(namespace, method, getParamList(args));

      Object ret = invoke(body);

      if (log.isDebugEnabled())
      {
         log.debug("Exit: Call::invoke(ns, meth, args)");
      }

      return ret;
   }

   /**
    * Convenience method to invoke a method with a default (empty)
    * namespace.  Calls invoke() above.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param method the method name
    * @param args   an array of Objects representing the arguments to the
    *               invoked method.  If any of these objects are RPCParams,
    *               Axis will use the embedded name of the RPCParam as the
    *               name of the parameter.  Otherwise, we will serialize
    *               each argument as an XML element called "arg<n>".
    * @return a deserialized Java Object containing the return value
    * @throws AxisFault
    */
   public Object invoke(String method, Object[] args) throws AxisFault
   {
      return invoke("", method, args);
   }

   /**
    * Invoke an RPC service with a pre-constructed RPCElement.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @param body an RPCElement containing all the information about
    *             this call.
    * @return a deserialized Java Object containing the return value
    * @throws AxisFault
    */
   public Object invoke(RPCElement body) throws AxisFault
   {

      if (log.isDebugEnabled())
      {
         log.debug("Enter: Call::invoke(RPCElement)");
      }

      /**
       * Since JAX-RPC requires us to specify a return type if we've set
       * parameter types, check for this case right now and toss a fault
       * if things don't look right.
       */
      if (!invokeOneWay && operation != null &&
              operation.getNumParams() > 0 && getReturnType() == null)
      {
         // TCK:
         // Issue an error if the return type was not set, but continue processing.
         //throw new AxisFault(Messages.getMessage("mustSpecifyReturnType"));
         log.error(Messages.getMessage("mustSpecifyReturnType"));
      }

      SOAPEnvelopeAxisImpl reqEnv =
              new SOAPEnvelopeAxisImpl(getMessageContext().getSOAPConstants(),
                      getMessageContext().getSchemaVersion());
      SOAPEnvelopeAxisImpl resEnv = null;
      Message reqMsg = new Message(reqEnv);
      Message resMsg = null;
      Vector resArgs = null;
      Object result = null;

      try
      {
         // Set the encoding style - in contradiction to BP-1.0/R1005
         // s1as interop tests expect this to be set in the <env:Body>
         // [TDI 11-Sep-2004]
         String encodingStyle = getEncodingStyle();
         String bodyEncStyle = reqEnv.getBody().getEncodingStyle();
         if (bodyEncStyle.equals(encodingStyle) == false)
         {
            if (bodyEncStyle.equals(""))
               reqEnv.getBody().setEncodingStyle(encodingStyle);
            else
               log.warn("Mixed encoding styles are not supported: " + bodyEncStyle + "!=" + encodingStyle);
         }

         setRequestMessage(reqMsg);

         reqEnv.addBodyElement(body);
         reqEnv.setMessageType(Message.REQUEST);

         invoke();
      }
      catch (Exception e)
      {
         log.debug(Messages.getMessage("toAxisFault00"), e);
         throw AxisFault.makeFault(e);
      }

      resMsg = getMessageContext().getResponseMessage();

      if (resMsg == null)
      {
         if (FAULT_ON_NO_RESPONSE)
         {
            throw new AxisFault(Messages.getMessage("nullResponse00"));
         }
         else
         {
            return null;
         }
      }

      resEnv = resMsg.getSOAPEnvelope();
      SOAPBodyElementAxisImpl bodyEl = resEnv.getFirstBody();
      if (bodyEl == null)
      {
         return null;
      }

      if (bodyEl instanceof RPCElement)
      {
         try
         {
            resArgs = ((RPCElement)bodyEl).getParams();
         }
         catch (Exception e)
         {
            log.error(Messages.getMessage("exception00"), e);
            throw AxisFault.makeFault(e);
         }

         if (resArgs != null && resArgs.size() > 0)
         {

            // If there is no return, then we start at index 0 to create the outParams Map.
            // If there IS a return, then we start with 1.
            int outParamStart = 0;

            // If we have resArgs and the returnType is specified, then the first
            // resArgs is the return.  If we have resArgs and neither returnType
            // nor paramXMLTypes are specified, then we assume that the caller is
            // following the non-JAX-RPC AXIS shortcut of not having to specify
            // the return, in which case we again assume the first resArgs is
            // the return.
            // NOTE 1:  the non-JAX-RPC AXIS shortcut allows a potential error
            // to escape notice.  If the caller IS NOT following the non-JAX-RPC
            // shortcut but instead intentionally leaves returnType and params
            // null (ie., a method that takes no parameters and returns nothing)
            // then, if we DO receive something it should be an error, but this
            // code passes it through.  The ideal solution here is to require
            // this caller to set the returnType to void, but there's no void
            // type in XML.
            // NOTE 2:  we should probably verify that the resArgs element
            // types match the expected returnType and paramXMLTypes, but I'm not
            // sure how to do that since the resArgs value is a Java Object
            // and the returnType and paramXMLTypes are QNames.

            // GD 03/15/02 : We're now checking for invalid metadata
            // config at the top of this method, so don't need to do it
            // here.  Check for void return, though.

            boolean findReturnParam = false;
            QName returnParamQName = null;
            if (operation != null)
               returnParamQName = operation.getReturnQName();

            if (!XMLType.AXIS_VOID.equals(getReturnType()))
            {
               if (returnParamQName == null)
               {
                  // Assume the first param is the return
                  RPCParam param = (RPCParam)resArgs.get(0);
                  result = param.getValue();
                  outParamStart = 1;
               }
               else
               {
                  // If the QName of the return value was given to us, look
                  // through the result arguments to find the right name
                  findReturnParam = true;
               }
            }

            // The following loop looks at the resargs and
            // converts the value to the appropriate return/out parameter
            // value.  If the return value is found, is value is
            // placed in result.  The remaining resargs are
            // placed in the outParams list (note that if a resArg
            // is found that does not match a operation parameter qname,
            // it is still placed in the outParms list).
            for (int i = outParamStart; i < resArgs.size(); i++)
            {
               RPCParam param = (RPCParam)resArgs.get(i);

               Class javaType = getJavaTypeForQName(param.getQName());
               Object value = param.getValue();

               // Convert type if needed
               if (javaType != null && value != null &&
                       !javaType.isAssignableFrom(value.getClass()))
               {
                  value = JavaUtils.convert(value, javaType);
               }

               //Convert in to proper array type if elements all of same
               if (javaType == null && value != null)
               {

                  //This is a hack due to the array type of unknown types being Object[]
                  if (value.getClass().isArray())
                  {
                     if (!value.getClass().getComponentType().isPrimitive())
                     {
                        int len = java.lang.reflect.Array.getLength(value);
                        Class type = null;
                        for (int x = 0; x < len; x++)
                        {
                           Object o = java.lang.reflect.Array.get(value, x);
                           if (o != null)
                           {
                              if (type == null)
                              {
                                 //TODO - if this works, need to then add code to handle arrays of arrays
                                 type = o.getClass();
                              }
                              else
                              {
                                 if (!type.getName().equals(o.getClass().getName()))
                                 {
                                    type = null;
                                    break;
                                 }
                              }
                           }
                        }
                        // did we find that all elements were of same type
                        if (type != null)
                        {
                           Object convertedArray = java.lang.reflect.Array.newInstance(type, len);
                           System.arraycopy(value, 0, convertedArray, 0, len);
                           value = convertedArray;
                        }
                     }
                  }
               }

               // Check if this parameter is our return
               // otherwise just add it to our outputs
               if (findReturnParam && returnParamQName.equals(param.getQName()))
               {
                  // found it!
                  result = value;
                  findReturnParam = false;
               }
               else
               {
                  outParams.put(param.getQName(), value);
                  outParamsList.add(value);
               }
            }

            // added by scheu:
            // If the return param is still not found, that means
            // the returned value did not have the expected qname.
            // The soap specification indicates that this should be
            // accepted (and we also fail interop tests if we are strict here).
            // Look through the outParms and find one that
            // does not match one of the operation parameters.
            if (findReturnParam)
            {
               Iterator it = outParams.keySet().iterator();
               while (it.hasNext() && findReturnParam)
               {
                  QName qname = (QName)it.next();
                  ParameterDesc paramDesc = operation.getOutputParamByQName(qname);
                  if (paramDesc == null)
                  {
                     // Doesn't match a paramter, so use this for the return
                     findReturnParam = false;
                     result = outParams.remove(qname);
                  }
               }
            }

            // If we were looking for a particular QName for the return and
            // still didn't find it, throw an exception
            if (findReturnParam)
            {
               String returnParamName = returnParamQName.toString();
               throw new AxisFault(Messages.getMessage("noReturnParam",
                       returnParamName));
            }
         }
      }
      else
      {
         // This is a SOAPBodyElement, try to treat it like a return value
         try
         {
            result = bodyEl.getValueAsType(getReturnType());
         }
         catch (Exception e)
         {
            // just return the SOAPElement
            result = bodyEl;
         }

      }

      if (log.isDebugEnabled())
      {
         log.debug("Exit: Call::invoke(RPCElement)");
      }

      // Convert type if needed
      if (operation != null && operation.getReturnClass() != null)
      {
         result = JavaUtils.convert(result, operation.getReturnClass());
      }

      return (result);
   }

   /**
    * Get the javaType for a given parameter.
    */
   private Class getJavaTypeForQName(QName name)
   {

      if (operation == null) return null;

      Class javaType = null;

      ParameterDesc param = operation.getOutputParamByQName(name);
      if (param != null)
      {
         javaType = param.getJavaType();
         log.debug("getJavaTypeForQName: " + name + " -> " + javaType);
      }
      else
      {
         log.debug("No output parameter for name: " + name);
      }

      return javaType;
   }

   /**
    * Set engine option.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    */
   public void setOption(String name, Object value)
   {
      service.getEngine().setOption(name, value);
   }

   /**
    * Invoke this Call with its established MessageContext
    * (perhaps because you called this.setRequestMessage())
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @throws AxisFault
    */
   public void invoke() throws AxisFault
   {
      if (log.isDebugEnabled())
      {
         log.debug("Enter: Call::invoke()");
      }

      // Clear the output params
      outParams = new HashMap();
      outParamsList = new ArrayList();

      Message reqMsg = null;
      SOAPEnvelopeAxisImpl reqEnv = null;

      getMessageContext().reset();
      getMessageContext().setResponseMessage(null);
      getMessageContext().setProperty(MessageContext.CALL, this);
      getMessageContext().setProperty(WSDL_SERVICE, service);
      getMessageContext().setProperty(WSDL_PORT_NAME, getPortName());
      if (isMsg)
         getMessageContext().setProperty(MessageContext.IS_MSG, "true");

      if (username != null)
      {
         getMessageContext().setUsername(username);
      }
      if (password != null)
      {
         getMessageContext().setPassword(password);
      }
      getMessageContext().setMaintainSession(maintainSession);

      if (operation != null)
      {
         getMessageContext().setOperation(operation);

         operation.setStyle(getOperationStyle());
         operation.setUse(getOperationUse());
      }

      if (useSOAPAction)
      {
         getMessageContext().setUseSOAPAction(true);
      }
      if (SOAPActionURI != null)
      {
         getMessageContext().setSOAPActionURI(SOAPActionURI);
      }
      if (timeout != null)
      {
         getMessageContext().setTimeout(timeout.intValue());
      }

      // Determine client target service
      if (myService != null)
      {
         // If we have a SOAPService kicking around, use that directly
         getMessageContext().setService(myService);
      }
      else
      {
         if (portName != null)
         {
            // No explicit service.  If we have a target service name,
            // try that.
            getMessageContext().setTargetService(portName.getLocalPart());
         }
         else
         {
            // No direct config, so try the namespace of the first body.
            reqMsg = getMessageContext().getRequestMessage();

            if (reqMsg != null)
            {
               reqEnv = reqMsg.getSOAPEnvelope();

               SOAPBodyElementAxisImpl body = reqEnv.getFirstBody();

               if (body != null)
               {
                  if (body.getNamespaceURI() == null)
                  {
                     throw new AxisFault("Call.invoke",
                             Messages.getMessage("cantInvoke00", body.getName()),
                             null, null);
                  }
                  else
                  {
                     getMessageContext().setTargetService(body.getNamespaceURI());
                  }
               }
            }
         }

         SOAPService svc = getMessageContext().getService();
         if (svc != null)
         {
            svc.setPropertyParent(myProperties);
         }
         else
         {
            getMessageContext().setPropertyParent(myProperties);
         }
      }
      if (log.isDebugEnabled())
      {
         log.debug(Messages.getMessage("targetService",
                 getMessageContext().getTargetService()));
      }

      Message requestMessage = getMessageContext().getRequestMessage();
      if (requestMessage != null)
      {
         reqEnv = requestMessage.getSOAPEnvelope();

         // If we have headers to insert, do so now.
         for (int i = 0; headers != null && i < headers.size(); i++)
         {
            SOAPHeaderElementAxisImpl header = (SOAPHeaderElementAxisImpl)headers.get(i);
            reqEnv.addHeader(header);
         }
      }

      // set up transport if there is one
      if (transport != null)
      {
         transport.setupMessageContext(msgContext, this, service.getEngine());
      }
      else
      {
         getMessageContext().setTransportName(transportName);
      }

      if (!invokeOneWay)
      {
         invokeEngine(msgContext);
      }
      else
      {
         invokeEngineOneWay(msgContext);
      }

      if (log.isDebugEnabled())
      {
         log.debug("Exit: Call::invoke()");
      }
   }

   private void invokeEngine(MessageContext msgContext) throws AxisFault
   {
      service.getEngine().invoke(msgContext);

      if (transport != null)
         transport.processReturnedMessageContext(msgContext);

      Message resMsg = getMessageContext().getResponseMessage();

      if (resMsg == null)
      {
         if (FAULT_ON_NO_RESPONSE)
         {
            throw new AxisFault(Messages.getMessage("nullResponse00"));
         }
         else
         {
            return;
         }
      }

      /** This must happen before deserialization...
       */
      resMsg.setMessageType(Message.RESPONSE);

      SOAPEnvelopeAxisImpl resEnv = resMsg.getSOAPEnvelope();

      SOAPBodyElementAxisImpl respBody = resEnv.getFirstBody();
      if (respBody instanceof SOAPFaultImpl)
      {
         if (operation == null ||
                 operation.getReturnClass() == null ||
                 operation.getReturnClass() !=
                 javax.xml.soap.SOAPMessage.class)
            throw ((SOAPFaultImpl)respBody).getFault();
      }
   }

   private void invokeEngineOneWay(final MessageContext msgContext)
   {
      Runnable runnable = new Runnable()
      {
         public void run()
         {
            try
            {
               service.getEngine().invoke(msgContext);
            }
            catch (AxisFault af)
            {
               log.debug(Messages.getMessage("exceptionPrinting"), af);
            }
         }
      };
      Thread thread = new Thread(runnable);
      thread.start();
   }

   /**
    * Get the output parameters (if any) from the last invocation.
    * <p/>
    * NOTE that the params returned are all RPCParams, containing
    * name and value - if you want the value, you'll need to call
    * param.getValue().
    *
    * @return Vector of RPCParams
    */
   public Map getOutputParams()
   {
      if (outParams != null)
      {
         return outParams;
      }
      else
      {
         // per 8.2.4.1 section of JSR 101 (JAXRPC 1.1), version 1.1 spec
         throw new JAXRPCException("Cannot get output parameters before invoke is called.");
      }
   }

   /**
    * Returns a List values for the output parameters of the last
    * invoked operation.
    *
    * @return Values for the output parameters. An empty List is
    *         returned if there are no output values.
    * @throws JAXRPCException - If this method is invoked for a
    *                         one-way operation or is invoked
    *                         before any invoke method has been called.
    */
   public List getOutputValues()
   {
      return outParamsList;
   }

   /**
    * Get the Service object associated with this Call object.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @return Service the Service object this Call object is associated with
    */
   public Service getService()
   {
      return this.service;
   }

   private SOAPService myService = null;

   /**
    *
    */
   public void setSOAPService(SOAPService service)
   {
      myService = service;
      if (service != null)
      {
         // Set the service so that it defers missing property gets to the
         // Call.  So when client-side Handlers get at the MessageContext,
         // the property scoping will be MC -> SOAPService -> Call
         service.setPropertyParent(myProperties);
         service.setEngine(this.service.getAxisClient());
      }
   }

   /**
    * Sets the client-side request and response Handlers.  This is handy
    * for programatically setting up client-side work without deploying
    * via WSDD or the EngineConfiguration mechanism.
    */
   public void setClientHandlers(Handler reqHandler, Handler respHandler)
   {
      // Create a SOAPService which will be used as the client-side service
      // handler.
      setSOAPService(new SOAPService(reqHandler, null, respHandler));
   }

   /**
    * This method adds an attachment.
    * <p/>
    * Note: Not part of JAX-RPC specification.
    *
    * @throws RuntimeException if there is no support for attachments.
    */
   public void addAttachmentPart(Object attachment)
   {
      attachmentParts.add(attachment);
   }

   /**
    * Add a fault for this operation
    * <p/>
    * Note: Not part of JAX-RPC specificaion
    */
   public void addFault(QName qname, Class cls,
                        QName xmlType, boolean isComplex)
   {
      if (operationSetManually)
      {
         throw new RuntimeException(Messages.getMessage("operationAlreadySet"));
      }

      if (operation == null)
         operation = new OperationDesc();

      FaultDesc fault = new FaultDesc();
      fault.setQName(qname);
      fault.setClassName(cls.getName());
      fault.setXmlType(xmlType);
      fault.setComplex(isComplex);
      operation.addFault(fault);
   }

   /**
    * Hand a complete OperationDesc to the Call, and note that this was
    * done so that others don't try to mess with it by calling addParameter,
    * setReturnType, etc.
    *
    * @param operation the OperationDesc to associate with this call.
    */
   public void setOperation(OperationDesc operation)
   {
      this.operation = operation;
      operationSetManually = true;
   }

   public OperationDesc getOperation()
   {
      return operation;
   }

   public void clearOperation()
   {
      operation = null;
      operationSetManually = false;
   }
}