mirror of
				https://github.com/espressif/ESP8266_RTOS_SDK.git
				synced 2025-10-25 05:25:06 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			895 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			895 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <stdio.h>
 | |
| #include "platform.h"
 | |
| 
 | |
| #include "mqtt_client.h"
 | |
| #include "mqtt_msg.h"
 | |
| #include "esp_transport.h"
 | |
| #include "esp_transport_tcp.h"
 | |
| #include "esp_transport_ssl.h"
 | |
| #include "esp_transport_ws.h"
 | |
| #include "platform.h"
 | |
| #include "mqtt_outbox.h"
 | |
| 
 | |
| /* using uri parser */
 | |
| #include "http_parser.h"
 | |
| 
 | |
| static const char *TAG = "MQTT_CLIENT";
 | |
| 
 | |
| typedef struct mqtt_state
 | |
| {
 | |
|     mqtt_connect_info_t *connect_info;
 | |
|     uint8_t *in_buffer;
 | |
|     uint8_t *out_buffer;
 | |
|     int in_buffer_length;
 | |
|     int out_buffer_length;
 | |
|     uint32_t message_length;
 | |
|     uint32_t message_length_read;
 | |
|     mqtt_message_t *outbound_message;
 | |
|     mqtt_connection_t mqtt_connection;
 | |
|     uint16_t pending_msg_id;
 | |
|     int pending_msg_type;
 | |
|     int pending_publish_qos;
 | |
|     int pending_msg_count;
 | |
| } mqtt_state_t;
 | |
| 
 | |
| typedef struct {
 | |
|     mqtt_event_callback_t event_handle;
 | |
|     int task_stack;
 | |
|     int task_prio;
 | |
|     char *uri;
 | |
|     char *host;
 | |
|     char *path;
 | |
|     char *scheme;
 | |
|     int port;
 | |
|     bool auto_reconnect;
 | |
|     void *user_context;
 | |
|     int network_timeout_ms;
 | |
| } mqtt_config_storage_t;
 | |
| 
 | |
| typedef enum {
 | |
|     MQTT_STATE_ERROR = -1,
 | |
|     MQTT_STATE_UNKNOWN = 0,
 | |
|     MQTT_STATE_INIT,
 | |
|     MQTT_STATE_CONNECTED,
 | |
|     MQTT_STATE_WAIT_TIMEOUT,
 | |
| } mqtt_client_state_t;
 | |
| 
 | |
| struct esp_mqtt_client {
 | |
|     esp_transport_list_handle_t transport_list;
 | |
|     esp_transport_handle_t transport;
 | |
|     mqtt_config_storage_t *config;
 | |
|     mqtt_state_t  mqtt_state;
 | |
|     mqtt_connect_info_t connect_info;
 | |
|     mqtt_client_state_t state;
 | |
|     long long keepalive_tick;
 | |
|     long long reconnect_tick;
 | |
|     int wait_timeout_ms;
 | |
|     int auto_reconnect;
 | |
|     esp_mqtt_event_t event;
 | |
|     bool run;
 | |
|     bool wait_for_ping_resp;
 | |
|     outbox_handle_t outbox;
 | |
|     EventGroupHandle_t status_bits;
 | |
| };
 | |
| 
 | |
| const static int STOPPED_BIT = BIT0;
 | |
| 
 | |
| static esp_err_t esp_mqtt_dispatch_event(esp_mqtt_client_handle_t client);
 | |
| static esp_err_t esp_mqtt_set_config(esp_mqtt_client_handle_t client, const esp_mqtt_client_config_t *config);
 | |
| static esp_err_t esp_mqtt_destroy_config(esp_mqtt_client_handle_t client);
 | |
| static esp_err_t esp_mqtt_connect(esp_mqtt_client_handle_t client, int timeout_ms);
 | |
| static esp_err_t esp_mqtt_abort_connection(esp_mqtt_client_handle_t client);
 | |
| static esp_err_t esp_mqtt_client_ping(esp_mqtt_client_handle_t client);
 | |
| static char *create_string(const char *ptr, int len);
 | |
| 
 | |
| static esp_err_t esp_mqtt_set_config(esp_mqtt_client_handle_t client, const esp_mqtt_client_config_t *config)
 | |
