|
|
@@ -1,5 +1,10 @@
|
|
|
package cn.iocoder.yudao.module.iot.core.messagebus.config;
|
|
|
|
|
|
+import cn.iocoder.yudao.framework.mq.redis.core.RedisMQTemplate;
|
|
|
+import cn.iocoder.yudao.framework.mq.redis.core.job.RedisPendingMessageResendJob;
|
|
|
+import cn.iocoder.yudao.framework.mq.redis.core.job.RedisStreamMessageCleanupJob;
|
|
|
+import cn.iocoder.yudao.framework.mq.redis.core.stream.AbstractRedisStreamMessage;
|
|
|
+import cn.iocoder.yudao.framework.mq.redis.core.stream.AbstractRedisStreamMessageListener;
|
|
|
import cn.iocoder.yudao.module.iot.core.messagebus.core.IotMessageBus;
|
|
|
import cn.iocoder.yudao.module.iot.core.messagebus.core.local.IotLocalMessageBus;
|
|
|
import cn.iocoder.yudao.module.iot.core.messagebus.core.redis.IotRedisMessageBus;
|
|
|
@@ -8,6 +13,7 @@ import cn.iocoder.yudao.module.iot.core.mq.producer.IotDeviceMessageProducer;
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
import org.apache.rocketmq.spring.autoconfigure.RocketMQProperties;
|
|
|
import org.apache.rocketmq.spring.core.RocketMQTemplate;
|
|
|
+import org.redisson.api.RedissonClient;
|
|
|
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
|
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
|
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
|
|
|
@@ -18,6 +24,10 @@ import org.springframework.context.annotation.Configuration;
|
|
|
import org.springframework.data.redis.core.RedisTemplate;
|
|
|
import org.springframework.data.redis.core.StringRedisTemplate;
|
|
|
|
|
|
+import java.util.List;
|
|
|
+
|
|
|
+import static cn.iocoder.yudao.framework.common.util.collection.CollectionUtils.convertList;
|
|
|
+
|
|
|
/**
|
|
|
* IoT 消息总线自动配置
|
|
|
*
|
|
|
@@ -40,7 +50,7 @@ public class IotMessageBusAutoConfiguration {
|
|
|
public static class IotLocalMessageBusConfiguration {
|
|
|
|
|
|
@Bean
|
|
|
- public IotMessageBus iotLocalMessageBus(ApplicationContext applicationContext) {
|
|
|
+ public IotLocalMessageBus iotLocalMessageBus(ApplicationContext applicationContext) {
|
|
|
log.info("[iotLocalMessageBus][创建 IoT Local 消息总线]");
|
|
|
return new IotLocalMessageBus(applicationContext);
|
|
|
}
|
|
|
@@ -55,8 +65,8 @@ public class IotMessageBusAutoConfiguration {
|
|
|
public static class IotRocketMQMessageBusConfiguration {
|
|
|
|
|
|
@Bean
|
|
|
- public IotMessageBus iotRocketMQMessageBus(RocketMQProperties rocketMQProperties,
|
|
|
- RocketMQTemplate rocketMQTemplate) {
|
|
|
+ public IotRocketMQMessageBus iotRocketMQMessageBus(RocketMQProperties rocketMQProperties,
|
|
|
+ RocketMQTemplate rocketMQTemplate) {
|
|
|
log.info("[iotRocketMQMessageBus][创建 IoT RocketMQ 消息总线]");
|
|
|
return new IotRocketMQMessageBus(rocketMQProperties, rocketMQTemplate);
|
|
|
}
|
|
|
@@ -65,17 +75,55 @@ public class IotMessageBusAutoConfiguration {
|
|
|
|
|
|
// ==================== Redis 实现 ====================
|
|
|
|
|
|
+ /**
|
|
|
+ * 特殊:由于 YudaoRedisMQConsumerAutoConfiguration 关于 Redis stream 的消费是动态注册,所以这里只能拷贝相关的逻辑!!!
|
|
|
+ *
|
|
|
+ * @see cn.iocoder.yudao.framework.mq.redis.config.YudaoRedisMQConsumerAutoConfiguration
|
|
|
+ */
|
|
|
@Configuration
|
|
|
@ConditionalOnProperty(prefix = "yudao.iot.message-bus", name = "type", havingValue = "redis")
|
|
|
@ConditionalOnClass(RedisTemplate.class)
|
|
|
public static class IotRedisMessageBusConfiguration {
|
|
|
|
|
|
@Bean
|
|
|
- public IotMessageBus iotRedisMessageBus(StringRedisTemplate redisTemplate) {
|
|
|
+ public IotRedisMessageBus iotRedisMessageBus(StringRedisTemplate redisTemplate) {
|
|
|
log.info("[iotRedisMessageBus][创建 IoT Redis 消息总线]");
|
|
|
return new IotRedisMessageBus(redisTemplate);
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * 创建 Redis Stream 重新消费的任务
|
|
|
+ */
|
|
|
+ @Bean
|
|
|
+ public RedisPendingMessageResendJob iotRedisPendingMessageResendJob(IotRedisMessageBus messageBus,
|
|
|
+ RedisMQTemplate redisTemplate,
|
|
|
+ RedissonClient redissonClient) {
|
|
|
+ List<AbstractRedisStreamMessageListener<?>> listeners = getListeners(messageBus);
|
|
|
+ return new RedisPendingMessageResendJob(listeners, redisTemplate, redissonClient);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 创建 Redis Stream 消息清理任务
|
|
|
+ */
|
|
|
+ @Bean
|
|
|
+ public RedisStreamMessageCleanupJob iotRedisStreamMessageCleanupJob(IotRedisMessageBus messageBus,
|
|
|
+ RedisMQTemplate redisTemplate,
|
|
|
+ RedissonClient redissonClient) {
|
|
|
+ List<AbstractRedisStreamMessageListener<?>> listeners = getListeners(messageBus);
|
|
|
+ return new RedisStreamMessageCleanupJob(listeners, redisTemplate, redissonClient);
|
|
|
+ }
|
|
|
+
|
|
|
+ private List<AbstractRedisStreamMessageListener<?>> getListeners(IotRedisMessageBus messageBus) {
|
|
|
+ return convertList(messageBus.getSubscribers(), subscriber ->
|
|
|
+ new AbstractRedisStreamMessageListener<>(subscriber.getTopic(), subscriber.getGroup()) {
|
|
|
+
|
|
|
+ @Override
|
|
|
+ public void onMessage(AbstractRedisStreamMessage message) {
|
|
|
+ throw new UnsupportedOperationException("不应该调用!!!");
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
}
|
|
|
|
|
|
}
|