【MQTT】在Linux使用MQTT上报温度到阿里云

前言

在上几篇文章中我们用MQTT.fx模拟客户端实现了与阿里云物联网平台的双向通信,接下来我们自己动手编程使用mosquitto库实现一个发布端。

iniparser配置文件

iniparser介绍
与阿里云进行通信时,我们需要如下一些信息,例如在使用MQTT.fx时:
在这里插入图片描述
阿里云配置和MQTT.fx使用
在上篇文章中阿里云配置文件另外之前发布和订阅的topic也要记下来:
在这里插入图片描述
这些都是我们在项目中要用到的配置信息,具体应用在:

#客户端id
clientid:
struct mosquitto *mosquitto_new( const char * id, bool clean_session, void * obj )
#连接阿里云的端口,主机名
brokeraddress and brokerport:
int mosquitto_connect( struct mosquitto * mosq, const char * host, int port, int keepalive )
#连接阿里云账号和密码
username and password:
int mosquitto_username_pw_set(struct mosquitto * mosq, const char *username, const char *password )
#订阅的主题
topic:
int mosquitto_publish( struct mosquitto * mosq, int * mid, const char * topic, int payloadlen, const void * payload, int qos, bool retain )

当然可以直接写死在代码里,但是这样一来,代码复用性不高,换一个设备那么就要改一次源码来更改设备配置信息。MQTT主机使用可以看这篇博客.

所以我们写一个.ini文件,这样当我们更换设备的时候只需要修改这个.ini文件中的信息即可。然后在代码中使用iniparser库中API。

cJSON

JSON(JavaScript Object Notation, JS 对象简谱) 是一种轻量级的数据交换格式。它基于 ECMAScript (欧洲计算机协会制定的js规范)的一个子集,采用完全独立于编程语言的文本格式来存储和表示数据。简洁和清晰的层次结构使得 JSON 成为理想的数据交换语言。 易于人阅读和编写,同时也易于机器解析和生成,并有效地提升网络传输效率。

阿里云上payload must be json format 负载(消息)必须为cJSON格式。
因为使用的阿里云提供的标准物模型,所以消息格式应该遵循阿里云的规范。这就是我们要使用cJSON的原因。

要是已其他格式发送,阿里云收到了消息,但是因为参数不匹配,无法对消息进行解析,也就无法获取到温度消息
学习cJSON可以参考博客

sqlite3数据库

SQLite是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。它是一个零配置的数据库,这意味着与其他数据库不一样,您不需要在系统中配置。
就像其他数据库,SQLite 引擎不是一个独立的进程,可以按应用程序需求进行静态或动态连接。SQLite 直接访问其存储文件。

sqlite3数据库优点

1.不需要一个单独的服务器进程或操作的系统(无服务器的)。
2.SQLite 不需要配置,这意味着不需要安装或管理。
3. 一个完整的 SQLite数据库是存储在一个单一的跨平台的磁盘文件。
4. SQLite 是非常小的,是轻量级的,完全配置时小于 400KiB,省略可选功能配置时小于250KiB。
5.SQLite 是自给自足的,这意味着不需要任何外部的依赖。
6 SQLite 事务是完全兼容ACID 的,允许从多个进程或线程安全访问。
7.SQLite 支持 SQL92(SQL2)标准的大多数查询语言的功能。
8. SQLite 使用ANSI-C 编写的,并提供了简单和易于使用的 API。
9. SQLite 可在 UNIX(Linux, Mac OS-X, Android,iOS)和 Windows(Win32, WinCE, WinRT)中运行。

sqlite3库函数

流程图

在这里插入图片描述

配置信息

#include "iniparser.h"
#include "mqtt_conf.h"
#include <stdio.h>
#include <string.h>

