multiconsumerjmsmessagereceiver.java
来自「提供ESB 应用mule源代码 提供ESB 应用mule源代码」· Java 代码 · 共 364 行
JAVA
364 行
/* * $Id: MultiConsumerJmsMessageReceiver.java 12427 2008-07-29 20:21:01Z tcarlson $ * -------------------------------------------------------------------------------------- * Copyright (c) MuleSource, Inc. All rights reserved. http://www.mulesource.com * * The software in this package is published under the terms of the CPAL v1.0 * license, a copy of which has been included with this distribution in the * LICENSE.txt file. */package org.mule.transport.jms;import org.mule.api.MuleException;import org.mule.api.MuleRuntimeException;import org.mule.api.endpoint.InboundEndpoint;import org.mule.api.lifecycle.CreateException;import org.mule.api.lifecycle.LifecycleException;import org.mule.api.service.Service;import org.mule.api.transaction.Transaction;import org.mule.api.transaction.TransactionException;import org.mule.api.transport.Connector;import org.mule.config.i18n.MessageFactory;import org.mule.transport.AbstractMessageReceiver;import org.mule.transport.AbstractReceiverWorker;import org.mule.transport.ConnectException;import org.mule.transport.jms.filters.JmsSelectorFilter;import org.mule.util.ClassUtils;import java.util.ArrayList;import javax.jms.Destination;import javax.jms.JMSException;import javax.jms.Message;import javax.jms.MessageConsumer;import javax.jms.MessageListener;import javax.jms.Session;import javax.jms.Topic;import javax.resource.spi.work.WorkException;import edu.emory.mathcs.backport.java.util.concurrent.BlockingDeque;import edu.emory.mathcs.backport.java.util.concurrent.LinkedBlockingDeque;import org.apache.commons.logging.Log;import org.apache.commons.logging.LogFactory;/** * This is an experimental rework of the <code>transacted.message.receiver</code> * (it may work with non-transacted sessions as well, * just hasn't been tested and load-tested yet. * <p/> * In Mule an endpoint corresponds to a single receiver. It's up to the receiver to do multithreaded consumption and * resource allocation, if needed. This class honors the <code>numberOfConcurrentTransactedReceivers</code> strictly * and will create exactly this number of consumers. */public class MultiConsumerJmsMessageReceiver extends AbstractMessageReceiver{ protected final BlockingDeque consumers; protected volatile int receiversCount; private final JmsConnector jmsConnector; public MultiConsumerJmsMessageReceiver(Connector connector, Service service, InboundEndpoint endpoint) throws CreateException { super(connector, service, endpoint); jmsConnector = (JmsConnector) connector; final boolean isTopic = jmsConnector.getTopicResolver().isTopic(endpoint, true); receiversCount = jmsConnector.getNumberOfConcurrentTransactedReceivers(); if (isTopic && receiversCount != 1) { if (logger.isInfoEnabled()) { logger.info("Destination " + getEndpoint().getEndpointURI() + " is a topic, but " + receiversCount + " receivers have been configured. Will configure only 1."); } receiversCount = 1; } if (logger.isDebugEnabled()) { logger.debug("Creating " + receiversCount + " sub-receivers for " + endpoint.getEndpointURI()); } consumers = new LinkedBlockingDeque(receiversCount); for (int i = 0; i < receiversCount; i++) { consumers.addLast(new SubReceiver()); } } protected void doStart() throws MuleException { logger.debug("doStart()"); for (int i = 0; i < receiversCount; i++) { SubReceiver sub = (SubReceiver) consumers.removeFirst(); sub.doStart(); consumers.addLast(sub); } } protected void doStop() throws MuleException { logger.debug("doStop()"); for (int i = 0; i < receiversCount; i++) { SubReceiver sub = (SubReceiver) consumers.removeFirst(); sub.doStop(); consumers.addLast(sub); } } protected void doConnect() throws Exception { logger.debug("doConnect()"); for (int i = 0; i < receiversCount; i++) { SubReceiver sub = (SubReceiver) consumers.removeFirst(); sub.doConnect(); consumers.addLast(sub); } } protected void doDisconnect() throws Exception { logger.debug("doDisconnect()"); for (int i = 0; i < receiversCount; i++) { SubReceiver sub = (SubReceiver) consumers.removeFirst(); sub.doDisconnect(); consumers.addLast(sub); } } protected void doDispose() { logger.debug("doDispose()"); } private class SubReceiver implements MessageListener { private final Log subLogger = LogFactory.getLog(getClass()); private volatile Session session; private volatile MessageConsumer consumer; private volatile boolean startOnConnect = false; protected void doConnect() throws Exception { subLogger.debug("SUB doConnect()"); createConsumer(); if (startOnConnect) { doStart(); } } protected void doDisconnect() throws Exception { closeConsumer(); } protected void closeConsumer() { jmsConnector.closeQuietly(consumer); consumer = null; jmsConnector.closeQuietly(session); session = null; } protected void doStart() throws MuleException { try { // If the consumer is null it means that the connection strategy is being // run in a separate thread (doThreading=true), and hasn't managed to connect // yet. This doStart will then be called from doConnect. if (consumer == null) { startOnConnect = true; } else { startOnConnect = false; consumer.setMessageListener(this); } } catch (JMSException e) { throw new LifecycleException(e, this); } } protected void doStop() throws MuleException { try { if (consumer != null) { consumer.setMessageListener(null); } } catch (JMSException e) { throw new LifecycleException(e, this); } } /** * Create a consumer for the jms destination. */ protected void createConsumer() throws Exception { try { JmsSupport jmsSupport = jmsConnector.getJmsSupport(); boolean topic = jmsConnector.getTopicResolver().isTopic(endpoint, true); // Create session if none exists if (session == null) { session = jmsConnector.getSession(endpoint); } // Create destination Destination dest = jmsSupport.createDestination(session, endpoint.getEndpointURI().getAddress(), topic); // Extract jms selector String selector = null; if (endpoint.getFilter() != null && endpoint.getFilter() instanceof JmsSelectorFilter) { selector = ((JmsSelectorFilter) endpoint.getFilter()).getExpression(); } else { if (endpoint.getProperties() != null) { // still allow the selector to be set as a property on the endpoint // to be backward compatable selector = (String) endpoint.getProperties().get(JmsConstants.JMS_SELECTOR_PROPERTY); } } String tempDurable = (String) endpoint.getProperties().get(JmsConstants.DURABLE_PROPERTY); boolean durable = jmsConnector.isDurable(); if (tempDurable != null) { durable = Boolean.valueOf(tempDurable).booleanValue(); } // Get the durable subscriber name if there is one String durableName = (String) endpoint.getProperties().get(JmsConstants.DURABLE_NAME_PROPERTY); if (durableName == null && durable && dest instanceof Topic) { durableName = "mule." + jmsConnector.getName() + "." + endpoint.getEndpointURI().getAddress(); logger.debug("Jms Connector for this receiver is durable but no durable name has been specified. Defaulting to: " + durableName); } // Create consumer consumer = jmsSupport.createConsumer(session, dest, selector, jmsConnector.isNoLocal(), durableName, topic); } catch (JMSException e) { throw new ConnectException(e, MultiConsumerJmsMessageReceiver.this); } } public void onMessage(final Message message) { try { // This must be the doWork() to preserve the transactional context. // We are already running in the consumer thread by this time. // The JmsWorker classe is a one-off executor which is abandoned after it's done and is // easily garbage-collected (confirmed with a profiler) getWorkManager().doWork(new JmsWorker(message, MultiConsumerJmsMessageReceiver.this, this)); } catch (WorkException e) { throw new MuleRuntimeException(MessageFactory.createStaticMessage( "Couldn't submit a work item to the WorkManager"), e); } } } protected class JmsWorker extends AbstractReceiverWorker { private final SubReceiver subReceiver; public JmsWorker(Message message, AbstractMessageReceiver receiver, SubReceiver subReceiver) { super(new ArrayList(1), receiver); this.subReceiver = subReceiver; messages.add(message); } //@Override protected Object preProcessMessage(Object message) throws Exception { Message m = (Message) message; if (logger.isDebugEnabled()) { logger.debug("Message received it is of type: " + ClassUtils.getSimpleName(message.getClass())); if (m.getJMSDestination() != null) { logger.debug("Message received on " + m.getJMSDestination() + " (" + m.getJMSDestination().getClass().getName() + ")"); } else { logger.debug("Message received on unknown destination"); } logger.debug("Message CorrelationId is: " + m.getJMSCorrelationID()); logger.debug("Jms Message Id is: " + m.getJMSMessageID()); } if (m.getJMSRedelivered()) { // lazily create the redelivery handler RedeliveryHandler redeliveryHandler = jmsConnector.getRedeliveryHandlerFactory().create(); redeliveryHandler.setConnector(jmsConnector); if (logger.isDebugEnabled()) { logger.debug("Message with correlationId: " + m.getJMSCorrelationID() + " has redelivered flag set, handing off to Exception Handler"); } redeliveryHandler.handleRedelivery(m); } return m; } protected void bindTransaction(Transaction tx) throws TransactionException { if (tx instanceof JmsTransaction) { if (logger.isDebugEnabled()) { logger.debug(">>> !! Binding " + subReceiver.session + " to " + jmsConnector.getConnection()); } tx.bindResource(jmsConnector.getConnection(), subReceiver.session); } else { if (tx instanceof JmsClientAcknowledgeTransaction) { //We should still bind the session to the transaction, but we also need the message itself //since that is the object that gets Acknowledged //tx.bindResource(jmsConnector.getConnection(), session); ((JmsClientAcknowledgeTransaction) tx).setMessage((Message) messages.get(0)); } } } }}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?