python MQTT在异常期间重新连接循环,如何构造脚本

ndh0cuux  于 2024-01-05  发布在  Python
关注(0)|答案(1)|浏览(121)

我试图为ESP 32编写一个温度控制器,它通过mqtt将传感器阅读发送到安装在rasperry pi上的mosquito。代码运行,只要wifi连接并且mosquito在raspberry pi上运行。只要ESP32没有连接到wifi或者mqtt broker无法到达,代码就会在重新连接循环中结束。这个想法是,代码的温度控制器端保持运行,即使有一个错误与wifi或mqtt.
Standard behaviour with good wifi / mqtt connectedError with mosquitto stopped
由于我对编程相当陌生,我怀疑我的代码相当混乱,因此我得到了这个错误。我尝试了不同的方法来排列我的函数和循环,但没有运气。
也许有人可以指出我在正确的方向上如何结构我的代码,使它可以处理的exeptions

  1. from machine import Pin, SoftI2C
  2. import ssd1306
  3. import machine
  4. import utime
  5. import onewire, ds18x20, time
  6. from umqtt.robust import MQTTClient
  7. import ubinascii
  8. from machine import RTC
  9. import os
  10. import micropython
  11. import network
  12. import esp
  13. esp.osdebug(None)
  14. import asyncio
  15. import gc
  16. gc.collect()
  17. ###################### define global variables##########################
  18. tempsetpoint = 30.0 # temperature set point
  19. last_message = 0
  20. message_interval = 5
  21. integral = 0
  22. lastupdate = utime.time()
  23. lasterror = 0
  24. output1=0
  25. output2=0
  26. checkin = 0
  27. maxtemperaturedifference = 10
  28. datalogging = False
  29. Kp=100. # 400 Proportional term - Basic steering (This is the first parameter you should tune for a particular setup)
  30. Ki=.01 # .05 Integral term - Compensate for heat loss by vessel
  31. Kd=0. # Derivative term - to prevent overshoot due to inertia - if it is zooming towards setpoint this
  32. # will cancel out the proportional term due to the large negative gradient
  33. output1= 0
  34. output2=0
  35. maxtemperaturedifference = 10
  36. ################### Connect to WIFI and MQTT server ##################################
  37. ssid = 'Inabnit'
  38. password = 'Buochserhorn'
  39. mqtt_server = '192.168.50.31'
  40. """
  41. station = network.WLAN(network.STA_IF)
  42. station.active(True)
  43. station.connect(ssid, password)
  44. while station.isconnected() == False:
  45. pass
  46. print('Connection to WIFI successful')
  47. """
  48. class wifi:
  49. def __init__(self, ssid, password):
  50. self.ssid = ssid
  51. self.password = password
  52. def connect(self):
  53. station = network.WLAN(network.STA_IF)
  54. station.active(True)
  55. station.connect(self.ssid, self.password)
  56. while station.isconnected() == False:
  57. pass
  58. print("Connection to %s successful" %self.ssid)
  59. credentials = wifi("Inabnit", "Buochserhorn")
  60. credentials.connect()
  61. ################### MQTT Setup ##################################
  62. client_id = ubinascii.hexlify(machine.unique_id())
  63. topic_pub_temp1 = b'esp/ds18b20/temperature1' #topic name, add more if needed
  64. topic_pub_temp2 = b'esp/ds18b20/temperature2' #topic name, add more if needed
  65. topic_pub_output1 = b'esp/mosfet/output1' #topic name, add more if needed
  66. topic_pub_output2 = b'esp/mosfet/output2' #topic name, add more if needed
  67. ################### RTC setup ##################################
  68. rtc=RTC()
  69. timeStamp=0
  70. ################### Mosfet setup ##################################
  71. #define mosfet1 pin
  72. p0 = machine.Pin(0)
  73. pwm0 = machine.PWM(p0)
  74. pwm0.freq(500)
  75. pwm0.duty_u16(0)
  76. #define mosfet2 pin
  77. p0 = machine.Pin(1)
  78. pwm1 = machine.PWM(p0)
  79. pwm1.freq(500)
  80. pwm1.duty_u16(0)
  81. ################### DS18b20 setup ##################################
  82. #define temp sensor pin
  83. ds_pin = machine.Pin(21)
  84. ds_sensor = ds18x20.DS18X20(onewire.OneWire(ds_pin))
  85. #roms = ds_sensor.scan()
  86. #print('Found DS devices: ', roms)
  87. Sensor1 = b"('/\x07\x00\x00\x00<"
  88. Sensor2 = b'\x10Fd\x16\x03\x08\x00\x91'
  89. Sensor3 = b'\x10Fd\x16\x03\x08\x00\x80' #update rom with actual address
  90. ################### OLED Setup ##################################
  91. #setup oled
  92. WIDTH = 128 # oled display width
  93. HEIGHT = 64 # oled display height
  94. i2c = SoftI2C(scl=Pin(8), sda=Pin(9))
  95. oled_width = 128
  96. oled_height = 64
  97. oled = ssd1306.SSD1306_I2C(oled_width, oled_height, i2c)
  98. ################### Functions ##################################
  99. ######################## Logging function#########################
  100. with open('savedata.txt', 'a') as f:
  101. f.write('Temperature Data Logging')
  102. f.write('\n')
  103. def logData():
  104. timeTuple=rtc.datetime()
  105. file_size=os.stat('/savedata.txt')
  106. if(file_size[6]<2000000):
  107. try:
  108. with open('savedata.txt', 'a') as f:
  109. f.write(str(timeTuple[4])+':'+str(timeTuple[5])+':'+str(timeTuple[6])+ ',')
  110. f.write(str(temp)+ ',')
  111. f.write(str(output1))
  112. #f.write(scaled_output)
  113. f.write('\n')
  114. print("Data Saved!")
  115. except:
  116. print("Error! Could not save")
  117. ################scaling function to scale X-XXX to X-XXX exapmple: scale_value(output, 0, 100, 0, 65535)#############
  118. def scale_value(value, in_min, in_max, out_min, out_max):
  119. scaled_value = (value - in_min) * (out_max - out_min) / (in_max - in_min) + out_min
  120. return int(scaled_value)
  121. ################################
  122. def checktemperature():
  123. global temp_max
  124. global temp1
  125. global temp2
  126. global tempdifference
  127. #global temp3 #uncomment if 3 sensors will be used
  128. ds_sensor.convert_temp()
  129. temp1 = round(ds_sensor.read_temp(Sensor1),1)
  130. temp2 = round(ds_sensor.read_temp(Sensor2),1)
  131. #temp3 = round(ds_sensor.read_temp(Sensor3),1) #uncomment if 3 sensors will be used
  132. temp_max = round(max(temp1, temp2),1)
  133. #temp_max = round(max(temp1, temp2, temp3),1) #uncomment if 3 sensors will be used
  134. #print(temp_max)
  135. tempdifference = abs(temp1-temp2)
  136. ################################
  137. def displayonOLED():
  138. #display data on the oled
  139. # Clear the oled display in case it has junk on it.
  140. oled.fill(0)
  141. oled.contrast(255) # bright
  142. # Add some text
  143. #oled.text("Temperature ",12,5)
  144. oled.text("Setpoint: ",1,5)
  145. oled.text(str(tempsetpoint),88,5)
  146. oled.text("Sensor 1: ",1,15)
  147. oled.text(str(temp1),88,15)
  148. oled.text("Sensor 2: ",1,25)
  149. oled.text(str(temp2),88,25)
  150. oled.text("Sensor Max: ",1,35)
  151. oled.text(str(temp_max),88,35)
  152. oled.text("Output 1: ",1,45)
  153. oled.text(str(output1),88,45)
  154. # Finally update the oled display so the image & text is displayed
  155. oled.show()
  156. ###################################
  157. def displayonOLED_tempdiff():
  158. #display data on the oled
  159. # Clear the oled display in case it has junk on it.
  160. oled.fill(0)
  161. """### Blink OLED###
  162. oled.show()
  163. utime.sleep_ms(200)
  164. oled.fill(1)
  165. oled.show()
  166. utime.sleep_ms(200)
  167. """
  168. oled.fill(0)
  169. oled.contrast(255) # bright
  170. # Add some text
  171. oled.text("Temp diff.> 10",1,5)
  172. oled.text("Sensor 1: ",1,15)
  173. oled.text(str(temp1),88,15)
  174. oled.text("Sensor 2: ",1,25)
  175. oled.text(str(temp2),88,25)
  176. # Finally update the oled display so the image & text is displayed
  177. oled.show()
  178. ####################### Connect to MQTT Broker###################
  179. def connect_mqtt():
  180. global client_id, mqtt_server
  181. try:
  182. client = MQTTClient(client_id, mqtt_server)
  183. #client = MQTTClient(client_id, mqtt_server, user=your_username, password=your_password)
  184. client.DEBUG = True
  185. client.connect()
  186. #client.loop_start()
  187. print('Connected to %s MQTT broker' % (mqtt_server))
  188. return client
  189. except:
  190. print("Error: MQTT client not reachable")
  191. ######################### restart MQTT
  192. def restart_and_reconnect():
  193. failsafe()
  194. print('Failed to connect to MQTT broker. Reconnecting...')
  195. #time.sleep(10)
  196. #machine.reset()
  197. ###########publish to mqtt#################
  198. def publishtomqtt():
  199. global client_id, mqtt_server
  200. temp1_mqtt = (b'{0:3.1f}'.format(temp1))
  201. temp2_mqtt = (b'{0:3.1f}'.format(temp2))
  202. output1_mqtt = (b'{0:3.1f}'.format(output1))
  203. output2_mqtt = (b'{0:3.1f}'.format(output2))
  204. try:
  205. client = MQTTClient(client_id, mqtt_server)
  206. #client = MQTTClient(client_id, mqtt_server, user=your_username, password=your_password)
  207. client.DEBUG = True
  208. client.connect()
  209. #client.loop_start()
  210. print('Connected to %s MQTT broker' % (mqtt_server))
  211. except OSError as e:
  212. #restart_and_reconnect()
  213. print("Error puplishtomqtt")
  214. #client = connect_mqtt()
  215. client.publish(topic_pub_temp1, temp1_mqtt)
  216. client.publish(topic_pub_temp2, temp2_mqtt)
  217. client.publish(topic_pub_output1, output1_mqtt)
  218. client.publish(topic_pub_output2, output2_mqtt)
  219. client.disconnect()
  220. #######################Heater Failsafe##############################
  221. def failsafe():
  222. pwm0.duty_u16(0)
  223. pwm1.duty_u16(0)
  224. print("Heating Stopped!")
  225. ################### Main Logic ##################################
  226. checktemperature()
  227. while True:
  228. if (time.ticks_ms()-timeStamp)>5000:
  229. timeStamp=time.ticks_ms()
  230. if tempdifference <= maxtemperaturedifference:
  231. try:
  232. checktemperature()
  233. displayonOLED()
  234. if (time.ticks_ms()-timeStamp)>5000:
  235. if datalogging == True :
  236. logData()
  237. try:
  238. publishtomqtt()
  239. except Exception as e:
  240. print( "An error has occurred!")
  241. timeStamp=time.ticks_ms()
  242. now = utime.time()
  243. dt= now-lastupdate
  244. if dt > checkin:
  245. error=tempsetpoint-temp1
  246. integral = integral + dt * error
  247. derivative = (error - lasterror)/dt
  248. output1 = Kp * error + Ki * integral + Kd * derivative
  249. #print(str(output)+"= Kp term: "+str(Kp*error)+" + Ki term:" + str(Ki*integral) + "+ Kd term: " + str(Kd*derivative))
  250. output1 = max(min(100, output1), 0) # Clamp output between 0 and 100
  251. scaled_output1 = scale_value(output1, 0, 100, 0, 65535)
  252. error=tempsetpoint-temp2
  253. integral = integral + dt * error
  254. derivative = (error - lasterror)/dt
  255. output2 = Kp * error + Ki * integral + Kd * derivative
  256. #print(str(output)+"= Kp term: "+str(Kp*error)+" + Ki term:" + str(Ki*integral) + "+ Kd term: " + str(Kd*derivative))
  257. output2 = max(min(100, output2), 0) # Clamp output between 0 and 100
  258. scaled_output2 = scale_value(output2, 0, 100, 0, 65535)
  259. print('Output1: ', round(output1),'% ', 'Output2: ', round(output2),'% ', 'Temp_max: ', temp_max,'°C ', 'Temp1: ', temp1, '°C ', 'Temp2: ', temp2, '°C')
  260. #print('Output: ', output,'% ', 'Temp_max: ', temp_max,'°C ', 'Temp1: ', temp1, '°C ', 'Temp2: ', temp2, '°C', 'Temp3: ', temp3, '°C') #uncomment if 3 sensors will be used
  261. if output1>0:
  262. pwm0.duty_u16(scaled_output1)
  263. if output2>0:
  264. pwm1.duty_u16(scaled_output2)
  265. else:
  266. pwm0.duty_u16(0)
  267. pwm1.duty_u16(0)
  268. #utime.sleep(.1)
  269. lastupdate = now
  270. lasterror = error
  271. except Exception as e:
  272. if (time.ticks_ms()-timeStamp)>1000:
  273. timeStamp=time.ticks_ms()
  274. failsafe()
  275. print('error encountered:'+str(e))
  276. utime.sleep(checkin)
  277. else:
  278. if (time.ticks_ms()-timeStamp)>1000:
  279. timeStamp=time.ticks_ms()
  280. print("Temperature difference >5!")
  281. checktemperature()
  282. displayonOLED_tempdiff()
  283. failsafe()

