发送带有 Spring 集成到 ActiveMQ Artemis 主题的消息
Sending message with Spring Integration to ActiveMQ Artemis topic
目标
我想向主题发送消息,稍后我将使用客户端应用程序处理该主题。为此,我使用 Spring Boot 和 Spring Integration Java DSL 及其 JMS 模块。作为消息代理,我使用原生的 ActiveMQ Artemis。
这是我的设置
DemoApplication.java
@SpringBootApplication
public class DemoApplication {
private static final Logger logger = LoggerFactory.getLogger(DemoApplication.class);
public interface StarGate {
void sendHello(String helloText);
}
@Autowired
private ConnectionFactory connectionFactory;
@Bean
public IntegrationFlow mainFlow() {
return IntegrationFlows
.from(StarGate.class)
.handle(Jms.outboundAdapter(connectionFactory)
.configureJmsTemplate(jmsTemplateSpec -> jmsTemplateSpec
.deliveryPersistent(true)
.pubSubDomain(true)
.sessionTransacted(true)
.sessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE)
.explicitQosEnabled(true)
)
.destination(new ActiveMQTopic("wormhole")))
.get();
}
public static void main(String[] args) {
ConfigurableApplicationContext context = SpringApplication.run(DemoApplication.class, args);
StarGate stargate = context.getBean(StarGate.class);
stargate.sendHello("Jaffa, kree!");
logger.info("Hello message sent.");
}
}
application.properties
spring.artemis.mode=native
spring.artemis.host=localhost
spring.artemis.port=61616
spring.artemis.user=artemis
spring.artemis.password=simetraehcapa
spring.jms.pub-sub-domain=true
spring.jms.template.delivery-mode=persistent
spring.jms.template.qos-enabled=true
spring.jms.listener.acknowledge-mode=client
logging.level.org.springframework=INFO
build.gradle(重要部分)
springBootVersion = '2.0.2.RELEASE'
dependencies {
compile('org.springframework.boot:spring-boot-starter-artemis')
compile('org.springframework.boot:spring-boot-starter-integration')
compile('org.springframework.integration:spring-integration-jms')
testCompile('org.springframework.boot:spring-boot-starter-test')
}
作为 ActiveMQ Artemis 服务器,我使用默认配置的 vromero/artemis (2.6.0) docker 映像。
问题
在生产者端,消息似乎已成功发送,但在消息代理端,消息丢失。地址已创建,但队列丢失。
话题名称以后会动态的,所以broker.xml不允许我手动创建话题。我依赖 Artemis 的自动队列创建功能。
为什么在这种情况下消息发送不起作用?
Nerd note: I'm aware that Star Gates are basically connected via wormholes in a point-to-point manner but for the sake of the question let's ignore this fact.
当消息被发送到主题并为地址和队列启用自动创建时,只会创建地址而不是队列。如果自动创建一个队列并将消息放入队列中,这将违反主题的语义。主题地址上的订阅队列仅在响应订阅者时创建。因此,在发送消息之前,您需要订阅该主题,否则消息将被丢弃(根据主题语义)。
目标
我想向主题发送消息,稍后我将使用客户端应用程序处理该主题。为此,我使用 Spring Boot 和 Spring Integration Java DSL 及其 JMS 模块。作为消息代理,我使用原生的 ActiveMQ Artemis。
这是我的设置
DemoApplication.java
@SpringBootApplication
public class DemoApplication {
private static final Logger logger = LoggerFactory.getLogger(DemoApplication.class);
public interface StarGate {
void sendHello(String helloText);
}
@Autowired
private ConnectionFactory connectionFactory;
@Bean
public IntegrationFlow mainFlow() {
return IntegrationFlows
.from(StarGate.class)
.handle(Jms.outboundAdapter(connectionFactory)
.configureJmsTemplate(jmsTemplateSpec -> jmsTemplateSpec
.deliveryPersistent(true)
.pubSubDomain(true)
.sessionTransacted(true)
.sessionAcknowledgeMode(Session.CLIENT_ACKNOWLEDGE)
.explicitQosEnabled(true)
)
.destination(new ActiveMQTopic("wormhole")))
.get();
}
public static void main(String[] args) {
ConfigurableApplicationContext context = SpringApplication.run(DemoApplication.class, args);
StarGate stargate = context.getBean(StarGate.class);
stargate.sendHello("Jaffa, kree!");
logger.info("Hello message sent.");
}
}
application.properties
spring.artemis.mode=native
spring.artemis.host=localhost
spring.artemis.port=61616
spring.artemis.user=artemis
spring.artemis.password=simetraehcapa
spring.jms.pub-sub-domain=true
spring.jms.template.delivery-mode=persistent
spring.jms.template.qos-enabled=true
spring.jms.listener.acknowledge-mode=client
logging.level.org.springframework=INFO
build.gradle(重要部分)
springBootVersion = '2.0.2.RELEASE'
dependencies {
compile('org.springframework.boot:spring-boot-starter-artemis')
compile('org.springframework.boot:spring-boot-starter-integration')
compile('org.springframework.integration:spring-integration-jms')
testCompile('org.springframework.boot:spring-boot-starter-test')
}
作为 ActiveMQ Artemis 服务器,我使用默认配置的 vromero/artemis (2.6.0) docker 映像。
问题
在生产者端,消息似乎已成功发送,但在消息代理端,消息丢失。地址已创建,但队列丢失。
话题名称以后会动态的,所以broker.xml不允许我手动创建话题。我依赖 Artemis 的自动队列创建功能。
为什么在这种情况下消息发送不起作用?
Nerd note: I'm aware that Star Gates are basically connected via wormholes in a point-to-point manner but for the sake of the question let's ignore this fact.
当消息被发送到主题并为地址和队列启用自动创建时,只会创建地址而不是队列。如果自动创建一个队列并将消息放入队列中,这将违反主题的语义。主题地址上的订阅队列仅在响应订阅者时创建。因此,在发送消息之前,您需要订阅该主题,否则消息将被丢弃(根据主题语义)。