blob: 28a5b5805efec56a14995274fa6980a1f6601372 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
/*
* Reflection based RPCMessageReceiver , request will be processed by looking at the method signature
* of the invocation method
*/
package org.apache.axis2.rpc.receivers;
import org.apache.axiom.om.OMElement;
import org.apache.axiom.om.OMNamespace;
import org.apache.axiom.om.OMAbstractFactory;
import org.apache.axiom.om.impl.builder.StAXOMBuilder;
import org.apache.axiom.soap.SOAPEnvelope;
import org.apache.axiom.soap.SOAPFactory;
import org.apache.axis2.AxisFault;
import org.apache.axis2.util.StreamWrapper;
import org.apache.axis2.databinding.utils.BeanUtil;
import org.apache.axis2.context.MessageContext;
import org.apache.axis2.description.AxisMessage;
import org.apache.axis2.description.AxisOperation;
import org.apache.axis2.description.AxisService;
import org.apache.axis2.description.Parameter;
import org.apache.axis2.description.WSDL2Constants;
import org.apache.axis2.description.java2wsdl.Java2WSDLConstants;
import org.apache.axis2.description.java2wsdl.TypeTable;
import org.apache.axis2.receivers.AbstractInOutMessageReceiver;
import org.apache.axis2.wsdl.WSDLConstants;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import javax.xml.namespace.QName;
import javax.xml.stream.XMLStreamReader;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class RPCMessageReceiver extends AbstractInOutMessageReceiver {
private static Log log = LogFactory.getLog(RPCMessageReceiver.class);
/**
* reflect and get the Java method - for each i'th param in the java method - get the first
* child's i'th child -if the elem has an xsi:type attr then find the deserializer for it - if
* not found, lookup deser for th i'th param (java type) - error if not found - deserialize &
* save in an object array - end for
* <p/>
* - invoke method and get the return value
* <p/>
* - look up serializer for return value based on the value and type
* <p/>
* - create response msg and add return value as grand child of <soap:body>
*
* @param inMessage incoming MessageContext
* @param outMessage outgoing MessageContext
* @throws AxisFault
*/
public void invokeBusinessLogic(MessageContext inMessage, MessageContext outMessage)
throws AxisFault {
Method method = null;
try {
// get the implementation class for the Web Service
Object obj = getTheImplementationObject(inMessage);
Class implClass = obj.getClass();
AxisOperation op = inMessage.getOperationContext().getAxisOperation();
method = (Method)(op.getParameterValue("myMethod"));
// If the declaring class has changed, then the cached method is invalid, so we need to
// reload it. This is to fix AXIS2-3947.
if (method != null && method.getDeclaringClass() != implClass) {
method = null;
}
AxisService service = inMessage.getAxisService();
OMElement methodElement = inMessage.getEnvelope().getBody()
.getFirstElement();
AxisMessage inAxisMessage = op.getMessage(WSDLConstants.MESSAGE_LABEL_IN_VALUE);
String messageNameSpace = null;
if (method == null) {
String methodName = op.getName().getLocalPart();
Method[] methods = implClass.getMethods();
for (Method method1 : methods) {
if (method1.isBridge()) {
continue;
}
if (method1.getName().equals(methodName)) {
method = method1;
op.addParameter("myMethod", method);
break;
}
}
if (method == null) {
throw new AxisFault("No such method '" + methodName +
"' in class " + implClass.getName());
}
}
Object resObject = null;
if (inAxisMessage != null) {
resObject = RPCUtil.invokeServiceClass(inAxisMessage,
method,
obj,
messageNameSpace,
methodElement,inMessage);
}
SOAPFactory fac = getSOAPFactory(inMessage);
// Handling the response
AxisMessage outaxisMessage = op.getMessage(WSDLConstants.MESSAGE_LABEL_OUT_VALUE);
if (outaxisMessage != null && outaxisMessage.getElementQName() !=null) {
messageNameSpace = outaxisMessage.getElementQName().getNamespaceURI();
} else {
messageNameSpace = service.getTargetNamespace();
}
OMNamespace ns = fac.createOMNamespace(messageNameSpace,
service.getSchemaTargetNamespacePrefix());
SOAPEnvelope envelope = fac.getDefaultEnvelope();
OMElement bodyContent = null;
if (WSDL2Constants.MEP_URI_ROBUST_IN_ONLY.equals(
op.getMessageExchangePattern())){
OMElement bodyChild = fac.createOMElement(outMessage.getAxisMessage().getName(), ns);
envelope.getBody().addChild(bodyChild);
outMessage.setEnvelope(envelope);
return;
}
Parameter generateBare = service.getParameter(Java2WSDLConstants.DOC_LIT_BARE_PARAMETER);
if (generateBare!=null && "true".equals(generateBare.getValue())) {
RPCUtil.processResonseAsDocLitBare(resObject, service,
envelope, fac, ns,
bodyContent, outMessage);
} else {
RPCUtil.processResponseAsDocLitWrapped(resObject, service,
method, envelope, fac, ns,
bodyContent, outMessage);
}
outMessage.setEnvelope(envelope);
} catch (InvocationTargetException e) {
String msg = null;
Throwable cause = e.getCause();
if (cause != null) {
msg = cause.getMessage();
}
if (msg == null) {
msg = "Exception occurred while trying to invoke service method " +
(method != null ? method.getName() : "null");
}
if (cause instanceof AxisFault) {
log.debug(msg, cause);
throw (AxisFault)cause;
}
Class[] exceptionTypes = method.getExceptionTypes();
for (Class exceptionType : exceptionTypes){
if (exceptionType.getName().equals(cause.getClass().getName())){
// this is an bussiness logic exception so handle it properly
String partQName = method.getName() + getSimpleClassName(exceptionType);
TypeTable typeTable = inMessage.getAxisService().getTypeTable();
QName elementQName = typeTable.getQNamefortheType(partQName);
SOAPFactory fac = getSOAPFactory(inMessage);
OMElement exceptionElement = fac.createOMElement(elementQName);
if (exceptionType.getName().equals(Exception.class.getName())){
// this is an exception class. so create a element by hand and add the message
OMElement innterExceptionElement = fac.createOMElement(elementQName);
OMElement messageElement = fac.createOMElement("Message", inMessage.getAxisService().getTargetNamespace(), null);
messageElement.setText(cause.getMessage());
innterExceptionElement.addChild(messageElement);
exceptionElement.addChild(innterExceptionElement);
} else {
// if it is a normal bussiness exception we need to generate the schema assuming it is a pojo
QName innerElementQName = new QName(elementQName.getNamespaceURI(), getSimpleClassName(exceptionType));
XMLStreamReader xr = BeanUtil.getPullParser(cause,
innerElementQName, typeTable, true, false);
StAXOMBuilder stAXOMBuilder = new StAXOMBuilder(OMAbstractFactory.getOMFactory(), new StreamWrapper(xr));
OMElement documentElement = stAXOMBuilder.getDocumentElement();
exceptionElement.addChild(documentElement);
}
AxisFault axisFault = new AxisFault(cause.getMessage());
axisFault.setDetail(exceptionElement);
throw axisFault;
}
}
log.error(msg, e);
throw new AxisFault(msg, e);
} catch(RuntimeException e) {
log.error(e.getMessage(), e);
throw AxisFault.makeFault(e);
} catch (Exception e) {
String msg = "Exception occurred while trying to invoke service method " +
(method != null ? method.getName() : "null");
log.error(msg, e);
throw AxisFault.makeFault(e);
}
}
private String getSimpleClassName(Class type) {
String simpleClassName = type.getName();
int idx = simpleClassName.lastIndexOf('.');
if (idx != -1 && idx < (simpleClassName.length() - 1)) {
simpleClassName = simpleClassName.substring(idx + 1);
}
return simpleClassName.replace('$', '_');
}
}