永久JMS连接失败,但没有错误或InactivityExceptions

时间:2019-05-28 13:27:33

标签: java spring jms activemq

我们有一个与此old question类似的问题。但是,我们的设置有些不同。例如,心跳应该已经存在,因为我们有ActiveMQ的默认InactivityMonitor。

我们有一个使用嵌入式经纪人的客户。嵌入式代理具有一个网络连接器,该连接器连接到作为独立服务在计算机上运行的远程代理。这样,我们可以使客户端和服务器之间的通信脱钩。嵌入式代理充当客户端的本地队列。

客户端将消息发送到嵌入式代理。这些消息要么通过网络连接器流到远程代理,要么(当连接暂时不可用时)留在嵌入式代理中,直到重新建立连接为止。

嵌入式代理和远程代理都是Apache ActiveMQ的实例。 JMS实现基于Spring JMS。

在实践中,有时我们会看到奇怪的行为(通常经过很长时间没有任何问题):

  • 网络连接器在远程代理的管理控制台的“连接”选项卡中列出。但是,并非所有消息都将在远程代理中传递。通常,字节消息卡在嵌入式代理中,而文本消息则传递到远程代理上的队列中。
  • 网络连接器在远程代理的管理控制台的“连接”选项卡中列出。但是,没有消息在远程代理处传递。

在远程代理上启用了不活动监视器。嵌入式代理使用以下代码创建(为简洁起见,省略了SSL代码)。

我们不知道是什么原因引起的问题,更重要的是,为什么设置不能自动恢复。我们希望此设置能够自动检测到不可用的连接,并设置新的连接。但是,这似乎没有发生。

有人知道我们可以从哪里开始寻找,或者甚至更好的知道可能是什么问题?

@Bean
public ActiveMQConnectionFactory activeMQConnectionFactory(final Optional<TransportListener> transportListener)
{
    final ActiveMQConnectionFactory activeMQConnectionFactory = new ActiveMQConnectionFactory();

    activeMQConnectionFactory.setClientIDPrefix(clientIDPrefix);
  activeMQConnectionFactory.setBrokerURL(this.env.getProperty("activemq.connection.url"));
    activeMQConnectionFactory.setUserName(this.env.getProperty("activemq.users.username"));
    activeMQConnectionFactory.setPassword(this.env.getProperty("activemq.users.password"));

    transportListener.ifPresent(activeMQConnectionFactory::setTransportListener);

    return activeMQConnectionFactory;
}

@Bean
public CachingConnectionFactory cachingConnectionFactory(final ActiveMQConnectionFactory activeMQConnectionFactory)
{
    return new CachingConnectionFactory(activeMQConnectionFactory);
}

@Bean
public JmsTemplate jmsTemplate(final CachingConnectionFactory cachingConnectionFactory)
{
    final JmsTemplate jmsTemplate = new JmsTemplate(cachingConnectionFactory);
    return jmsTemplate;
}

@Bean
public DefaultJmsListenerContainerFactory jmsListenerContainerFactory(final ActiveMQConnectionFactory activeMQConnectionFactory)
{
    final DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
    factory.setConnectionFactory(activeMQConnectionFactory);
    factory.setConcurrency("3-10");

    return factory;
}

@Bean
public MessageSender messageSender()
{
    return new MessageSender();
}

@Bean(initMethod = "start", destroyMethod = "stop")
public BrokerService bufferingBroker() throws Exception
{
    final BrokerService broker = new BrokerService();

    String brokerName = UUID.randomUUID().toString();
    broker.setBrokerName(brokerName);

    final NetworkConnector networkConnector = new DiscoveryNetworkConnector(
        new URI("static://" + this.env.getProperty("activemq.connection.url")));
    networkConnector.setUserName(this.env.getProperty("activemq.users.username"));
    networkConnector.setPassword(this.env.getProperty("activemq.users.password"));
    networkConnector.setNetworkTTL(5);
    broker.addNetworkConnector(networkConnector);

    return broker;
}

@Bean
public ActiveMQConnectionFactory embeddedActiveMQConnectionFactory(final BrokerService bufferingBroker) throws Exception
{
    final ActiveMQConnectionFactory activeMQConnectionFactory = new ActiveMQConnectionFactory();
    activeMQConnectionFactory.setBrokerURL("vm://" + bufferingBroker.getBrokerName() + "?create=false");

    return activeMQConnectionFactory;
}

@Bean
public CachingConnectionFactory embeddedCachingConnectionFactory(final ActiveMQConnectionFactory embeddedActiveMQConnectionFactory) throws Exception
{
    return new CachingConnectionFactory(embeddedActiveMQConnectionFactory);
}

@Bean
public JmsTemplate embeddedJmsTemplate(final CachingConnectionFactory embeddedCachingConnectionFactory) throws Exception
{
    return new JmsTemplate(embeddedCachingConnectionFactory);
}

@Bean
public DefaultJmsListenerContainerFactory embeddedJmsListenerContainerFactory(final ActiveMQConnectionFactory embeddedActiveMQConnectionFactory) throws Exception
{
    final DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
    factory.setConnectionFactory(embeddedActiveMQConnectionFactory);
    factory.setConcurrency("3-10");

    return factory;
}

@Bean
public BufferedMessageSender bufferedMessageSender()
{
    return new BufferedMessageSender();
}

1 个答案:

答案 0 :(得分:0)

事实证明,外部防火墙阻止了字节消息,而文本消息可以毫无问题地通过。