/*
* The Apache Software License, Version 1.1
*
*
* Copyright (c) 2001 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.apache.axis.wsdl.toJava;
import org.apache.axis.enum.Style;
import org.apache.axis.enum.Use;
import org.apache.axis.utils.Messages;
import org.apache.axis.utils.JavaUtils;
import org.apache.axis.wsdl.symbolTable.BindingEntry;
import org.apache.axis.wsdl.symbolTable.CollectionTE;
import org.apache.axis.wsdl.symbolTable.Element;
import org.apache.axis.wsdl.symbolTable.FaultInfo;
import org.apache.axis.wsdl.symbolTable.Parameter;
import org.apache.axis.wsdl.symbolTable.Parameters;
import org.apache.axis.wsdl.symbolTable.SymbolTable;
import org.apache.axis.wsdl.symbolTable.TypeEntry;
import org.apache.axis.wsdl.symbolTable.MimeInfo;
import org.apache.axis.Constants;
import javax.wsdl.Binding;
import javax.wsdl.BindingOperation;
import javax.wsdl.Fault;
import javax.wsdl.Message;
import javax.wsdl.Operation;
import javax.wsdl.OperationType;
import javax.wsdl.Part;
import javax.wsdl.PortType;
import javax.wsdl.extensions.soap.SOAPOperation;
import javax.wsdl.extensions.soap.SOAPBinding;
import javax.wsdl.extensions.UnknownExtensibilityElement;
import javax.xml.namespace.QName;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Vector;
/**
* This is Wsdl2java's stub writer. It writes the <BindingName>Stub.java
* file which contains the <bindingName>Stub class.
*/
public class JavaStubWriter extends JavaClassWriter {
private BindingEntry bEntry;
private Binding binding;
private SymbolTable symbolTable;
static String [] modeStrings = new String [] { "",
"org.apache.axis.description.ParameterDesc.IN",
"org.apache.axis.description.ParameterDesc.OUT",
"org.apache.axis.description.ParameterDesc.INOUT" };
static HashMap styles = new HashMap();
static HashMap uses = new HashMap();
static {
styles.put(Style.DOCUMENT, "org.apache.axis.enum.Style.DOCUMENT");
styles.put(Style.RPC, "org.apache.axis.enum.Style.RPC");
styles.put(Style.MESSAGE, "org.apache.axis.enum.Style.MESSAGE");
styles.put(Style.WRAPPED, "org.apache.axis.enum.Style.WRAPPED");
uses.put(Use.ENCODED, "org.apache.axis.enum.Use.ENCODED");
uses.put(Use.LITERAL, "org.apache.axis.enum.Use.LITERAL");
}
/**
* Constructor.
*/
protected JavaStubWriter(
Emitter emitter,
BindingEntry bEntry,
SymbolTable symbolTable) {
super(emitter, bEntry.getName() + "Stub", "stub");
this.bEntry = bEntry;
this.binding = bEntry.getBinding();
this.symbolTable = symbolTable;
} // ctor
/**
* Returns "extends org.apache.axis.client.Stub ".
*/
protected String getExtendsText() {
return "extends org.apache.axis.client.Stub ";
} // getExtendsText
/**
* Returns "implements <SEI> ".
*/
protected String getImplementsText() {
return "implements " + bEntry.getDynamicVar(JavaBindingWriter.INTERFACE_NAME) + " ";
} // getImplementsText
/**
* Write the body of the binding's stub file.
*/
protected void writeFileBody(PrintWriter pw) throws IOException {
PortType portType = binding.getPortType();
HashSet types = getTypesInPortType(portType);
boolean hasMIME = Utils.hasMIME(bEntry);
if (types.size() > 0 || hasMIME) {
pw.println(" private java.util.Vector cachedSerClasses = new java.util.Vector();");
pw.println(" private java.util.Vector cachedSerQNames = new java.util.Vector();");
pw.println(" private java.util.Vector cachedSerFactories = new java.util.Vector();");
pw.println(" private java.util.Vector cachedDeserFactories = new java.util.Vector();");
}
pw.println();
pw.println(" static org.apache.axis.description.OperationDesc [] _operations;");
pw.println();
writeOperationMap(pw);
pw.println();
pw.println(" public " + className + "() throws org.apache.axis.AxisFault {");
pw.println(" this(null);");
pw.println(" }");
pw.println();
pw.println(" public " + className + "(java.net.URL endpointURL, javax.xml.rpc.Service service) throws org.apache.axis.AxisFault {");
pw.println(" this(service);");
pw.println(" super.cachedEndpoint = endpointURL;");
pw.println(" }");
pw.println();
pw.println(" public " + className + "(javax.xml.rpc.Service service) throws org.apache.axis.AxisFault {");
pw.println(" if (service == null) {");
pw.println(" super.service = new org.apache.axis.client.Service();");
pw.println(" } else {");
pw.println(" super.service = service;");
pw.println(" }");
// keep track of how many type mappings we write out
int typeMappingCount = 0;
if (types.size() > 0) {
Iterator it = types.iterator();
while (it.hasNext()) {
TypeEntry type = (TypeEntry) it.next();
// Note this same check is repeated in JavaDeployWriter.
// 1) Don't register types that are base (primitive) types.
// If the baseType != null && getRefType() != null this
// is a simpleType that must be registered.
// 2) Don't register the special types for collections
// (indexed properties) or elements
// 3) Don't register types that are not referenced
// or only referenced in a literal context.
if ((type.getBaseType() != null && type.getRefType() == null) ||
type instanceof CollectionTE ||
type instanceof Element ||
!type.isReferenced() ||
type.isOnlyLiteralReferenced()) {
continue;
}
// Write out serializer declarations
if (typeMappingCount == 0) {
writeSerializationDecls(pw, hasMIME, binding.getQName().getNamespaceURI());
}
// write the type mapping for this type
writeSerializationInit(pw, type);
// increase the number of type mappings count
typeMappingCount++;
}
}
// We need to write out the MIME mapping, even if we don't have
// any type mappings
if (typeMappingCount == 0 && hasMIME) {
writeSerializationDecls(pw, hasMIME, binding.getQName().getNamespaceURI());
typeMappingCount++;
}
pw.println(" }");
pw.println();
pw.println(" private org.apache.axis.client.Call createCall() throws java.rmi.RemoteException {");
pw.println(" try {");
pw.println(" org.apache.axis.client.Call _call =");
pw.println(" (org.apache.axis.client.Call) super.service.createCall();");
pw.println(" if (super.maintainSessionSet) {");
pw.println(" _call.setMaintainSession(super.maintainSession);");
pw.println(" }");
pw.println(" if (super.cachedUsername != null) {");
pw.println(" _call.setUsername(super.cachedUsername);");
pw.println(" }");
pw.println(" if (super.cachedPassword != null) {");
pw.println(" _call.setPassword(super.cachedPassword);");
pw.println(" }");
pw.println(" if (super.cachedEndpoint != null) {");
pw.println(" _call.setTargetEndpointAddress(super.cachedEndpoint);");
pw.println(" }");
pw.println(" if (super.cachedTimeout != null) {");
pw.println(" _call.setTimeout(super.cachedTimeout);");
pw.println(" }");
pw.println(" if (super.cachedPortName != null) {");
pw.println(" _call.setPortName(super.cachedPortName);");
pw.println(" }");
pw.println(" java.util.Enumeration keys = super.cachedProperties.keys();");
pw.println(" while (keys.hasMoreElements()) {");
pw.println(" java.lang.String key = (java.lang.String) keys.nextElement();");
pw.println(" _call.setProperty(key, super.cachedProperties.get(key));");
pw.println(" }");
if (typeMappingCount > 0) {
pw.println(" // " + Messages.getMessage("typeMap00"));
pw.println(" // " + Messages.getMessage("typeMap01"));
pw.println(" // " + Messages.getMessage("typeMap02"));
pw.println(" // " + Messages.getMessage("typeMap03"));
pw.println(" // " + Messages.getMessage("typeMap04"));
pw.println(" synchronized (this) {");
pw.println(" if (firstCall()) {");
// Hack alert - we need to establish the encoding style before we register type mappings due
// to the fact that TypeMappings key off of encoding style
pw.println(" // "
+ Messages.getMessage("mustSetStyle"));
if (bEntry.hasLiteral()) {
pw.println(" _call.setEncodingStyle(null);");
} else {
Iterator iterator = bEntry.getBinding().getExtensibilityElements().iterator();
while (iterator.hasNext()) {
Object obj = iterator.next();
if (obj instanceof SOAPBinding) {
pw.println(" _call.setSOAPVersion(org.apache.axis.soap.SOAPConstants.SOAP11_CONSTANTS);");
pw.println(" _call.setEncodingStyle(org.apache.axis.Constants.URI_SOAP11_ENC);");
} else if (obj instanceof UnknownExtensibilityElement) {
//TODO: After WSDL4J supports soap12, change this code
UnknownExtensibilityElement unkElement = (UnknownExtensibilityElement) obj;
QName name = unkElement.getElementType();
if(name.getNamespaceURI().equals(Constants.URI_WSDL12_SOAP) &&
name.getLocalPart().equals("binding")){
pw.println(" _call.setSOAPVersion(org.apache.axis.soap.SOAPConstants.SOAP12_CONSTANTS);");
pw.println(" _call.setEncodingStyle(org.apache.axis.Constants.URI_SOAP12_ENC);");
}
}
}
}
pw.println(" for (int i = 0; i < cachedSerFactories.size(); ++i) {");
pw.println(" java.lang.Class cls = (java.lang.Class) cachedSerClasses.get(i);");
pw.println(" javax.xml.namespace.QName qName =");
pw.println(" (javax.xml.namespace.QName) cachedSerQNames.get(i);");
pw.println(" java.lang.Class sf = (java.lang.Class)");
pw.println(" cachedSerFactories.get(i);");
pw.println(" java.lang.Class df = (java.lang.Class)");
pw.println(" cachedDeserFactories.get(i);");
pw.println(" _call.registerTypeMapping(cls, qName, sf, df, false);");
pw.println(" }");
pw.println(" }");
pw.println(" }");
}
pw.println(" return _call;");
pw.println(" }");
pw.println(" catch (java.lang.Throwable t) {");
pw.println(" throw new org.apache.axis.AxisFault(\""
+ Messages.getMessage("badCall01") + "\", t);");
pw.println(" }");
pw.println(" }");
pw.println();
List operations = binding.getBindingOperations();
for (int i = 0; i < operations.size(); ++i) {
BindingOperation operation = (BindingOperation) operations.get(i);
Parameters parameters =
bEntry.getParameters(operation.getOperation());
// Get the soapAction from the <soap:operation>
String soapAction = "";
String opStyle = null;
Iterator operationExtensibilityIterator = operation.getExtensibilityElements().iterator();
for (; operationExtensibilityIterator.hasNext();) {
Object obj = operationExtensibilityIterator.next();
if (obj instanceof SOAPOperation) {
soapAction = ((SOAPOperation) obj).getSoapActionURI();
opStyle = ((SOAPOperation) obj).getStyle();
break;
} else if (obj instanceof UnknownExtensibilityElement) {
//TODO: After WSDL4J supports soap12, change this code
UnknownExtensibilityElement unkElement = (UnknownExtensibilityElement) obj;
QName name = unkElement.getElementType();
if(name.getNamespaceURI().equals(Constants.URI_WSDL12_SOAP) &&
name.getLocalPart().equals("operation")){
if (unkElement.getElement().getAttribute("soapAction") != null) {
soapAction = unkElement.getElement().getAttribute("soapAction");
}
opStyle = unkElement.getElement().getAttribute("style");
}
}
}
Operation ptOperation = operation.getOperation();
OperationType type = ptOperation.getStyle();
// These operation types are not supported. The signature
// will be a string stating that fact.
if (type == OperationType.NOTIFICATION
|| type == OperationType.SOLICIT_RESPONSE) {
pw.println(parameters.signature);
pw.println();
}
else {
writeOperation(pw, operation, parameters, soapAction, opStyle,
type == OperationType.ONE_WAY, i);
}
}
} // writeFileBody
private void writeOperationMap(PrintWriter pw) {
List operations = binding.getBindingOperations();
pw.println(" static {");
pw.println(" _operations = new org.apache.axis.description.OperationDesc[" +
operations.size() + "];");
pw.println(" org.apache.axis.description.OperationDesc oper;");
for (int i = 0; i < operations.size(); ++i) {
BindingOperation operation = (BindingOperation) operations.get(i);
Parameters parameters =
bEntry.getParameters(operation.getOperation());
// Get the soapAction from the <soap:operation>
String opStyle = null;
Iterator operationExtensibilityIterator = operation.getExtensibilityElements().iterator();
for (; operationExtensibilityIterator.hasNext();) {
Object obj = operationExtensibilityIterator.next();
if (obj instanceof SOAPOperation) {
opStyle = ((SOAPOperation) obj).getStyle();
break;
} else if (obj instanceof UnknownExtensibilityElement) {
//TODO: After WSDL4J supports soap12, change this code
UnknownExtensibilityElement unkElement = (UnknownExtensibilityElement) obj;
QName name = unkElement.getElementType();
if(name.getNamespaceURI().equals(Constants.URI_WSDL12_SOAP) &&
name.getLocalPart().equals("operation")){
opStyle = unkElement.getElement().getAttribute("style");
}
}
}
Operation ptOperation = operation.getOperation();
OperationType type = ptOperation.getStyle();
// These operation types are not supported. The signature
// will be a string stating that fact.
if (type == OperationType.NOTIFICATION
|| type == OperationType.SOLICIT_RESPONSE) {
pw.println(parameters.signature);
pw.println();
}
String operName = operation.getName();
String indent = " ";
pw.println(indent + "oper = new org.apache.axis.description.OperationDesc();");
pw.println(indent + "oper.setName(\"" + operName + "\");");
// 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);
// Set the javaType to the name of the type
String javaType = null;
if (p.getMIMEInfo() != null) {
MimeInfo mimeInfo = p.getMIMEInfo();
javaType = JavaUtils.mimeToJava(mimeInfo.getType()) + mimeInfo.getDimensions() + ".class, ";
}
else {
javaType = p.getType().getName();
if (javaType != null) {
javaType += ".class, ";
} else {
javaType = "null, ";
}
}
// Get the text representing newing a QName for the name and type
String paramNameText = Utils.getNewQName(p.getQName());
String paramTypeText = Utils.getNewQName(paramType);
// Generate the addParameter call with the
// name qname, typeQName, optional javaType, and mode
boolean isInHeader = p.isInHeader();
boolean isOutHeader = p.isOutHeader();
pw.println(" oper.addParameter(" + paramNameText
+ ", " + paramTypeText + ", "
+ javaType + modeStrings[p.getMode()]
+ ", " + isInHeader + ", " + isOutHeader + ");");
}
// set output type
if (parameters.returnParam != null) {
// Get the QName for the return Type
QName returnType = Utils.getXSIType(parameters.returnParam);
// Get the javaType
String javaType = null;
if (parameters.returnParam.getMIMEInfo() != null) {
MimeInfo mimeInfo = parameters.returnParam.getMIMEInfo();
javaType = JavaUtils.mimeToJava(mimeInfo.getType()) + mimeInfo.getDimensions();
}
else {
javaType = parameters.returnParam.getType().getName();
}
if (javaType == null) {
javaType = "";
}
else {
javaType = javaType + ".class";
}
pw.println(" oper.setReturnType(" +
Utils.getNewQName(returnType) + ");");
pw.println(" oper.setReturnClass("
+ javaType + ");");
QName returnQName = parameters.returnParam.getQName();
if (returnQName != null) {
pw.println(" oper.setReturnQName(" +
Utils.getNewQName(returnQName) + ");");
}
if (parameters.returnParam.isOutHeader()) {
pw.println(" oper.setReturnHeader(true);");
}
}
else {
pw.println(" oper.setReturnType(org.apache.axis.encoding.XMLType.AXIS_VOID);");
}
boolean hasMIME = Utils.hasMIME(bEntry, operation);
Style style = Style.getStyle(opStyle, bEntry.getBindingStyle());
Use use = bEntry.getInputBodyType(operation.getOperation());
if (style == Style.DOCUMENT && symbolTable.isWrapped()) {
style = Style.WRAPPED;
}
if (!hasMIME) {
pw.println(" oper.setStyle(" + styles.get(style) + ");");
pw.println(" oper.setUse(" + uses.get(use) + ");");
}
// Register fault/exception information for this operation
writeFaultInfo(pw, operation);
pw.println(indent + "_operations[" + i + "] = oper;");
pw.println("");
}
pw.println(" }");
}
/**
* This method returns a set of all the TypeEntry in a given PortType.
* The elements of the returned HashSet are Types.
*/
private HashSet getTypesInPortType(PortType portType) {
HashSet types = new HashSet();
HashSet firstPassTypes = new HashSet();
// Get all the types from all the operations
List operations = portType.getOperations();
for (int i = 0; i < operations.size(); ++i) {
Operation op = (Operation) operations.get(i);
firstPassTypes.addAll(getTypesInOperation(op));
}
// Add all the types nested and derived from the types
// in the first pass.
Iterator i = firstPassTypes.iterator();
while (i.hasNext()) {
TypeEntry type = (TypeEntry) i.next();
if (!types.contains(type)) {
types.add(type);
types.addAll(
Utils.getNestedTypes(type, symbolTable, true));
}
}
return types;
} // getTypesInPortType
/**
* This method returns a set of all the TypeEntry in a given Operation.
* The elements of the returned HashSet are TypeEntry.
*/
private HashSet getTypesInOperation(Operation operation) {
HashSet types = new HashSet();
Vector v = new Vector();
Parameters params = bEntry.getParameters(operation);
// Loop over parameter types for this operation
for (int i=0; i < params.list.size(); i++) {
Parameter p = (Parameter) params.list.get(i);
v.add(p.getType());
}
// Add the return type
if (params.returnParam != null)
v.add(params.returnParam.getType());
// Collect all the types in faults
Map faults = operation.getFaults();
if (faults != null) {
Iterator i = faults.values().iterator();
while (i.hasNext()) {
Fault f = (Fault) i.next();
partTypes(v,
f.getMessage().getOrderedParts(null));
}
}
// Put all these types into a set. This operation eliminates all duplicates.
for (int i = 0; i < v.size(); i++)
types.add(v.get(i));
return types;
} // getTypesInOperation
/**
* This method returns a vector of TypeEntry for the parts.
*/
private void partTypes(Vector v, Collection parts) {
Iterator i = parts.iterator();
while (i.hasNext()) {
Part part = (Part) i.next();
QName qType = part.getTypeName();
if (qType != null) {
v.add(symbolTable.getType(qType));
} else {
qType = part.getElementName();
if (qType != null) {
v.add(symbolTable.getElement(qType));
}
}
} // while
} // partTypes
/**
* This function writes the regsiterFaultInfo API calls
*/
private void writeFaultInfo(PrintWriter pw, BindingOperation bindOp) {
Map faultMap = bEntry.getFaults();
// Get the list of faults for this operation
ArrayList faults = (ArrayList) faultMap.get(bindOp);
// 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();
Message message = info.getMessage();
// if no parts in fault, skip it!
if (qname == null) {
continue;
}
// Get the Exception class name
String className = Utils.getFullExceptionName(message, symbolTable);
// output the registration API call
pw.println(" oper.addFault(new org.apache.axis.description.FaultDesc(");
pw.println(" " + Utils.getNewQName(qname) + ",");
pw.println(" \"" + className + "\",");
pw.println(" " + Utils.getNewQName(info.getXMLType()) + ", ");
pw.println(" " + Utils.isFaultComplex(message, symbolTable));
pw.println(" ));");
}
}
/**
* In the stub constructor, write the serializer code for the complex types.
*/
private void writeSerializationDecls(PrintWriter pw, boolean hasMIME,
String namespace) {
pw.println(" java.lang.Class cls;" );
pw.println(" javax.xml.namespace.QName qName;" );
pw.println(" java.lang.Class beansf = org.apache.axis.encoding.ser.BeanSerializerFactory.class;");
pw.println(" java.lang.Class beandf = org.apache.axis.encoding.ser.BeanDeserializerFactory.class;");
pw.println(" java.lang.Class enumsf = org.apache.axis.encoding.ser.EnumSerializerFactory.class;");
pw.println(" java.lang.Class enumdf = org.apache.axis.encoding.ser.EnumDeserializerFactory.class;");
pw.println(" java.lang.Class arraysf = org.apache.axis.encoding.ser.ArraySerializerFactory.class;");
pw.println(" java.lang.Class arraydf = org.apache.axis.encoding.ser.ArrayDeserializerFactory.class;");
pw.println(" java.lang.Class simplesf = org.apache.axis.encoding.ser.SimpleSerializerFactory.class;");
pw.println(" java.lang.Class simpledf = org.apache.axis.encoding.ser.SimpleDeserializerFactory.class;");
if (hasMIME) {
pw.println(" java.lang.Class mimesf = org.apache.axis.encoding.ser.JAFDataHandlerSerializerFactory.class;");
pw.println(" java.lang.Class mimedf = org.apache.axis.encoding.ser.JAFDataHandlerDeserializerFactory.class;");
pw.println();
QName qname = new QName(namespace, "DataHandler");
pw.println(" qName = new javax.xml.namespace.QName(\""
+ qname.getNamespaceURI() + "\", \"" + qname.getLocalPart()
+ "\");");
pw.println(" cachedSerQNames.add(qName);");
pw.println(" cls = javax.activation.DataHandler.class;");
pw.println(" cachedSerClasses.add(cls);");
pw.println(" cachedSerFactories.add(mimesf);");
pw.println(" cachedDeserFactories.add(mimedf);");
pw.println();
}
} // writeSerializationDecls
private void writeSerializationInit(PrintWriter pw, TypeEntry type) {
QName qname = type.getQName();
pw.println(" qName = new javax.xml.namespace.QName(\""
+ qname.getNamespaceURI() + "\", \"" + qname.getLocalPart()
+ "\");");
pw.println(" cachedSerQNames.add(qName);");
pw.println(" cls = " + type.getName() + ".class;");
pw.println(" cachedSerClasses.add(cls);");
if (type.getName().endsWith("[]")) {
pw.println(" cachedSerFactories.add(arraysf);");
pw.println(" cachedDeserFactories.add(arraydf);");
} else if (type.getNode() != null &&
Utils.getEnumerationBaseAndValues(
type.getNode(), symbolTable) != null) {
pw.println(" cachedSerFactories.add(enumsf);");
pw.println(" cachedDeserFactories.add(enumdf);");
} else if (type.isSimpleType()) {
pw.println(" cachedSerFactories.add(simplesf);");
pw.println(" cachedDeserFactories.add(simpledf);");
} else if (type.getBaseType() != null) {
// serializers are not required for types derived from base types
// java type to qname mapping is anyway established by default
// note that we have to add null to the serfactories vector to
// keep the order of other entries, this is not going to screw
// up because if type mapping returns null for a serialization
// factory, it is assumed to be not-defined and the delegate
// will be checked, the end delegate is DefaultTypeMappingImpl
// that'll get it right with the base type name
pw.println(" cachedSerFactories.add(null);");
pw.println(" cachedDeserFactories.add(simpledf);");
} else {
pw.println(" cachedSerFactories.add(beansf);");
pw.println(" cachedDeserFactories.add(beandf);");
}
pw.println();
} // writeSerializationInit
/**
* Write the stub code for the given operation.
*/
private void writeOperation(
PrintWriter pw,
BindingOperation operation,
Parameters parms,
String soapAction,
String opStyle,
boolean oneway,
int opIndex) {
writeComment(pw, operation.getDocumentationElement());
pw.println(parms.signature + " {");
pw.println(" if (super.cachedEndpoint == null) {");
pw.println(" throw new org.apache.axis.NoEndPointException();");
pw.println(" }");
pw.println(" org.apache.axis.client.Call _call = createCall();");
pw.println(" _call.setOperation(_operations[" + opIndex + "]);");
// SoapAction
if (soapAction != null) {
pw.println(" _call.setUseSOAPAction(true);");
pw.println(" _call.setSOAPActionURI(\"" + soapAction + "\");");
}
boolean hasMIME = Utils.hasMIME(bEntry, operation);
// Encoding: literal or encoded use.
Use use = bEntry.getInputBodyType(operation.getOperation());
if (use == Use.LITERAL) {
// Turn off encoding
pw.println(" _call.setEncodingStyle(null);");
// turn off XSI types
pw.println(" _call.setProperty(org.apache.axis.client.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.
pw.println(" _call.setProperty(org.apache.axis.AxisEngine.PROP_DOMULTIREFS, Boolean.FALSE);");
}
Style style = Style.getStyle(opStyle, bEntry.getBindingStyle());
if (style == Style.DOCUMENT && symbolTable.isWrapped()) {
style = Style.WRAPPED;
}
Iterator iterator = bEntry.getBinding().getExtensibilityElements().iterator();
while (iterator.hasNext()) {
Object obj = iterator.next();
if (obj instanceof SOAPBinding) {
pw.println(" _call.setSOAPVersion(org.apache.axis.soap.SOAPConstants.SOAP11_CONSTANTS);");
} else if (obj instanceof UnknownExtensibilityElement) {
//TODO: After WSDL4J supports soap12, change this code
UnknownExtensibilityElement unkElement = (UnknownExtensibilityElement) obj;
QName name = unkElement.getElementType();
if(name.getNamespaceURI().equals(Constants.URI_WSDL12_SOAP) &&
name.getLocalPart().equals("binding")){
pw.println(" _call.setSOAPVersion(org.apache.axis.soap.SOAPConstants.SOAP12_CONSTANTS);");
}
}
}
// 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 = operation.getOperation().getInput().getMessage().getParts();
Part p = (Part)partsMap.values().iterator().next();
QName q = p.getElementName();
pw.println(" _call.setOperationName(" + Utils.getNewQName(q) + ");" );
} else {
QName elementQName =
Utils.getOperationQName(operation, bEntry, symbolTable);
if (elementQName != null) {
pw.println(" _call.setOperationName(" +
Utils.getNewQName(elementQName) + ");" );
}
}
pw.println();
// Set the headers
pw.println(" setRequestHeaders(_call);");
// Set the attachments
pw.println(" setAttachments(_call);");
// Set DIME flag if needed
if(bEntry.isOperationDIME(operation.getOperation().getName())) {
pw.println(" _call.setProperty(_call.ATTACHMENT_ENCAPSULATION_FORMAT, _call.ATTACHMENT_ENCAPSULATION_FORMAT_DIME);");
}
// Invoke the operation
if (oneway) {
pw.print(" _call.invokeOneWay(");
}
else {
pw.print(" java.lang.Object _resp = _call.invoke(");
}
pw.print("new java.lang.Object[] {");
writeParameters(pw, parms);
pw.println("});");
pw.println();
if (!oneway) {
writeResponseHandling(pw, parms);
}
pw.println(" }");
pw.println();
} // writeOperation
private void writeParameters(PrintWriter pw, Parameters parms) {
// Write the input and inout parameter list
boolean needComma = false;
for (int i = 0; i < parms.list.size(); ++i) {
Parameter p = (Parameter) parms.list.get(i);
if (p.getMode() != Parameter.OUT) {
if (needComma) {
pw.print(", ");
}
else {
needComma = true;
}
String javifiedName = Utils.xmlNameToJava(p.getName());
if (p.getMode() != Parameter.IN) {
javifiedName += ".value";
}
if (p.getMIMEInfo() == null) {
javifiedName = Utils.wrapPrimitiveType(
p.getType(), javifiedName);
}
pw.print(javifiedName);
}
}
} // writeParamters
private void writeResponseHandling(PrintWriter pw, Parameters parms) {
pw.println(" if (_resp instanceof java.rmi.RemoteException) {");
pw.println(" throw (java.rmi.RemoteException)_resp;");
pw.println(" }");
int allOuts = parms.outputs + parms.inouts;
if (allOuts > 0) {
pw.println(" else {");
pw.println(" getResponseHeaders(_call);");
pw.println(" extractAttachments(_call);");
if (allOuts == 1) {
if (parms.returnParam != null) {
writeOutputAssign(pw, "return ", parms.returnParam.getType(),
parms.returnParam.getMIMEInfo() , "_resp");
}
else {
// The resp object must go into a holder
int i = 0;
Parameter p = (Parameter) parms.list.get(i);
while (p.getMode() == Parameter.IN) {
p = (Parameter) parms.list.get(++i);
}
String javifiedName = Utils.xmlNameToJava(p.getName());
String qnameName = Utils.getNewQName(p.getQName());
pw.println(" java.util.Map _output;");
pw.println(" _output = _call.getOutputParams();");
writeOutputAssign(pw, javifiedName + ".value = ",
p.getType(), p.getMIMEInfo(),
"_output.get(" + qnameName + ")");
}
}
else {
// There is more than 1 output. Get the outputs from getOutputParams.
pw.println(" java.util.Map _output;");
pw.println(" _output = _call.getOutputParams();");
for (int i = 0; i < parms.list.size (); ++i) {
Parameter p = (Parameter) parms.list.get (i);
String javifiedName = Utils.xmlNameToJava(p.getName());
String qnameName = Utils.getNewQName(p.getQName());
if (p.getMode() != Parameter.IN) {
writeOutputAssign(pw, javifiedName + ".value = ",
p.getType(), p.getMIMEInfo(),
"_output.get(" + qnameName + ")");
}
}
if (parms.returnParam != null) {
writeOutputAssign(pw, "return ", parms.returnParam.getType(),
parms.returnParam.getMIMEInfo(), "_resp");
}
}
pw.println(" }");
} else {
pw.println(" getResponseHeaders(_call);");
pw.println(" extractAttachments(_call);");
}
} // writeResponseHandling
/**
* writeOutputAssign
* @param target (either "return" or "something ="
* @param type (source TypeEntry)
* @param source (source String)
*
*/
private void writeOutputAssign(PrintWriter pw, String target,
TypeEntry type, MimeInfo mimeInfo,
String source) {
if (type != null && type.getName() != null) {
// Try casting the output to the expected output.
// If that fails, use JavaUtils.convert()
pw.println(" try {");
pw.println(" " + target +
Utils.getResponseString(type, mimeInfo, source));
pw.println(" } catch (java.lang.Exception _exception) {");
pw.println(" " + target +
Utils.getResponseString(type, mimeInfo,
"org.apache.axis.utils.JavaUtils.convert(" +
source + ", " + type.getName() + ".class)"));
pw.println(" }");
} else {
pw.println(" " + target +
Utils.getResponseString(type, mimeInfo, source));
}
}
} // class JavaStubWriter