字符串

pxy2qtax

pxy2qtax1#

我只是尝试构建你的代码,并提供了一些处理代码中异常的例子。

  1. from machine import Pin, SoftI2C
  2. import ssd1306
  3. import machine
  4. import utime
  5. import onewire
  6. import ds18x20
  7. from umqtt.robust import MQTTClient
  8. import ubinascii
  9. from machine import RTC
  10. import os
  11. import network
  12. import esp
  13. # ... (Other imports)
  14. ###################### Define Global Variables ##########################
  15. tempsetpoint = 30.0
  16. integral = 0
  17. lastupdate = utime.time()
  18. lasterror = 0
  19. output1 = 0
  20. output2 = 0
  21. checkin = 0
  22. maxtemperaturedifference = 10
  23. datalogging = False
  24. Kp = 100.0
  25. Ki = 0.01
  26. Kd = 0.0
  27. # ... (Other global variables)
  28. ssid = 'Inabnit'
  29. password = 'Buochserhorn'
  30. mqtt_server = '192.168.50.31'
  31. # ... (Other configurations)
  32. class Wifi:
  33. def __init__(self, ssid, password):
  34. self.ssid = ssid
  35. self.password = password
  36. def connect(self):
  37. station = network.WLAN(network.STA_IF)
  38. station.active(True)
  39. station.connect(self.ssid, self.password)
  40. while not station.isconnected():
  41. pass
  42. print("Connection to %s successful" % self.ssid)
  43. credentials = Wifi("Inabnit", "Buochserhorn")
  44. credentials.connect()
  45. # ... (Other configurations)
  46. client_id = ubinascii.hexlify(machine.unique_id())
  47. topic_pub_temp1 = b'esp/ds18b20/temperature1'
  48. topic_pub_temp2 = b'esp/ds18b20/temperature2'
  49. topic_pub_output1 = b'esp/mosfet/output1'
  50. topic_pub_output2 = b'esp/mosfet/output2'
  51. # ... (Other configurations)
  52. ###################### Functions ##########################
  53. def connect_mqtt():
  54. global client_id, mqtt_server
  55. try:
  56. client = MQTTClient(client_id, mqtt_server)
  57. client.DEBUG = True
  58. client.connect()
  59. print('Connected to %s MQTT broker' % mqtt_server)
  60. return client
  61. except Exception as e:
  62. print("Error: MQTT client not reachable")
  63. return None
  64. def restart_and_reconnect():
  65. failsafe()
  66. print('Failed to connect to MQTT broker. Reconnecting...')
  67. def publish_to_mqtt(client):
  68. if client is not None:
  69. temp1_mqtt = b'{0:3.1f}'.format(temp1)
  70. temp2_mqtt = b'{0:3.1f}'.format(temp2)
  71. output1_mqtt = b'{0:3.1f}'.format(output1)
  72. output2_mqtt = b'{0:3.1f}'.format(output2)
  73. try:
  74. client.publish(topic_pub_temp1, temp1_mqtt)
  75. client.publish(topic_pub_temp2, temp2_mqtt)
  76. client.publish(topic_pub_output1, output1_mqtt)
  77. client.publish(topic_pub_output2, output2_mqtt)
  78. except Exception as e:
  79. print("Error publishing to MQTT:", e)
  80. finally:
  81. client.disconnect()
  82. # ... (Other functions)
  83. ###################### Main Logic ##########################
  84. checktemperature()
  85. while True:
  86. if (utime.ticks_ms() - timeStamp) > 5000:
  87. timeStamp = utime.ticks_ms()
  88. if tempdifference <= maxtemperaturedifference:
  89. try:
  90. checktemperature()
  91. displayonOLED()
  92. if (utime.ticks_ms() - timeStamp) > 5000:
  93. if datalogging:
  94. logData()
  95. try:
  96. mqtt_client = connect_mqtt()
  97. if mqtt_client:
  98. publishtomqtt(mqtt_client)
  99. except Exception as e:
  100. print("An error has occurred:", e)
  101. timeStamp = utime.ticks_ms()
  102. now = utime.time()
  103. dt = now - lastupdate
  104. if dt > checkin:
  105. error = tempsetpoint - temp1
  106. integral = integral + dt * error
  107. derivative = (error - lasterror) / dt
  108. output1 = Kp * error + Ki * integral + Kd * derivative
  109. output1 = max(min(100, output1), 0)
  110. scaled_output1 = scale_value(output1, 0, 100, 0, 65535)
  111. error = tempsetpoint - temp2
  112. integral = integral + dt * error
  113. derivative = (error - lasterror) / dt
  114. output2 = Kp * error + Ki * integral + Kd * derivative
  115. output2 = max(min(100, output2), 0)
  116. scaled_output2 = scale_value(output2, 0, 100, 0, 65535)
  117. print('Output1: ', round(output1), '% ', 'Output2: ', round(output2), '% ',
  118. 'Temp_max: ', temp_max, '°C ', 'Temp1: ', temp1, '°C ', 'Temp2: ', temp2, '°C')
  119. if output1 > 0:
  120. pwm0.duty_u16(scaled_output1)
  121. if output2 > 0:
  122. pwm1.duty_u16(scaled_output2)
  123. else:
  124. pwm0.duty_u16(0)
  125. pwm1.duty_u16(0)
  126. lastupdate = now
  127. lasterror = error
  128. except Exception as e:
  129. if (utime.ticks_ms() - timeStamp) > 1000:
  130. timeStamp = utime.ticks_ms()
  131. failsafe()
  132. print('Error encountered:', e)
  133. utime.sleep(checkin)
  134. else:
  135. if (utime.ticks_ms() - timeStamp) > 1000:
  136. timeStamp = utime.ticks_ms()
  137. print("Temperature difference > 5!")
  138. checktemperature()
  139. displayonOLED_tempdiff()
  140. failsafe()

