mosquitto/lib/send_connect.c

214 lines
5.9 KiB
C
Raw Normal View History

2014-05-07 22:27:00 +00:00
/*
Copyright (c) 2009-2020 Roger Light <roger@atchoo.org>
2014-05-07 22:27:00 +00:00
All rights reserved. This program and the accompanying materials
2020-11-25 17:34:21 +00:00
are made available under the terms of the Eclipse Public License 2.0
2014-05-07 22:27:00 +00:00
and Eclipse Distribution License v1.0 which accompany this distribution.
2014-05-07 22:27:00 +00:00
The Eclipse Public License is available at
2020-11-25 17:34:21 +00:00
https://www.eclipse.org/legal/epl-2.0/
2014-05-07 22:27:00 +00:00
and the Eclipse Distribution License is available at
http://www.eclipse.org/org/documents/edl-v10.php.
2020-12-01 18:21:59 +00:00
SPDX-License-Identifier: EPL-2.0 OR EDL-1.0
2014-05-07 22:27:00 +00:00
Contributors:
Roger Light - initial implementation and documentation.
*/
#include "config.h"
2014-05-07 22:27:00 +00:00
#include <assert.h>
2017-03-06 21:19:53 +00:00
#include <string.h>
2014-05-07 22:27:00 +00:00
2018-04-16 10:48:42 +00:00
#ifdef WITH_BROKER
# include "mosquitto_broker_internal.h"
#endif
2015-04-29 20:37:47 +00:00
#include "logging_mosq.h"
#include "memory_mosq.h"
#include "mosquitto.h"
#include "mosquitto_internal.h"
#include "mqtt_protocol.h"
2015-04-29 20:37:47 +00:00
#include "packet_mosq.h"
#include "property_mosq.h"
2014-05-07 22:27:00 +00:00
int send__connect(struct mosquitto *mosq, uint16_t keepalive, bool clean_session, const mosquitto_property *properties)
2014-05-07 22:27:00 +00:00
{
struct mosquitto__packet *packet = NULL;
2020-10-17 00:23:08 +00:00
uint32_t payloadlen;
2014-05-07 22:27:00 +00:00
uint8_t will = 0;
uint8_t byte;
int rc;
2015-01-27 00:32:20 +00:00
uint8_t version;
2014-08-16 22:14:41 +00:00
char *clientid, *username, *password;
2020-10-17 00:23:08 +00:00
uint32_t headerlen;
uint32_t proplen = 0, varbytes;
2019-01-08 12:27:19 +00:00
mosquitto_property *local_props = NULL;
uint16_t receive_maximum;
2014-05-07 22:27:00 +00:00
assert(mosq);
if(mosq->protocol == mosq_p_mqtt31 && !mosq->id) return MOSQ_ERR_PROTOCOL;
2014-05-07 22:27:00 +00:00
2014-06-10 22:30:15 +00:00
#if defined(WITH_BROKER) && defined(WITH_BRIDGE)
if(mosq->bridge){
2014-08-16 22:14:41 +00:00
clientid = mosq->bridge->remote_clientid;
username = mosq->bridge->remote_username;
password = mosq->bridge->remote_password;
2014-06-10 22:30:15 +00:00
}else{
clientid = mosq->id;
2014-08-16 22:14:41 +00:00
username = mosq->username;
password = mosq->password;
2014-06-10 22:30:15 +00:00
}
#else
clientid = mosq->id;
2014-08-16 22:14:41 +00:00
username = mosq->username;
password = mosq->password;
2014-06-10 22:30:15 +00:00
#endif
if(mosq->protocol == mosq_p_mqtt5){
2019-01-08 12:27:19 +00:00
/* Generate properties from options */
if(!mosquitto_property_read_int16(properties, MQTT_PROP_RECEIVE_MAXIMUM, &receive_maximum, false)){
rc = mosquitto_property_add_int16(&local_props, MQTT_PROP_RECEIVE_MAXIMUM, mosq->msgs_in.inflight_maximum);
2019-01-08 12:27:19 +00:00
if(rc) return rc;
}else{
mosq->msgs_in.inflight_maximum = receive_maximum;
mosq->msgs_in.inflight_quota = receive_maximum;
2019-01-08 12:27:19 +00:00
}
version = MQTT_PROTOCOL_V5;
headerlen = 10;
2019-01-08 12:27:19 +00:00
proplen = 0;
proplen += property__get_length_all(properties);
proplen += property__get_length_all(local_props);
varbytes = packet__varint_bytes(proplen);
headerlen += proplen + varbytes;
2015-01-27 00:32:20 +00:00
}else if(mosq->protocol == mosq_p_mqtt311){
version = MQTT_PROTOCOL_V311;
headerlen = 10;
}else if(mosq->protocol == mosq_p_mqtt31){
version = MQTT_PROTOCOL_V31;
headerlen = 12;
2015-01-27 00:32:20 +00:00
}else{
return MOSQ_ERR_INVAL;
}
packet = mosquitto__calloc(1, sizeof(struct mosquitto__packet));
2014-05-07 22:27:00 +00:00
if(!packet) return MOSQ_ERR_NOMEM;
if(clientid){
2020-10-17 00:23:08 +00:00
payloadlen = (uint32_t)(2U+strlen(clientid));
}else{
2020-10-17 00:23:08 +00:00
payloadlen = 2U;
}
#ifdef WITH_BROKER
if(mosq->will && (mosq->bridge == NULL || mosq->bridge->notifications_local_only == false)){
#else
2014-05-07 22:27:00 +00:00
if(mosq->will){
#endif
2014-05-07 22:27:00 +00:00
will = 1;
2018-10-24 13:07:09 +00:00
assert(mosq->will->msg.topic);
2014-05-07 22:27:00 +00:00
2020-10-17 00:23:08 +00:00
payloadlen += (uint32_t)(2+strlen(mosq->will->msg.topic) + 2+(uint32_t)mosq->will->msg.payloadlen);
if(mosq->protocol == mosq_p_mqtt5){
payloadlen += property__get_remaining_length(mosq->will->properties);
}
2014-05-07 22:27:00 +00:00
}
/* After this check we can be sure that the username and password are
* always valid for the current protocol, so there is no need to check
* username before checking password. */
if(mosq->protocol == mosq_p_mqtt31 || mosq->protocol == mosq_p_mqtt311){
if(password != NULL && username == NULL){
mosquitto__free(packet);
return MOSQ_ERR_INVAL;
}
}
2014-08-16 22:14:41 +00:00
if(username){
2020-10-17 00:23:08 +00:00
payloadlen += (uint32_t)(2+strlen(username));
}
if(password){
2020-10-17 00:23:08 +00:00
payloadlen += (uint32_t)(2+strlen(password));
2014-05-07 22:27:00 +00:00
}
packet->command = CMD_CONNECT;
packet->remaining_length = headerlen + payloadlen;
2015-05-16 13:16:40 +00:00
rc = packet__alloc(packet);
2014-05-07 22:27:00 +00:00
if(rc){
mosquitto__free(packet);
2014-05-07 22:27:00 +00:00
return rc;
}
/* Variable header */
2015-01-27 00:32:20 +00:00
if(version == MQTT_PROTOCOL_V31){
2015-05-16 13:16:40 +00:00
packet__write_string(packet, PROTOCOL_NAME_v31, strlen(PROTOCOL_NAME_v31));
}else{
packet__write_string(packet, PROTOCOL_NAME, strlen(PROTOCOL_NAME));
2015-01-27 00:32:20 +00:00
}
2014-05-07 22:27:00 +00:00
#if defined(WITH_BROKER) && defined(WITH_BRIDGE)
if(mosq->bridge && mosq->bridge->protocol_version != mosq_p_mqtt5 && mosq->bridge->try_private && mosq->bridge->try_private_accepted){
2014-05-07 22:27:00 +00:00
version |= 0x80;
}else{
}
#endif
2015-05-16 13:16:40 +00:00
packet__write_byte(packet, version);
2014-05-07 22:27:00 +00:00
byte = (clean_session&0x1)<<1;
if(will){
2020-10-17 00:23:08 +00:00
byte = byte | (uint8_t)(((mosq->will->msg.qos&0x3)<<3) | ((will&0x1)<<2));
if(mosq->retain_available){
2020-10-17 00:23:08 +00:00
byte |= (uint8_t)((mosq->will->msg.retain&0x1)<<5);
}
2014-05-07 22:27:00 +00:00
}
2014-08-16 22:14:41 +00:00
if(username){
2014-05-07 22:27:00 +00:00
byte = byte | 0x1<<7;
}
if(mosq->password){
byte = byte | 0x1<<6;
2014-05-07 22:27:00 +00:00
}
2015-05-16 13:16:40 +00:00
packet__write_byte(packet, byte);
packet__write_uint16(packet, keepalive);
2014-05-07 22:27:00 +00:00
if(mosq->protocol == mosq_p_mqtt5){
/* Write properties */
2019-01-08 12:27:19 +00:00
packet__write_varint(packet, proplen);
property__write_all(packet, properties, false);
property__write_all(packet, local_props, false);
}
mosquitto_property_free_all(&local_props);
2014-05-07 22:27:00 +00:00
/* Payload */
if(clientid){
2020-10-17 00:23:08 +00:00
packet__write_string(packet, clientid, (uint16_t)strlen(clientid));
}else{
packet__write_uint16(packet, 0);
}
2014-05-07 22:27:00 +00:00
if(will){
if(mosq->protocol == mosq_p_mqtt5){
/* Write will properties */
2018-12-20 15:32:43 +00:00
property__write_all(packet, mosq->will->properties, true);
}
2020-10-17 00:23:08 +00:00
packet__write_string(packet, mosq->will->msg.topic, (uint16_t)strlen(mosq->will->msg.topic));
packet__write_string(packet, (const char *)mosq->will->msg.payload, (uint16_t)mosq->will->msg.payloadlen);
2014-05-07 22:27:00 +00:00
}
2014-08-16 22:14:41 +00:00
if(username){
2020-10-17 00:23:08 +00:00
packet__write_string(packet, username, (uint16_t)strlen(username));
}
if(password){
2020-10-17 00:23:08 +00:00
packet__write_string(packet, password, (uint16_t)strlen(password));
2014-05-07 22:27:00 +00:00
}
mosq->keepalive = keepalive;
#ifdef WITH_BROKER
# ifdef WITH_BRIDGE
2015-05-18 07:53:21 +00:00
log__printf(mosq, MOSQ_LOG_DEBUG, "Bridge %s sending CONNECT", clientid);
2014-05-07 22:27:00 +00:00
# endif
#else
2015-05-18 07:53:21 +00:00
log__printf(mosq, MOSQ_LOG_DEBUG, "Client %s sending CONNECT", clientid);
2014-05-07 22:27:00 +00:00
#endif
2015-05-16 13:16:40 +00:00
return packet__queue(mosq, packet);
2014-05-07 22:27:00 +00:00
}