Package org.apache.synapse.transport.amqp

Source Code of org.apache.synapse.transport.amqp.AMQPSender

package org.apache.synapse.transport.amqp;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import org.apache.axiom.om.OMOutputFormat;
import org.apache.axiom.om.util.UUIDGenerator;
import org.apache.axis2.AxisFault;
import org.apache.axis2.context.ConfigurationContext;
import org.apache.axis2.context.MessageContext;
import org.apache.axis2.description.TransportOutDescription;
import org.apache.axis2.transport.MessageFormatter;
import org.apache.axis2.transport.OutTransportInfo;
import org.apache.axis2.transport.TransportUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.qpidity.api.Message;
import org.apache.qpidity.nclient.Client;
import org.apache.qpidity.nclient.Connection;
import org.apache.qpidity.nclient.Session;
import org.apache.qpidity.nclient.util.MessagePartListenerAdapter;
import org.apache.qpidity.transport.DeliveryProperties;
import org.apache.qpidity.transport.MessageProperties;
import org.apache.qpidity.transport.Option;
import org.apache.qpidity.transport.ReplyTo;
import org.apache.synapse.transport.base.AbstractTransportSender;
import org.apache.synapse.transport.base.BaseConstants;
import org.apache.synapse.transport.base.BaseUtils;
import org.apache.synapse.transport.jms.JMSUtils;

public class AMQPSender extends AbstractTransportSender {

    public static final String TRANSPORT_NAME = "amqp";
    private static final Log log = LogFactory.getLog(AMQPSender.class);
    private Map<String,ConnectionDetails> _connectionDetails = new HashMap<String,ConnectionDetails>();

    @Override
    public void init(ConfigurationContext cfgCtx, TransportOutDescription transportOut) throws AxisFault {
        setTransportName(TRANSPORT_NAME);
        super.init(cfgCtx, transportOut);
        // init connections
    }

    /**
     * Needs a more robust strategy to cache connections and sessions
     * For efficiency I assume that the reply to exchange,queue and the binding already exists.
     *
     * For synchrouns request/reponse a temp queue will be create and bound to the direct exchange.
     */
    @Override
    public void sendMessage(MessageContext msgCtx, String targetEPR, OutTransportInfo outTransportInfo) throws AxisFault{

        AMQPOutTransportInfo amqpTransportInfo = null;
        ConnectionDetails conDetails = null;
        Session session = null;

        // If targetEPR is not null, determine the addressing info from it
        if (targetEPR != null) {
            amqpTransportInfo = new AMQPOutTransportInfo(targetEPR);
        }
        // If not try to get the addressing info from the transport description
        else if (outTransportInfo != null && outTransportInfo instanceof AMQPOutTransportInfo) {
            amqpTransportInfo = (AMQPOutTransportInfo) outTransportInfo;
        }

        if (_connectionDetails.containsKey(amqpTransportInfo.getConnectionURL())){
            conDetails = _connectionDetails.get(amqpTransportInfo.getConnectionURL());
        }else{
            // else create a new connection
            Connection con = Client.createConnection();
            try{
                con.connect(amqpTransportInfo.getConnectionURL());
            }catch(Exception e){
                throw new AMQPSynapseException("Error creating a connection to the broker",e);
            }
            _connectionDetails.put(amqpTransportInfo.getConnectionURL(), new ConnectionDetails(con));
        }

        if (conDetails != null) {
            session = conDetails.getSession();
        }

        byte[] message = null;
        try {
            message = createMessageData(msgCtx);
        } catch (AMQPSynapseException e) {
            handleException("Error creating a message from the axis message context", e);
        }

        // should we wait for a synchronous response on this same thread?
        boolean waitForResponse = waitForSynchronousResponse(msgCtx);

        DeliveryProperties deliveryProps = new DeliveryProperties();
        MessageProperties msgProps = new MessageProperties();
        fillMessageHeaders(msgCtx,amqpTransportInfo,session,waitForResponse,deliveryProps,msgProps);

        synchronized(session){
            session.header(msgProps,deliveryProps);
            session.data(message);
            session.endData();
        }

        // if we are expecting a synchronous response back for the message sent out
        if (waitForResponse) {
            waitForResponseAndProcess(session, msgProps, msgCtx);
        }
    }