字符串
在修改后的代码中,我添加了一些异常处理,以提高处理潜在错误的鲁棒性。下面是代码各个部分处理的异常的摘要:
MQTT连接异常:
在connect_mqtt函数中,如果连接到MQTT代理时出现问题,则会捕获异常。

  1. try:
  2. # MQTT connection logic
  3. except Exception as e:
  4. print("Error: MQTT client not reachable")


MQTT发布异常:
在publish_to_mqtt函数中,如果在将消息发布到MQTT主题的过程中出现错误,则会捕获异常。

  1. try:
  2. # MQTT publishing logic
  3. except Exception as e:
  4. print("Error publishing to MQTT:", e)
  5. finally:
  6. # Disconnect the MQTT client even if an exception occurred
  7. client.disconnect()


主要逻辑异常:
在主逻辑循环中,添加了一个通用异常处理程序,用于捕获在主循环执行期间可能发生的任何意外错误。

  1. try:
  2. # Main logic
  3. except Exception as e:
  4. if (utime.ticks_ms() - timeStamp) > 1000:
  5. timeStamp = utime.ticks_ms()
  6. failsafe()
  7. print('Error encountered:', e)
  8. utime.sleep(checkin)


这些异常旨在捕获错误,例如连接到MQTT代理时的网络连接问题或程序主逻辑中的意外错误。特定异常类型(Exception)用于简化,在生产环境中,您可能希望根据潜在错误的性质处理更特定的异常。

展开查看全部

相关问题