mosquitto/src/context.c

295 lines
7.1 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.
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>
2014-09-14 17:08:09 +00:00
#include <time.h>
2014-05-07 22:27:00 +00:00
#include "mosquitto_broker_internal.h"
2019-01-18 21:30:34 +00:00
#include "alias_mosq.h"
2015-04-29 20:37:47 +00:00
#include "memory_mosq.h"
2015-04-29 20:23:59 +00:00
#include "packet_mosq.h"
#include "property_mosq.h"
2015-04-29 20:37:47 +00:00
#include "time_mosq.h"
2019-09-25 11:17:17 +00:00
#include "util_mosq.h"
#include "will_mosq.h"
2014-05-07 22:27:00 +00:00
#include "uthash.h"
struct mosquitto *context__init(mosq_sock_t sock)
2014-05-07 22:27:00 +00:00
{
struct mosquitto *context;
char address[1024];
context = mosquitto__calloc(1, sizeof(struct mosquitto));
2014-05-07 22:27:00 +00:00
if(!context) return NULL;
#ifdef WITH_EPOLL
context->ident = id_client;
#endif
2018-03-08 12:43:41 +00:00
context->pollfd_index = -1;
mosquitto__set_state(context, mosq_cs_new);
2014-05-07 22:27:00 +00:00
context->sock = sock;
context->last_msg_in = db.now_s;
context->next_msg_out = db.now_s + 60;
2014-05-07 22:27:00 +00:00
context->keepalive = 60; /* Default to 60s */
context->clean_start = true;
2014-05-07 22:27:00 +00:00
context->id = NULL;
context->last_mid = 0;
context->will = NULL;
context->username = NULL;
context->password = NULL;
context->listener = NULL;
context->acl_list = NULL;
context->retain_available = true;
2014-05-07 22:27:00 +00:00
/* is_bridge records whether this client is a bridge or not. This could be
* done by looking at context->bridge for bridges that we create ourself,
* but incoming bridges need some other way of being recorded. */
context->is_bridge = false;
context->in_packet.payload = NULL;
2015-05-16 13:16:40 +00:00
packet__cleanup(&context->in_packet);
2014-05-07 22:27:00 +00:00
context->out_packet = NULL;
context->current_out_packet = NULL;
context->address = NULL;
2014-10-26 21:17:08 +00:00
if((int)sock >= 0){
2015-05-16 17:43:06 +00:00
if(!net__socket_get_address(sock, address, 1024)){
context->address = mosquitto__strdup(address);
2014-05-07 22:27:00 +00:00
}
if(!context->address){
/* getpeername and inet_ntop failed and not a bridge */
mosquitto__free(context);
2014-05-07 22:27:00 +00:00
return NULL;
}
}
context->bridge = NULL;
context->msgs_in.inflight_maximum = db.config->max_inflight_messages;
context->msgs_out.inflight_maximum = db.config->max_inflight_messages;
context->msgs_in.inflight_quota = db.config->max_inflight_messages;
context->msgs_out.inflight_quota = db.config->max_inflight_messages;
context->max_qos = 2;
2014-05-07 22:27:00 +00:00
#ifdef WITH_TLS
context->ssl = NULL;
#endif
2014-10-26 21:17:08 +00:00
if((int)context->sock >= 0){
HASH_ADD(hh_sock, db.contexts_by_sock, sock, sizeof(context->sock), context);
}
2014-05-07 22:27:00 +00:00
return context;
}
/*
* This will result in any outgoing packets going unsent. If we're disconnected
* forcefully then it is usually an error condition and shouldn't be a problem,
* but it will mean that CONNACK messages will never get sent for bad protocol
* versions for example.
*/
void context__cleanup(struct mosquitto *context, bool force_free)
2014-05-07 22:27:00 +00:00
{
struct mosquitto__packet *packet;
2014-05-07 22:27:00 +00:00
if(!context) return;
if(force_free){
context->clean_start = true;
}
2014-05-07 22:27:00 +00:00
#ifdef WITH_BRIDGE
if(context->bridge){
bridge__cleanup(context);
2014-05-07 22:27:00 +00:00
}
#endif
2019-01-18 21:30:34 +00:00
alias__free_all(context);
mosquitto__free(context->auth_method);
context->auth_method = NULL;
mosquitto__free(context->username);
context->username = NULL;
mosquitto__free(context->password);
context->password = NULL;
net__socket_close(context);
if(force_free){
sub__clean_session(context);
2014-05-07 22:27:00 +00:00
}
db__messages_delete(context, force_free);
mosquitto__free(context->address);
context->address = NULL;
2014-06-29 22:16:10 +00:00
context__send_will(context);
2014-05-07 22:27:00 +00:00
if(context->id){
context__remove_from_by_id(context);
mosquitto__free(context->id);
2014-05-07 22:27:00 +00:00
context->id = NULL;
}
2015-05-16 13:16:40 +00:00
packet__cleanup(&(context->in_packet));
if(context->current_out_packet){
2015-05-16 13:16:40 +00:00
packet__cleanup(context->current_out_packet);
mosquitto__free(context->current_out_packet);
context->current_out_packet = NULL;
}
2014-05-07 22:27:00 +00:00
while(context->out_packet){
2015-05-16 13:16:40 +00:00
packet__cleanup(context->out_packet);
2014-05-07 22:27:00 +00:00
packet = context->out_packet;
context->out_packet = context->out_packet->next;
mosquitto__free(packet);
2014-05-07 22:27:00 +00:00
}
#if defined(WITH_BROKER) && defined(__GLIBC__) && defined(WITH_ADNS)
if(context->adns){
gai_cancel(context->adns);
mosquitto__free((struct addrinfo *)context->adns->ar_request);
mosquitto__free(context->adns);
}
#endif
if(force_free){
mosquitto__free(context);
2014-05-07 22:27:00 +00:00
}
}
void context__send_will(struct mosquitto *ctxt)
2014-05-07 22:27:00 +00:00
{
if(ctxt->state != mosq_cs_disconnecting && ctxt->will){
2019-02-25 22:28:51 +00:00
if(ctxt->will_delay_interval > 0){
will_delay__add(ctxt);
2019-02-25 22:28:51 +00:00
return;
}
if(mosquitto_acl_check(ctxt,
2018-10-24 13:07:09 +00:00
ctxt->will->msg.topic,
2020-10-17 00:23:08 +00:00
(uint32_t)ctxt->will->msg.payloadlen,
2018-10-24 13:07:09 +00:00
ctxt->will->msg.payload,
2020-10-17 00:23:08 +00:00
(uint8_t)ctxt->will->msg.qos,
2018-10-24 13:07:09 +00:00
ctxt->will->msg.retain,
MOSQ_ACL_WRITE) == MOSQ_ERR_SUCCESS){
/* Unexpected disconnect, queue the client will. */
db__messages_easy_queue(ctxt,
2018-10-24 13:07:09 +00:00
ctxt->will->msg.topic,
2020-10-17 00:23:08 +00:00
(uint8_t)ctxt->will->msg.qos,
(uint32_t)ctxt->will->msg.payloadlen,
2018-10-24 13:07:09 +00:00
ctxt->will->msg.payload,
ctxt->will->msg.retain,
ctxt->will->expiry_interval,
&ctxt->will->properties);
}
2014-05-07 22:27:00 +00:00
}
will__clear(ctxt);
}
void context__disconnect(struct mosquitto *context)
{
if(mosquitto__get_state(context) == mosq_cs_disconnected){
return;
}
plugin__handle_disconnect(context, -1);
net__socket_close(context);
2019-03-12 22:47:48 +00:00
context__send_will(context);
if(context->session_expiry_interval == 0){
/* Client session is due to be expired now */
#ifdef WITH_BRIDGE
if(context->bridge == NULL)
#endif
{
if(context->will_delay_interval == 0){
2019-03-12 22:47:48 +00:00
/* This will be done later, after the will is published for delay>0. */
context__add_to_disused(context);
}
}
}else{
session_expiry__add(context);
}
keepalive__remove(context);
mosquitto__set_state(context, mosq_cs_disconnected);
2014-05-07 22:27:00 +00:00
}
void context__add_to_disused(struct mosquitto *context)
{
2019-03-12 22:47:48 +00:00
if(context->state == mosq_cs_disused) return;
mosquitto__set_state(context, mosq_cs_disused);
2019-03-12 22:47:48 +00:00
if(context->id){
context__remove_from_by_id(context);
mosquitto__free(context->id);
context->id = NULL;
}
context->for_free_next = db.ll_for_free;
db.ll_for_free = context;
}
void context__free_disused(void)
{
struct mosquitto *context, *next;
#ifdef WITH_WEBSOCKETS
struct mosquitto *last = NULL;
#endif
context = db.ll_for_free;
db.ll_for_free = NULL;
while(context){
#ifdef WITH_WEBSOCKETS
if(context->wsi){
/* Don't delete yet, lws hasn't finished with it */
if(last){
last->for_free_next = context;
}else{
db.ll_for_free = context;
}
next = context->for_free_next;
context->for_free_next = NULL;
last = context;
context = next;
}else
#endif
{
next = context->for_free_next;
context__cleanup(context, true);
context = next;
}
}
}
void context__remove_from_by_id(struct mosquitto *context)
{
struct mosquitto *context_found;
if(context->removed_from_by_id == false && context->id){
HASH_FIND(hh_id, db.contexts_by_id, context->id, strlen(context->id), context_found);
if(context_found){
HASH_DELETE(hh_id, db.contexts_by_id, context_found);
}
context->removed_from_by_id = true;
}
}
2019-03-15 21:31:11 +00:00