diff --git a/boards/nologo_esp32c3_super_mini.json b/boards/nologo_esp32c3_super_mini.json new file mode 100644 index 0000000..3b2924c --- /dev/null +++ b/boards/nologo_esp32c3_super_mini.json @@ -0,0 +1,48 @@ +{ + "build": { + "arduino": { + "ldscript": "esp32c3_out.ld" + }, + "core": "esp32", + "extra_flags": [ + "-DARDUINO_XIAO_ESP32C3", + "-DARDUINO_USB_MODE=1", + "-DARDUINO_USB_CDC_ON_BOOT=1" + ], + "f_cpu": "160000000L", + "f_flash": "80000000L", + "flash_mode": "qio", + "hwids": [ + [ + "0x2886", + "0x0046" + ], + [ + "0x303a", + "0x1001" + ] + ], + "mcu": "esp32c3", + "variant": "nologo_esp32c3_super_mini" + }, + "connectivity": [ + "wifi" + ], + "debug": { + "openocd_target": "esp32c3.cfg" + }, + "frameworks": [ + "arduino", + "espidf" + ], + "name": "No logo ESP32C3 super mini", + "upload": { + "flash_size": "4MB", + "maximum_ram_size": 327680, + "maximum_size": 4194304, + "require_upload_port": true, + "speed": 460800 + }, + "url": "", + "vendor": "No logo" +} diff --git a/boards/seeed_xiao_esp32s3.json b/boards/seeed_xiao_esp32s3.json new file mode 100644 index 0000000..5c7171c --- /dev/null +++ b/boards/seeed_xiao_esp32s3.json @@ -0,0 +1,54 @@ +{ + "build": { + "arduino": { + "ldscript": "esp32s3_out.ld", + "partitions": "default_8MB.csv", + "memory_type": "qio_opi" + }, + "core": "esp32", + "extra_flags": [ + "-DARDUINO_XIAO_ESP32S3", + "-DBOARD_HAS_PSRAM", + "-DARDUINO_USB_MODE=1", + "-DARDUINO_USB_CDC_ON_BOOT=1", + "-DARDUINO_RUNNING_CORE=1", + "-DARDUINO_EVENT_RUNNING_CORE=1" + ], + "f_cpu": "240000000L", + "f_flash": "80000000L", + "flash_mode": "qio", + "hwids": [ + [ + "0x2886", + "0x0056" + ], + [ + "0x2886", + "0x8056" + ] + ], + "mcu": "esp32s3", + "variant": "XIAO_ESP32S3" + }, + "connectivity": [ + "bluetooth", + "wifi" + ], + "debug": { + "openocd_target": "esp32s3.cfg" + }, + "frameworks": [ + "arduino", + "espidf" + ], + "name": "Seeed Studio XIAO ESP32S3", + "upload": { + "flash_size": "8MB", + "maximum_ram_size": 327680, + "maximum_size": 8388608, + "require_upload_port": true, + "speed": 460800 + }, + "url": "https://www.seeedstudio.com/XIAO-ESP32S3-p-5627.html", + "vendor": "Seeed Studio" +} diff --git a/boards/variants/XIAO_ESP32S3/bootloader-tinyuf2.bin b/boards/variants/XIAO_ESP32S3/bootloader-tinyuf2.bin new file mode 100644 index 0000000..b11e5b2 Binary files /dev/null and b/boards/variants/XIAO_ESP32S3/bootloader-tinyuf2.bin differ diff --git a/boards/variants/XIAO_ESP32S3/partitions-8MB.csv b/boards/variants/XIAO_ESP32S3/partitions-8MB.csv new file mode 100644 index 0000000..4026378 --- /dev/null +++ b/boards/variants/XIAO_ESP32S3/partitions-8MB.csv @@ -0,0 +1,10 @@ +# ESP-IDF Partition Table +# Name, Type, SubType, Offset, Size, Flags +# bootloader.bin,, 0x1000, 32K +# partition table,, 0x8000, 4K +nvs, data, nvs, 0x9000, 20K, +otadata, data, ota, 0xe000, 8K, +ota_0, 0, ota_0, 0x10000, 2048K, +ota_1, 0, ota_1, 0x210000, 2048K, +uf2, app, factory,0x410000, 256K, +ffat, data, fat, 0x450000, 3776K, diff --git a/boards/variants/XIAO_ESP32S3/pins_arduino.h b/boards/variants/XIAO_ESP32S3/pins_arduino.h new file mode 100644 index 0000000..ea46b27 --- /dev/null +++ b/boards/variants/XIAO_ESP32S3/pins_arduino.h @@ -0,0 +1,82 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include + +#define USB_VID 0x2886 +#define USB_PID 0x0056 + +static const uint8_t LED_BUILTIN = 21; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN + +static const uint8_t TX = 43; +static const uint8_t RX = 44; + +/* +static const uint8_t SDA = 5; +static const uint8_t SCL = 6; + +static const uint8_t SS = 44; +static const uint8_t MOSI = 9; +static const uint8_t MISO = 8; +static const uint8_t SCK = 7; +*/ + +static const uint8_t SDA = 6; +static const uint8_t SCL = 5; + +static const uint8_t SS = 3; +static const uint8_t MOSI = 2; +static const uint8_t MISO = 1; +static const uint8_t SCK = 10; + +static const uint8_t A0 = 1; +static const uint8_t A1 = 2; +static const uint8_t A2 = 3; +static const uint8_t A3 = 4; +static const uint8_t A4 = 5; +static const uint8_t A5 = 6; +static const uint8_t A8 = 7; +static const uint8_t A9 = 8; +static const uint8_t A10 = 9; + +static const uint8_t D0 = 1; +static const uint8_t D1 = 2; +static const uint8_t D2 = 3; +static const uint8_t D3 = 4; +static const uint8_t D4 = 5; + +static const uint8_t A0 = 1; +static const uint8_t A1 = 2; +static const uint8_t A2 = 3; +static const uint8_t A3 = 4; +static const uint8_t A4 = 5; +static const uint8_t A5 = 6; +static const uint8_t A8 = 7; +static const uint8_t A9 = 8; +static const uint8_t A10 = 9; + +static const uint8_t D0 = 1; +static const uint8_t D1 = 2; +static const uint8_t D2 = 3; +static const uint8_t D3 = 4; +static const uint8_t D4 = 5; +static const uint8_t D5 = 6; +static const uint8_t D6 = 43; +static const uint8_t D7 = 44; +static const uint8_t D8 = 7; +static const uint8_t D9 = 8; +static const uint8_t D10 = 9; + +static const uint8_t T1 = 1; +static const uint8_t T2 = 2; +static const uint8_t T3 = 3; +static const uint8_t T4 = 4; +static const uint8_t T5 = 5; +static const uint8_t T6 = 6; +static const uint8_t T7 = 7; +static const uint8_t T8 = 8; +static const uint8_t T9 = 9; + +#endif /* Pins_Arduino_h */ diff --git a/boards/variants/XIAO_ESP32S3/tinyuf2.bin b/boards/variants/XIAO_ESP32S3/tinyuf2.bin new file mode 100644 index 0000000..86d981f Binary files /dev/null and b/boards/variants/XIAO_ESP32S3/tinyuf2.bin differ diff --git a/boards/variants/nologo_esp32c3_super_mini/pins_arduino.h b/boards/variants/nologo_esp32c3_super_mini/pins_arduino.h new file mode 100644 index 0000000..d4e98a0 --- /dev/null +++ b/boards/variants/nologo_esp32c3_super_mini/pins_arduino.h @@ -0,0 +1,28 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include + +static const uint8_t LED_BUILTIN = 8; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN + +static const uint8_t TX = 21; +static const uint8_t RX = 20; + +static const uint8_t SDA = 10; +static const uint8_t SCL = 9; + +static const uint8_t SS = 7; +static const uint8_t MOSI = 6; +static const uint8_t MISO = 5; +static const uint8_t SCK = 4; + +static const uint8_t A0 = 0; +static const uint8_t A1 = 1; +static const uint8_t A2 = 2; +static const uint8_t A3 = 3; +static const uint8_t A4 = 4; +static const uint8_t A5 = 5; + +#endif /* Pins_Arduino_h */ diff --git a/platformio.ini b/platformio.ini index 917bc53..472d055 100644 --- a/platformio.ini +++ b/platformio.ini @@ -10,24 +10,28 @@ [platformio] ;default_envs = esp32_cc1101 -default_envs = esp32c3_cdc_cc1101 -;default_envs = rp2040 +;default_envs = esp32c3_cdc_cc1101 +default_envs = esp32s3_cdc_sx1278 +;default_envs = esp32c3_cdc_sx1278 boards_dir = boards [libraries] arduinolog = https://github.com/1technophile/Arduino-Log.git#d13cd80 arduinojson = ArduinoJson + finitespace/BME280 ; ArduinoJson @ 7.0.4 ; ArduinoJson @ 6.21.5 ; ArduinoJson @ 5.13.4 ; deprecated -rtl_433_ESP = shchuko/rtl_433_ESP +;rtl_433_ESP = shchuko/rtl_433_ESP +rtl_433_ESP = https://github.com/NorthernMan54/rtl_433_ESP [env] framework = arduino monitor_filters = esp32_exception_decoder ;platform = espressif32@3.5.0 -platform = espressif32@6.1.0 +;platform = espressif32@6.1.0 +platform = espressif32 lib_ldf_mode = deep+ lib_deps = sui77/rc-switch @@ -35,8 +39,72 @@ lib_deps = ${libraries.arduinojson} ${libraries.rtl_433_ESP} +[env:esp32s3_cdc_sx1278] +board = seeed_xiao_esp32s3 +build_flags = + '-DCONFIG_ESP_CONSOLE_UART=1' ; settings for esp32c3 without uart + '-DARDUINO_USB_MODE=1' + '-DARDUINO_USB_CDC_ON_BOOT=1' + '-DLOG_LEVEL=LOG_LEVEL_TRACE' + ;'-DOOK_FIXED_THRESHOLD=0x6E' + ;'-DAUTOOOKFIX=1' + ;'-DPUBLISH_UNPARSED=true' + ;'-DSIGNAL_RSSI=true' + ;'-DDEMOD_DEBUG=true' + '-DOOK_MODULATION=true' ; False is FSK, True is OOK + '-DRF_SX1278="SX1278"' ; CC1101 Transceiver Module + '-DRF_MODULE_DIO0=8' ; CC1101 pin GDO0 + '-DRF_MODULE_DIO1=44' ; CC1101 pin GDO2 + '-DRF_MODULE_DIO2=7' + '-DRF_MODULE_RST=43' ; CC1101 pin GDO2 + ;'-DRF_MODULE_SCK=9' ; SPI Clock + ;'-DRF_MODULE_MISO=1' ; SPI Serial Output + ;'-DRF_MODULE_MOSI=2' ; SPI Serial Input + '-DRF_MODULE_CS=3' ; SPI Chip select + '-DRF_MODULE_INIT_STATUS=true' ; Display transceiver config during startup +targets = upload +monitor_port = /dev/ttyACM0 +monitor_speed = 115200 +upload_port = /dev/ttyACM0 +monitor_filters = + default ; Remove typical terminal control codes from input + time ; Add timestamp with milliseconds for each new line +; log2file ; Log data to a file “platformio-device-monitor-*.log” located in the current working directory + +[env:esp32c3_cdc_sx1278] +board = nologo_esp32c3_super_mini +build_flags = + '-DCONFIG_ESP_CONSOLE_UART=1' ; settings for esp32c3 without uart + '-DARDUINO_USB_MODE=1' + '-DARDUINO_USB_CDC_ON_BOOT=1' + '-DLOG_LEVEL=LOG_LEVEL_TRACE' + ;'-DOOK_FIXED_THRESHOLD=0x6E' + ;'-DAUTOOOKFIX=1' + ;'-DPUBLISH_UNPARSED=true' + ;'-DSIGNAL_RSSI=true' + ;'-DDEMOD_DEBUG=true' + '-DOOK_MODULATION=true' ; False is FSK, True is OOK + '-DRF_SX1278="SX1278"' ; CC1101 Transceiver Module + '-DRF_MODULE_DIO0=8' ; CC1101 pin GDO0 + '-DRF_MODULE_DIO1=44' ; CC1101 pin GDO2 + '-DRF_MODULE_DIO2=7' + '-DRF_MODULE_RST=44' ; CC1101 pin GDO2 + ;'-DRF_MODULE_SCK=9' ; SPI Clock + ;'-DRF_MODULE_MISO=1' ; SPI Serial Output + ;'-DRF_MODULE_MOSI=2' ; SPI Serial Input + '-DRF_MODULE_CS=3' ; SPI Chip select + '-DRF_MODULE_INIT_STATUS=true' ; Display transceiver config during startup +targets = upload +monitor_port = /dev/ttyACM0 +monitor_speed = 115200 +upload_port = /dev/ttyACM0 +monitor_filters = + default ; Remove typical terminal control codes from input + time ; Add timestamp with milliseconds for each new line +; log2file ; Log data to a file “platformio-device-monitor-*.log” located in the current working directory + [env:esp32c3_cdc_cc1101] -board = esp32-c3-devkitm-1 +board = nologo_esp32c3_super_mini build_flags = '-DCONFIG_ESP_CONSOLE_UART=1' ; settings for esp32c3 without uart '-DARDUINO_USB_MODE=1' @@ -76,9 +144,3 @@ build_flags = targets = upload monitor_port = /dev/ttyACM0 monitor_speed = 115200 -upload_port = /dev/ttyACM0 -monitor_filters = - default ; Remove typical terminal control codes from input - time ; Add timestamp with milliseconds for each new line -; log2file ; Log data to a file “platformio-device-monitor-*.log” located in the current working directory - diff --git a/python/mqtt-systray.py b/python/mqtt-systray.py old mode 100644 new mode 100755 index a48a6ae..a967093 --- a/python/mqtt-systray.py +++ b/python/mqtt-systray.py @@ -7,7 +7,7 @@ import signal import paho.mqtt.client as mqtt gi.require_version('Gtk', '3.0') -from gi.repository import Gtk +from gi.repository import Gtk, Gdk gi.require_version('AppIndicator3', '0.1') from gi.repository import AppIndicator3 as AppIndicator @@ -15,9 +15,6 @@ from gi.repository import AppIndicator3 as AppIndicator from PIL import Image, ImageDraw, ImageFont import yaml -mqttprefix = "devices" -mqtthost = '192.168.67.1' - oldtmpfile = '' oldalertcount = 0 menu = Gtk.Menu() @@ -26,17 +23,18 @@ fp = open("mqtt-systray.yaml") conf = yaml.safe_load(fp) fp.close() +mqttprefix = conf['config']['mqtt_prefix'] +mqtthost = conf['config']['mqtt_host'] + appindicator = AppIndicator.Indicator.new( 'mqttalert', '', AppIndicator.IndicatorCategory.APPLICATION_STATUS) -def create_image(width, height, count): +def create_image(width, height, count, color_circle): # Generate an image and draw a pattern image = Image.new('RGB', (width, height), 'white') dc = ImageDraw.Draw(image) - color_circle = 'green' - if count > 0: color_circle = 'red' #dc.circle((width // 2, height // 2), height // 2, fill=color2) radius = height // 2 ellipse_xy = (0, 0, radius*2, radius*2) @@ -54,42 +52,76 @@ def quit(): Gtk.main_quit() def signal_handler(sig, frame): - print('You pressed Ctrl+C!') + #print('You pressed Ctrl+C!') quit() def gtkquit(source): - quit() + quit() -def get_alert(devicea): - for sensor in conf[devicea]['values'].keys(): - if 'value' in conf[devicea]['values'][sensor] and (conf[devicea]['values'][sensor]['value'] > conf[devicea]['values'][sensor]['max'] or conf[devicea]['values'][sensor]['value'] < conf[devicea]['values'][sensor]['min']): - return True - return False +def gtkdashboard(source): + Gtk.show_uri_on_window(None, conf['config']['dashboard_url'],int(Gdk.CURRENT_TIME)) -def get_alert2(devicea, sensora): +def get_alert_text_warning(devicea, sensora): alert_text = '' - if 'value' in conf[devicea]['values'][sensora]: - if conf[devicea]['values'][sensora]['value'] > conf[devicea]['values'][sensora]['max']: - alert_text = '!!HIGH!! ' - elif conf[devicea]['values'][sensora]['value'] < conf[devicea]['values'][sensora]['min']: - alert_text = '!!LOW!! ' + if 'value' in conf['devices'][devicea]['values'][sensora]: + if 'max_warning' in conf['devices'][devicea]['values'][sensora] and conf['devices'][devicea]['values'][sensora]['value'] > conf['devices'][devicea]['values'][sensora]['max_warning']: + alert_text = '!!high!! ' + elif 'min_warning' in conf['devices'][devicea]['values'][sensora] and conf['devices'][devicea]['values'][sensora]['value'] < conf['devices'][devicea]['values'][sensora]['min_warning']: + alert_text = '!!low!! ' return alert_text +def get_alert_text_critical(devicea, sensora): + alert_text = '' + if 'value' in conf['devices'][devicea]['values'][sensora]: + if 'max_critical' in conf['devices'][devicea]['values'][sensora] and conf['devices'][devicea]['values'][sensora]['value'] > conf['devices'][devicea]['values'][sensora]['max_critical']: + alert_text = '!!HIGH!! ' + elif 'min_critical' in conf['devices'][devicea]['values'][sensora] and conf['devices'][devicea]['values'][sensora]['value'] < conf['devices'][devicea]['values'][sensora]['min_critical']: + alert_text = '!!LOW!! ' + return alert_text + +def get_alert_text(devicea, sensora): + alert_text = '' + if get_alert_text_critical(devicea, sensora) != '': + alert_text = get_alert_text_critical(devicea, sensora) + elif get_alert_text_warning(devicea, sensora) != '': + alert_text = get_alert_text_warning(devicea, sensora) + return alert_text + +def alert_count_critical(): + count = 0 + for device in conf['devices']: + for sensor in conf['devices'][device]['values'].keys(): + if get_alert_text_critical(device, sensor) != '': count = count + 1 + return count + +def alert_count_warning(): + count = 0 + for device in conf['devices']: + for sensor in conf['devices'][device]['values'].keys(): + if get_alert_text_warning(device, sensor) != '': count = count + 1 + return count + def alert_count(): - count = 0 - for device in conf: - for sensor in conf[device]['values'].keys(): - if get_alert2(device, sensor) != '': count = count + 1 - return count + count = 0 + for device in conf['devices']: + for sensor in conf['devices'][device]['values'].keys(): + if get_alert_text_critical(device, sensor) != '' or get_alert_text_warning(device, sensor) != '': count = count + 1 + return count def get_icon(): - global oldtmpfile - if oldtmpfile!='': os.remove(oldtmpfile) - icon_path = tempfile.mktemp() - oldtmpfile=icon_path - with open(icon_path, 'wb') as f: - create_image(64, 64, alert_count()).save(f, 'PNG') - return icon_path + global oldtmpfile + if oldtmpfile!='': os.remove(oldtmpfile) + icon_path = tempfile.mktemp() + oldtmpfile=icon_path + color = 'green' + if alert_count_critical() > 0: + color = 'red' + elif alert_count_warning() > 0: + color = 'yellow' + + with open(icon_path, 'wb') as f: + create_image(64, 64, alert_count(), color).save(f, 'PNG') + return icon_path #def menun2(deviceg): # itemn2 = [] @@ -107,17 +139,20 @@ def get_icon(): def build_menu(): menu = Gtk.Menu() - for device1 in conf: - devname = conf[device1]['name'] - conf[device1]['menuitem'] = Gtk.MenuItem(devname) + for device1 in conf['devices']: + devname = conf['devices'][device1]['name'] + conf['devices'][device1]['menuitem'] = Gtk.MenuItem(label=devname) submenu = Gtk.Menu() - for sensor in conf[device1]['values'].keys(): - conf[device1]['values'][sensor]['menuitem'] = Gtk.MenuItem(sensor) - submenu.append(conf[device1]['values'][sensor]['menuitem']) - conf[device1]['menuitem'].set_submenu(submenu) - menu.append(conf[device1]['menuitem']) + for sensor in conf['devices'][device1]['values'].keys(): + conf['devices'][device1]['values'][sensor]['menuitem'] = Gtk.MenuItem(label=conf['devices'][device1]['values'][sensor]['name']) + submenu.append(conf['devices'][device1]['values'][sensor]['menuitem']) + conf['devices'][device1]['menuitem'].set_submenu(submenu) + menu.append(conf['devices'][device1]['menuitem']) - item_quit = Gtk.MenuItem('Quit') + item_dashboard = Gtk.MenuItem(label='Open Dashboard') + item_dashboard.connect('activate', gtkdashboard) + menu.append(item_dashboard) + item_quit = Gtk.MenuItem(label='Quit') item_quit.connect('activate', gtkquit) menu.append(item_quit) @@ -160,59 +195,59 @@ def build_menu(): # return menu def update_menu(): - appindicator.set_menu(build_menu()) + appindicator.set_menu(build_menu()) def set_value(topic, value): - global oldalertcount - topicds = topic.rsplit("/", 1) - devices=topicds[0] - devices = devices.removeprefix(mqttprefix+'/') - conf[devices]['values'][topicds[1]]['value'] = float(value) - print("menu") - #update_menu() + global oldalertcount + topicds = topic.rsplit("/", 1) + devices=topicds[0] + devices = devices.removeprefix(mqttprefix+'/') + conf['devices'][devices]['values'][topicds[1]]['value'] = float(value) + #update_menu() - alert_text = get_alert2(devices, topicds[1]) - text='{}{}: {}'.format(alert_text, topicds[1], conf[devices]['values'][topicds[1]]['value']) - conf[devices]['values'][topicds[1]]['menuitem'].set_label(text) - #conf[devices]['values'][topicds[1]]['menuitem'].set_label("{}".format(conf[devices]['values'][topicds[1]]['value'])) - devname = conf[devices]['name'] - #print(get_alert(devices)) - #if get_alert(devices): devname = "! " + devname - #if get_alert(devices): devname = "!!ALERT!! " + devname - #conf[devices]['menuitem'].set_label(devname) + alert_text = get_alert_text(devices, topicds[1]) + text='{}{}: {} {}'.format(alert_text, conf['devices'][devices]['values'][topicds[1]]['name'], conf['devices'][devices]['values'][topicds[1]]['value'], conf['devices'][devices]['values'][topicds[1]]['unit']) + conf['devices'][devices]['values'][topicds[1]]['menuitem'].set_label(text) + #conf['devices'][devices]['values'][topicds[1]]['menuitem'].set_label("{}".format(conf['devices'][devices]['values'][topicds[1]]['value'])) + devname = conf['devices'][devices]['name'] + #print(get_alert(devices)) + #if get_alert(devices): devname = "! " + devname + #if get_alert(devices): devname = "!!ALERT!! " + devname + #conf['devices'][devices]['menuitem'].set_label(devname) - if oldalertcount != alert_count(): - oldalertcount = alert_count() - appindicator.set_icon(get_icon()) - os.system("notify-send '{}' '{}'".format(conf[devices]['name'], text)) + if oldalertcount != (alert_count()): + oldalertcount = alert_count() + appindicator.set_icon_full(get_icon(), "mqtt-icon") + os.system("notify-send '{}' '{}'".format(conf['devices'][devices]['name'], text)) def subscribe_mqtt(mqttc): - for device in conf: - #print(conf[device]) - for sensor in conf[device]['values'].keys(): - #print(sensor) - tosubscribe = "{}/{}/{}".format(mqttprefix, device, sensor) - mqttc.subscribe(tosubscribe, 0) - print(tosubscribe) + for device in conf['devices']: + #print(conf[device]) + for sensor in conf['devices'][device]['values'].keys(): + #print(sensor) + tosubscribe = "{}/{}/{}".format(mqttprefix, device, sensor) + mqttc.subscribe(tosubscribe, 0) + print(tosubscribe) def on_connect(mqttc, obj, flags, rc): print("rc: " + str(rc)) #def on_subscribe(mqttc, obj, mid, granted_qos): # print("Subscribed: " + str(mid) + " " + str(granted_qos)) def on_message(client, userdata, msg): - print(msg.topic+" "+str(msg.payload)) - set_value(msg.topic, msg.payload) + print(msg.topic+" "+str(msg.payload)) + set_value(msg.topic, msg.payload) signal.signal(signal.SIGINT, signal_handler) -appindicator.set_icon(get_icon()) +appindicator.set_icon_full(get_icon(), "mqtt-icon") appindicator.set_title('MQTT Alert') appindicator.set_status(AppIndicator.IndicatorStatus.ACTIVE) appindicator.set_menu(build_menu()) -set_value("model1/1/1/temperature1C", "10.1") -print(conf) +set_value("model1/1/1/temperature1C", "25.1") +#set_value("model2/2/2/humidity2", "75") +#print(conf) mqttc = mqtt.Client() mqttc.on_message = on_message @@ -220,13 +255,13 @@ mqttc.on_connect = on_connect #mqttc.on_subscribe = on_subscribe connected = False while connected == False: - try: - mqttc.connect(mqtthost, 1883, 60) - except: - time.sleep(5) - connected=False - else: - connected=True + try: + mqttc.connect(mqtthost, 1883, 60) + except: + time.sleep(5) + connected=False + else: + connected=True mqttc.loop_start() subscribe_mqtt(mqttc) diff --git a/python/mqtt-systray.yaml b/python/mqtt-systray.yaml index 7850279..0f098c3 100644 --- a/python/mqtt-systray.yaml +++ b/python/mqtt-systray.yaml @@ -1,22 +1,35 @@ -model1/1/1: - name: "toto" - values: - temperature1C: - name: "Temperature" - min: 5 - max: 20 - humidity1: - name: "Humidity" - min: 30 - max: 70 -model2/2/2: - name: "tata" - values: - temperature2C: - name: "Temperature" - min: 5 - max: 20 - humidity2: - name: "Humidity" - min: 30 - max: 70 +config: + mqtt_host: "192.168.67.1" + mqtt_prefix: "devices" + dashboard_url: "https://github.com/NorthernMan54/rtl_433_ESP" +devices: + model1/1/1: + name: "toto" + values: + temperature1C: + name: "Temperature" + unit: "°C" + humidity1: + name: "Humidity" + unit: "%" + min_critical: 20 + min_warning: 30 + max_warning: 70 + max_critical: 80 + model2/2/2: + name: "tata" + values: + temperature2C: + name: "Temperature" + unit: "°C" + min_critical: 0 + min_warning: 5 + max_warning: 20 + max_critical: 30 + humidity2: + name: "Humidity" + unit: "%" + min_critical: 20 + min_warning: 30 + max_warning: 70 + max_critical: 80 diff --git a/python/rtl_433_json2mqtt.py b/python/rtl_433_json2mqtt.py index 9c64bf1..0c3c58d 100755 --- a/python/rtl_433_json2mqtt.py +++ b/python/rtl_433_json2mqtt.py @@ -11,12 +11,19 @@ TOPIC_KEYS = [ "type", "model", "subtype", "channel", "id" ] #default="devices[/type][/model][/subtype][/channel][/id]" prefix = "home/rtl_433" -broker = "localhost" +broker = "192.168.67.1" port = 1883 +altitude = 150 + serialdev = "/dev/ttyACM0" serialspeed = 115200 +def get_humidex(tdb, rh): + return round(tdb + 5 / 9 * ((6.112 * 10 ** (7.5 * tdb / (237.7 + tdb)) * rh / 100) - 10)) + +def get_sealevel_pressure(pres, altitude, temp): + return (pres / pow(1 - ((0.0065 *altitude) / (temp + (0.0065 *altitude) + 273.15)), 5.257)); def generate_topic(jsonin): topic = prefix+'/devices' @@ -59,7 +66,15 @@ while True: print("Error JSON, received {}".format(input)) continue #print(data) - mqtt.publish(prefix+'/events', input.rstrip()) + #mqtt.publish(prefix+'/events', input.rstrip()) + + if 'humidity' in data and 'temperature_C' in data: + data['humidex'] = get_humidex(data['temperature_C'], data['humidity']) + if 'pressure' in data: + data['pressure'] = get_sealevel_pressure(data['pressure'], altitude, data['temperature_C']) + data['pressure'] = round(data['pressure'], 1) + + mqtt.publish(prefix+'/events', json.dumps(data)) prefix_device = generate_topic(data) publish(data, prefix_device) diff --git a/python/rtl_433_mqtt2vm.py b/python/rtl_433_mqtt2vm.py index d584d35..f0d31e9 100755 --- a/python/rtl_433_mqtt2vm.py +++ b/python/rtl_433_mqtt2vm.py @@ -82,6 +82,16 @@ def on_message(mqttc, obj, msg): if 'humidity' not in recorddb[ref] or recorddb[ref]['humidity'] != value: measures += create_measure("humidity", currentts, tags, value) recorddb[ref]['humidity'] = value + if 'moisture' in jsonin: + value = int(jsonin['moisture']) + if 'moisture' not in recorddb[ref] or recorddb[ref]['moisture'] != value: + measures += create_measure("moisture", currentts, tags, value) + recorddb[ref]['moisture'] = value + if 'pressure' in jsonin: + value = float(jsonin['pressure']) + if 'pressure' not in recorddb[ref] or recorddb[ref]['pressure'] != value: + measures += create_measure("pressure", currentts, tags, value) + recorddb[ref]['humidity'] = value if 'battery_ok' in jsonin: value = int(jsonin['battery_ok']) if 'battery_ok' not in recorddb[ref] or recorddb[ref]['battery_ok'] != value: diff --git a/src/main.ino b/src/main.ino index bf1031c..7240c3b 100644 --- a/src/main.ino +++ b/src/main.ino @@ -1,14 +1,22 @@ #include #include #include +#include +#include +#include #include "Dictionary.h" +#ifndef RF_MODULE_FREQUENCY +# define RF_MODULE_FREQUENCY 433.92 +#endif #define _DICT_PACK_STRUCTURES -#define LED 8 +#define LED LED_BUILTIN -CC1101 radiotx = RADIO_LIB_MODULE; +#define RCSWITCH_PIN 43 + +//CC1101 radiotx = RADIO_LIB_MODULE; RCSwitch mySwitch = RCSwitch(); #ifndef RF_MODULE_FREQUENCY @@ -17,6 +25,8 @@ RCSwitch mySwitch = RCSwitch(); #define JSON_MSG_BUFFER 512 #define MINDELAY 10L +#define ALTITUDE 150.0 +#define DELAYLOCALSENSORS 30L Dictionary d1; @@ -24,15 +34,35 @@ char messageBuffer[JSON_MSG_BUFFER]; rtl_433_ESP rf; // use -1 to disable transmitter +BME280I2C bme; +bool sensors_enabled; + int count = 0; +uint32_t lastts_localsensors = 0; + void rtl_433_Callback(char* message) { + //Serial.println("RTL433 Callback"); JsonDocument jsonDocument; deserializeJson(jsonDocument,message); logJson(jsonDocument); count++; } +JsonDocument analyzeTH_type1(JsonArray payloadrh) { + JsonDocument doc; + doc["moisture"] = payloadrh[5].as(); + int16_t payloadtemp = payloadrh[4].as(); + payloadtemp += payloadrh[3].as() << 8; + doc["temperature_C"] = payloadtemp / 100.0; + doc["battery_ok"] = payloadrh[0].as() & 0b00000001; + uint16_t payloadbat = payloadrh[2].as(); + payloadbat += payloadrh[1].as() << 8; + doc["battery_V"] = ((payloadbat & 0b1111111110000000) >> 7) / 100.0; + doc["battery_percent"] = (payloadbat & 0b0000000001111111); + return doc; +} + void logJson(JsonDocument jsondata) { if(jsondata["model"].is()) { @@ -42,6 +72,31 @@ void logJson(JsonDocument jsondata) { if(!d1.contains(ref.c_str()) || millis() > (d1.get(ref.c_str()) + (MINDELAY * 1000)) || d1.get(ref.c_str()) > millis()) { //Serial.println(ref.c_str()); //if(d1.contains(ref.c_str())) { Serial.println(d1.get(ref.c_str())); } + + if(jsondata["model"].is() && jsondata["model"] == "RadioHead-ASK") { + JsonArray payloadrh = jsondata["payload"]; + bool treatrh = 0; + if(jsondata["len"].as() == 6 && ((payloadrh[0].as() & 0b00010000 ) > 0)) { + treatrh = true; + //Serial.println("Analyse RH Payload"); + JsonDocument adoc = analyzeTH_type1(payloadrh); + jsondata["moisture"] = adoc["moisture"]; + jsondata["battery_ok"] = adoc["battery_ok"]; + jsondata["battery_V"] = adoc["battery_V"]; + jsondata["battery_percent"] = adoc["battery_percent"]; + jsondata["temperature_C"] = adoc["temperature_C"]; + } + if(treatrh == true) { + jsondata.remove("from"); + jsondata.remove("to"); + jsondata.remove("payload"); + jsondata.remove("len"); + jsondata.remove("flags"); + jsondata.remove("mic"); + jsondata.remove("duration"); + } + } + #if defined(ESP8266) || defined(ESP32) || defined(__AVR_ATmega2560__) || defined(__AVR_ATmega1280__) char JSONmessageBuffer[measureJson(jsondata) + 1]; serializeJson(jsondata, JSONmessageBuffer, measureJson(jsondata) + 1); @@ -50,12 +105,97 @@ void logJson(JsonDocument jsondata) { serializeJson(jsondata, JSONmessageBuffer, JSON_MSG_BUFFER); #endif d1.set(ref.c_str(), millis()); - ledblink(); Serial.println(JSONmessageBuffer); + ledblink(); } } } + +/* +void logJsonRH(JsonDocument jsondata) { + if(jsondata["model"].is() && jsondata["model"] == "RadioHead-ASK") { + JsonArray payloadrh = jsondata["payload"]; + bool treatrh = 0; + if(jsondata["len"].as() == 6 && ((payloadrh[0].as() & 0b00010000 ) > 0)) { + treatrh = true; + //Serial.println("Analyse RH Payload"); + JsonDocument adoc = analyzeTH_type1(payloadrh); + jsondata["moisture"] = adoc["moisture"]; + jsondata["battery_ok"] = adoc["battery_ok"]; + jsondata["battery_V"] = adoc["battery_V"]; + jsondata["battery_percent"] = adoc["battery_percent"]; + jsondata["temperature_C"] = adoc["temperature_C"]; + } + if(treatrh == true) { + jsondata.remove("from"); + jsondata.remove("to"); + jsondata.remove("payload"); + jsondata.remove("len"); + jsondata.remove("flags"); + jsondata.remove("mic"); + jsondata.remove("duration"); + } + } + #if defined(ESP8266) || defined(ESP32) || defined(__AVR_ATmega2560__) || defined(__AVR_ATmega1280__) + char JSONmessageBuffer[measureJson(jsondata) + 1]; + serializeJson(jsondata, JSONmessageBuffer, measureJson(jsondata) + 1); + #else + char JSONmessageBuffer[JSON_MSG_BUFFER]; + serializeJson(jsondata, JSONmessageBuffer, JSON_MSG_BUFFER); + #endif + Serial.println(JSONmessageBuffer); + //Serial.print(JSONmessageBuffer); + //Serial.println(); + //ledblink(); +} + +void logJsonAll(JsonDocument jsondata) { + // Serial.println(count); + #if defined(ESP8266) || defined(ESP32) || defined(__AVR_ATmega2560__) || defined(__AVR_ATmega1280__) + char JSONmessageBuffer[measureJson(jsondata) + 1]; + serializeJson(jsondata, JSONmessageBuffer, measureJson(jsondata) + 1); + #else + char JSONmessageBuffer[JSON_MSG_BUFFER]; + serializeJson(jsondata, JSONmessageBuffer, JSON_MSG_BUFFER); + #endif + Serial.println(JSONmessageBuffer); + //ledblink(); +} +*/ + +void sendBME280() { + JsonDocument jsonDocument; + jsonDocument["model"]="LocalSensors"; + jsonDocument["protocol"]="Local Sensors"; + //jsonDocument["channel"]=1; + //jsonDocument["id"]=1; + float temp(NAN), hum(NAN), pres(NAN); + + BME280::TempUnit tempUnit(BME280::TempUnit_Celsius); + BME280::PresUnit presUnit(BME280::PresUnit_hPa); + EnvironmentCalculations::AltitudeUnit envAltUnit = EnvironmentCalculations::AltitudeUnit_Meters; + EnvironmentCalculations::TempUnit envTempUnit = EnvironmentCalculations::TempUnit_Celsius; + + bme.read(pres, temp, hum, tempUnit, presUnit); + + if(!isnan(pres) && !isnan(temp) && !isnan(hum)) { + + //float temp2 = (int)(temp * 100 + .5); + //jsonDocument["temperature_C"]=(float)temp2 / 100; + jsonDocument["temperature_C"]=temp; + jsonDocument["humidity"]=(int8_t) roundf(hum); + //jsonDocument["pressure"]=EnvironmentCalculations::EquivalentSeaLevelPressure(ALTITUDE, temp, pres, envAltUnit, envTempUnit); + //float pres2 = (int)(pres * 10 + .5); + //jsonDocument["pressure"]=(float)pres2 / 10; + jsonDocument["pressure"]=pres; + + logJson(jsonDocument); + + } + +} + void ledblink() { digitalWrite(LED, LOW); delay(150); @@ -63,19 +203,21 @@ void ledblink() { } void setup() { + Serial.setTxBufferSize(512); + //Serial.setTxTimeoutMs(100); Serial.begin(115200); delay(1000); - for (int i=0 ; i<10; i++) { - Serial.print("Hello"); - delay(1000); - } - if(SS != 21 || MOSI != 20 || MISO != 10 || SCK != 7) + //for (int i=0 ; i<10; i++) { + // Serial.print("Hello"); + // delay(1000); + //} + /* if(SS != 21 || MOSI != 20 || MISO != 10 || SCK != 7) { for ( ; ; ) { Serial.println("Please define pin assignment for SPI in pins_arduino.h (~/.platformio/packages/framework-arduinoespressif32/variants/esp32c3) / SS: 21 MOSI : 20 MISO : 10 SCK : 7"); delay(1000); } - } + } */ pinMode(LED, OUTPUT); digitalWrite(LED, HIGH); rf.initReceiver(RF_MODULE_RECEIVER_GPIO, RF_MODULE_FREQUENCY); @@ -86,11 +228,17 @@ void setup() { //d = new Dictionary(25); //d("aa", "bb"); //if(d1("aa")) { Serial.println("toto"); } + Wire.begin(); + sensors_enabled = bme.begin(); } void loop() { rf.loop(); + if(sensors_enabled == true && (millis() > (lastts_localsensors + (DELAYLOCALSENSORS*1000)) || lastts_localsensors > millis())) { + lastts_localsensors = millis(); + sendBME280(); + } if (Serial.available()) { // Read the JSON document from the "link" serial port @@ -106,15 +254,16 @@ void loop() { if(doc["cmd"].as().compareTo(String("rcsend"))==0) { Serial.println("Transmit RCSwitch"); ledblink(); - //rf.getModuleStatus(); - rf.disableReceiver(); - radiotx.SPIsendCommand(RADIOLIB_CC1101_CMD_TX); - mySwitch.enableTransmit(RF_MODULE_GDO0); - mySwitch.setRepeatTransmit(8); + //rf.disableReceiver(); + //radiotx.SPIsendCommand(RADIOLIB_CC1101_CMD_TX); + //mySwitch.enableTransmit(RF_MODULE_GDO0); + mySwitch.enableTransmit(RCSWITCH_PIN); + //mySwitch.setRepeatTransmit(8); + mySwitch.setRepeatTransmit(5); mySwitch.send(doc["value"].as(), 24); - mySwitch.disableTransmit(); - radiotx.SPIsendCommand(RADIOLIB_CC1101_CMD_RX); - rf.enableReceiver(); + //mySwitch.disableTransmit(); + //radiotx.SPIsendCommand(RADIOLIB_CC1101_CMD_RX); + //rf.enableReceiver(); //rf.getModuleStatus(); } }