我试图为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
from machine import Pin, SoftI2C
import ssd1306
import machine
import utime
import onewire, ds18x20, time
from umqtt.robust import MQTTClient
import ubinascii
from machine import RTC
import os
import micropython
import network
import esp
esp.osdebug(None)
import asyncio
import gc
gc.collect()
###################### define global variables##########################
tempsetpoint = 30.0 # temperature set point
last_message = 0
message_interval = 5
integral = 0
lastupdate = utime.time()
lasterror = 0
output1=0
output2=0
checkin = 0
maxtemperaturedifference = 10
datalogging = False
Kp=100. # 400 Proportional term - Basic steering (This is the first parameter you should tune for a particular setup)
Ki=.01 # .05 Integral term - Compensate for heat loss by vessel
Kd=0. # Derivative term - to prevent overshoot due to inertia - if it is zooming towards setpoint this
# will cancel out the proportional term due to the large negative gradient
output1= 0
output2=0
maxtemperaturedifference = 10
################### Connect to WIFI and MQTT server ##################################
ssid = 'Inabnit'
password = 'Buochserhorn'
mqtt_server = '192.168.50.31'
"""
station = network.WLAN(network.STA_IF)
station.active(True)
station.connect(ssid, password)
while station.isconnected() == False:
pass
print('Connection to WIFI successful')
"""
class wifi:
def __init__(self, ssid, password):
self.ssid = ssid
self.password = password
def connect(self):
station = network.WLAN(network.STA_IF)
station.active(True)
station.connect(self.ssid, self.password)
while station.isconnected() == False:
pass
print("Connection to %s successful" %self.ssid)
credentials = wifi("Inabnit", "Buochserhorn")
credentials.connect()
################### MQTT Setup ##################################
client_id = ubinascii.hexlify(machine.unique_id())
topic_pub_temp1 = b'esp/ds18b20/temperature1' #topic name, add more if needed
topic_pub_temp2 = b'esp/ds18b20/temperature2' #topic name, add more if needed
topic_pub_output1 = b'esp/mosfet/output1' #topic name, add more if needed
topic_pub_output2 = b'esp/mosfet/output2' #topic name, add more if needed
################### RTC setup ##################################
rtc=RTC()
timeStamp=0
################### Mosfet setup ##################################
#define mosfet1 pin
p0 = machine.Pin(0)
pwm0 = machine.PWM(p0)
pwm0.freq(500)
pwm0.duty_u16(0)
#define mosfet2 pin
p0 = machine.Pin(1)
pwm1 = machine.PWM(p0)
pwm1.freq(500)
pwm1.duty_u16(0)
################### DS18b20 setup ##################################
#define temp sensor pin
ds_pin = machine.Pin(21)
ds_sensor = ds18x20.DS18X20(onewire.OneWire(ds_pin))
#roms = ds_sensor.scan()
#print('Found DS devices: ', roms)
Sensor1 = b"('/\x07\x00\x00\x00<"
Sensor2 = b'\x10Fd\x16\x03\x08\x00\x91'
Sensor3 = b'\x10Fd\x16\x03\x08\x00\x80' #update rom with actual address
################### OLED Setup ##################################
#setup oled
WIDTH = 128 # oled display width
HEIGHT = 64 # oled display height
i2c = SoftI2C(scl=Pin(8), sda=Pin(9))
oled_width = 128
oled_height = 64
oled = ssd1306.SSD1306_I2C(oled_width, oled_height, i2c)
################### Functions ##################################
######################## Logging function#########################
with open('savedata.txt', 'a') as f:
f.write('Temperature Data Logging')
f.write('\n')
def logData():
timeTuple=rtc.datetime()
file_size=os.stat('/savedata.txt')
if(file_size[6]<2000000):
try:
with open('savedata.txt', 'a') as f:
f.write(str(timeTuple[4])+':'+str(timeTuple[5])+':'+str(timeTuple[6])+ ',')
f.write(str(temp)+ ',')
f.write(str(output1))
#f.write(scaled_output)
f.write('\n')
print("Data Saved!")
except:
print("Error! Could not save")
################scaling function to scale X-XXX to X-XXX exapmple: scale_value(output, 0, 100, 0, 65535)#############
def scale_value(value, in_min, in_max, out_min, out_max):
scaled_value = (value - in_min) * (out_max - out_min) / (in_max - in_min) + out_min
return int(scaled_value)
################################
def checktemperature():
global temp_max
global temp1
global temp2
global tempdifference
#global temp3 #uncomment if 3 sensors will be used
ds_sensor.convert_temp()
temp1 = round(ds_sensor.read_temp(Sensor1),1)
temp2 = round(ds_sensor.read_temp(Sensor2),1)
#temp3 = round(ds_sensor.read_temp(Sensor3),1) #uncomment if 3 sensors will be used
temp_max = round(max(temp1, temp2),1)
#temp_max = round(max(temp1, temp2, temp3),1) #uncomment if 3 sensors will be used
#print(temp_max)
tempdifference = abs(temp1-temp2)
################################
def displayonOLED():
#display data on the oled
# Clear the oled display in case it has junk on it.
oled.fill(0)
oled.contrast(255) # bright
# Add some text
#oled.text("Temperature ",12,5)
oled.text("Setpoint: ",1,5)
oled.text(str(tempsetpoint),88,5)
oled.text("Sensor 1: ",1,15)
oled.text(str(temp1),88,15)
oled.text("Sensor 2: ",1,25)
oled.text(str(temp2),88,25)
oled.text("Sensor Max: ",1,35)
oled.text(str(temp_max),88,35)
oled.text("Output 1: ",1,45)
oled.text(str(output1),88,45)
# Finally update the oled display so the image & text is displayed
oled.show()
###################################
def displayonOLED_tempdiff():
#display data on the oled
# Clear the oled display in case it has junk on it.
oled.fill(0)
"""### Blink OLED###
oled.show()
utime.sleep_ms(200)
oled.fill(1)
oled.show()
utime.sleep_ms(200)
"""
oled.fill(0)
oled.contrast(255) # bright
# Add some text
oled.text("Temp diff.> 10",1,5)
oled.text("Sensor 1: ",1,15)
oled.text(str(temp1),88,15)
oled.text("Sensor 2: ",1,25)
oled.text(str(temp2),88,25)
# Finally update the oled display so the image & text is displayed
oled.show()
####################### Connect to MQTT Broker###################
def connect_mqtt():
global client_id, mqtt_server
try:
client = MQTTClient(client_id, mqtt_server)
#client = MQTTClient(client_id, mqtt_server, user=your_username, password=your_password)
client.DEBUG = True
client.connect()
#client.loop_start()
print('Connected to %s MQTT broker' % (mqtt_server))
return client
except:
print("Error: MQTT client not reachable")
######################### restart MQTT
def restart_and_reconnect():
failsafe()
print('Failed to connect to MQTT broker. Reconnecting...')
#time.sleep(10)
#machine.reset()
###########publish to mqtt#################
def publishtomqtt():
global client_id, mqtt_server
temp1_mqtt = (b'{0:3.1f}'.format(temp1))
temp2_mqtt = (b'{0:3.1f}'.format(temp2))
output1_mqtt = (b'{0:3.1f}'.format(output1))
output2_mqtt = (b'{0:3.1f}'.format(output2))
try:
client = MQTTClient(client_id, mqtt_server)
#client = MQTTClient(client_id, mqtt_server, user=your_username, password=your_password)
client.DEBUG = True
client.connect()
#client.loop_start()
print('Connected to %s MQTT broker' % (mqtt_server))
except OSError as e:
#restart_and_reconnect()
print("Error puplishtomqtt")
#client = connect_mqtt()
client.publish(topic_pub_temp1, temp1_mqtt)
client.publish(topic_pub_temp2, temp2_mqtt)
client.publish(topic_pub_output1, output1_mqtt)
client.publish(topic_pub_output2, output2_mqtt)
client.disconnect()
#######################Heater Failsafe##############################
def failsafe():
pwm0.duty_u16(0)
pwm1.duty_u16(0)
print("Heating Stopped!")
################### Main Logic ##################################
checktemperature()
while True:
if (time.ticks_ms()-timeStamp)>5000:
timeStamp=time.ticks_ms()
if tempdifference <= maxtemperaturedifference:
try:
checktemperature()
displayonOLED()
if (time.ticks_ms()-timeStamp)>5000:
if datalogging == True :
logData()
try:
publishtomqtt()
except Exception as e:
print( "An error has occurred!")
timeStamp=time.ticks_ms()
now = utime.time()
dt= now-lastupdate
if dt > checkin:
error=tempsetpoint-temp1
integral = integral + dt * error
derivative = (error - lasterror)/dt
output1 = Kp * error + Ki * integral + Kd * derivative
#print(str(output)+"= Kp term: "+str(Kp*error)+" + Ki term:" + str(Ki*integral) + "+ Kd term: " + str(Kd*derivative))
output1 = max(min(100, output1), 0) # Clamp output between 0 and 100
scaled_output1 = scale_value(output1, 0, 100, 0, 65535)
error=tempsetpoint-temp2
integral = integral + dt * error
derivative = (error - lasterror)/dt
output2 = Kp * error + Ki * integral + Kd * derivative
#print(str(output)+"= Kp term: "+str(Kp*error)+" + Ki term:" + str(Ki*integral) + "+ Kd term: " + str(Kd*derivative))
output2 = max(min(100, output2), 0) # Clamp output between 0 and 100
scaled_output2 = scale_value(output2, 0, 100, 0, 65535)
print('Output1: ', round(output1),'% ', 'Output2: ', round(output2),'% ', 'Temp_max: ', temp_max,'°C ', 'Temp1: ', temp1, '°C ', 'Temp2: ', temp2, '°C')
#print('Output: ', output,'% ', 'Temp_max: ', temp_max,'°C ', 'Temp1: ', temp1, '°C ', 'Temp2: ', temp2, '°C', 'Temp3: ', temp3, '°C') #uncomment if 3 sensors will be used
if output1>0:
pwm0.duty_u16(scaled_output1)
if output2>0:
pwm1.duty_u16(scaled_output2)
else:
pwm0.duty_u16(0)
pwm1.duty_u16(0)
#utime.sleep(.1)
lastupdate = now
lasterror = error
except Exception as e:
if (time.ticks_ms()-timeStamp)>1000:
timeStamp=time.ticks_ms()
failsafe()
print('error encountered:'+str(e))
utime.sleep(checkin)
else:
if (time.ticks_ms()-timeStamp)>1000:
timeStamp=time.ticks_ms()
print("Temperature difference >5!")
checktemperature()
displayonOLED_tempdiff()
failsafe()
字符串
1条答案
按热度按时间pxy2qtax1#
我只是尝试构建你的代码,并提供了一些处理代码中异常的例子。
字符串
在修改后的代码中,我添加了一些异常处理,以提高处理潜在错误的鲁棒性。下面是代码各个部分处理的异常的摘要:
MQTT连接异常:
在connect_mqtt函数中,如果连接到MQTT代理时出现问题,则会捕获异常。
型
MQTT发布异常:
在publish_to_mqtt函数中,如果在将消息发布到MQTT主题的过程中出现错误,则会捕获异常。
型
主要逻辑异常:
在主逻辑循环中,添加了一个通用异常处理程序,用于捕获在主循环执行期间可能发生的任何意外错误。
型
这些异常旨在捕获错误,例如连接到MQTT代理时的网络连接问题或程序主逻辑中的意外错误。特定异常类型(Exception)用于简化,在生产环境中,您可能希望根据潜在错误的性质处理更特定的异常。