我使用Mosquitto和Paho的Python实现来尝试传达几个程序。当我使用最后一个功能时,我遇到了一些麻烦。我的代码是这样的:
会员键:
import paho.mqtt.client as mqtt
def on_message(client, userdata, msg):
print 'Received: ' + msg.payload
client = mqtt.Client()
client.on_message = on_message
client.connect('localhost', 1883)
client.subscribe('hello/#')
client.loop_forever()
出版商:
import paho.mqtt.client as mqtt
client = mqtt.Client()
client.will_set('hello/will', 'Last will', 0, False)
client.connect('localhost', 1883)
client.publish('hello/world', 'Regular msg', 0, False)
client.disconnect()
输出:
Received: Last will
我应该只接收常规消息,因为我使用client.disconnect()
来关闭连接。如果我评论will_set
行,我会收到常规消息。我也尝试在同一主题上发布它们并且它不起作用。
答案 0 :(得分:6)
最后遗嘱和遗嘱功能需要遵循一些规则。
client.will_set
在will_set
之前拨打client.connect
。
client = mqtt.Client()
client.will_set("stack/clientstatus", "LOST_CONNECTION", 0, False)
client.connect(host, port, timeout)
连接后调用will_set
对我不起作用。
client.will_set
次尝试多次使用will_set
是可能的,但只有最后一个有效,之前的将被忽略。
client.disconnect
完成 client.disconnect
实际上是请求发送" DISCONNECT"给经纪人的消息。
由于消息是异步发送的,因此您的程序可能会更快结束,然后就是" DISCONNECT"消息已完成。
在client.loop_*()
之后使用client.disconnect
这将允许客户完成发送" DISCONNECT"消息。
以下组合为我工作:
loop_start()
- > loop_stop()
loop_start()
- > loop_forever()
(在发送DISCONNECT消息后完成)我没有尝试其他组合。
从这些选项中,第一个似乎对我更好(保持loop_start
与loop_stop
配对)。第二种选择可能完全错误。
我的工作代码:
import paho.mqtt.client as mqtt
host = "test.mosquitto.org"
port = 1883
timeout = 10
client = mqtt.Client()
client.will_set("stack/clientstatus", "LOST_CONNECTION", 0, False)
client.connect(host, port, timeout)
client.publish("stack/clientstatus", "ON-LINE")
client.loop_start()
for msg in ["msg1", "msg2", "msg3"]:
client.publish("stack/message", msg)
print msg
client.publish("stack/clientstatus", "OFF-LINE")
client.disconnect()
client.loop_stop() # waits, until DISCONNECT message is sent out
print "Now I am done."
sleep
,添加on_disconnect
)添加time.sleep
可能会有所帮助,但在其他情况下(不可靠的连接)可能会失败。
添加client.on_disconnect = on_disconnect
在我的情况下没有帮助,因为看起来,这个回调是异步处理的,并不会阻止client.disconnect
调用。
答案 1 :(得分:3)
disconnect()
必须向经纪人发送DISCONNECT
命令,这就是经纪人知道不发送遗嘱的方式。如果您只是致电disconnect()
,那么无法保证会发生这种情况。您需要调用loop*()
来处理传出的网络流量。您可以通过检查正在调用的on_disconnect()
回调来确保已发送DISCONNECT消息。您也可以在loop_forever()
之后调用disconnect()
,这样会有效。
import paho.mqtt.client as mqtt
client = mqtt.Client()
client.will_set('hello/will', 'Last will', 0, False)
client.connect('localhost', 1883)
client.publish('hello/world', 'Regular msg', 0, False)
client.disconnect()
client.loop_forever()
或者使用paho.mqtt.publish.single()
作为@hardillb建议。
答案 2 :(得分:2)
您可以尝试使用单一方法发布一条消息,如下所示:
import paho.mqtt.publish as publish
publish.single('hello/world', 'Regular msg', 0, False, 'localhost' , 1883, 'publisher', 10, {'topic': 'hello/will', 'payload': 'Will msg', 'qos': 0, 'retain': False})
https://pypi.python.org/pypi/paho-mqtt#single
我猜测问题是你在发布实际完成之前断开连接,这可能就是你看到遗嘱信息的原因。
编辑 - 当我使用mosquitto_sub -v -t'你好/#'运行代码时我看到正常的信息和意志传递。
EDIT2 -
这对我来说很好用:
import paho.mqtt.client as mqtt
client = mqtt.Client()
client.will_set('hello/will', 'Last will', 0, False)
client.connect('localhost', 1883)
client.publish('hello/world', 'Regular msg', 0, False)
client.loop();
client.disconnect()
client.loop();