/* 载入设置参数 */
int get_config(char *path,mqtt_user_data *mqtt,int mode)
{
    dictionary      *ini = NULL;
    const char      *brokeraddress;
    int             brokerport;
    const char      *username;
    const char      *password;
    const char      *clientid;
    const char      *topic;
    int             Qos;


    const char      *method;
    const char      *time;
    const char      *jsonid;
    const char      *identifier;
    const char      *version;




    if(!path || !mqtt)
    {
        printf("invalid_argument: %s\n",__FUNCTION__);
        return -1;
    }
    ini = iniparser_load(path);

    if(!ini)
    {
        printf("iniparser_load failed!\n");
        return -1;
    }


    brokeraddress = iniparser_getstring(ini,"address:host",DEFAULT_BROKER_ADDRESS);
    brokerport= iniparser_getint(ini,"address:port",DEFAULT_BROKER_PORT);
    username = iniparser_getstring(ini,"user_name_pwd:username",DEFAULT_USERNAME);
    password = iniparser_getstring(ini,"user_name_pwd:pwd",DEFAULT_PASSWORD);
    clientid = iniparser_getstring(ini,"clientid:id",DEFAULT_CLIENTID);
    identifier = iniparser_getstring(ini,"ali_json:identifier",DEFAULT_JSONID);



    if(mode == SUB)
    {
        topic = iniparser_getstring(ini,"sub_topic:topic",DEFAULT_SUB_TOPIC);

    }

    else if(mode == PUB)
    {

        topic = iniparser_getstring(ini,"pub_topic:topic",DEFAULT_PUB_TOPIC);
        method = iniparser_getstring(ini,"ali_json:method",DEFAULT_METHOD);
        //time = iniparser_getstring(ini,"ali_time:time",DEFAULT_TIME);
       //jsonid = iniparser_getstring(ini,"ali_json:id",DEFAULT_JSONID);
        version = iniparser_getstring(ini,"ali_json:version",DEFAULT_VERSION);
    }

    else if(mode != SUB && mode != PUB)
    {
        printf("invalid_argument : %s mode!\n",__FUNCTION__);
        return -1;
    }

    strncpy(mqtt->brokeraddress,brokeraddress,SIZE);
    mqtt->brokerport = brokerport;
    mqtt->Qos        = Qos;
    strncpy(mqtt->username,username,SIZE);
    strncpy(mqtt->password,password,SIZE);
    strncpy(mqtt->clientid,clientid,SIZE);
    strncpy(mqtt->topic,topic,SIZE);          
    strncpy(mqtt->identifier,identifier,SIZE);

    if(mode == PUB)
    {
        strncpy(mqtt->method,method,SIZE);
        //strncpy(mqtt->time,time,SIZE);
       //strncpy(mqtt->jsonid,jsonid,SIZE);
        strncpy(mqtt->version,version,SIZE);
    }


    iniparser_freedict(ini);
    return 0;

}
#ifndef MQTT_CONF_H
#define MQTT_CONF_H


#define  SIZE                1024
#define KEEP_ALIVE          60

enum{

    SUB ,
    PUB
};


typedef struct mqtt_user_data
{
    char    brokeraddress[SIZE] ;
    int     brokerport;
    char    username[SIZE];
    char    password[SIZE];
    char    clientid[SIZE];
    char    topic[SIZE];
    int     Qos;

    char    method[SIZE] ;
    char    time[SIZE];
    char    jsonid[SIZE] ;
    char    identifier[SIZE] ;
    char    version[SIZE] ;
    


}mqtt_user_data;


int set_config(char *path,char *host,int port, char *clientid,char *uname,char *pwd,char *topic);

int get_config(char *path, mqtt_user_data *mqtt, int mode);

#endif
[address]
host    =       iot-06z00i99uliom9a.mqtt.iothub.aliyuncs.com
port    =       1883


[user_name_pwd]
username    =   ds18b20&gr71iZ9S48c
pwd         =   2a59fde4b9765b403df5bfe7ce63b06cfb79e7d57cb373a0dfa501f79623cc83


[clientid]
id          =    gr71iZ9S48c.ds18b20|securemode=2,signmethod=hmacsha256,timestamp=2524608000000|

[pub_topic]
topic       =   /sys/gr71iZ9S48c/ds18b20/thing/event/property/post


[sub_topic]
topic       =   /sys/gr71iZ9S48c/ds18b20/thing/service/property/set


[ali_json]
method      =   thing.service.property.set
id          =   ip    
identifier  =   CurrentTemperature
version     =   1.0.0


[KEEP_ALIVE]
alive   =   60

[ali_Qos]
Qos     =   0

发布端代码实现

#include <stdio.h>
#include <errno.h>
#include <unistd.h>
#include <libgen.h>
#include <getopt.h>
#include <string.h>
#include <mosquitto.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <dirent.h>
#include <signal.h>
#include <time.h>


#include "mqtt_conf.h"
#include "cJSON.h"
#include "dictionary.h"
#include "iniparser.h"
#include "ifconfig.h"
#include "db_sqlite3.h"


#define  PROG_VERSION "1.0.0"
#define  INI_PATH "./mqtt_aliyun.ini"



static int g_stop = 0;

/* 调用信号 */
void sig_handler(int sig_num)
{
    if(sig_num == SIGUSR1)
    g_stop = 1;

}

int db_init(sqlite3  *db,char *zErrMsg);
int db_table(char *zErrMsg, sqlite3 *db);
int db_store(sqlite3 *db,char *zErrMsg);
int get_temperature(float *temper);
int get_time (char *datetime,int len);
static inline void print_usage(char *progname);
char *create_json(void *obj);
void connect_callback(struct mosquitto *mosq,void *obj,int rc);