    private void fillMessageHeaders(MessageContext msgCtx, AMQPOutTransportInfo amqpTransportInfo,
                                    Session session, boolean waitForResponse,
                                    DeliveryProperties deliveryProps, MessageProperties msgProps){
        // Routing info
        deliveryProps.setExchange(amqpTransportInfo.getExchangeName());
        deliveryProps.setRoutingKey(amqpTransportInfo.getRoutingKey());

        // Content type
        OMOutputFormat format = BaseUtils.getOMOutputFormat(msgCtx);
        MessageFormatter messageFormatter = null;
        try {
            messageFormatter = TransportUtils.getMessageFormatter(msgCtx);
        } catch (AxisFault axisFault) {
            throw new AMQPSynapseException("Unable to get the message formatter to use");
        }

        String contentType = messageFormatter.getContentType(
                msgCtx, format, msgCtx.getSoapAction());
        msgProps.setContentType(contentType);

        // Custom properties - SOAP ACTION
        Map<String,Object> props = new HashMap<String,Object>();

        if (msgCtx.isServerSide()) {
            // set SOAP Action as a property on the message
            props.put(BaseConstants.SOAPACTION,(String)msgCtx.getProperty(BaseConstants.SOAPACTION));

        } else {
            String action = msgCtx.getOptions().getAction();
            if (action != null) {
                props.put(BaseConstants.SOAPACTION, action);
            }
        }

        msgProps.setApplicationHeaders(props);

        // transport headers
        Map headerMap = (Map) msgCtx.getProperty(MessageContext.TRANSPORT_HEADERS);

        if (headerMap != null){
            Iterator iter = headerMap.keySet().iterator();
            while (iter.hasNext()) {

                String name = (String) iter.next();

                if (AMQPConstants.AMQP_CORELATION_ID.equals(name)) {
                    msgProps.setCorrelationId((String) headerMap.get(AMQPConstants.AMQP_CORELATION_ID));
                    // If it's request/response, then we need to fill in corelation id and reply to properties
                }
                else if (AMQPConstants.AMQP_DELIVERY_MODE.equals(name)) {
                    Object o = headerMap.get(AMQPConstants.AMQP_DELIVERY_MODE);
                    if (o instanceof Short) {
                        deliveryProps.setDeliveryMode(((Short) o).shortValue());
                    }else if (o instanceof Integer) {
                        deliveryProps.setDeliveryMode(((Integer) o).shortValue());
                    }else if (o instanceof String) {
                        try {
                            deliveryProps.setDeliveryMode(Short.parseShort((String) o));
                        } catch (NumberFormatException nfe) {
                            log.warn("Invalid delivery mode ignored : " + o, nfe);
                        }
                    } else {
                        log.warn("Invalid delivery mode ignored : " + o);
                    }
                }
                else if (AMQPConstants.AMQP_EXPIRATION.equals(name)) {
                    deliveryProps.setExpiration(
                        Long.parseLong((String) headerMap.get(AMQPConstants.AMQP_EXPIRATION)));
                }
                else if (AMQPConstants.AMQP_MESSAGE_ID.equals(name)) {
                    msgProps.setMessageId((String) headerMap.get(AMQPConstants.AMQP_MESSAGE_ID));
                }
                else if (AMQPConstants.AMQP_PRIORITY.equals(name)) {
                    deliveryProps.setPriority(
                        Short.parseShort((String) headerMap.get(AMQPConstants.AMQP_PRIORITY)));
                }
                else if (AMQPConstants.AMQP_TIMESTAMP.equals(name)) {
                    deliveryProps.setTimestamp(
                        Long.parseLong((String) headerMap.get(AMQPConstants.AMQP_TIMESTAMP)));
                }else {
                    // custom app props
                    Object value = headerMap.get(name);
                    props.put(name, value);
                }
            }
        }

        /* For efficiency I assume that the reply to exchange and destination is already created
         *  If the reply is for the same service, then this should be the queue that the service is listening to.
         *  Blindly creating these exchanges,queues and bindings is sub optimal and can be avoid if the administrator
         *  creates the nessacery exchanges,queues and bindings before hand.
         *
         *  If the service hasn't specify and it's a request/reply MEP then a temporary queue
         *  (which is auto-deleted) is created and bound to the amq.direct exchange.
         */
         if (msgCtx.getProperty(AMQPConstants.AMQP_REPLY_TO_EXCHANGE_NAME) != null){
             String replyExchangeName = (String) msgCtx.getProperty(AMQPConstants.AMQP_REPLY_TO_EXCHANGE_NAME);
             String replyRoutingKey = msgCtx.getProperty(AMQPConstants.AMQP_REPLY_TO_ROUTING_KEY)!= null?(String) msgCtx.getProperty(AMQPConstants.AMQP_REPLY_TO_ROUTING_KEY):null;

             // for fannout exchange or some other custom exchange, the routing key maybe null
             msgProps.setReplyTo(new ReplyTo(replyExchangeName,replyRoutingKey));
         }

         // If it's request/response, then we need to fill in reply to properties and correlation_id
         if (waitForResponse){

             if (waitForResponse && msgProps.getCorrelationId() == null) {
                 if (msgCtx.getProperty(AMQPConstants.AMQP_CORELATION_ID) != null){
                     msgProps.setCorrelationId((String)msgCtx.getProperty(AMQPConstants.AMQP_CORELATION_ID));
                 }else{
                     msgProps.setCorrelationId(UUIDGenerator.getUUID());
                 }

             }

             if (msgProps.getReplyTo() == null){
                 //We need to use a temp queue here.
                 String tempQueueName = "Queue_" + msgProps.getCorrelationId();
                 synchronized(session){
                     session.queueDeclare(tempQueueName, null, null, Option.AUTO_DELETE,Option.EXCLUSIVE);
                     session.queueBind(tempQueueName, "amq.direct", tempQueueName, null);
                     session.sync();
                 }
                 msgProps.replyTo(new ReplyTo("amq.direct",tempQueueName));
             }
         }
    }

