Fix for #575314.
Incoming QoS 2 messages that had not completed the QoS flow were not being checked for ACL access when a clean session=False client was reconnecting. Closes #575314.pull/2343/head
parent
9d95cba95e
commit
37b5aedcb6
@ -0,0 +1,69 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
# A clean start=False client connects, and publishes to a topic it has access
|
||||||
|
# to with QoS 2 - but does not send a PUBREL. It closes the connection. The
|
||||||
|
# access to the topic is revoked, the client reconnects and it attempts to
|
||||||
|
# complete the flow. Is the publish allowed? It should not be.
|
||||||
|
|
||||||
|
from mosq_test_helper import *
|
||||||
|
|
||||||
|
def write_config(filename, port, plugin_ver):
|
||||||
|
with open(filename, 'w') as f:
|
||||||
|
f.write("listener %d\n" % (port))
|
||||||
|
f.write("auth_plugin c/auth_plugin_acl_change.so\n")
|
||||||
|
f.write("allow_anonymous true\n")
|
||||||
|
|
||||||
|
def do_test(plugin_ver):
|
||||||
|
port = mosq_test.get_port()
|
||||||
|
conf_file = os.path.basename(__file__).replace('.py', '.conf')
|
||||||
|
write_config(conf_file, port, plugin_ver)
|
||||||
|
|
||||||
|
rc = 1
|
||||||
|
connect1_packet = mosq_test.gen_connect("acl-change-test", clean_session=False)
|
||||||
|
connack1_packet = mosq_test.gen_connack(rc=0)
|
||||||
|
|
||||||
|
connect2_packet = mosq_test.gen_connect("acl-change-test", clean_session=False)
|
||||||
|
connack2_packet = mosq_test.gen_connack(rc=0,flags=1)
|
||||||
|
|
||||||
|
mid = 1
|
||||||
|
subscribe_packet = mosq_test.gen_subscribe(mid, "#", 0)
|
||||||
|
suback_packet = mosq_test.gen_suback(mid, 0)
|
||||||
|
|
||||||
|
mid = 2
|
||||||
|
publish1_packet = mosq_test.gen_publish("publish/topic", qos=2, mid=mid, payload="message")
|
||||||
|
pubrec1_packet = mosq_test.gen_pubrec(mid)
|
||||||
|
pubrel1_packet = mosq_test.gen_pubrel(mid)
|
||||||
|
pubcomp1_packet = mosq_test.gen_pubcomp(mid)
|
||||||
|
|
||||||
|
broker = mosq_test.start_broker(filename=os.path.basename(__file__), use_conf=True, port=port)
|
||||||
|
|
||||||
|
try:
|
||||||
|
sock = mosq_test.do_client_connect(connect1_packet, connack1_packet, timeout=20, port=port)
|
||||||
|
|
||||||
|
mosq_test.do_send_receive(sock, subscribe_packet, suback_packet, "suback 1")
|
||||||
|
mosq_test.do_send_receive(sock, publish1_packet, pubrec1_packet, "pubrec")
|
||||||
|
sock.close()
|
||||||
|
|
||||||
|
# ACL has changed
|
||||||
|
sock = mosq_test.do_client_connect(connect2_packet, connack2_packet, timeout=20, port=port)
|
||||||
|
mosq_test.do_send_receive(sock, subscribe_packet, suback_packet, "suback 2")
|
||||||
|
mosq_test.do_send_receive(sock, pubrel1_packet, pubcomp1_packet, "pubcomp")
|
||||||
|
mosq_test.do_ping(sock)
|
||||||
|
|
||||||
|
rc = 0
|
||||||
|
|
||||||
|
sock.close()
|
||||||
|
except mosq_test.TestError:
|
||||||
|
pass
|
||||||
|
except Exception as err:
|
||||||
|
print(err)
|
||||||
|
finally:
|
||||||
|
os.remove(conf_file)
|
||||||
|
broker.terminate()
|
||||||
|
broker.wait()
|
||||||
|
(stdo, stde) = broker.communicate()
|
||||||
|
if rc:
|
||||||
|
print(stde.decode('utf-8'))
|
||||||
|
exit(rc)
|
||||||
|
|
||||||
|
do_test(4)
|
@ -0,0 +1,52 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <mosquitto.h>
|
||||||
|
#include <mosquitto_broker.h>
|
||||||
|
#include <mosquitto_plugin.h>
|
||||||
|
|
||||||
|
int mosquitto_auth_acl_check_v5(int event, void *event_data, void *user_data);
|
||||||
|
int mosquitto_auth_unpwd_check_v5(int event, void *event_data, void *user_data);
|
||||||
|
|
||||||
|
static mosquitto_plugin_id_t *plg_id;
|
||||||
|
|
||||||
|
static int login_count = 0;
|
||||||
|
|
||||||
|
int mosquitto_plugin_version(int supported_version_count, const int *supported_versions)
|
||||||
|
{
|
||||||
|
return 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mosquitto_plugin_init(mosquitto_plugin_id_t *identifier, void **user_data, struct mosquitto_opt *auth_opts, int auth_opt_count)
|
||||||
|
{
|
||||||
|
plg_id = identifier;
|
||||||
|
|
||||||
|
mosquitto_callback_register(plg_id, MOSQ_EVT_ACL_CHECK, mosquitto_auth_acl_check_v5, NULL, NULL);
|
||||||
|
mosquitto_callback_register(plg_id, MOSQ_EVT_BASIC_AUTH, mosquitto_auth_unpwd_check_v5, NULL, NULL);
|
||||||
|
|
||||||
|
return MOSQ_ERR_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mosquitto_plugin_cleanup(void *user_data, struct mosquitto_opt *auth_opts, int auth_opt_count)
|
||||||
|
{
|
||||||
|
mosquitto_callback_unregister(plg_id, MOSQ_EVT_ACL_CHECK, mosquitto_auth_acl_check_v5, NULL);
|
||||||
|
mosquitto_callback_unregister(plg_id, MOSQ_EVT_BASIC_AUTH, mosquitto_auth_unpwd_check_v5, NULL);
|
||||||
|
|
||||||
|
return MOSQ_ERR_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mosquitto_auth_acl_check_v5(int event, void *event_data, void *user_data)
|
||||||
|
{
|
||||||
|
struct mosquitto_evt_acl_check *ed = event_data;
|
||||||
|
|
||||||
|
if(login_count == 2 && ed->access == MOSQ_ACL_WRITE){
|
||||||
|
return MOSQ_ERR_ACL_DENIED;
|
||||||
|
}else{
|
||||||
|
return MOSQ_ERR_SUCCESS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int mosquitto_auth_unpwd_check_v5(int event, void *event_data, void *user_data)
|
||||||
|
{
|
||||||
|
login_count++;
|
||||||
|
return MOSQ_ERR_SUCCESS;
|
||||||
|
}
|
Loading…
Reference in New Issue