int main (int argc, char **argv)
{

    char                        *progname = basename(argv[0]);
    int                         brokerport = 1883;
    char                        *brokeraddress = NULL;
    char                        *clientid = NULL;
    char                        *username = NULL;
    char                        *password = NULL;
    char                        *topic = NULL;

    struct mosquitto            *mosq = NULL;
    int                         daemon_run = 0;
    int                         opt = -1;
    int                         rv;
    int                         log_fd;

    struct mqtt_user_data       mqtt;

    struct option options[] = 
    {
        {"daemon",no_argument,NULL,'d'},
        {"topic", required_argument,NULL,'t'},
        {"brokeraddress", required_argument,NULL,'H'},
        {"clientid", required_argument, NULL, 'i'},
        {"brokerport",required_argument,NULL,'p'},
        {"help",no_argument,NULL,'h'},
        {"username",required_argument,NULL,'u'},
        {"password",required_argument,NULL,'P'},
        {NULL,0,NULL,0}
    };

    while((opt = getopt_long(argc,argv,"dhp:t:i:u:P:H:",options,NULL)) != -1)
    {
        switch(opt)
        {
            case 't':
                topic = optarg;
                break;

            case 'i':
                clientid = optarg;
                break;

            case 'H':
                brokeraddress = optarg;
                break;
            case 'u':
                username = optarg;
                break;
            case 'P':
                password = optarg;
                break;
            case 'd':
                daemon_run = 1;
                break;
            case 'p':
                brokerport = atoi(optarg);
                break;
            case 'h':
                print_usage(argv[0]);
                return 0;
            default:
                break;
        }
    }


    /* 创建日志 */
    if(daemon_run)
    {
        printf("program %s running in backgrund\n", progname);
        if( (log_fd = open("client.log", O_CREAT|O_RDWR, 0666)) < 0)
        {
            printf("open() failed:%s\n", strerror(errno)) ;
            return -2;
        }

        dup2(log_fd, STDOUT_FILENO);
        dup2(log_fd, STDERR_FILENO);
        
        daemon(1,1);
    }

    /* 安装信号 */
    signal(SIGUSR1,sig_handler);

    
    /* 载入配置文件 */
    memset(&mqtt,0,sizeof(mqtt));
    rv = get_config(INI_PATH,&mqtt,PUB);

    /* MQTT 初始化 */
    mosquitto_lib_init();


    /* 创建MQTT 客户端 */
    mosq = mosquitto_new(mqtt.clientid,true,(void *)&mqtt);

    if(!mosq)
    {
        printf("mosquitto_new() failed: %s\n",strerror(errno));
        goto cleanup;
        return -1;
    }
    

    /* 设置账号密码 */
    if(mosquitto_username_pw_set(mosq,mqtt.username,mqtt.password) != MOSQ_ERR_SUCCESS)
    {
        printf("mosquitto_username_pw_set failed: %s\n",strerror(errno));
        goto cleanup;
    }
    printf("Create mosquitto successfully!\n");

    /* 回调函数 */
    mosquitto_connect_callback_set(mosq,connect_callback);

    while(!g_stop)
    {
        /* 连接MQTT服务器,ip,端口,时间 */ 
        if(mosquitto_connect(mosq, mqtt.brokeraddress,mqtt.brokerport,KEEP_ALIVE) != MOSQ_ERR_SUCCESS)
        {
            printf("mosquitto_connect() failed: %s\n",strerror(errno));
            goto cleanup;
        }
        printf("connect successfully\n");

        /*无阻塞 断线连接 */
        mosquitto_loop_forever(mosq,-1,1);
        sleep(3);
    }

/* 释放客户端,清除 */
cleanup:
    close(log_fd);
    mosquitto_destroy(mosq);
    mosquitto_lib_cleanup();
    return 0;
} 


/* 帮助信息 */
void print_usage(char *progname)
{
    printf("%s usage:\n",progname);
    printf("Example: %s -h ${brokeraddress} -p ${brokerport} -i ${clientid} -u ${username} -p ${password} -t${topic} -h ${help} -d ${daemon}\n",progname);
    printf("-h(--host): sepcify brokeraddress.\n");
    printf("-p(--port): sepcify brokerport.\n");
    printf("-h(--Help): print this help information.\n");
    printf("-d(--daemon): set program running on background.\n");
    printf("-i(--clientid): sepcify the clientid.\n");
    printf("-u(--username): sepcify username of the client.\n");
    printf("-p(--password): sepcify password of the username.\n");
    printf("-t(--topic): sepcify topic of the client.\n");
    printf("-d(--daemon): running in backgrund.\n");
}


