spring-boot - RabbitHandler 在 Spring 中创建消费者并重试致命异常,以便在侦听 RabbitMQ 时排队
问题描述
我正在使用 Spring AMQP RabbitHandler 并编写了以下代码:
@RabbitListener(queues = "#{testQueue.name}")
public class Tut4Receiver {
@RabbitHandler
public void receiveMessage(String message){
System.out.println("Message received "+message);
}
}
队列定义如下: -
@Bean
public Queue testQueue() {
return new AnonymousQueue();
}
我正在使用单独的代码来初始化连接工厂。
我的问题是,如果 RabbitMQ 关闭了一段时间,它会继续重试创建消费者,但前提是它收到ConnectionRefused
错误。但是假设该用户在 RabbitMQ 中不存在,并且将创建一个新用户,然后它从 RabbitMQ 收到一个致命错误并且它永远不会重试,结果是自动删除队列将在 RabbitMQ 上创建而没有任何消费者。
堆栈跟踪:
SimpleMessageListenerContainer] [SimpleAsyncTaskExecutor-11] [|] [|||] Consumer received fatal exception on startup
org.springframework.amqp.rabbit.listener.exception.FatalListenerStartupException: Authentication failure
at org.springframework.amqp.rabbit.listener.BlockingQueueConsumer.start(BlockingQueueConsumer.java:476)
at org.springframework.amqp.rabbit.listener.SimpleMessageListenerContainer$AsyncMessageProcessingConsumer.run(SimpleMessageListenerContainer.java:1280)
at java.lang.Thread.run(Thread.java:748)
Caused by: org.springframework.amqp.AmqpAuthenticationException: com.rabbitmq.client.AuthenticationFailureException: ACCESS_REFUSED - Login was refused using authentication mechanism PLAIN. For details see the broker logfile.
at org.springframework.amqp.rabbit.support.RabbitExceptionTranslator.convertRabbitAccessException(RabbitExceptionTranslator.java:65)
at org.springframework.amqp.rabbit.connection.AbstractConnectionFactory.createBareConnection(AbstractConnectionFactory.java:309)
at org.springframework.amqp.rabbit.connection.CachingConnectionFactory.createConnection(CachingConnectionFactory.java:547)
at org.springframework.amqp.rabbit.connection.ConnectionFactoryUtils$1.createConnection(ConnectionFactoryUtils.java:90)
at org.springframework.amqp.rabbit.connection.ConnectionFactoryUtils.doGetTransactionalResourceHolder(ConnectionFactoryUtils.java:140)
at org.springframework.amqp.rabbit.connection.ConnectionFactoryUtils.getTransactionalResourceHolder(ConnectionFactoryUtils.java:76)
at org.springframework.amqp.rabbit.listener.BlockingQueueConsumer.start(BlockingQueueConsumer.java:472)
... 2 common frames omitted
Caused by: com.rabbitmq.client.AuthenticationFailureException: ACCESS_REFUSED - Login was refused using authentication mechanism PLAIN. For details see the broker logfile.
at com.rabbitmq.client.impl.AMQConnection.start(AMQConnection.java:339)
at com.rabbitmq.client.ConnectionFactory.newConnection(ConnectionFactory.java:813)
at com.rabbitmq.client.ConnectionFactory.newConnection(ConnectionFactory.java:767)
at com.rabbitmq.client.ConnectionFactory.newConnection(ConnectionFactory.java:887)
at org.springframework.amqp.rabbit.connection.AbstractConnectionFactory.createBareConnection(AbstractConnectionFactory.java:300)
SimpleMessageListenerContainer] [SimpleAsyncTaskExecutor-11] [|] [|||] Stopping container from aborted consumer
[|] [|||] Waiting for workers to finish.
[|] [|||] Successfully waited for workers to finish.
有什么方法可以重试致命异常,例如用户不存在时?
解决方案
默认情况下,身份验证失败被认为是致命的,不会重试。
possibleAuthenticationFailureFatal
您可以通过在侦听器容器 ( )上设置属性来覆盖此行为。该属性不能用作引导属性,因此您必须覆盖引导的容器工厂...
@Bean(name = "rabbitListenerContainerFactory")
public SimpleRabbitListenerContainerFactory simpleRabbitListenerContainerFactory(
SimpleRabbitListenerContainerFactoryConfigurer configurer, ConnectionFactory connectionFactory) {
SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
configurer.configure(factory, connectionFactory);
factory.setContainerConfigurer(smlc -> smlc.setPossibleAuthenticationFailureFatal(false));
return factory;
}
推荐阅读
- c - 如何限制用户输入
- python - 为什么在 Kivy ScrollView 中将 minimum_height 绑定到 self.setter('height')
- apache-spark - 选择最佳超参数组合后,SparkML CrossValidator 是否会重新适应完整的训练数据集?
- django - django 连接过滤器模型
- java - Java是强类型的
- html - 如何停止嵌入 youtube 视频的声音?
- linux - 我可以在传输数据时指定linux内核套接字将用户数据复制到DMA内存吗?
- java - MVEL 验证没有值的算术表达式
- apache-spark-sql - 使用sql查询替换除零以外的所有数字
- java - 正确使用 void 方法