大家好,今天小白给大家简单介绍下Mosquitto,它是一个开源的MQTT broker,欢迎一起讨论学习。
Mosquitto 是一个开源的 MQTT broker ,目前支持 v3.1 和 v3.1.1 协议 ,同时提供了一个 C 语言动态链接库 libmosquitto ,用于实现 mqtt 客户端
1、主要的几个源码目录
源码目录层级比较多,但是需关注的目录有/mosquitto-1.5.8/src、/mosquitto-1.5.8/lib、/mosquitto-1.5.8/client三个目录,其中/src和/lib目录下主要放置服务端(Broker)的实现代码以及部分底层与网络相关的操作,client目录主要是订阅客户端和发布客户端的实现源码。
2、客户端
订阅客户端实现源码主要是在/client/sub_client.c文件中,main函数主要流程:
a.调用client_config_load函数加载配置文件,初始化cfg变量以及命令行参数解析,cfg变量类型是struct mosq_config,客户端(订阅端和发布端)配置相关的变量。
b. 调用client_id_generate函数,生成client id;id形如:mosqsub|pid-hostname。其中pid为该客户端进程的进程号,hostname是主机名。
c. 调用mosquitto_new函数,为订阅客户端分配空间,返回一个客户端的操作句柄。
d. 调用client_opts_set接口,根据cfg变量的标志位(will_topic,username,max_inflightprotocol_version等)去初始化客户端遗嘱相关项,用户名,密码,支持的协议版本等。
e. 接下来就是给客户端绑定回调函数,即给客户端的函数指针成员变量(on_log,on_subscribe, on_connect_with_flags, on_message)赋值。
f. 然后调用client_connect函数,这个函数是用于初始化客户端host,port,keepalive,sockpairR,sockpairW,state等成员,并创建sock成员处理连接,以及在send__connect函数中组织连接请求的报文,并放到报文队列,然后依次发送出去。
d. 最后就是进入到mosquitto_loop_forever函数,该函数是一个死循环,用于处理与服务端的数据交互。详细交互过程可以追踪源码。
发布客户端实现的源码主要是在/client/pub_client.c文件中,main函数主要流程:
a. 同样的调用client_config_load函数,初始化cfg变量以及做命令行参数的解析。
b. 初始化一些全局变量,供回调函数使用;如:topic,qos,retain,username,password等。
c. 调用client_id_generate函数,生成发布客户端id。
d. 调用mosquitto_new,为发布客户端分配空间,返回一个客户端的操作句柄。
e. 接下来也是设置一些回调;
f. 调用client_opts_set接口,同理目的是为了初始化发布客户端。
g. 调用client_connect函数,同理处理连接相关操作。
h. 最后是一个do while循环,处理与服务端交互相关的操作。
综上可以看出来,订阅客户端和发布客户端有部分操作是相似的,相同的处理逻辑放在:clinet.share.c中,阅读代码时可以对比理解。
3、服务端(broker,即代理)
服务端的实现逻辑主要是在/lib和/src目录下,main函数所在文件是:mosquitto.c,其大致流程是:
a. 调用net__broker_init函数,创建套接字。
b. 调用config__init函数,初始化服务端配置相关项。配置相关的结构体:struct mosquitto__config。
c. 调用config__parse_args,根据配置文件和命令行参数初始化服务端配置。
d. 调用db__open函数,主要是创建了订阅树。该变量是服务端最重要的结构体,维护了订阅树根节点,订阅客户端的索引,消息链表等。
e. 调用mosquitto_security_module_init函数,给db中安全认证相关的成员变量初始化。
f. 调用mosquitto_security_init,主要是函数指针调函数(上一步已经给函数指针赋值),实现安全认证相关的操作。
g. 根据config.listener_count数目,循环调用net__socket_listen函数,在该函数中创建监听套接字,设置套接字属性,并开始监听。
h. 接下来将所有config.listener[i]的套接字统一放到listensock数组中去管理;
i. 调用drop_privileges函数,如果以root身份运行,此函数将尝试更改为非特权用户和组。
j. 接下来是调用signal函数,为信号注册信号处理函数,实现消息的异步通知。
k. 最后进入到mosquitto_main_loop函数,该函数的主要功能就是处理客户端的订阅以及发布等请求。
函数调用流程:
调用sysconf函数获取pollfd的最大值pollfd_max,病据此pollfds分配空间;
while大循环 中:
填充pollfds;
根据客户端套接字遍历哈希表,并返回查找到的客户端;
db__message_write //根据client的inflight_msgs消息的state进行相应的处理。
根据客户端id遍历哈希表,并返回查找到的客户端;
调用poll函数,获取准备好的描述符。
loop_handle_reads_writes
HASH_ITER(遍历哈希表)
packet__write
net_write
write packet__read --->packet_mosq.c
handle__packet --->/src/read_handle.c (处理连接,订阅,发布等等)
4、总结
本篇主要是简单粗略的介绍了Mosquitto源码的中订阅客户端,发布客户端,服务端(broker,即代理)的源码调用流程,欢迎一起交流学习。