1 MQTT基础知识
1.1 MQTT的概念
– MQTT即英文“Message Queuing Telemetry Transport”的缩写,中文翻译为“消息队列遥测传输”
– MQTT协议通过轻量级的消息发布和订阅的模型实现消息传递
– MQTT协议适用于物联网的消息传递(低功率传感器或移动设备,如电话、嵌入式计算机或微控制器)
1.2 MQTT架构的原理
– MQTT协议基于消息发布和订阅主题(pub和sub)的模型实现
– 多个客户端连接到代理并订阅他们感兴趣的主题
– 多个客户端可订阅相同的主题并根据自己的需求处理信息
– 客户端亦可连接到代理并将消息发布到主题
– 代理(broker)与MQTT充当连接所有内容的简单通用接口
关于以上的特性,意味着,
– 如果将订阅的消息转存储到数据库、Twitter、Cosm或简单文本文件的客户端
– 那么将新传感器或其他数据添加到数据库、Twiter等将变得十分简单
1.3 MQTT主题的层次结构
sensors/COMPUTER_NAME/temperature/HARDDRIVE_NAME
如上所示,
– 主题是通过“/”来分隔的体现父子之间的层次关系(类似文件夹的层次结构)
– 以上层次结构可体现不同Computer的不同的硬盘温度
1.4 MQTT订阅消息的匹配方式
1.4.1 显式主题匹配
由于显式主题的订阅是完全匹配的,所以该订阅仅接收到完全匹配的主题消息
1.4.2 模式主题匹配
– 即主题可使用通配符(“+”或者“#”)来模糊匹配主题
– “+”可用于单层次结构的通配符
– “#”可用于剩余层次结构的通配符(注意,必须是订阅主题最后面的层次结构才能匹配)
1.4.3 单层次结构的通配符使用
使用范例1,
sensors/+/temperature/+
如上所示,
– 通过“COMPUTER_NAME”与“HARDDRIVE_NAME”层使用单层次的通配符“+”
– 该模式可获取包含任意计算机名称和任意硬盘驱动器名称的主题消息
使用范例2,
a/b/c/d
如上所示,该主题可以通过以下任意一种模式匹配成功,
+/b/c/d a/+/c/d a/+/+/d +/+/+/+
以下模式的任意一种不能匹配成功,
a/b/c b/+/c/d +/+/+
1.4.4 剩余层次结构的通配符使用
使用范例,
a/b/c/d
如上所示,该主题可以通过以下任意一种模式匹配成功,
# a/# a/b/# a/b/c/# +/b/c/#
1.5 MQTT的QoS
1.5.1 QoS的概念
– QoS即Quality of service的英文缩写,中文翻译服务质量
– QoS用于定义代理与客户端尝试确保接收消息的难度级别
1.5.2 QoS的级别
– MQTT的QoS分三个级别的,分别用QoS 0、QoS 1和QoS2表示
– 更高级别的QoS意味着需要更低的延迟和更高的带宽
– QoS 0,代理与客户端将只传递一次消息且不确认消息是否接受成功
– QoS 1,代理与客户端将只传递一次消息并且确认消息是否接受成功
– QoS 2,代理与客户端将使用四步握手的方式进行一次性消息传递
1.5.3 QoS的接收规则
– 如果以QoS 2发布的消息,客户端以QoS 0订阅,消息以QoS 0传递给客户端
– 如果以QoS 2发布的消息,客户端以QoS 2订阅,消息以QoS 2传递给客户端
– 如果以QoS 0发布的消息,客户端以QoS 2订阅,消息以QoS 0传递给客户端
注:笔者通过以上实例注意到就低原则
1.6 MQTT的消息性能
– MQTT支持将所有消息设置为保留
– 设置保留后,消息发送给所有当前订阅者后,代理将保留该消息
– 如果新订阅与保留消息匹配,则消息将发送到客户端
– 保留设置适用于不经常更新的主题
– 相对于没有保留消息的设置,客户端将需要等待更长的时间才能接收到消息
1.7 MQTT的会话模式
– 建立连接时,客户端设置“clean session”标志(也称“clean start”)
– 如果“clean session”设置为“false”,则建立持久连接
– 如果“clean session”设置为“true”,则建立非持久连接
– 持久连接模式下,客户端断开连接,则保留客户端的所有订阅并存储后续的QoS 1或2消息,直到再次连接
– 非持久连接模式下,客户端断开连接,则删除客户端的所有订阅
1.8 MQTT的消息嘱咐
– 客户端可以向代理申请嘱咐
– 代理在客户端意外断开时发送具有与任何其他消息相同的主题、QoS和保留状态
2 Eclipse Mosquitto的基础知识
2.1 Mosquitto的介绍
– Mosquitto是一个开源消息代理
– Mosquitto基于MQTT协议的5.0、3.1.1和3.1版
– Mosquitto从低功率的单板计算机到服务器都适用
– Mosquitto是Eclipse Foundation的一部分
– Mosquitto是一个iot.eclipse.org项目
2.2 Mosquitto的客户端
– Mosquitto提供用户实现MQTT客户端的C库
– Mosquitto提供流行的命令行MQTT客户端(即mosquitto_pub和mosquitto_sub)
3 最佳实践
3.1 系统环境
3.1.1 系统信息
OS = CentOS 7.6 x86_64
IP Address = 10.168.0.91
HostName = mosquitto.cmdschool.org
3.1.2 安装常用的工具
yum install -y wget vim
3.2 软件环境
3.2.1 软装编译工具
yum -y install gcc gcc-c++ cmake
3.2.2 下载软件包
cd ~ wget https://mosquitto.org/files/source/mosquitto-1.5.8.tar.gz
如果你需要其他版本,请从以下链接下载,
https://mosquitto.org/files/source/
3.2.3 解压软件包
cd ~ tar -xf mosquitto-1.5.8.tar.gz
3.3 编译软件包
3.3.1 预编译软件包
cd ~/mosquitto-1.5.8 cmake . -DCMAKE_INSTALL_BINDIR=/usr/bin \ -DCMAKE_INSTALL_SBINDIR=/usr/sbin \ -DCMAKE_INSTALL_LIBEXECDIR=/usr/libexec \ -DCMAKE_INSTALL_SYSCONFDIR=/etc \ -DCMAKE_INSTALL_LOCALSTATEDIR=/var \ -DCMAKE_INSTALL_LIBDIR=/usr/lib64 \ -DCMAKE_INSTALL_INCLUDEDIR=/usr/include \ -DCMAKE_INSTALL_DATAROOTDIR=/usr/share \ -DCMAKE_INSTALL_INFODIR=/usr/share/info \ -DCMAKE_INSTALL_LOCALEDIR=/usr/share/locale \ -DCMAKE_INSTALL_MANDIR=/usr/share/man \ -DCMAKE_INSTALL_DOCDIR=/usr/share/doc/mosquitto
如果遇到如下错误提示,
CMake Error at /usr/share/cmake/Modules/FindPackageHandleStandardArgs.cmake:108 (message): Could NOT find OpenSSL, try to set the path to OpenSSL root folder in the system variable OPENSSL_ROOT_DIR (missing: OPENSSL_LIBRARIES OPENSSL_INCLUDE_DIR) [...]
你可能需要安装以下包解决依赖关系,
yum install -y openssl-devel
3.3.2 编译并安装软件包
cd ~/mosquitto-1.5.8 make make install
执行完以上命令后,你可能会发现软件并没有按照预编译的目录部署(可能是一个bug),但我们可以使用以下命令手动修正此错误,
mkdir -p /etc/mosquitto mkdir -p /usr/local/share/pkgconfig/ cp /usr/local/etc/mosquitto/mosquitto.conf /etc/mosquitto/mosquitto.conf cp /usr/local/etc/mosquitto/aclfile.example /etc/mosquitto/aclfile.example cp /usr/local/etc/mosquitto/pskfile.example /etc/mosquitto/pskfile.example cp /usr/local/etc/mosquitto/pwfile.example /etc/mosquitto/pwfile.example cp /usr/local/lib/pkgconfig/libmosquitto.pc /usr/local/share/pkgconfig/libmosquitto.pc cp /usr/local/lib/pkgconfig/libmosquittopp.pc /usr/local/share/pkgconfig/libmosquittopp.pc cp /usr/local/lib/libmosquitto.so.1.5.8 /usr/lib64/libmosquitto.so.1.5.8 cp /usr/local/lib/libmosquitto.so.1 /usr/lib64/libmosquitto.so.1 cp /usr/local/lib/libmosquitto.so /usr/lib64/libmosquitto.so cp /usr/local/include/mosquitto.h /usr/include/mosquitto.h cp /usr/local/lib/libmosquittopp.so.1.5.8 /usr/lib64/libmosquittopp.so.1.5.8 cp /usr/local/lib/libmosquittopp.so.1 /usr/lib64/libmosquittopp.so.1 cp /usr/local/lib/libmosquittopp.so /usr/lib64/libmosquittopp.so cp /usr/local/include/mosquittopp.h /usr/include/mosquittopp.h cp /usr/local/bin/mosquitto_pub /usr/bin/mosquitto_pub cp /usr/local/bin/mosquitto_sub /usr/bin/mosquitto_sub cp /usr/local/sbin/mosquitto /usr/sbin/mosquitto cp /usr/local/include/mosquitto_broker.h /usr/include/mosquitto_broker.h cp /usr/local/include/mosquitto_plugin.h /usr/include/mosquitto_plugin.h cp /usr/local/bin/mosquitto_passwd /usr/bin/mosquitto_passwd cp /usr/local/share/man/man1/mosquitto_passwd.1 /usr/share/man/man1/mosquitto_passwd.1 cp /usr/local/share/man/man1/mosquitto_pub.1 /usr/share/man/man1/mosquitto_pub.1 cp /usr/local/share/man/man1/mosquitto_sub.1 /usr/share/man/man1/mosquitto_sub.1 cp /usr/local/share/man/man3/libmosquitto.3 /usr/share/man/man3/libmosquitto.3 cp /usr/local/share/man/man5/mosquitto.conf.5 /usr/share/man/man5/mosquitto.conf.5 cp /usr/local/share/man/man7/mosquitto-tls.7 /usr/share/man/man7/mosquitto-tls.7 cp /usr/local/share/man/man7/mqtt.7 /usr/share/man/man7/mqtt.7 cp /usr/local/share/man/man8/mosquitto.8 /usr/share/man/man8/mosquitto.8
修正软件的安装部署目录后,我们建议你清理异常安装部署的软件,执行以下命令即可,
cd ~/mosquitto-1.5.8 cat install_manifest.txt | xargs rm -rf
3.4 配置软件包
3.4.1 创建运行用户
groupadd -g 309 mosquitto useradd -u 309 -g 309 -d /var/lib/mosquitto/ -s /sbin/nologin mosquitto
配置完毕后,我们建议你使用如下命令验证安装,
mosquitto -v
正常情况下,进程启动你可以看到如下提示,
1564714032: mosquitto version 1.5.8 starting 1564714032: Using default config. 1564714032: Opening ipv4 listen socket on port 1883. 1564714032: Opening ipv6 listen socket on port 1883.
注:按下【Ctrl+C】结束进程启动
3.4.2 定义配置文件
vim /etc/mosquitto/mosquitto.conf
定义如下参数,
pid_file /var/run/mosquitto.pid persistence true persistence_location /var/lib/mosquitto/ log_dest file /var/log/mosquitto.log include_dir /etc/mosquitto/conf.d/
修改完以上配置后,你还需要创建配置文件定义的目录或文件,
mkdir -p /etc/mosquitto/conf.d/ touch /var/log/mosquitto.log chown mosquitto:mosquitto /var/log/mosquitto.log
3.4.3 手动运行测试启动
mosquitto -d -c /etc/mosquitto/mosquitto.conf
关于异常参数,
-d参数,声明mosquitto以守护进程的方式启动
-c参数,声明mosquitto守护进程的配置文件路径
运行完毕后,我们建议你通过如下命令确认mosquitto的启动,
pgrep -a -u mosquitto mosquitto
如果你看到如下输出侧启动正常,
10242 mosquitto -d -c /etc/mosquitto/mosquitto.conf
另外,我们还建议你使用如下命令观察守护进程倾听的端口,
netstat -antp | grep mosquitto
如果你看到如下输出侧启动正常,
tcp 0 0 0.0.0.0:1883 0.0.0.0:* LISTEN 10242/mosquitto tcp6 0 0 :::1883 :::* LISTEN 10242/mosquitto
如果你需要手动结束退出以上守护进程,我建议你使用如下命令,
kill 2 `pgrep -u mosquitto mosquitto`
3.4.4 创建服务控制脚本
vim /usr/lib/systemd/system/mosquitto.service
加入如下配置,
[Unit] Description=Eclipse Mosquitto manager Wants=network.target Before=network.target After=network-pre.target ConditionPathExists=/etc/mosquitto/mosquitto.conf Documentation=man:mosquitto [Service] Type=forking PIDFile=/var/run/mosquitto.pid ExecStart=/usr/sbin/mosquitto -d -c /etc/mosquitto/mosquitto.conf Restart=on-abort [Install] WantedBy=multi-user.target
修改完毕后,你需要重新载入脚本,
systemctl daemon-reload
3.4.5 测试启动脚本
systemctl start mosquitto.service systemctl stop mosquitto.service systemctl restart mosquitto.service systemctl status mosquitto.service
3.4.6 配置服务自启动
systemctl enable mosquitto.service
3.4.7 配置防火墙
firewall-cmd --permanent --add-port=1883/tcp firewall-cmd --reload firewall-cmd --list-all
3.5 测试服务
3.5.1 订阅主题
mosquitto_sub -t 'test/topic' -v
注:以上命令启动后会一直倾听,请不要按【Ctrl+C】等键关闭
3.5.2 发布消息
mosquitto_pub -t 'test/topic' -m 'hello world'
以上需要另外开一个终端,命令执行后,之前订阅主题的终端会接收到发布的消息,具体效果请参阅如下截图,
3.6 卸载服务
3.6.1 清理安装文件
rm -f /etc/mosquitto/mosquitto.conf rm -f /etc/mosquitto/aclfile.example rm -f /etc/mosquitto/pskfile.example rm -f /etc/mosquitto/pwfile.example rm -f /usr/local/share/pkgconfig/libmosquitto.pc rm -f /usr/local/share/pkgconfig/libmosquittopp.pc rm -f /usr/lib64/libmosquitto.so.1.5.8 rm -f /usr/lib64/libmosquitto.so.1 rm -f /usr/lib64/libmosquitto.so rm -f /usr/include/mosquitto.h rm -f /usr/lib64/libmosquittopp.so.1.5.8 rm -f /usr/lib64/libmosquittopp.so.1 rm -f /usr/lib64/libmosquittopp.so rm -f /usr/include/mosquittopp.h rm -f /usr/bin/mosquitto_pub rm -f /usr/bin/mosquitto_sub rm -f /usr/sbin/mosquitto rm -f /usr/include/mosquitto_broker.h rm -f /usr/include/mosquitto_plugin.h rm -f /usr/bin/mosquitto_passwd rm -f /usr/share/man/man1/mosquitto_passwd.1 rm -f /usr/share/man/man1/mosquitto_pub.1 rm -f /usr/share/man/man1/mosquitto_sub.1 rm -f /usr/share/man/man3/libmosquitto.3 rm -f /usr/share/man/man5/mosquitto.conf.5 rm -f /usr/share/man/man7/mosquitto-tls.7 rm -f /usr/share/man/man7/mqtt.7 rm -f /usr/share/man/man8/mosquitto.8
3.6.2 清理服务控制脚本
systemctl disable mosquitto.service rm -rf /usr/lib/systemd/system/mosquitto.service systemctl daemon-reload
参阅文档
===================
Mosquitto的GitHub
—————–
https://github.com/eclipse/mosquitto
下载页面
———–
https://mosquitto.org/download/
版本日志
———–
https://mosquitto.org/blog/
官方首页
———-
https://mosquitto.org/
MQTT的官方首页
————-
http://mqtt.org/
MQTT手册
————-
https://mosquitto.org/man/mqtt-7.html
用于测试的Demo
————-
https://test.mosquitto.org/
cmake的目录配置选项
——————
https://cmake.org/cmake/help/v2.8.12/cmake.html#module:GNUInstallDirs
Mosquitto的守护进程使用手册与配置文件
—————————
https://mosquitto.org/man/mosquitto-8.html
https://mosquitto.org/man/mosquitto-conf-5.html
推荐的UID与GID
——————
https://slackbuilds.org/uid_gid.txt
没有评论