springboot整合mqtt进行消息的发布及订阅
一、.导入maven包
<!--mqtt-->
<dependency>
<groupId>org.springframework.integration</groupId>
<artifactId>spring-integration-mqtt</artifactId>
</dependency>
二、添加mqtt的配置
#mqtt的配置
mqtt:
server:
url: 192.168.137.59,192.168.137.4
port: 1883
username: admin
password: 111111
client:
consumerId: consumerCo
publishId: publishCo
default:
topic: topic
completionTimeout: 3000
三、mqtt客户端配置
MqttConfig订阅消息(动态添加topic,不用写死):
import com.cecjx.common.utils.RedisUtil;
import com.cecjx.device.domain.TopicDO;
import lombok.extern.slf4j.Slf4j;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.integration.annotation.ServiceActivator;
import org.springframework.integration.channel.DirectChannel;
import org.springframework.integration.core.MessageProducer;
import org.springframework.integration.mqtt.core.DefaultMqttPahoClientFactory;
import org.springframework.integration.mqtt.core.MqttPahoClientFactory;
import org.springframework.integration.mqtt.inbound.MqttPahoMessageDrivenChannelAdapter;
import org.springframework.integration.mqtt.outbound.MqttPahoMessageHandler;
import org.springframework.integration.mqtt.support.DefaultPahoMessageConverter;
import org.springframework.messaging.MessageChannel;
import org.springframework.messaging.MessageHandler;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutionException;
/**
* @author huangquanguang
* @date 2020/1/3 16:33
* @description mqtt配置类
* outbound 是发送消息到mqtt broker inbound是接收mqtt消息进行处理
*/
@Slf4j
@Configuration
public class MqttConfig {
@Autowired
private MongoTemplate mongoTemplate;
@Value("${mqtt.server.url}")
private String url = "192.168.137.1";
@Value("${mqtt.server.port}")
private String port = "1883";
@Value("${mqtt.server.username}")
private String username = "admin";
@Value("${mqtt.server.password}")
private String password = "admin";
@Value("${mqtt.client.consumerId}")
private String consumerId = "consumerClient";
@Value("${mqtt.client.publishId}")
private String publishId = "publishClient";
@Value("${mqtt.default.topic}")
private String topic = "topic";
@Value("${mqtt.default.completionTimeout}")
private Integer completionTimeout = 3000;
/**
* @author huangquanguang
* @date 2020/1/3 16:34
* @description 创建MqttPahoClientFactory,设置MQTT Broker连接属性,如果使用SSL验证,也在这里设置。
*/
@Bean
public MqttPahoClientFactory mqttClientFactory() {
DefaultMqttPahoClientFactory factory = new DefaultMqttPahoClientFactory();
MqttConnectOptions options = new MqttConnectOptions();
options.setUserName(username);
options.setPassword(password.toCharArray());
//多个服务器地址时处理
String[] strings = url.split(",");
String[] serverUrls = new String[strings.length];
if (strings.length > 0) {
for (int i = 0; i < strings.length; i++) {
String serverUrl = "tcp://" + strings[i] + ":" + port;
serverUrls[i] = serverUrl;
}
}
options.setServerURIs(serverUrls);
factory.setConnectionOptions(options);
return factory;
}
//--------------------------------- 接收消息部分 ------------------------------------------
/**
* @param
* @return adapter
* 接收消息 配置及订阅消息
* @author huangquanguang
* @date 2020/1/3 16:34
*/
@Bean
public MessageProducer inbound() {
//这里要设置接收的topic,要写成动态添加的方式,后续新增设备时不用设置
//获取所有要订阅的topic
// List<TopicDO> topicDOS = topicService.listByMap(map);
List<TopicDO> topicDOS = new ArrayList<>();
//启动项目执行:把所有当前主题保存到redis缓存
RedisUtil redisUtil = new RedisUtil();
redisUtil.addObject("topics", topicDOS);
String[] topics = new String[topicDOS.size()];
if (topicDOS != null && topicDOS.size() > 0) {
for (int i = 0; i < topicDOS.size(); i++) {
topics[i] = topicDOS.get(i).getTopic();
}
}
MqttPahoMessageDrivenChannelAdapter adapter = new MqttPahoMessageDrivenChannelAdapter(consumerId,
mqttClientFactory(), topics);
adapter.setCompletionTimeout(completionTimeout);
adapter.setConverter(new DefaultPahoMessageConverter());
// 设置服务质量
// 0 最多一次,数据可能丢失;
// 1 至少一次,数据可能重复;
// 2 只有一次,有且只有一次;最耗性能
adapter.setQos(1);
adapter.setOutputChannel(mqttInputChannel());
return adapter;
}
@Bean
public MessageChannel mqttInputChannel() {
return new DirectChannel();
}
/**
* @author huangquanguang
* @date 2020/1/3 16:35
* @description 消费消息
* ServiceActivator注解表明当前方法用于处理MQTT消息,inputChannel参数指定了用于消费消息的channel。
*/
@Bean
@ServiceActivator(inputChannel = "mqttInputChannel")
public MessageHandler handler() {
return message -> {
String payload = message.getPayload().toString();
String topic = message.getHeaders().get("mqtt_receivedTopic").toString();
//用线程池处理订阅到的消息
try {
dealMessageByThreadPool(payload, topic);
} catch (ExecutionException e) {
e.printStackTrace();
log.error("处理订阅出错:" + e.getMessage());
} catch (InterruptedException e) {
e.printStackTrace();
}
};
}
/**
* @param
* @return 多线程处理收到的传感器推送, 线程处理参考:
* https://www.bbsmax.com/A/mo5kxvPvdw/(这里注释掉线程的使用,保持数据的一致性
* 在处理某一次上报的数据时可以用多线程,提高效率,不能每次订阅数据都起一个线程让它单独处理)
* 要根据厂商提交的设备id及主题先把所有设备的主题管理起来,一个设备对应一个或多个主题
* @author huangquanguang
* @date 2020/2/21 11:54
*/
public void dealMessageByThreadPool(String payload, String topic) throws ExecutionException, InterruptedException {
//保存mongodb部分
log.info(": 处理消息 " + payload);
//通用保存到mongodb的方法(集合存在则新增,不存在则先创建再新增)
mongoTemplate.insert(payload, topic);
}
//--------------------------------发送消息部分-------------------------------------------
/**
* @author huangquanguang
* @date 2020/1/3 16:37
* @description 发送消息
*/
@Bean
public MessageChannel mqttOutboundChannel() {
return new DirectChannel();
}
/*****
* 发送消息和消费消息Channel可以使用相同MqttPahoClientFactory
* @return
*/
@Bean
@ServiceActivator(inputChannel = "mqttOutboundChannel")
public MessageHandler outbound() {
// 在这里进行mqttOutboundChannel的相关设置
MqttPahoMessageHandler messageHandler =
new MqttPahoMessageHandler(publishId, mqttClientFactory());
messageHandler.setAsync(true); //如果设置成true,发送消息时将不会阻塞。
messageHandler.setDefaultTopic(topic);
return messageHandler;
}
}
四、发布消息
1.MqttController :
import com.cecjx.common.annotation.Log;
import com.cecjx.common.utils.R;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;
import java.util.Map;
/**
* @author huangquanguang
* @date 2020/1/3 16:30
* @description mqtt发布消息
*/
@Controller
@RequestMapping(value = "mqtt")
public class MqttController {
@Autowired
private MqttGateway mqttGateway;
/**
* @param topic:主题 message:内容
* @return send message
* @author huangquanguang
* @date 2020/1/3 16:31
*/
@RequestMapping("/send")
@ResponseBody
public R send(@RequestParam Map params) {
// 发送消息到指定topic
try {
String topic = params.get("topic").toString();
String message = params.get("message").toString();
mqttGateway.sendToMqtt(topic, message);
return R.ok().put("message","send message :"+message);
} catch (Exception e) {
e.printStackTrace();
return R.error();
}
}
}
2.MqttGateway :
import org.springframework.integration.annotation.MessagingGateway;
import org.springframework.integration.mqtt.support.MqttHeaders;
import org.springframework.messaging.handler.annotation.Header;
import org.springframework.stereotype.Component;
/**
* @author huangquanguang
* @date 2020/1/3 16:41
* @description mqtt网关
* 定义重载方法,用于消息发送
*/
@Component
@MessagingGateway(defaultRequestChannel = "mqttOutboundChannel")
public interface MqttGateway {
// 定义重载方法,用于消息发送
void sendToMqtt(String payload);
// 指定topic进行消息发送
void sendToMqtt(@Header(MqttHeaders.TOPIC) String topic, String payload);
void sendToMqtt(@Header(MqttHeaders.TOPIC) String topic, @Header(MqttHeaders.QOS) int qos, String payload);
}
五、测试
1.请求MqttController 的send接口,发布消息
2.在MqttConfig订阅消息中可以打印订阅到的消息并保存到mongodb