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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章