Unexpected behaviour of cached messages of Paho MQTT client on Mosquitto server - mqtt

I do use the PAHO C client library in my application. I do subscribe topics with MQTTAsync_subscribe() and QoS set to 1. From what I understand is that 1 means, that a message is send to the client at least one time.
I disconnect the client that subscribed the topic and the client that published the topic is still sending messages to the Mosquitto broker. If I start the subscriber lets say hours later, I get all the buffered messages beginning from the last one when the subscriber was shut down. So far so good! But the problem is, that the messages arrive in the same interval as the publisher sends new messages. By doing so, you will never get the latest message queued up by the publisher.
What I expect is that the Mosquitto broker tries to send all pending messages one after the other to the client instead of sending one old message when a new one is published.
Maybe someone can help me to understand why this happens or maybe how to overcome this situation?

Queued messages should all be batch delivered when the client reconnects.
They are not queued up and delivered with the same time offset, mosquitto wouldn't even keep time of arrival to be able to do time offset delayed delivery (Time of arrival is kept for MQTT v5 messages because they can now have TTL value, but this is only used to remove expired messages from the queue, not to delay late delivery).

Related

MQTT broker when send PUBACK packet

Assume there is a mqtt broker , a topic has 10000 subscriber at QoS 1 called topic_A .Now one publisher publish a message on topic_A,how the broker deal this message?
I think of a way is:
1.save the message
2.send PUBACK to publisher
3.dispatch message to 10000 subscriber
3.1 save one subscriber's message
3.2 publish to one subscriber
3.3 wait puback message from subscriber
3.4 delete the message saved in 3.1
4.delete saved message in 1
but in step 3.Suppose the broker machine is powered off,at this time, 1000 subscriber push completed(3.4 is done),4000 subscriber is waitting from PUBACK(3.3),5000 subscriber haven't started pushing yet(not start 3.1).After a while the broker restart,how to deal with this situation? How to set the publish DUP flag? Is the first one thousand suscriber need push once more after broker restart?
The MQTT spec provides guidance on how this should be done:
When a Server takes ownership of an incoming Application Message it MUST add it to the Session state of those clients that have matching Subscriptions. Matching rules are defined in Section 4.7.
The session state consists of:
· The existence of a Session, even if the rest of the Session state is empty.
· The Client’s subscriptions.
· QoS 1 and QoS 2 messages which have been sent to the Client, but have not been completely acknowledged.
· QoS 1 and QoS 2 messages pending transmission to the Client.
· QoS 2 messages which have been received from the Client, but have not been completely acknowledged.
· Optionally, QoS 0 messages pending transmission to the Client.
So when the server receives a message it effectively adds it to a queue held for each client with a matching subscription (the message may be sent immediately if the client is currently connected). It's important to note that while the message body sent to each client will be identical the headers may differ (different message ID, possibly different QOS etc) and the server must adhere to rules around message ordering. The server knows if the message has already been sent to the client due to the session state meaning it can add the DUP flag appropriately.
I thought it might be worth pointing out a few weaknesses in the algorithm you proposed because it helps explain why the above process is used:
Its much more efficient to send messages in parallel; receive PUB, send PUB to all subscribed clients simultaneously (subject to ordering rules).
If one client is disconnected (cleansession = 0) at the time a message comes in then the message needs to be delivered when it reconnects (your algorithm does not really support this).
If one client does not respond then delivery to other clients would be delayed.
How would the server coordinate messages arriving from multiple clients on one topic (remembering that message ordering is important).

MQTT - Getting the notification when message retention period is over

I have a requirement where I want to get some kind of notification while the message retention period is over and the message is about to be discarded from the MQTT topic.
So the actual requirement is, we have Bluetooth bands, which are send their presence through a centralized agent and an MQTT broker. Now we got a requirement where we need to upgrade the band firmware. For doing so, we will send a message to the topic with a message and a specific retention period. Infra will receive the message notification and look for the band. If the band is found then it's ok otherwise it will wait for new bands to be available. Once the retention period is over, in some cases we have to retry, so to implement the retry mechanism, I wanted to receive the notification from the MQTT broker if a message retention period is over.
Please help me if this is even possible into MQTT?
The broker won't tell you when it drops messages, but since you know when you sent the message and what expiry time you set there is nothing to stop you implementing this yourself.