| {
 | |
|     //Copy user configurations to client context
 | |
|     esp_err_t err = ESP_OK;
 | |
|     mqtt_config_storage_t *cfg = calloc(1, sizeof(mqtt_config_storage_t));
 | |
|     ESP_MEM_CHECK(TAG, cfg, return ESP_ERR_NO_MEM);
 | |
| 
 | |
|     client->config = cfg;
 | |
| 
 | |
|     cfg->task_prio = config->task_prio;
 | |
|     if (cfg->task_prio <= 0) {
 | |
|         cfg->task_prio = MQTT_TASK_PRIORITY;
 | |
|     }
 | |
| 
 | |
|     cfg->task_stack = config->task_stack;
 | |
|     if (cfg->task_stack == 0) {
 | |
|         cfg->task_stack = MQTT_TASK_STACK;
 | |
|     }
 | |
|     err = ESP_ERR_NO_MEM;
 | |
|     if (config->host) {
 | |
|         cfg->host = strdup(config->host);
 | |
|         ESP_MEM_CHECK(TAG, cfg->host, goto _mqtt_set_config_failed);
 | |
|     }
 | |
|     cfg->port = config->port;
 | |
| 
 | |
|     if (config->username) {
 | |
|         client->connect_info.username = strdup(config->username);
 | |
|         ESP_MEM_CHECK(TAG, client->connect_info.username, goto _mqtt_set_config_failed);
 | |
|     }
 | |
| 
 | |
|     if (config->password) {
 | |
|         client->connect_info.password = strdup(config->password);
 | |
|         ESP_MEM_CHECK(TAG, client->connect_info.password, goto _mqtt_set_config_failed);
 | |
|     }
 | |
| 
 | |
|     if (config->client_id) {
 | |
|         client->connect_info.client_id = strdup(config->client_id);
 | |
|     } else {
 | |
|         client->connect_info.client_id = platform_create_id_string();
 | |
|     }
 | |
|     ESP_MEM_CHECK(TAG, client->connect_info.client_id, goto _mqtt_set_config_failed);
 | |
|     ESP_LOGD(TAG, "MQTT client_id=%s", client->connect_info.client_id);
 | |
| 
 | |
|     if (config->uri) {
 | |
|         cfg->uri = strdup(config->uri);
 | |
|         ESP_MEM_CHECK(TAG, cfg->uri, goto _mqtt_set_config_failed);
 | |
|     }
 | |
| 
 | |
|     if (config->lwt_topic) {
 | |
|         client->connect_info.will_topic = strdup(config->lwt_topic);
 | |
|         ESP_MEM_CHECK(TAG, client->connect_info.will_topic, goto _mqtt_set_config_failed);
 | |
|     }
 | |
| 
 | |
|     if (config->lwt_msg_len) {
 | |
|         client->connect_info.will_message = malloc(config->lwt_msg_len);
 | |
|         ESP_MEM_CHECK(TAG, client->connect_info.will_message, goto _mqtt_set_config_failed);
 | |
|         memcpy(client->connect_info.will_message, config->lwt_msg, config->lwt_msg_len);
 | |
|         client->connect_info.will_length = config->lwt_msg_len;
 | |
|     } else if (config->lwt_msg) {
 | |
|         client->connect_info.will_message = strdup(config->lwt_msg);
 | |
|         ESP_MEM_CHECK(TAG, client->connect_info.will_message, goto _mqtt_set_config_failed);
 | |
|         client->connect_info.will_length = strlen(config->lwt_msg);
 | |
|     }
 | |
| 
 | |
|     client->connect_info.will_qos = config->lwt_qos;
 | |
|     client->connect_info.will_retain = config->lwt_retain;
 | |
| 
 | |
|     client->connect_info.clean_session = 1;
 | |
|     if (config->disable_clean_session) {
 | |
|         client->connect_info.clean_session = false;
 | |
|     }
 | |
|     client->connect_info.keepalive = config->keepalive;
 | |
|     if (client->connect_info.keepalive == 0) {
 | |
|         client->connect_info.keepalive = MQTT_KEEPALIVE_TICK;
 | |
|     }
 | |
|     cfg->network_timeout_ms = MQTT_NETWORK_TIMEOUT_MS;
 | |
|     cfg->user_context = config->user_context;
 | |
|     cfg->event_handle = config->event_handle;
 | |
|     cfg->auto_reconnect = true;
 | |
|     if (config->disable_auto_reconnect) {
 | |
|         cfg->auto_reconnect = false;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     return err;
 | |
| _mqtt_set_config_failed:
 | |
|     esp_mqtt_destroy_config(client);
 | |
|     return err;
 | |
| }
 | |
| 
 | |
| static esp_err_t esp_mqtt_destroy_config(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     mqtt_config_storage_t *cfg = client->config;
 | |
|     free(cfg->host);
 | |
|     free(cfg->uri);
 | |
|     free(cfg->path);
 | |
|     free(cfg->scheme);
 | |
|     free(client->connect_info.will_topic);
 | |
|     free(client->connect_info.will_message);
 | |
|     free(client->connect_info.client_id);
 | |
|     free(client->connect_info.username);
 | |
|     free(client->connect_info.password);
 | |
|     free(client->config);
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| static esp_err_t esp_mqtt_connect(esp_mqtt_client_handle_t client, int timeout_ms)
 | |
| {
 | |
|     int write_len, read_len, connect_rsp_code;
 | |
|     client->wait_for_ping_resp = false;
 | |
|     mqtt_msg_init(&client->mqtt_state.mqtt_connection,
 | |
|                   client->mqtt_state.out_buffer,
 | |
|                   client->mqtt_state.out_buffer_length);
 | |
|     client->mqtt_state.outbound_message = mqtt_msg_connect(&client->mqtt_state.mqtt_connection,
 | |
|                                           client->mqtt_state.connect_info);
 | |
|     client->mqtt_state.pending_msg_type = mqtt_get_type(client->mqtt_state.outbound_message->data);
 | |
|     client->mqtt_state.pending_msg_id = mqtt_get_id(client->mqtt_state.outbound_message->data,
 | |
|                                         client->mqtt_state.outbound_message->length);
 | |
|     ESP_LOGI(TAG, "Sending MQTT CONNECT message, type: %d, id: %04X",
 | |
|              client->mqtt_state.pending_msg_type,
 | |
|              client->mqtt_state.pending_msg_id);
 | |
| 
 | |
|     write_len = esp_transport_write(client->transport,
 | |
|                                 (char *)client->mqtt_state.outbound_message->data,
 | |
|                                 client->mqtt_state.outbound_message->length,
 | |
|                                 client->config->network_timeout_ms);
 | |
|     if (write_len < 0) {
 | |
|         ESP_LOGE(TAG, "Writing failed, errno= %d", errno);
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
|     read_len = esp_transport_read(client->transport,
 | |
|                               (char *)client->mqtt_state.in_buffer,
 | |
|                               client->mqtt_state.in_buffer_length,
 | |
|                               client->config->network_timeout_ms);
 | |
|     if (read_len < 0) {
 | |
|         ESP_LOGE(TAG, "Error network response");
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
| 
 | |
|     if (mqtt_get_type(client->mqtt_state.in_buffer) != MQTT_MSG_TYPE_CONNACK) {
 | |
|         ESP_LOGE(TAG, "Invalid MSG_TYPE response: %d, read_len: %d", mqtt_get_type(client->mqtt_state.in_buffer), read_len);
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
|     connect_rsp_code = mqtt_get_connect_return_code(client->mqtt_state.in_buffer);
 | |
|     switch (connect_rsp_code) {
 | |
|         case CONNECTION_ACCEPTED:
 | |
|             ESP_LOGD(TAG, "Connected");
 | |
|             return ESP_OK;
 | |
|         case CONNECTION_REFUSE_PROTOCOL:
 | |
|             ESP_LOGW(TAG, "Connection refused, bad protocol");
 | |
|             return ESP_FAIL;
 | |
|         case CONNECTION_REFUSE_SERVER_UNAVAILABLE:
 | |
|             ESP_LOGW(TAG, "Connection refused, server unavailable");
 | |
|             return ESP_FAIL;
 | |
|         case CONNECTION_REFUSE_BAD_USERNAME:
 | |
|             ESP_LOGW(TAG, "Connection refused, bad username or password");
 | |
|             return ESP_FAIL;
 | |
|         case CONNECTION_REFUSE_NOT_AUTHORIZED:
 | |
|             ESP_LOGW(TAG, "Connection refused, not authorized");
 | |
|             return ESP_FAIL;
 | |
|         default:
 | |
|             ESP_LOGW(TAG, "Connection refused, Unknow reason");
 | |
|             return ESP_FAIL;
 | |
|     }
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| static esp_err_t esp_mqtt_abort_connection(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     esp_transport_close(client->transport);
 | |
|     client->wait_timeout_ms = MQTT_RECONNECT_TIMEOUT_MS;
 | |
|     client->reconnect_tick = platform_tick_get_ms();
 | |
|     client->state = MQTT_STATE_WAIT_TIMEOUT;
 | |
|     ESP_LOGI(TAG, "Reconnect after %d ms", client->wait_timeout_ms);
 | |
|     client->event.event_id = MQTT_EVENT_DISCONNECTED;
 | |
|     client->wait_for_ping_resp = false;
 | |
|     esp_mqtt_dispatch_event(client);
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| esp_mqtt_client_handle_t esp_mqtt_client_init(const esp_mqtt_client_config_t *config)
 | |
| {
 | |
|     esp_mqtt_client_handle_t client = calloc(1, sizeof(struct esp_mqtt_client));
 | |
|     ESP_MEM_CHECK(TAG, client, return NULL);
 | |
| 
 | |
|     esp_mqtt_set_config(client, config);
 | |
| 
 | |
|     client->transport_list = esp_transport_list_init();
 | |
|     ESP_MEM_CHECK(TAG, client->transport_list, goto _mqtt_init_failed);
 | |
| 
 | |
|     esp_transport_handle_t tcp = esp_transport_tcp_init();
 | |
|     ESP_MEM_CHECK(TAG, tcp, goto _mqtt_init_failed);
 | |
|     esp_transport_set_default_port(tcp, MQTT_TCP_DEFAULT_PORT);
 | |
|     esp_transport_list_add(client->transport_list, tcp, "mqtt");
 | |
|     if (config->transport == MQTT_TRANSPORT_OVER_TCP) {
 | |
|         client->config->scheme = create_string("mqtt", 4);
 | |
|         ESP_MEM_CHECK(TAG, client->config->scheme, goto _mqtt_init_failed);
 | |
|     }
 | |
| 
 | |
| #if MQTT_ENABLE_WS
 | |
|     esp_transport_handle_t ws = esp_transport_ws_init(tcp);
 | |
|     ESP_MEM_CHECK(TAG, ws, goto _mqtt_init_failed);
 | |
|     esp_transport_set_default_port(ws, MQTT_WS_DEFAULT_PORT);
 | |
|     esp_transport_list_add(client->transport_list, ws, "ws");
 | |
|     if (config->transport == MQTT_TRANSPORT_OVER_WS) {
 | |
|         client->config->scheme = create_string("ws", 2);
 | |
|         ESP_MEM_CHECK(TAG, client->config->scheme, goto _mqtt_init_failed);
 | |
|     }
 | |
| #endif
 | |
| 
 | |
| #if MQTT_ENABLE_SSL
 | |
|     esp_transport_handle_t ssl = esp_transport_ssl_init();
 | |
|     ESP_MEM_CHECK(TAG, ssl, goto _mqtt_init_failed);
 | |
|     esp_transport_set_default_port(ssl, MQTT_SSL_DEFAULT_PORT);
 | |
|     if (config->cert_pem) {
 | |
|         esp_transport_ssl_set_cert_data(ssl, config->cert_pem, strlen(config->cert_pem));
 | |
|     }
 | |
|     if (config->client_cert_pem) {
 | |
|         esp_transport_ssl_set_client_cert_data(ssl, config->client_cert_pem, strlen(config->client_cert_pem));
 | |
|     }
 | |
|     if (config->client_key_pem) {
 | |
|         esp_transport_ssl_set_client_key_data(ssl, config->client_key_pem, strlen(config->client_key_pem));
 | |
|     }
 | |
|     esp_transport_list_add(client->transport_list, ssl, "mqtts");
 | |
|     if (config->transport == MQTT_TRANSPORT_OVER_SSL) {
 | |
|         client->config->scheme = create_string("mqtts", 5);
 | |
|         ESP_MEM_CHECK(TAG, client->config->scheme, goto _mqtt_init_failed);
 | |
|     }
 | |
| #endif
 | |
| 
 | |
| #if MQTT_ENABLE_WSS
 | |
|     esp_transport_handle_t wss = esp_transport_ws_init(ssl);
 | |
|     ESP_MEM_CHECK(TAG, wss, goto _mqtt_init_failed);
 | |
|     esp_transport_set_default_port(wss, MQTT_WSS_DEFAULT_PORT);
 | |
|     esp_transport_list_add(client->transport_list, wss, "wss");
 | |
|     if (config->transport == MQTT_TRANSPORT_OVER_WSS) {
 | |
|         client->config->scheme = create_string("wss", 3);
 | |
|         ESP_MEM_CHECK(TAG, client->config->scheme, goto _mqtt_init_failed);
 | |
|     }
 | |
| #endif
 | |
|     if (client->config->uri) {
 | |
|         if (esp_mqtt_client_set_uri(client, client->config->uri) != ESP_OK) {
 | |
|             goto _mqtt_init_failed;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (client->config->scheme == NULL) {
 | |
|         client->config->scheme = create_string("mqtt", 4);
 | |
|         ESP_MEM_CHECK(TAG, client->config->scheme, goto _mqtt_init_failed);
 | |
|     }
 | |
| 
 | |
|     client->keepalive_tick = platform_tick_get_ms();
 | |
|     client->reconnect_tick = platform_tick_get_ms();
 | |
|     client->wait_for_ping_resp = false;
 | |
|     int buffer_size = config->buffer_size;
 | |
|     if (buffer_size <= 0) {
 | |
|         buffer_size = MQTT_BUFFER_SIZE_BYTE;
 | |
|     }
 | |
| 
 | |
|     client->mqtt_state.in_buffer = (uint8_t *)malloc(buffer_size);
 | |
|     ESP_MEM_CHECK(TAG, client->mqtt_state.in_buffer, goto _mqtt_init_failed);
 | |
|     client->mqtt_state.in_buffer_length = buffer_size;
 | |
|     client->mqtt_state.out_buffer = (uint8_t *)malloc(buffer_size);
 | |
|     ESP_MEM_CHECK(TAG, client->mqtt_state.out_buffer, goto _mqtt_init_failed);
 | |
| 
 | |
|     client->mqtt_state.out_buffer_length = buffer_size;
 | |
|     client->mqtt_state.connect_info = &client->connect_info;
 | |
|     client->outbox = outbox_init();
 | |
|     ESP_MEM_CHECK(TAG, client->outbox, goto _mqtt_init_failed);
 | |
|     client->status_bits = xEventGroupCreate();
 | |
|     ESP_MEM_CHECK(TAG, client->status_bits, goto _mqtt_init_failed);
 | |
|     return client;
 | |
| _mqtt_init_failed:
 | |
|     esp_mqtt_client_destroy(client);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| esp_err_t esp_mqtt_client_destroy(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     esp_mqtt_client_stop(client);
 | |
|     esp_mqtt_destroy_config(client);
 | |
|     esp_transport_list_destroy(client->transport_list);
 | |
|     outbox_destroy(client->outbox);
 | |
|     vEventGroupDelete(client->status_bits);
 | |
|     free(client->mqtt_state.in_buffer);
 | |
|     free(client->mqtt_state.out_buffer);
 | |
|     free(client);
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| static char *create_string(const char *ptr, int len)
 | |
| {
 | |
|     char *ret;
 | |
|     if (len <= 0) {
 | |
|         return NULL;
 | |
|     }
 | |
|     ret = calloc(1, len + 1);
 | |
|     ESP_MEM_CHECK(TAG, ret, return NULL);
 | |
|     memcpy(ret, ptr, len);
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| esp_err_t esp_mqtt_client_set_uri(esp_mqtt_client_handle_t client, const char *uri)
 | |
| {
 | |
|     struct http_parser_url puri;
 | |
|     http_parser_url_init(&puri);
 | |
|     int parser_status = http_parser_parse_url(uri, strlen(uri), 0, &puri);
 | |
|     if (parser_status != 0) {
 | |
|         ESP_LOGE(TAG, "Error parse uri = %s", uri);
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
| 
 | |
|     if (client->config->scheme == NULL) {
 | |
|         client->config->scheme = create_string(uri + puri.field_data[UF_SCHEMA].off, puri.field_data[UF_SCHEMA].len);
 | |
|     }
 | |
| 
 | |
|     if (client->config->host == NULL) {
 | |
|         client->config->host = create_string(uri + puri.field_data[UF_HOST].off, puri.field_data[UF_HOST].len);
 | |
|     }
 | |
| 
 | |
|     if (client->config->path == NULL) {
 | |
|         client->config->path = create_string(uri + puri.field_data[UF_PATH].off, puri.field_data[UF_PATH].len);
 | |
|     }
 | |
|     if (client->config->path) {
 | |
| #if MQTT_ENABLE_WSS || MQTT_ENABLE_WS
 | |
|         esp_transport_handle_t trans = esp_transport_list_get_transport(client->transport_list, "ws");
 | |
|         if (trans) {
 | |
|             esp_transport_ws_set_path(trans, client->config->path);
 | |
|         }
 | |
|         trans = esp_transport_list_get_transport(client->transport_list, "wss");
 | |
|         if (trans) {
 | |
|             esp_transport_ws_set_path(trans, client->config->path);
 | |
|         }
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     if (puri.field_data[UF_PORT].len) {
 | |
|         client->config->port = strtol((const char*)(uri + puri.field_data[UF_PORT].off), NULL, 10);
 | |
|     }
 | |
| 
 | |
|     char *user_info = create_string(uri + puri.field_data[UF_USERINFO].off, puri.field_data[UF_USERINFO].len);
 | |
|     if (user_info) {
 | |
|         char *pass = strchr(user_info, ':');
 | |
|         if (pass) {
 | |
|             pass[0] = 0; //terminal username
 | |
|             pass ++;
 | |
|             client->connect_info.password = strdup(pass);
 | |
|         }
 | |
|         client->connect_info.username = strdup(user_info);
 | |
| 
 | |
|         free(user_info);
 | |
|     }
 | |
| 
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| static esp_err_t mqtt_write_data(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     int write_len = esp_transport_write(client->transport,
 | |
|                                     (char *)client->mqtt_state.outbound_message->data,
 | |
|                                     client->mqtt_state.outbound_message->length,
 | |
|                                     client->config->network_timeout_ms);
 | |
|     // client->mqtt_state.pending_msg_type = mqtt_get_type(client->mqtt_state.outbound_message->data);
 | |
|     if (write_len <= 0) {
 | |
|         ESP_LOGE(TAG, "Error write data or timeout, written len = %d", write_len);
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
|     /* we've just sent a mqtt control packet, update keepalive counter
 | |
|      * [MQTT-3.1.2-23]
 | |
|      */
 | |
|     client->keepalive_tick = platform_tick_get_ms();
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| static esp_err_t esp_mqtt_dispatch_event(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     client->event.msg_id = mqtt_get_id(client->mqtt_state.in_buffer, client->mqtt_state.in_buffer_length);
 | |
|     client->event.user_context = client->config->user_context;
 | |
|     client->event.client = client;
 | |
| 
 | |
|     if (client->config->event_handle) {
 | |
|         return client->config->event_handle(&client->event);
 | |
|     }
 | |
|     return ESP_FAIL;
 | |
| }
 | |
| 
 | |
| 
 | |
| typedef struct {
 | |
|     char *path;
 | |
|     char *buffer;
 | |
|     esp_transport_handle_t parent;
 | |
| } transport_ws_t;
 | |
| 
 | |
| static void deliver_publish(esp_mqtt_client_handle_t client, uint8_t *message, int length)
 | |
| {
 | |
|     const char *mqtt_topic, *mqtt_data;
 | |
|     uint32_t mqtt_topic_length, mqtt_data_length;
 | |
|     uint32_t mqtt_len, mqtt_offset = 0, total_mqtt_len = 0;
 | |
|     int len_read;
 | |
|     esp_transport_handle_t transport = client->transport;
 | |
| 
 | |
|     do
 | |
|     {
 | |
|         if (total_mqtt_len == 0) {
 | |
|             mqtt_topic_length = length;
 | |
|             mqtt_topic = mqtt_get_publish_topic(message, &mqtt_topic_length);
 | |
|             mqtt_data_length = length;
 | |
|             mqtt_data = mqtt_get_publish_data(message, &mqtt_data_length);
 | |
|             total_mqtt_len = client->mqtt_state.message_length - client->mqtt_state.message_length_read + mqtt_data_length;
 | |
|             mqtt_len = mqtt_data_length;
 | |
|             /* any further reading only the underlying payload */
 | |
|             transport = esp_transport_get_payload_transport_handle(transport);
 | |
|         } else {
 | |
|             mqtt_len = len_read;
 | |
|             mqtt_data = (const char*)client->mqtt_state.in_buffer;
 | |
|             mqtt_topic = NULL;
 | |
|             mqtt_topic_length = 0;
 | |
|         }
 | |
| 
 | |
|         ESP_LOGD(TAG, "Get data len= %d, topic len=%d", mqtt_len, mqtt_topic_length);
 | |
|         client->event.event_id = MQTT_EVENT_DATA;
 | |
|         client->event.data = (char *)mqtt_data;
 | |
|         client->event.data_len = mqtt_len;
 | |
|         client->event.total_data_len = total_mqtt_len;
 | |
|         client->event.current_data_offset = mqtt_offset;
 | |
|         client->event.topic = (char *)mqtt_topic;
 | |
|         client->event.topic_len = mqtt_topic_length;
 | |
|         esp_mqtt_dispatch_event(client);
 | |
| 
 | |
|         mqtt_offset += mqtt_len;
 | |
|         if (client->mqtt_state.message_length_read >= client->mqtt_state.message_length) {
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         len_read = esp_transport_read(transport,
 | |
|                                   (char *)client->mqtt_state.in_buffer,
 | |
|                                   client->mqtt_state.message_length - client->mqtt_state.message_length_read > client->mqtt_state.in_buffer_length ?
 | |
|                                   client->mqtt_state.in_buffer_length : client->mqtt_state.message_length - client->mqtt_state.message_length_read,
 | |
|                                   client->config->network_timeout_ms);
 | |
|         if (len_read <= 0) {
 | |
|             ESP_LOGE(TAG, "Read error or timeout: %d", errno);
 | |
|             break;
 | |
|         }
 | |
|         client->mqtt_state.message_length_read += len_read;
 | |
|     } while (1);
 | |
| 
 | |
| 
 | |
| }
 | |
| 
 | |
| static bool is_valid_mqtt_msg(esp_mqtt_client_handle_t client, int msg_type, int msg_id)
 | |
| {
 | |
|     ESP_LOGD(TAG, "pending_id=%d, pending_msg_count = %d", client->mqtt_state.pending_msg_id, client->mqtt_state.pending_msg_count);
 | |
|     if (client->mqtt_state.pending_msg_count == 0) {
 | |
|         return false;
 | |
|     }
 | |
|     if (outbox_delete(client->outbox, msg_id, msg_type) == ESP_OK) {
 | |
|         client->mqtt_state.pending_msg_count --;
 | |
|         return true;
 | |
|     }
 | |
|     if (client->mqtt_state.pending_msg_type == msg_type && client->mqtt_state.pending_msg_id == msg_id) {
 | |
|         client->mqtt_state.pending_msg_count --;
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| static void mqtt_enqueue(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     ESP_LOGD(TAG, "mqtt_enqueue id: %d, type=%d successful",
 | |
|              client->mqtt_state.pending_msg_id, client->mqtt_state.pending_msg_type);
 | |
|     //lock mutex
 | |
|     if (client->mqtt_state.pending_msg_count > 0) {
 | |
|         //Copy to queue buffer
 | |
|         outbox_enqueue(client->outbox,
 | |
|                        client->mqtt_state.outbound_message->data,
 | |
|                        client->mqtt_state.outbound_message->length,
 | |
|                        client->mqtt_state.pending_msg_id,
 | |
|                        client->mqtt_state.pending_msg_type,
 | |
|                        platform_tick_get_ms());
 | |
|     }
 | |
|     //unlock
 | |
| }
 | |
| 
 | |
| static esp_err_t mqtt_process_receive(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     int read_len;
 | |
|     uint8_t msg_type;
 | |
|     uint8_t msg_qos;
 | |
|     uint16_t msg_id;
 | |
| 
 | |
|     read_len = esp_transport_read(client->transport, (char *)client->mqtt_state.in_buffer, client->mqtt_state.in_buffer_length, 1000);
 | |
| 
 | |
|     if (read_len < 0) {
 | |
|         ESP_LOGE(TAG, "Read error or end of stream");
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
| 
 | |
|     if (read_len == 0) {
 | |
|         return ESP_OK;
 | |
|     }
 | |
| 
 | |
|     msg_type = mqtt_get_type(client->mqtt_state.in_buffer);
 | |
|     msg_qos = mqtt_get_qos(client->mqtt_state.in_buffer);
 | |
|     msg_id = mqtt_get_id(client->mqtt_state.in_buffer, client->mqtt_state.in_buffer_length);
 | |
| 
 | |
|     ESP_LOGD(TAG, "msg_type=%d, msg_id=%d", msg_type, msg_id);
 | |
|     switch (msg_type)
 | |
|     {
 | |
|         case MQTT_MSG_TYPE_SUBACK:
 | |
|             if (is_valid_mqtt_msg(client, MQTT_MSG_TYPE_SUBSCRIBE, msg_id)) {
 | |
|                 ESP_LOGD(TAG, "Subscribe successful");
 | |
|                 client->event.event_id = MQTT_EVENT_SUBSCRIBED;
 | |
|                 esp_mqtt_dispatch_event(client);
 | |
|             }
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_UNSUBACK:
 | |
|             if (is_valid_mqtt_msg(client, MQTT_MSG_TYPE_UNSUBSCRIBE, msg_id)) {
 | |
|                 ESP_LOGD(TAG, "UnSubscribe successful");
 | |
|                 client->event.event_id = MQTT_EVENT_UNSUBSCRIBED;
 | |
|                 esp_mqtt_dispatch_event(client);
 | |
|             }
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_PUBLISH:
 | |
|             if (msg_qos == 1) {
 | |
|                 client->mqtt_state.outbound_message = mqtt_msg_puback(&client->mqtt_state.mqtt_connection, msg_id);
 | |
|             }
 | |
|             else if (msg_qos == 2) {
 | |
|                 client->mqtt_state.outbound_message = mqtt_msg_pubrec(&client->mqtt_state.mqtt_connection, msg_id);
 | |
|             }
 | |
| 
 | |
|             if (msg_qos == 1 || msg_qos == 2) {
 | |
|                 ESP_LOGD(TAG, "Queue response QoS: %d", msg_qos);
 | |
| 
 | |
|                 if (mqtt_write_data(client) != ESP_OK) {
 | |
|                     ESP_LOGE(TAG, "Error write qos msg repsonse, qos = %d", msg_qos);
 | |
|                     // TODO: Shoule reconnect?
 | |
|                     // return ESP_FAIL;
 | |
|                 }
 | |
|             }
 | |
|             client->mqtt_state.message_length_read = read_len;
 | |
|             client->mqtt_state.message_length = mqtt_get_total_length(client->mqtt_state.in_buffer, client->mqtt_state.message_length_read);
 | |
|             ESP_LOGI(TAG, "deliver_publish, message_length_read=%d, message_length=%d", read_len, client->mqtt_state.message_length);
 | |
| 
 | |
|             deliver_publish(client, client->mqtt_state.in_buffer, client->mqtt_state.message_length_read);
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_PUBACK:
 | |
|             if (is_valid_mqtt_msg(client, MQTT_MSG_TYPE_PUBLISH, msg_id)) {
 | |
|                 ESP_LOGD(TAG, "received MQTT_MSG_TYPE_PUBACK, finish QoS1 publish");
 | |
|                 client->event.event_id = MQTT_EVENT_PUBLISHED;
 | |
|                 esp_mqtt_dispatch_event(client);
 | |
|             }
 | |
| 
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_PUBREC:
 | |
|             ESP_LOGD(TAG, "received MQTT_MSG_TYPE_PUBREC");
 | |
|             client->mqtt_state.outbound_message = mqtt_msg_pubrel(&client->mqtt_state.mqtt_connection, msg_id);
 | |
|             mqtt_write_data(client);
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_PUBREL:
 | |
|             ESP_LOGD(TAG, "received MQTT_MSG_TYPE_PUBREL");
 | |
|             client->mqtt_state.outbound_message = mqtt_msg_pubcomp(&client->mqtt_state.mqtt_connection, msg_id);
 | |
|             mqtt_write_data(client);
 | |
| 
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_PUBCOMP:
 | |
|             ESP_LOGD(TAG, "received MQTT_MSG_TYPE_PUBCOMP");
 | |
|             if (is_valid_mqtt_msg(client, MQTT_MSG_TYPE_PUBLISH, msg_id)) {
 | |
|                 ESP_LOGD(TAG, "Receive MQTT_MSG_TYPE_PUBCOMP, finish QoS2 publish");
 | |
|                 client->event.event_id = MQTT_EVENT_PUBLISHED;
 | |
|                 esp_mqtt_dispatch_event(client);
 | |
|             }
 | |
|             break;
 | |
|         case MQTT_MSG_TYPE_PINGRESP:
 | |
|             ESP_LOGD(TAG, "MQTT_MSG_TYPE_PINGRESP");
 | |
|             client->wait_for_ping_resp = false;
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| static void esp_mqtt_task(void *pv)
 | |
| {
 | |
|     esp_mqtt_client_handle_t client = (esp_mqtt_client_handle_t) pv;
 | |
|     client->run = true;
 | |
| 
 | |
|     //get transport by scheme
 | |
|     client->transport = esp_transport_list_get_transport(client->transport_list, client->config->scheme);
 | |
| 
 | |
|     if (client->transport == NULL) {
 | |
|         ESP_LOGE(TAG, "There are no transports valid, stop mqtt client, config scheme = %s", client->config->scheme);
 | |
|         client->run = false;
 | |
|     }
 | |
|     //default port
 | |
|     if (client->config->port == 0) {
 | |
|         client->config->port = esp_transport_get_default_port(client->transport);
 | |
|     }
 | |
| 
 | |
|     client->state = MQTT_STATE_INIT;
 | |
|     xEventGroupClearBits(client->status_bits, STOPPED_BIT);
 | |
|     while (client->run) {
 | |
| 
 | |
|         switch ((int)client->state) {
 | |
|             case MQTT_STATE_INIT:
 | |
|                 if (client->transport == NULL) {
 | |
|                     ESP_LOGE(TAG, "There are no transport");
 | |
|                     client->run = false;
 | |
|                 }
 | |
| 
 | |
|                 if (esp_transport_connect(client->transport,
 | |
|                                       client->config->host,
 | |
|                                       client->config->port,
 | |
|                                       client->config->network_timeout_ms) < 0) {
 | |
|                     ESP_LOGE(TAG, "Error transport connect");
 | |
|                     esp_mqtt_abort_connection(client);
 | |
|                     break;
 | |
|                 }
 | |
|                 ESP_LOGD(TAG, "Transport connected to %s://%s:%d", client->config->scheme, client->config->host, client->config->port);
 | |
|                 if (esp_mqtt_connect(client, client->config->network_timeout_ms) != ESP_OK) {
 | |
|                     ESP_LOGI(TAG, "Error MQTT Connected");
 | |
|                     esp_mqtt_abort_connection(client);
 | |
|                     break;
 | |
|                 }
 | |
|                 client->event.event_id = MQTT_EVENT_CONNECTED;
 | |
|                 client->event.session_present = mqtt_get_connect_session_present(client->mqtt_state.in_buffer);
 | |
|                 client->state = MQTT_STATE_CONNECTED;
 | |
|                 esp_mqtt_dispatch_event(client);
 | |
| 
 | |
|                 break;
 | |
|             case MQTT_STATE_CONNECTED:
 | |
|                 // receive and process data
 | |
|                 if (mqtt_process_receive(client) == ESP_FAIL) {
 | |
|                     esp_mqtt_abort_connection(client);
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 if (platform_tick_get_ms() - client->keepalive_tick > client->connect_info.keepalive * 1000 / 2) {
 | |
|                     //No ping resp from last ping => Disconnected
 | |
|                 	if(client->wait_for_ping_resp){
 | |
|                     	ESP_LOGE(TAG, "No PING_RESP, disconnected");
 | |
|                     	esp_mqtt_abort_connection(client);
 | |
|                     	client->wait_for_ping_resp = false;
 | |
|                     	break;
 | |
|                     }
 | |
|                 	if (esp_mqtt_client_ping(client) == ESP_FAIL) {
 | |
|                         ESP_LOGE(TAG, "Can't send ping, disconnected");
 | |
|                         esp_mqtt_abort_connection(client);
 | |
|                         break;
 | |
|                     } else {
 | |
|                     	client->wait_for_ping_resp = true;
 | |
|                     }
 | |
|                 	ESP_LOGD(TAG, "PING sent");
 | |
|                 }
 | |
| 
 | |
|                 //Delete mesaage after 30 senconds
 | |
|                 outbox_delete_expired(client->outbox, platform_tick_get_ms(), OUTBOX_EXPIRED_TIMEOUT_MS);
 | |
|                 //
 | |
|                 outbox_cleanup(client->outbox, OUTBOX_MAX_SIZE);
 | |
|                 break;
 | |
|             case MQTT_STATE_WAIT_TIMEOUT:
 | |
| 
 | |
|                 if (!client->config->auto_reconnect) {
 | |
|                     client->run = false;
 | |
|                     break;
 | |
|                 }
 | |
|                 if (platform_tick_get_ms() - client->reconnect_tick > client->wait_timeout_ms) {
 | |
|                     client->state = MQTT_STATE_INIT;
 | |
|                     client->reconnect_tick = platform_tick_get_ms();
 | |
|                     ESP_LOGD(TAG, "Reconnecting...");
 | |
|                 }
 | |
|                 vTaskDelay(client->wait_timeout_ms / 2 / portTICK_RATE_MS);
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
|     esp_transport_close(client->transport);
 | |
|     xEventGroupSetBits(client->status_bits, STOPPED_BIT);
 | |
| 
 | |
|     vTaskDelete(NULL);
 | |
| }
 | |
| 
 | |
| esp_err_t esp_mqtt_client_start(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     if (client->state >= MQTT_STATE_INIT) {
 | |
|         ESP_LOGE(TAG, "Client has started");
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
| #if MQTT_CORE_SELECTION_ENABLED
 | |
|         ESP_LOGD(TAG, "Core selection enabled on %u", MQTT_TASK_CORE);
 | |
|         if (xTaskCreatePinnedToCore(esp_mqtt_task, "mqtt_task", client->config->task_stack, client, client->config->task_prio, NULL, MQTT_TASK_CORE) != pdTRUE) {
 | |
|             ESP_LOGE(TAG, "Error create mqtt task");
 | |
|             return ESP_FAIL;
 | |
|         }
 | |
| #else
 | |
|         ESP_LOGD(TAG, "Core selection disabled");
 | |
|         if (xTaskCreate(esp_mqtt_task, "mqtt_task", client->config->task_stack, client, client->config->task_prio, NULL) != pdTRUE) {
 | |
|             ESP_LOGE(TAG, "Error create mqtt task");
 | |
|             return ESP_FAIL;
 | |
|         }
 | |
| #endif
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| esp_err_t esp_mqtt_client_stop(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     if (client->run) {
 | |
|         client->run = false;
 | |
|         xEventGroupWaitBits(client->status_bits, STOPPED_BIT, false, true, portMAX_DELAY);
 | |
|         client->state = MQTT_STATE_UNKNOWN;
 | |
|         return ESP_OK;
 | |
|     } else {
 | |
|         ESP_LOGW(TAG, "Client asked to stop, but was not started");
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static esp_err_t esp_mqtt_client_ping(esp_mqtt_client_handle_t client)
 | |
| {
 | |
|     client->mqtt_state.outbound_message = mqtt_msg_pingreq(&client->mqtt_state.mqtt_connection);
 | |
| 
 | |
|     if (mqtt_write_data(client) != ESP_OK) {
 | |
|         ESP_LOGE(TAG, "Error sending ping");
 | |
|         return ESP_FAIL;
 | |
|     }
 | |
|     ESP_LOGD(TAG, "Sent PING successful");
 | |
|     return ESP_OK;
 | |
| }
 | |
| 
 | |
| int esp_mqtt_client_subscribe(esp_mqtt_client_handle_t client, const char *topic, int qos)
 | |
| {
 | |
|     if (client->state != MQTT_STATE_CONNECTED) {
 | |
|         ESP_LOGE(TAG, "Client has not connected");
 | |
|         return -1;
 | |
|     }
 | |
|     mqtt_enqueue(client); //move pending msg to outbox (if have)
 | |
|     client->mqtt_state.outbound_message = mqtt_msg_subscribe(&client->mqtt_state.mqtt_connection,
 | |
|                                           topic, qos,
 | |
|                                           &client->mqtt_state.pending_msg_id);
 | |
| 
 | |
|     client->mqtt_state.pending_msg_type = mqtt_get_type(client->mqtt_state.outbound_message->data);
 | |
|     client->mqtt_state.pending_msg_count ++;
 | |
| 
 | |
|     if (mqtt_write_data(client) != ESP_OK) {
 | |
|         ESP_LOGE(TAG, "Error to subscribe topic=%s, qos=%d", topic, qos);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     ESP_LOGD(TAG, "Sent subscribe topic=%s, id: %d, type=%d successful", topic, client->mqtt_state.pending_msg_id, client->mqtt_state.pending_msg_type);
 | |
|     return client->mqtt_state.pending_msg_id;
 | |
| }
 | |
| 
 | |
| int esp_mqtt_client_unsubscribe(esp_mqtt_client_handle_t client, const char *topic)
 | |
| {
 | |
|     if (client->state != MQTT_STATE_CONNECTED) {
 | |
|         ESP_LOGE(TAG, "Client has not connected");
 | |
|         return -1;
 | |
|     }
 | |
|     mqtt_enqueue(client);
 | |
|     client->mqtt_state.outbound_message = mqtt_msg_unsubscribe(&client->mqtt_state.mqtt_connection,
 | |
|                                           topic,
 | |
|                                           &client->mqtt_state.pending_msg_id);
 | |
|     ESP_LOGD(TAG, "unsubscribe, topic\"%s\", id: %d", topic, client->mqtt_state.pending_msg_id);
 | |
| 
 | |
|     client->mqtt_state.pending_msg_type = mqtt_get_type(client->mqtt_state.outbound_message->data);
 | |
|     client->mqtt_state.pending_msg_count ++;
 | |
| 
 | |
|     if (mqtt_write_data(client) != ESP_OK) {
 | |
|         ESP_LOGE(TAG, "Error to unsubscribe topic=%s", topic);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     ESP_LOGD(TAG, "Sent Unsubscribe topic=%s, id: %d, successful", topic, client->mqtt_state.pending_msg_id);
 | |
|     return client->mqtt_state.pending_msg_id;
 | |
| }
 | |
| 
 | |
| int esp_mqtt_client_publish(esp_mqtt_client_handle_t client, const char *topic, const char *data, int len, int qos, int retain)
 | |
| {
 | |
|     uint16_t pending_msg_id = 0;
 | |
|     if (client->state != MQTT_STATE_CONNECTED) {
 | |
|         ESP_LOGE(TAG, "Client has not connected");
 | |
|         return -1;
 | |
|     }
 | |
|     if (len <= 0) {
 | |
|         len = strlen(data);
 | |
|     }
 | |
| 
 | |
|     mqtt_message_t *publish_msg = mqtt_msg_publish(&client->mqtt_state.mqtt_connection,
 | |
|                                   topic, data, len,
 | |
|                                   qos, retain,
 | |
|                                   &pending_msg_id);
 | |
| 
 | |
|     /* We have to set as pending all the qos>0 messages) */
 | |
|     if (qos > 0) {
 | |
|         mqtt_enqueue(client);
 | |
|         client->mqtt_state.outbound_message = publish_msg;
 | |
|         client->mqtt_state.pending_msg_type = mqtt_get_type(client->mqtt_state.outbound_message->data);
 | |
|         client->mqtt_state.pending_msg_id = pending_msg_id;
 | |
|         client->mqtt_state.pending_msg_count ++;
 | |
|     } else {
 | |
|         client->mqtt_state.outbound_message = publish_msg;
 | |
|     }
 | |
| 
 | |
|     if (mqtt_write_data(client) != ESP_OK) {
 | |
|         ESP_LOGE(TAG, "Error to public data to topic=%s, qos=%d", topic, qos);
 | |
|         return -1;
 | |
|     }
 | |
|     return pending_msg_id;
 | |
| }
 | |
| 
 | |
| 
 | 