    private byte[] createMessageData(MessageContext msgContext){
        OMOutputFormat format = BaseUtils.getOMOutputFormat(msgContext);
        MessageFormatter messageFormatter = null;
        try {
            messageFormatter = TransportUtils.getMessageFormatter(msgContext);
        } catch (AxisFault axisFault) {
            throw new AMQPSynapseException("Unable to get the message formatter to use",axisFault);
        }

        String contentType = messageFormatter.getContentType(
            msgContext, format, msgContext.getSoapAction());

        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        try {
            messageFormatter.writeTo(msgContext, format, baos, true);
            baos.flush();
            return baos.toByteArray();
        } catch (IOException e) {
            throw new AMQPSynapseException("IO Error while creating message", e);
        }
    }

    private void waitForResponseAndProcess(Session session, MessageProperties msgProps,
            MessageContext msgCtx) throws AxisFault {

        long timeout = AMQPConstants.DEFAULT_AMQP_TIMEOUT;
        String waitReply = (String) msgCtx.getProperty(AMQPConstants.AMQP_WAIT_REPLY);
        if (waitReply != null) {
            timeout = Long.valueOf(waitReply).longValue();
        }
        // We are using the routing key (which is the queue name) as the destination
        String destination = msgProps.getReplyTo().getRoutingKey();
        MessageManager listener = new MessageManager(session,destination,msgProps.getCorrelationId());
        session.messageSubscribe(msgProps.getReplyTo().getRoutingKey(),
                                 destination,
                                 Session.TRANSFER_CONFIRM_MODE_REQUIRED,
                                 Session.TRANSFER_ACQUIRE_MODE_PRE_ACQUIRE,
                                 new MessagePartListenerAdapter(listener), null, Option.NO_OPTION);

        Message reply = listener.receive(timeout);

        if (reply != null) {
            processSyncResponse(msgCtx, reply);

        } else {
            log.warn("Did not receive a response within " +
                timeout + " ms to destination : " + msgProps.getReplyTo().getRoutingKey() +
                " with correlation ID : " + msgProps.getCorrelationId());
        }
    }

    private void processSyncResponse(MessageContext outMsgCtx, Message message) throws AxisFault {

        MessageContext responseMsgCtx = createResponseMessageContext(outMsgCtx);

        // load any transport headers from received message
        Map map = AMQPUtils.getTransportHeaders(message);
        responseMsgCtx.setProperty(MessageContext.TRANSPORT_HEADERS, map);

        // workaround for Axis2 TransportUtils.createSOAPMessage() issue, where a response
        // of content type "text/xml" is thought to be REST if !MC.isServerSide(). This
        // question is still under debate and due to the timelines, I am commiting this
        // workaround as Axis2 1.2 is about to be released and Synapse 1.0
        responseMsgCtx.setServerSide(false);

        String contentType = JMSUtils.getInstace().getProperty(message, BaseConstants.CONTENT_TYPE);

        AMQPUtils.getInstace().setSOAPEnvelope(message, responseMsgCtx, contentType);
        responseMsgCtx.setServerSide(true);

        handleIncomingMessage(
            responseMsgCtx,
            map,
            (String)map.get(BaseConstants.SOAPACTION),
            contentType
        );
    }

    private class ConnectionDetails{
        private Connection _conn;
        //private Map _sessions = new HashMap();
        private Session _session;

        public ConnectionDetails(Connection conn){
            _conn = conn;
        }

        public Session getSession(){
          if (_session != null){
              _session = _conn.createSession(0);
          }
          return _session;
       }
    }

}
TOP

Related Classes of org.apache.synapse.transport.amqp.AMQPSender

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