MQTT - How to know is message received by device

We have mqtt producer and consumer.
MQTT producer is at client level.
When we push message to producer, if the device is switched on, then it will receive the message.
If the device is switched off, then it wont receive the message until it turned on.
We need to know, when message sent to MQTT in server, if server is switched off, then we need to know the status as, it is queued or not received by server.
Based on it, we will send message for the user as, Please turn on device to do specific action.
Is there any better approach to know the status in MQTT to find is message is delivered or failed or queued to know server is active or not.
There is no end to end delivery notification in the MQTT protocol. Part of the pub/sub paradigm is that the publisher should be decoupled from the subscriber, there can be anywhere from 0 to many subscribers to a given topic.
There are 2 approaches to how to potentially work round this.
Have the subscriber respond on a separate topic to acknowledge that it has received the message. You will need to include a unique identifier in the message payload so it can be used in the response message.
You can use the Last Will and Testament feature of MQTT to have the subscriber maintain it's current status. When it starts it publishes a retained message to a known topic, e.g. publishes true to consumer/12345/status and sets a LWT to publish false if the device goes offline unexpectedly. It should also publish false if it cleanly shuts down. That way the publisher can check the status of the subscriber before deciding to publish the message.
I don't know which broker you are using. But in EMQ X MQTT broker, when QoS > 1, the message terminated delivered or ack broker will notify the server
The plugin: emqx-web-hook

MQTT paho client publish message and delete immediately the receiver consumes it

I have been trying to work with paho mqtt client to publish and receive messages with mosquitto as the broker and works fine. My use case although involves the sender publishing a message to the broker and disconnects, at this point, the receiver whether connected or disconnected should consume this message and delete it immediately. I have played with all the properties e.g QOS, retained messages, clean sessions, etc but none is yielding the result I want. Please help.
Assuming a Publish and Subscription at QOS2 the message will only ever be delivered to the subscriber once, there is nothing to delete from anywhere.
If you are trying to ensure that the message is only ever consumed by one specific client then I think you have a misunderstanding about what MQTT is.
MQTT is a PUB/SUB protocol, and as such is designed to totally decouple the subscriber from the publisher. The publisher doesn't know how many subscribers there are, just that it has published a message to a given topic.
0 to N (where N can be any number) of clients can subscribe to the topic. Using QOS, persistent subscriptions and the clean session flag, a client can indicate to the broker that it would like to receive any messages published since was last connected, but this will not influence any other clients that may have also subscribed to that topic.
Starting at MQTT protocol v5 (most brokers and clients currently still only support v3 as of Sept 2018) includes something called Shared Subscriptions* that can be used to round-robin deliver messages on a give topic to a group of clients so only 1 of the set will receive this message, but this does not prevent clients not part of the group from also receiving the message.
The last message with the retained flag set published to a topic will be delivered to all clients at the point they subscribe to the topic. This message can be cleared by publishing a new message with a null payload and the retained flag set. A client could publish a message like this as soon as it receives the retained message but there would still be a timing window where other clients may subscribe and receive the retained message.
*some v3 brokers have implemented propriety versions of this.

get entire past mqtt message queue?

When in QOS 1 & 2 it replays all past messages. Is there a way in standard implementations to receive the entire past queue (as array) when becoming live again? (Of course only subscribed ones)
When a client has subscribed to a topic at QOS 1 or 2 and then disconnects. If when that client reconnects with the same client id and with the clean session flag set to false, the broker should replay any missed messages.
The broker should not replay any messages that had already been sent during the first connected period (with the possible exception of any QOS 1 messages that may have been in flight at the time of the disconnect)

Resources