/* cJSON打包数据 */
void connect_callback(struct mosquitto *mosq,void *obj,int rc)
{

    char                    ipaddr[16];
    char                    *interface="eth0";
    char                    datetime[64];
    cJSON                   *root;
    cJSON                   *item;
    char                    *msg;
    struct mqtt_user_data   *mqtt;
    
    char                    buf[2048];

    printf("Connection successful cJSON call packaging\n");

    float temper = 0.00000000;

    if(get_temperature(&temper) < 0)
    {
        printf("get_temperature failed.\n");
        return;
    }

    if(get_time(datetime,sizeof(datetime))<0)
    {
        printf("get_time failure\n");
        return ;
    }

    memset(ipaddr,0,sizeof(ipaddr));
    if(get_ipaddr(interface,ipaddr,sizeof(ipaddr))<0)
    {
        printf("ERROR:get ip address failure\n");
        return ;
    }

    snprintf(buf,sizeof(buf),"%s/%s/%f",ipaddr,datetime,temper);
    sqlite_tem(buf);
    printf("Data transfer database successfully!\n");
    printf("\n");

    root = cJSON_CreateObject();
    item = cJSON_CreateObject();


    if(!obj)
    {
        printf("invalid_argument in %s\n",__FUNCTION__);
        return;
    }

    mqtt = (mqtt_user_data *)obj;

    cJSON_AddItemToObject(root,"method",cJSON_CreateString(mqtt->method));
    cJSON_AddItemToObject(root,"id",cJSON_CreateString(ipaddr));
    cJSON_AddItemToObject(root,"time",cJSON_CreateString(datetime));
    cJSON_AddItemToObject(root,"params",item);
    cJSON_AddItemToObject(item,"CurrentTemperature",cJSON_CreateNumber(temper));
    cJSON_AddItemToObject(root,"version",cJSON_CreateString(mqtt->version));
   
    msg = cJSON_Print(root);
    printf("%s\n",msg);


    if(!rc)
    {
        if(mosquitto_publish(mosq,NULL,mqtt->topic,strlen(msg),msg,mqtt->Qos,NULL) != MOSQ_ERR_SUCCESS)
        {
            printf("mosquitto_publish failed: %s\n",strerror(errno));
            return;
        }
    }

    mosquitto_disconnect(mosq);

}


/* 获取温度 */
int get_temperature(float *temper)
{
    char            w1_path[64] = "/sys/bus/w1/devices/";
    char            chip[32];
    char            buf[128];
    DIR            *dirp;
    struct dirent  *direntp;
    int             fd = -1;
    char           *ptr;
    int             found = 0;

    if( !temper )
    {
        return -1;
    }

    if((dirp = opendir(w1_path)) == NULL)
    {
        printf("Opendir '%s' error: %s\n", w1_path, strerror(errno));
        return -2;
    }

    while((direntp = readdir(dirp)) != NULL)
    {
        if(strstr(direntp->d_name,"28-"))
        {
            strncpy(chip, direntp->d_name, sizeof(chip));
            found = 1;
            break;
        }
    }
    closedir(dirp);

    if( !found )
    {
        printf("Can not find ds18b20 in '%s'\n", w1_path);
        return -3;
    }

    strncat(w1_path, chip, sizeof(w1_path)-strlen(w1_path));
    strncat(w1_path, "/w1_slave", sizeof(w1_path)-strlen(w1_path));

    if( (fd=open(w1_path, O_RDONLY)) < 0 ) 
    { 
        printf("open %s error: %s\n", w1_path, strerror(errno)); 
        return -4;
    } 

    if(read(fd, buf, sizeof(buf)) < 0)
    {
        printf("read %s error: %s\n", w1_path, strerror(errno)); 
        return -5;
    } 

    ptr = strstr(buf, "t=");
    if( !ptr )
    {
        printf("ERROR: Can not get temperature\n");
        return -6;
    }

    *temper = atof(ptr+=2)/1000;

    close(fd);
    return 0;
}

/* 获取时间 */
int get_time(char *datime,int len)
{
    time_t          t;
    struct tm       *p;

    time(&t);
    p = gmtime(&t);

    snprintf(datime, 32, "%d-%d-%d %d:%d:%d", 1900+p->tm_year,1+p->tm_mon, p->tm_mday, (p->tm_hour + 8), p->tm_min, p->tm_sec);
    return 0 ;
}

运行结果

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
可以看到温度在实时更新(我设置睡眠时间短,所有没有显示,温度变化).

项目源码地址

posted @ 2022-09-25 19:39  西故黄鹤楼  阅读(582)  评论(0编辑  收藏  举报