簡體   English   中英

Spring AMQP - 匿名隊列聲明

[英]Spring AMQP - anonymous queue declaration

我正在嘗試使用 Spring AMQP 注釋聲明匿名隊列,但它似乎不起作用。

@RabbitListener(
  id = "b1",
  bindings =
      @QueueBinding(
          value =
              @Queue(
                  name = "",
                  durable = "false",
                  exclusive = "true",
                  autoDelete = "true",
                  admins = "amqpAdmin1"),
          exchange =
              @Exchange(
                  value = "${my.rabbitmq[0].exchange}", declare = "false",
                  ignoreDeclarationExceptions = "true",
                  type = ExchangeTypes.TOPIC,
                  admins = "amqpAdmin1"),
          key = "*.*.*.*.*.*.*.-",
          admins = "amqpAdmin1"),
  admin = "amqpAdmin1",
  containerFactory = "lf1",
  autoStartup = "true")

我讀到我可以使用空name參數,它應該創建匿名隊列——但問題是我需要服務器生成的名稱(我沒有權限命名我自己的隊列)。 目前使用 spring AMQP lib 是否有可能?

我在它當前使用的代碼中看到

private String declareQueue(org.springframework.amqp.rabbit.annotation.Queue bindingQueue,
        Collection<Declarable> declarables) {
    String queueName = (String) resolveExpression(bindingQueue.value());
    boolean isAnonymous = false;
    if (!StringUtils.hasText(queueName)) {
        queueName = Base64UrlNamingStrategy.DEFAULT.generateName(); //<---- HERE
        // default exclusive/autodelete and non-durable when anonymous
        isAnonymous = true;
    }

但這里不是Base64UrlNamingStrategy.DEFAULT.generateName(); -> 我需要 spring 到 go 到 RabbitMQ 經紀人並獲取 RabbitMQ 生成的名稱並使用該名稱。

更新

@Bean
  public Queue q2(){
    Queue q = QueueBuilder.nonDurable().exclusive().autoDelete().build();
    //Queue q = new Queue("", false, true, true); --> same
    q.setAdminsThatShouldDeclare(amqpAdmin2());
    return q;
  }

  @Bean
  public Exchange e2(@Value("${my.rabbitmq[1].exchange}") String name){
    Exchange e = ExchangeBuilder.topicExchange(name).ignoreDeclarationExceptions().build();
    e.setAdminsThatShouldDeclare(amqpAdmin2());
    return e;
  }

  @Bean
  public Binding bin2(Queue q2, Exchange e2){
    Binding b = BindingBuilder.bind(q2).to(e2).with("*.*.*.*.*.*.*.-").noargs();
    b.setAdminsThatShouldDeclare(amqpAdmin2());
    return b;
  }

  @RabbitListener(id = "b2", queues = "#{q2}",
      admin = "amqpAdmin2",
      containerFactory = "lf2", autoStartup = "true")
  public void listen2(GenericMessage<?> msg) {
    comparator.recordForComparison(props.getRabbitmq().get(1).getId(), msg.getPayload());
  }

這是在日志中:

Auto-declaring a non-durable, auto-delete, or exclusive Queue () durable:false, auto-delete:true, exclusive:true. It will be redeclared if the broker stops and is restarted while the connection factory is alive, but all messages will be lost.
2022-02-10 09:22:43.660  INFO 30076 --- [           main] o.s.a.r.l.DirectMessageListenerContainer : Container initialized for queues: [spring.gen-ZAVana-pSSW2WaanrkVZXw_awaiting_declaration]
2022-02-10 09:22:43.661  INFO 30076 --- [           main] o.s.a.r.c.CachingConnectionFactory       : Attempting to connect to: my.server.com:5671
2022-02-10 09:22:43.765 ERROR 30076 --- [           b2-1] o.s.a.r.l.DirectMessageListenerContainer : Queue not present, scheduling consumer SimpleConsumer [queue=spring.gen-ZAVana-pSSW2WaanrkVZXw_awaiting_declaration, index=0, consumerTag=null identity=3f844994] for restart

這是我在刪除@RabbitListener后添加的內容。 現在它適用於:

@Bean
  public MessageListenerContainer c2(DirectRabbitListenerContainerFactory lf2, Queue q2){
    DirectMessageListenerContainer lc = lf2.createListenerContainer();
    lc.setQueues(q2);
    lc.setAutoStartup(true);
    MessageListenerAdapter a = new MessageListenerAdapter();
    a.setMessageConverter(converter());
    a.setDefaultListenerMethod("receive");
    a.setDelegate(new MessageReceiver(props.getRabbitmq().get(1).getId()));
    lc.setupMessageListener(a);
    lc.afterPropertiesSet();
    return lc;
  }

class MessageReceiver{

    private final int id;

    public MessageReceiver(int id){
      this.id= id;
    }

    void receive(Object o){
      comparator.recordForComparison(this.id, o);
    }
  }

我認為當我們實現這個https://jira.spring.io/browse/AMQP-816時,這個應該被視為一個錯誤和缺失的部分。 因此,使用常規Queue bean 定義,您可以將名稱設置為空字符串,並且actualName將在通過RabbitAdmin在代理上聲明后設置。

將其視為您的用例的解決方法。 您必須將此@QueueBinding移至BindingQueueExchange bean 定義。 然后您只需使用queues = "#{myQueueBean}" 有關詳細信息,請參閱文檔: https://docs.spring.io/spring-amqp/docs/current/reference/html/#builder-api

請為此錯誤提出 GH 問題,我們將盡快修復它。

我發現了問題 - @RabbitListener bean 后處理器在聲明名稱之前就獲取了名稱。

這是一個解決方法(手動注入隊列):

@SpringBootApplication
public class So71033589Application {

    public static void main(String[] args) {
        SpringApplication.run(So71033589Application.class, args);
    }

    @RabbitListener(id = "foo", autoStartup = "false")
    public void listen(String in) {
        System.out.println(in);
    }

}

@Configuration
class Config{

    @Bean
    Queue q() {
        return new Queue("", false, true, true);
    }

    @Bean
    ApplicationRunner runner(RabbitListenerEndpointRegistry registry, Queue q) {
        return args -> {
            MessageListenerContainer container = registry.getListenerContainer("foo");
            ((AbstractMessageListenerContainer) container).setQueues(q);
            container.start();
        };
    }

}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM