commit d377fa89a4a4ba41d8f23539a4fc34c0cc75f3da Author: Nigreon Date: Sat Feb 22 23:27:11 2025 +0100 Initial commit diff --git a/esp32-c3.jpg b/esp32-c3.jpg new file mode 100644 index 0000000..42c2e58 Binary files /dev/null and b/esp32-c3.jpg differ diff --git a/esp32-c3.md b/esp32-c3.md new file mode 100644 index 0000000..3d0502a --- /dev/null +++ b/esp32-c3.md @@ -0,0 +1 @@ +https://docs.espressif.com/projects/arduino-esp32/en/latest/index.html diff --git a/platformio.ini b/platformio.ini new file mode 100644 index 0000000..917bc53 --- /dev/null +++ b/platformio.ini @@ -0,0 +1,84 @@ +; PlatformIO Project Configuration File +; +; Build options: build flags, source filter +; Upload options: custom upload port, speed and extra flags +; Library options: dependencies, extra library storages +; Advanced options: extra scripting +; +; Please visit documentation for the other options and examples +; https://docs.platformio.org/page/projectconf.html + +[platformio] +;default_envs = esp32_cc1101 +default_envs = esp32c3_cdc_cc1101 +;default_envs = rp2040 +boards_dir = boards + +[libraries] +arduinolog = https://github.com/1technophile/Arduino-Log.git#d13cd80 +arduinojson = + ArduinoJson + ; ArduinoJson @ 7.0.4 + ; ArduinoJson @ 6.21.5 + ; ArduinoJson @ 5.13.4 ; deprecated +rtl_433_ESP = shchuko/rtl_433_ESP + +[env] +framework = arduino +monitor_filters = esp32_exception_decoder +;platform = espressif32@3.5.0 +platform = espressif32@6.1.0 +lib_ldf_mode = deep+ +lib_deps = + sui77/rc-switch + ${libraries.arduinolog} + ${libraries.arduinojson} + ${libraries.rtl_433_ESP} + +[env:esp32c3_cdc_cc1101] +board = esp32-c3-devkitm-1 +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' +; '-DONBOARD_LED=13' ; LED_D4 +; *** rtl_433_ESP Options *** +; '-DRF_MODULE_FREQUENCY=915.00' + '-DOOK_MODULATION=true' ; False is FSK, True is OOK +; '-DRTL_DEBUG=4' ; rtl_433 verbose mode +; '-DRTL_VERBOSE=74' ; LaCrosse TX141-Bv2, TX141TH-Bv2, TX141-Bv3, TX141W, TX145wsdth sensor +; '-DRAW_SIGNAL_DEBUG=true' ; display raw received messages +; '-DMEMORY_DEBUG=true' ; display memory usage information +; '-DDEMOD_DEBUG=true' ; display signal debug info +; '-DMY_DEVICES=true' ; subset of devices +; '-DPUBLISH_UNPARSED=true' ; publish unparsed signal details + '-DDISABLERSSITHRESHOLD=true' + '-DMINRSSI=-82' +; '-DMINRSSI=-60' +; '-DRSSI_THRESHOLD=12' ; Apply a delta of 12 to average RSSI level +; '-DAVERAGE_RSSI=5000' ; Display RSSI floor ( Average of 5000 samples ) +; '-DSIGNAL_RSSI=true' ; Display during signal receive +; '-DOOK_MODULATION=false' ; False is FSK, True is OOK +; *** RF Module Options *** + '-DRF_CC1101="CC1101"' ; CC1101 Transceiver Module + '-DRF_MODULE_CS=21' ; pin to be used as chip select + '-DRF_MODULE_GDO0=6' ; CC1101 pin GDO0 + '-DRF_MODULE_GDO2=5' ; CC1101 pin GDO2 + '-DRF_MODULE_INIT_STATUS=true' ; Display transceiver config during startup +; *** RadioLib Options *** +; '-DRADIOLIB_DEBUG=true' +; '-DRADIOLIB_VERBOSE=true' +; *** FSK Setting Testing *** +; '-DsetBitrate' +; '-DsetFreqDev' +; '-DsetRxBW' +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 new file mode 100644 index 0000000..a48a6ae --- /dev/null +++ b/python/mqtt-systray.py @@ -0,0 +1,234 @@ +import gi +import tempfile +import time +import os +import signal + +import paho.mqtt.client as mqtt + +gi.require_version('Gtk', '3.0') +from gi.repository import Gtk + +gi.require_version('AppIndicator3', '0.1') +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() + +fp = open("mqtt-systray.yaml") +conf = yaml.safe_load(fp) +fp.close() + +appindicator = AppIndicator.Indicator.new( + 'mqttalert', + '', + AppIndicator.IndicatorCategory.APPLICATION_STATUS) + +def create_image(width, height, count): + # 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) + dc.ellipse(ellipse_xy, color_circle) + + if count > 0: + #font = ImageFont.truetype("OpenSans-Regular.ttf", int(width)) + font = ImageFont.load_default(float(width)) + dc.text((12, -12),str(count),(0,0,0),font=font) + + return image + +def quit(): + os.remove(oldtmpfile) + Gtk.main_quit() + +def signal_handler(sig, frame): + print('You pressed Ctrl+C!') + quit() + +def gtkquit(source): + 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 get_alert2(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!! ' + return alert_text + +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 + +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 + +#def menun2(deviceg): +# itemn2 = [] +# submenu = Gtk.Menu() +# for sensor in conf[deviceg]['values'].keys(): +# text='' +# if 'value' in conf[deviceg]['values'][sensor]: +# alert_text = get_alert2(deviceg, sensor) +# text='{}{}: {}'.format(alert_text, sensor, conf[deviceg]['values'][sensor]['value']) +# else: +# text=sensor +# submenuitem = Gtk.MenuItem(text) +# submenu.append(submenuitem) +# return submenu + +def build_menu(): + menu = Gtk.Menu() + for device1 in conf: + devname = conf[device1]['name'] + conf[device1]['menuitem'] = Gtk.MenuItem(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']) + + item_quit = Gtk.MenuItem('Quit') + item_quit.connect('activate', gtkquit) + menu.append(item_quit) + + menu.show_all() + return menu + + +#def build_menu_old(): +# global menu +# global quitcount +# global item_quit +# #menu = Gtk.Menu() +# menu.connect('popup-menu', menu_show) +# #menu.connect('destroy', menu_hide) +# for device1 in conf: +# devname = conf[device1]['name'] +# if get_alert(device1): devname = "!!ALERT!! " + devname +# menuitem = Gtk.MenuItem(devname) +# menuitem.set_submenu(menun2(device1)) +# menu.append(menuitem) +# +# if quitcount == 0: +# item_quit = Gtk.MenuItem('Quit') +# item_quit.connect('select', menu_show) +# print('Quit0') +# menu.append(item_quit) +# quitcount = quitcount + 1 +# else: +# #item_quit = Gtk.MenuItem('Quit') +# item_quit.set_label(str(quitcount)) +# print('QuitX') +# #menu.append(item_quit) +# #item_quit.show() +# quitcount = quitcount + 1 +# +# print("Alert Count: {}".format(alert_count())) +# +# menu.show_all() +# #menu.show_now() +# return menu + +def update_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() + + 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) + + if oldalertcount != alert_count(): + oldalertcount = alert_count() + appindicator.set_icon(get_icon()) + os.system("notify-send '{}' '{}'".format(conf[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) + +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) + + +signal.signal(signal.SIGINT, signal_handler) + +appindicator.set_icon(get_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) + +mqttc = mqtt.Client() +mqttc.on_message = on_message +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 +mqttc.loop_start() + +subscribe_mqtt(mqttc) + +Gtk.main() diff --git a/python/mqtt-systray.yaml b/python/mqtt-systray.yaml new file mode 100644 index 0000000..7850279 --- /dev/null +++ b/python/mqtt-systray.yaml @@ -0,0 +1,22 @@ +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 diff --git a/python/rtl_433_json2mqtt.py b/python/rtl_433_json2mqtt.py new file mode 100755 index 0000000..9c64bf1 --- /dev/null +++ b/python/rtl_433_json2mqtt.py @@ -0,0 +1,65 @@ +import json +#import fileinput +import serial + +import paho.mqtt.client as paho + +#input = '{"model":"Oregon-THGR810","id":226,"channel":10,"battery_ok":0,"temperature_C":19.7,"humidity":24,"protocol":"Oregon Scientific Weather Sensor"}' +SKIP_KEYS = [ "type", "model", "subtype", "channel", "id", "mic", "mod", "freq", "sequence_num", "message_type", "exception", "raw_msg", "protocol", "duration", "sync", "flags", "status" ] +TOPIC_KEYS = [ "type", "model", "subtype", "channel", "id" ] + +#default="devices[/type][/model][/subtype][/channel][/id]" +prefix = "home/rtl_433" + +broker = "localhost" +port = 1883 + +serialdev = "/dev/ttyACM0" +serialspeed = 115200 + + +def generate_topic(jsonin): + topic = prefix+'/devices' + for t in TOPIC_KEYS: + if t in jsonin: + topic += '/' + str(jsonin[t]) + return topic + +def publish(jsonin, prefix_device): + for t in jsonin: + if t not in SKIP_KEYS: + topic = prefix_device + '/' + t + value = jsonin[t] + #print("{} {}".format(topic, value)) + mqtt.publish(topic, value) + +def on_connect(client, userdata, flags, reason_code, properties): + print(f"Connected to MQTT") + client.subscribe(prefix+"/cmd") + +def on_message(client, userdata, msg): + print(msg.topic+" "+str(msg.payload)) + if(str(msg.topic) == prefix+"/cmd"): + ser.write(msg.payload) + +mqtt=paho.Client(paho.CallbackAPIVersion.VERSION2) +mqtt.on_connect = on_connect +mqtt.on_message = on_message +mqtt.connect(broker,port) +mqtt.loop_start() + +ser = serial.Serial(serialdev, serialspeed) + +#for input in fileinput.input(): +while True: + input = ser.readline() + try: + data = json.loads(input) + except json.decoder.JSONDecodeError: + print("Error JSON, received {}".format(input)) + continue + #print(data) + mqtt.publish(prefix+'/events', input.rstrip()) + prefix_device = generate_topic(data) + publish(data, prefix_device) + diff --git a/python/rtl_433_mqtt2influxdb.py b/python/rtl_433_mqtt2influxdb.py new file mode 100755 index 0000000..71f0696 --- /dev/null +++ b/python/rtl_433_mqtt2influxdb.py @@ -0,0 +1,133 @@ +#!/usr/bin/python3 +# -*- coding: utf-8 -*- + +# Copyright (c) 2010-2013 Roger Light +# +# All rights reserved. This program and the accompanying materials +# are made available under the terms of the Eclipse Distribution License v1.0 +# which accompanies this distribution. +# +# The Eclipse Distribution License is available at +# http://www.eclipse.org/org/documents/edl-v10.php. +# +# Contributors: +# Roger Light - initial implementation +# Copyright (c) 2010,2011 Roger Light +# All rights reserved. + +# This shows a simple example of an MQTT subscriber. + +import signal +import sys +import json +import datetime +import paho.mqtt.client as mqtt +from influxdb import InfluxDBClient + +def signal_handler(sig, frame): + influxc.close() + mqttc.disconnect() + sys.exit(0) + +prefix = 'home/rtl_433' +#prefix = 'rtl_433/fedora' +TAGS_KEYS = ['type','model','subtype','channel','id'] + +DBNAME = "rtl433" +DELAYRECORD=300 +recorddb = {} + +def on_connect(mqttc, obj, flags, rc): + print("rc: " + str(rc)) + + +def on_message(mqttc, obj, msg): + #print(msg.topic + " " + str(msg.qos) + " " + str(msg.payload)) + currentdt=datetime.datetime.now(datetime.UTC) + dateinflux=currentdt.strftime("%Y-%m-%dT%H:%M:%SZ") + jsonin=json.loads(msg.payload) + ref="" + if "model" in jsonin: + ref+=str(jsonin["model"]) + if "channel" in jsonin: + ref+=str(jsonin["channel"]) + if "id" in jsonin: + ref+=str(jsonin["id"]) + if ref not in recorddb: + recorddb[ref] = {} + recorddb[ref]["lastsend"]=0 + #print(jsonin) + tags = {} + for t in TAGS_KEYS: + if t in jsonin: + tags[t] = jsonin[t] + points = [] + fields = {} + + changed = False + if 'temperature_C' in jsonin: + fields = {'value': float(jsonin['temperature_C'])} + measurement = 'temperature_C' + points.append({'measurement': measurement, 'tags': tags, 'time': dateinflux, 'fields': fields}) + if 'temperature_C' not in recorddb[ref] or recorddb[ref]['temperature_C'] != fields["value"]: + changed = True + recorddb[ref]['temperature_C'] = fields["value"] + if 'humidity' in jsonin: + fields = {'value': int(jsonin['humidity'])} + measurement = 'humidity' + points.append({'measurement': measurement, 'tags': tags, 'time': dateinflux, 'fields': fields}) + if 'humidity' not in recorddb[ref] or recorddb[ref]['humidity'] != fields["value"]: + changed = True + recorddb[ref]['humidity'] = fields["value"] + if 'battery_ok' in jsonin: + fields = {'value': True if jsonin['battery_ok']==1 else False} + measurement = 'battery_ok' + points.append({'measurement': measurement, 'tags': tags, 'time': dateinflux, 'fields': fields}) + if 'battery_ok' not in recorddb[ref] or recorddb[ref]['battery_ok'] != fields["value"]: + changed = True + recorddb[ref]['battery_ok'] = fields["value"] + + current_ts = currentdt.timestamp() + if changed == True and (current_ts < recorddb[ref]["lastsend"] or current_ts > (recorddb[ref]["lastsend"] + DELAYRECORD)): + recorddb[ref]["lastsend"] = current_ts + #print("Write") + #print(ref) + print(points) + influxc.write_points(points) + #else: + # print("Not Write") + # print(ref) + # print(recorddb[ref]["lastsend"]) + + + +def on_subscribe(mqttc, obj, mid, granted_qos): + print("Subscribed: " + str(mid) + " " + str(granted_qos)) + +signal.signal(signal.SIGINT, signal_handler) + +# If you want to use a specific client id, use +# mqttc = mqtt.Client("client-id") +# but note that the client id must be unique on the broker. Leaving the client +# id parameter empty will generate a random id for you. +mqttc = mqtt.Client() +mqttc.on_message = on_message +mqttc.on_connect = on_connect +mqttc.on_subscribe = on_subscribe +# Uncomment to enable debug messages +# mqttc.on_log = on_log +mqttc.connect("localhost", 1883, 60) +mqttc.subscribe(prefix+"/events", 0) + +influxc = InfluxDBClient(host="localhost", port=8428) + +dbs = influxc.get_list_database() +dbs_list = [] +for db in dbs: + dbs_list.append(db.get("name")) + +if DBNAME not in dbs_list: + influxc.create_database(DBNAME) +influxc.switch_database(DBNAME) + +mqttc.loop_forever() diff --git a/python/rtl_433_mqtt2vm.py b/python/rtl_433_mqtt2vm.py new file mode 100755 index 0000000..d584d35 --- /dev/null +++ b/python/rtl_433_mqtt2vm.py @@ -0,0 +1,119 @@ +#!/usr/bin/python3 +# -*- coding: utf-8 -*- + +# Copyright (c) 2010-2013 Roger Light +# +# All rights reserved. This program and the accompanying materials +# are made available under the terms of the Eclipse Distribution License v1.0 +# which accompanies this distribution. +# +# The Eclipse Distribution License is available at +# http://www.eclipse.org/org/documents/edl-v10.php. +# +# Contributors: +# Roger Light - initial implementation +# Copyright (c) 2010,2011 Roger Light +# All rights reserved. + +# This shows a simple example of an MQTT subscriber. + +import signal +import sys +import json +import datetime +import paho.mqtt.client as mqtt +import requests + +def signal_handler(sig, frame): + mqttc.disconnect() + sys.exit(0) + +prefix = 'home/rtl_433' +vmhost = 'localhost' +mqtthost = 'localhost' +#prefix = 'rtl_433/fedora' +TAGS_KEYS = ['type','model','subtype','channel','id'] + +DELAYRECORD = 300 +MINTS = 1735686000 # 01/01/2025 00:00:00 +recorddb = {} + +def on_connect(mqttc, obj, flags, rc): + print("rc: " + str(rc)) + + +def create_measure(name, timestamp, tags, value): + #jsonmeasure = { "metric": { "__name__": name, "instance": "rtl433", "job": "rtl433tovm" }, "values": [value], "timestamps":[timestamp*1000] } + jsonmeasure = { "metric": { "__name__": name }, "values": [value], "timestamps":[timestamp*1000] } + jsonmeasure["metric"].update(tags) + return json.dumps(jsonmeasure) + '\n' + +def on_message(mqttc, obj, msg): + #print(msg.topic + " " + str(msg.qos) + " " + str(msg.payload)) + currentts=int(datetime.datetime.now(datetime.UTC).timestamp()) + jsonin=json.loads(msg.payload) + ref="" + if "model" in jsonin: + ref+=str(jsonin["model"]) + if "channel" in jsonin: + ref+=str(jsonin["channel"]) + if "id" in jsonin: + ref+=str(jsonin["id"]) + if ref not in recorddb: + recorddb[ref] = {} + recorddb[ref]["lastsend"]=0 + + + if currentts > MINTS and (currentts < recorddb[ref]["lastsend"] or currentts > (recorddb[ref]["lastsend"] + DELAYRECORD)): + tags = {} + for t in TAGS_KEYS: + if t in jsonin: + tags[t] = str(jsonin[t]) + + measures = "" + + if 'temperature_C' in jsonin: + value = float(jsonin['temperature_C']) + if 'temperature' not in recorddb[ref] or recorddb[ref]['temperature'] != value: + measures += create_measure("temperature", currentts, tags, value) + recorddb[ref]['temperature'] = value + if 'humidity' in jsonin: + value = int(jsonin['humidity']) + if 'humidity' not in recorddb[ref] or recorddb[ref]['humidity'] != value: + measures += create_measure("humidity", 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: + measures += create_measure("battery_ok", currentts, tags, value) + recorddb[ref]['battery_ok'] = value + + if len(measures) > 0: + recorddb[ref]["lastsend"] = currentts + #print(measures) + try: + requests.post("http://{}:8428/api/v1/import".format(vmhost), data=measures) + except: + print("VictoriaMetrics communication error") + + + +def on_subscribe(mqttc, obj, mid, granted_qos): + print("Subscribed: " + str(mid) + " " + str(granted_qos)) + +signal.signal(signal.SIGINT, signal_handler) + +# If you want to use a specific client id, use +# mqttc = mqtt.Client("client-id") +# but note that the client id must be unique on the broker. Leaving the client +# id parameter empty will generate a random id for you. +mqttc = mqtt.Client() +mqttc.on_message = on_message +mqttc.on_connect = on_connect +mqttc.on_subscribe = on_subscribe +# Uncomment to enable debug messages +# mqttc.on_log = on_log +mqttc.connect(mqtthost, 1883, 60) +mqttc.subscribe(prefix+"/events", 0) + +mqttc.loop_forever() diff --git a/python/rtl_433_mqtt_hass.py b/python/rtl_433_mqtt_hass.py new file mode 100755 index 0000000..4440a9b --- /dev/null +++ b/python/rtl_433_mqtt_hass.py @@ -0,0 +1,1127 @@ +#!/usr/bin/env python +# coding=utf-8 + +from __future__ import print_function +from __future__ import with_statement + +AP_DESCRIPTION=""" +Publish Home Assistant MQTT auto discovery topics for rtl_433 devices. + +rtl_433_mqtt_hass.py connects to MQTT and subscribes to the rtl_433 +event stream that is published to MQTT by rtl_433. The script publishes +additional MQTT topics that can be used by Home Assistant to automatically +discover and minimally configure new devices. + +The configuration topics published by this script tell Home Assistant +what MQTT topics to subscribe to in order to receive the data published +as device topics by MQTT. +""" + +AP_EPILOG=""" +It is strongly recommended to run rtl_433 with "-C si". +This script requires rtl_433 to publish both event messages and device +messages. If you've changed the device topic in rtl_433, use the same device +topic with the "-T" parameter. + +MQTT Username and Password can be set via the cmdline or passed in the +environment: MQTT_USERNAME and MQTT_PASSWORD. + +Prerequisites: + +1. rtl_433 running separately publishing events and devices messages to MQTT. + +2. Python installation +* Python 3.x preferred. +* Needs Paho-MQTT https://pypi.python.org/pypi/paho-mqtt + + Debian/raspbian: apt install python3-paho-mqtt + Or + pip install paho-mqtt +* Optional for running as a daemon see PEP 3143 - Standard daemon process library + (use Python 3.x or pip install python-daemon) + + +Running: + +This script can run continually as a daemon, where it will publish +a configuration topic for the device events sent to MQTT by rtl_433 +every 10 minutes. + +Alternatively if the rtl_433 devices in your environment change infrequently +this script can use the MQTT retain flag to make the configuration topics +persistent. The script will only need to be run when things change or if +the MQTT server loses its retained messages. + +Getting rtl_433 devices back after Home Assistant restarts will happen +more quickly if MQTT retain is enabled. Note however that definitions +for any transitient devices/false positives will retained indefinitely. + +If your sensor values change infrequently and you prefer to write the most +recent value even if not changed set -f to append "force_update = true" to +all configs. This is useful if you're graphing the sensor data or want to +alert on missing data. + +If you have changed the topic structure from the default topics in the rtl433 +configuration use the -T parameter to set the same topic structure here. + +Suggestions: + +Running this script will cause a number of Home Assistant entities (sensors +and binary sensors) to be created. These entities can linger for a while unless +the topic is republished with an empty config string. To avoid having to +do a lot of clean up When running this initially or debugging, set this +script to publish to a topic other than the one Home Assistant users (homeassistant). + +MQTT Explorer (http://mqtt-explorer.com/) is a very nice GUI for +working with MQTT. It is free, cross platform, and OSS. The structured +hierarchical view makes it easier to understand what rtl_433 is publishing +and how this script works with Home Assistant. + +MQTT Explorer also makes it easy to publish an empty config topic to delete an +entity from Home Assistant. + + +As of 2020-10, Home Assistant MQTT auto discovery doesn't currently support +supplying "friendly name", and "area" key, so some configuration must be +done in Home Assistant. + +There is a single global set of field mappings to Home Assistant meta data. + +""" + + + +# import daemon + + +import os +import argparse +import logging +import time +import json +import paho.mqtt.client as mqtt +import re + + +discovery_timeouts = {} + +# Fields that get ignored when publishing to Home Assistant +# (reduces noise to help spot missing field mappings) +SKIP_KEYS = [ "type", "model", "subtype", "channel", "id", "mic", "mod", + "freq", "sequence_num", "message_type", "exception", "raw_msg" ] + + +# Global mapping of rtl_433 field names to Home Assistant metadata. +# @todo - should probably externalize to a config file +# @todo - Model specific definitions might be needed + +mappings = { + "temperature_C": { + "device_type": "sensor", + "object_suffix": "T", + "config": { + "device_class": "temperature", + "name": "Temperature", + "unit_of_measurement": "°C", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + "temperature_1_C": { + "device_type": "sensor", + "object_suffix": "T1", + "config": { + "device_class": "temperature", + "name": "Temperature 1", + "unit_of_measurement": "°C", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + "temperature_2_C": { + "device_type": "sensor", + "object_suffix": "T2", + "config": { + "device_class": "temperature", + "name": "Temperature 2", + "unit_of_measurement": "°C", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + "temperature_3_C": { + "device_type": "sensor", + "object_suffix": "T3", + "config": { + "device_class": "temperature", + "name": "Temperature 3", + "unit_of_measurement": "°C", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + "temperature_4_C": { + "device_type": "sensor", + "object_suffix": "T4", + "config": { + "device_class": "temperature", + "name": "Temperature 4", + "unit_of_measurement": "°C", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + "temperature_F": { + "device_type": "sensor", + "object_suffix": "F", + "config": { + "device_class": "temperature", + "name": "Temperature", + "unit_of_measurement": "°F", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + + # This diagnostic sensor is useful to see when a device last sent a value, + # even if the value didn't change. + # https://community.home-assistant.io/t/send-metrics-to-influxdb-at-regular-intervals/9096 + # https://github.com/home-assistant/frontend/discussions/13687 + "time": { + "device_type": "sensor", + "object_suffix": "UTC", + "config": { + "device_class": "timestamp", + "name": "Timestamp", + "entity_category": "diagnostic", + "enabled_by_default": False, + "icon": "mdi:clock-in" + } + }, + + "battery_ok": { + "device_type": "sensor", + "object_suffix": "B", + "config": { + "device_class": "battery", + "name": "Battery", + "unit_of_measurement": "%", + "value_template": "{{ ((float(value) * 99)|round(0)) + 1 }}", + "state_class": "measurement", + "entity_category": "diagnostic" + } + }, + + "battery_mV": { + "device_type": "sensor", + "object_suffix": "mV", + "config": { + "device_class": "voltage", + "name": "Battery mV", + "unit_of_measurement": "mV", + "value_template": "{{ float(value) }}", + "state_class": "measurement", + "entity_category": "diagnostic" + } + }, + + "supercap_V": { + "device_type": "sensor", + "object_suffix": "V", + "config": { + "device_class": "voltage", + "name": "Supercap V", + "unit_of_measurement": "V", + "value_template": "{{ float(value) }}", + "state_class": "measurement", + "entity_category": "diagnostic" + } + }, + + "humidity": { + "device_type": "sensor", + "object_suffix": "H", + "config": { + "device_class": "humidity", + "name": "Humidity", + "unit_of_measurement": "%", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + "humidity_1": { + "device_type": "sensor", + "object_suffix": "H1", + "config": { + "device_class": "humidity", + "name": "Humidity 1", + "unit_of_measurement": "%", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + "humidity_2": { + "device_type": "sensor", + "object_suffix": "H2", + "config": { + "device_class": "humidity", + "name": "Humidity 2", + "unit_of_measurement": "%", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "moisture": { + "device_type": "sensor", + "object_suffix": "M", + "config": { + "device_class": "moisture", + "name": "Moisture", + "unit_of_measurement": "%", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "detect_wet": { + "device_type": "binary_sensor", + "object_suffix": "moisture", + "config": { + "name": "Water Sensor", + "device_class": "moisture", + "force_update": "true", + "payload_on": "1", + "payload_off": "0" + } + }, + + "pressure_hPa": { + "device_type": "sensor", + "object_suffix": "P", + "config": { + "device_class": "pressure", + "name": "Pressure", + "unit_of_measurement": "hPa", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "pressure_kPa": { + "device_type": "sensor", + "object_suffix": "P", + "config": { + "device_class": "pressure", + "name": "Pressure", + "unit_of_measurement": "kPa", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "wind_speed_km_h": { + "device_type": "sensor", + "object_suffix": "WS", + "config": { + "device_class": "wind_speed", + "name": "Wind Speed", + "unit_of_measurement": "km/h", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "wind_avg_km_h": { + "device_type": "sensor", + "object_suffix": "WS", + "config": { + "device_class": "wind_speed", + "name": "Wind Speed", + "unit_of_measurement": "km/h", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "wind_avg_mi_h": { + "device_type": "sensor", + "object_suffix": "WS", + "config": { + "device_class": "wind_speed", + "name": "Wind Speed", + "unit_of_measurement": "mi/h", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "wind_avg_m_s": { + "device_type": "sensor", + "object_suffix": "WS", + "config": { + "device_class": "wind_speed", + "name": "Wind Average", + "unit_of_measurement": "km/h", + "value_template": "{{ (float(value|float) * 3.6) | round(2) }}", + "state_class": "measurement" + } + }, + + "wind_speed_m_s": { + "device_type": "sensor", + "object_suffix": "WS", + "config": { + "device_class": "wind_speed", + "name": "Wind Speed", + "unit_of_measurement": "km/h", + "value_template": "{{ float(value|float) * 3.6 }}", + "state_class": "measurement" + } + }, + + "gust_speed_km_h": { + "device_type": "sensor", + "object_suffix": "GS", + "config": { + "device_class": "wind_speed", + "name": "Gust Speed", + "unit_of_measurement": "km/h", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "wind_max_km_h": { + "device_type": "sensor", + "object_suffix": "GS", + "config": { + "device_class": "wind_speed", + "name": "Wind max speed", + "unit_of_measurement": "km/h", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "wind_max_m_s": { + "device_type": "sensor", + "object_suffix": "GS", + "config": { + "device_class": "wind_speed", + "name": "Wind max", + "unit_of_measurement": "km/h", + "value_template": "{{ (float(value|float) * 3.6) | round(2) }}", + "state_class": "measurement" + } + }, + + "gust_speed_m_s": { + "device_type": "sensor", + "object_suffix": "GS", + "config": { + "device_class": "wind_speed", + "name": "Gust Speed", + "unit_of_measurement": "km/h", + "value_template": "{{ float(value|float) * 3.6 }}", + "state_class": "measurement" + } + }, + + "wind_dir_deg": { + "device_type": "sensor", + "object_suffix": "WD", + "config": { + "name": "Wind Direction", + "unit_of_measurement": "°", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "rain_mm": { + "device_type": "sensor", + "object_suffix": "RT", + "config": { + "device_class": "precipitation", + "name": "Rain Total", + "unit_of_measurement": "mm", + "value_template": "{{ value|float|round(2) }}", + "state_class": "total_increasing" + } + }, + + "rain_rate_mm_h": { + "device_type": "sensor", + "object_suffix": "RR", + "config": { + "device_class": "precipitation_intensity", + "name": "Rain Rate", + "unit_of_measurement": "mm/h", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "rain_in": { + "device_type": "sensor", + "object_suffix": "RT", + "config": { + "device_class": "precipitation", + "name": "Rain Total", + "unit_of_measurement": "in", + "value_template": "{{ value|float|round(2) }}", + "state_class": "total_increasing" + } + }, + + "rain_rate_in_h": { + "device_type": "sensor", + "object_suffix": "RR", + "config": { + "device_class": "precipitation_intensity", + "name": "Rain Rate", + "unit_of_measurement": "in/h", + "value_template": "{{ value|float|round(2) }}", + "state_class": "measurement" + } + }, + + "reed_open": { + "device_type": "binary_sensor", + "object_suffix": "reed_open", + "config": { + "device_class": "safety", + "force_update": "true", + "payload_on": "1", + "payload_off": "0", + "entity_category": "diagnostic" + } + }, + + "contact_open": { + "device_type": "binary_sensor", + "object_suffix": "contact_open", + "config": { + "device_class": "safety", + "force_update": "true", + "payload_on": "1", + "payload_off": "0", + "entity_category": "diagnostic" + } + }, + + "tamper": { + "device_type": "binary_sensor", + "object_suffix": "tamper", + "config": { + "device_class": "safety", + "force_update": "true", + "payload_on": "1", + "payload_off": "0", + "entity_category": "diagnostic" + } + }, + + "alarm": { + "device_type": "binary_sensor", + "object_suffix": "alarm", + "config": { + "device_class": "safety", + "force_update": "true", + "payload_on": "1", + "payload_off": "0", + "entity_category": "diagnostic" + } + }, + + "rssi": { + "device_type": "sensor", + "object_suffix": "rssi", + "config": { + "device_class": "signal_strength", + "unit_of_measurement": "dB", + "value_template": "{{ value|float|round(2) }}", + "state_class": "measurement", + "entity_category": "diagnostic" + } + }, + + "snr": { + "device_type": "sensor", + "object_suffix": "snr", + "config": { + "device_class": "signal_strength", + "unit_of_measurement": "dB", + "value_template": "{{ value|float|round(2) }}", + "state_class": "measurement", + "entity_category": "diagnostic" + } + }, + + "noise": { + "device_type": "sensor", + "object_suffix": "noise", + "config": { + "device_class": "signal_strength", + "unit_of_measurement": "dB", + "value_template": "{{ value|float|round(2) }}", + "state_class": "measurement", + "entity_category": "diagnostic" + } + }, + + "depth_cm": { + "device_type": "sensor", + "object_suffix": "D", + "config": { + "name": "Depth", + "unit_of_measurement": "cm", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "power_W": { + "device_type": "sensor", + "object_suffix": "watts", + "config": { + "device_class": "power", + "name": "Power", + "unit_of_measurement": "W", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "energy_kWh": { + "device_type": "sensor", + "object_suffix": "kwh", + "config": { + "device_class": "energy", + "name": "Energy", + "unit_of_measurement": "kWh", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "current_A": { + "device_type": "sensor", + "object_suffix": "A", + "config": { + "device_class": "current", + "name": "Current", + "unit_of_measurement": "A", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "voltage_V": { + "device_type": "sensor", + "object_suffix": "V", + "config": { + "device_class": "voltage", + "name": "Voltage", + "unit_of_measurement": "V", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + "light_lux": { + "device_type": "sensor", + "object_suffix": "lux", + "config": { + "device_class": "illuminance", + "name": "Outside Luminance", + "unit_of_measurement": "lx", + "value_template": "{{ value|int }}", + "state_class": "measurement" + } + }, + "lux": { + "device_type": "sensor", + "object_suffix": "lux", + "config": { + "device_class": "illuminance", + "name": "Outside Luminance", + "unit_of_measurement": "lx", + "value_template": "{{ value|int }}", + "state_class": "measurement" + } + }, + + "uv": { + "device_type": "sensor", + "object_suffix": "uv", + "config": { + "name": "UV Index", + "unit_of_measurement": "UV Index", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + "uvi": { + "device_type": "sensor", + "object_suffix": "uvi", + "config": { + "name": "UV Index", + "unit_of_measurement": "UV Index", + "value_template": "{{ value|float|round(1) }}", + "state_class": "measurement" + } + }, + + "storm_dist_km": { + "device_type": "sensor", + "object_suffix": "stdist", + "config": { + "name": "Lightning Distance", + "unit_of_measurement": "km", + "value_template": "{{ value|int }}", + "state_class": "measurement" + } + }, + + "storm_dist": { + "device_type": "sensor", + "object_suffix": "stdist", + "config": { + "name": "Lightning Distance", + "unit_of_measurement": "mi", + "value_template": "{{ value|int }}", + "state_class": "measurement" + } + }, + + "strike_distance": { + "device_type": "sensor", + "object_suffix": "stdist", + "config": { + "name": "Lightning Distance", + "unit_of_measurement": "mi", + "value_template": "{{ value|int }}", + "state_class": "measurement" + } + }, + + "strike_count": { + "device_type": "sensor", + "object_suffix": "strcnt", + "config": { + "name": "Lightning Strike Count", + "value_template": "{{ value|int }}", + "state_class": "total_increasing" + } + }, + + "consumption_data": { + "device_type": "sensor", + "object_suffix": "consumption", + "config": { + "name": "SCM Consumption Value", + "value_template": "{{ value|int }}", + "state_class": "total_increasing", + } + }, + + "consumption": { + "device_type": "sensor", + "object_suffix": "consumption", + "config": { + "name": "SCMplus Consumption Value", + "value_template": "{{ value|int }}", + "state_class": "total_increasing", + } + }, + + "channel": { + "device_type": "device_automation", + "object_suffix": "CH", + "config": { + "automation_type": "trigger", + "type": "button_short_release", + "subtype": "button_1", + } + }, + + "button": { + "device_type": "device_automation", + "object_suffix": "BTN", + "config": { + "automation_type": "trigger", + "type": "button_short_release", + "subtype": "button_2", + } + }, + + # WH45, WH290 + "pm2_5_ug_m3": { + "device_type": "sensor", + "object_suffix": "PM25", + "config": { + "device_class": "pm25", + "name": "PM 2.5 Concentration", + "unit_of_measurement": "µg/m³", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + # WH45 + "pm10_ug_m3": { + "device_type": "sensor", + "object_suffix": "PM10", + "config": { + "device_class": "pm10", + "name": "PM 10 Concentration", + "unit_of_measurement": "µg/m³", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + # WH290 + "estimated_pm10_0_ug_m3": { + "device_type": "sensor", + "object_suffix": "PM10", + "config": { + "device_class": "pm10", + "name": "Estimated PM 10 Concentration", + "unit_of_measurement": "µg/m³", + "value_template": "{{ value|float }}", + "state_class": "measurement" + } + }, + + # WH45 + "co2_ppm": { + "device_type": "sensor", + "object_suffix": "CO2", + "config": { + "device_class": "carbon_dioxide", + "name": "CO2 Concentration", + "unit_of_measurement": "ppm", + "value_template": "{{ value|int }}", + "state_class": "measurement" + } + }, + + "ext_power": { + "device_type": "binary_sensor", + "object_suffix": "extpwr", + "config": { + "device_class": "power", + "name": "External Power", + "payload_on": "1", + "payload_off": "0", + "entity_category": "diagnostic" + } + }, + +} + +# Use secret_knock to trigger device automations for Honeywell ActivLink +# doorbells. We have this outside of mappings as we need to configure two +# different configuration topics. +secret_knock_mappings = [ + + { + "device_type": "device_automation", + "object_suffix": "Knock", + "config": { + "automation_type": "trigger", + "type": "button_short_release", + "subtype": "button_1", + "payload": 0, + } + }, + + { + "device_type": "device_automation", + "object_suffix": "Secret-Knock", + "config": { + "automation_type": "trigger", + "type": "button_triple_press", + "subtype": "button_1", + "payload": 1, + } + }, + +] + +TOPIC_PARSE_RE = re.compile(r'\[(?P/?)(?P[^\]:]+):?(?P[^\]:]*)\]') + +def mqtt_connect(client, userdata, flags, rc): + """Callback for MQTT connects.""" + + logging.info("MQTT connected: " + mqtt.connack_string(rc)) + if rc != 0: + logging.error("Could not connect. Error: " + str(rc)) + else: + logging.info("Subscribing to: " + args.rtl_topic) + client.subscribe(args.rtl_topic) + + +def mqtt_disconnect(client, userdata, rc): + """Callback for MQTT disconnects.""" + logging.info("MQTT disconnected: " + mqtt.connack_string(rc)) + + +def mqtt_message(client, userdata, msg): + """Callback for MQTT message PUBLISH.""" + logging.debug("MQTT message: " + json.dumps(msg.payload.decode())) + + try: + # Decode JSON payload + data = json.loads(msg.payload.decode()) + + except json.decoder.JSONDecodeError: + logging.error("JSON decode error: " + msg.payload.decode()) + return + + topicprefix = "/".join(msg.topic.split("/", 2)[:2]) + bridge_event_to_hass(client, topicprefix, data) + + +def sanitize(text): + """Sanitize a name for Graphite/MQTT use.""" + return (text + .replace(" ", "_") + .replace("/", "_") + .replace(".", "_") + .replace("&", "")) + +def rtl_433_device_info(data, topic_prefix): + """Return rtl_433 device topic to subscribe to for a data element, based on the + rtl_433 device topic argument, as well as the device identifier""" + + path_elements = [] + id_elements = [] + last_match_end = 0 + # The default for args.device_topic_suffix is the same topic structure + # as set by default in rtl433 config + for match in re.finditer(TOPIC_PARSE_RE, args.device_topic_suffix): + path_elements.append(args.device_topic_suffix[last_match_end:match.start()]) + key = match.group(2) + if key in data: + # If we have this key, prepend a slash if needed + if match.group(1): + path_elements.append('/') + element = sanitize(str(data[key])) + path_elements.append(element) + id_elements.append(element) + elif match.group(3): + path_elements.append(match.group(3)) + last_match_end = match.end() + + path = ''.join(list(filter(lambda item: item, path_elements))) + id = '-'.join(id_elements) + return (f"{topic_prefix}/{path}", id) + + +def publish_config(mqttc, topic, model, object_id, mapping, key=None): + """Publish Home Assistant auto discovery data.""" + global discovery_timeouts + + device_type = mapping["device_type"] + object_suffix = mapping["object_suffix"] + object_name = "-".join([object_id, object_suffix]) + + path = "/".join([args.discovery_prefix, device_type, object_id, object_name, "config"]) + + # check timeout + now = time.time() + if path in discovery_timeouts: + if discovery_timeouts[path] > now: + logging.debug("Discovery timeout in the future for: " + path) + return False + + discovery_timeouts[path] = now + args.discovery_interval + + config = mapping["config"].copy() + + # Device Automation configuration is in a different structure compared to + # all other mqtt discovery types. + # https://www.home-assistant.io/integrations/device_trigger.mqtt/ + if device_type == 'device_automation': + config["topic"] = topic + config["platform"] = 'mqtt' + else: + readable_name = mapping["config"]["name"] if "name" in mapping["config"] else key + config["state_topic"] = topic + config["unique_id"] = object_name + config["name"] = readable_name + config["device"] = { "identifiers": [object_id], "name": object_id, "model": model, "manufacturer": "rtl_433" } + + if args.force_update: + config["force_update"] = "true" + + if args.expire_after: + config["expire_after"] = args.expire_after + + logging.debug(path + ":" + json.dumps(config)) + + mqttc.publish(path, json.dumps(config), retain=args.retain) + + return True + +def bridge_event_to_hass(mqttc, topic_prefix, data): + """Translate some rtl_433 sensor data to Home Assistant auto discovery.""" + + if "model" not in data: + # not a device event + logging.debug("Model is not defined. Not sending event to Home Assistant.") + return + + model = sanitize(data["model"]) + + skipped_keys = [] + published_keys = [] + + base_topic, device_id = rtl_433_device_info(data, topic_prefix) + if not device_id: + # no unique device identifier + logging.warning("No suitable identifier found for model: %s", model) + return + + if args.ids and "id" in data and data.get("id") not in args.ids: + # not in the safe list + logging.debug("Device (%s) is not in the desired list of device ids: [%s]" % (data["id"], ids)) + return + + # detect known attributes + for key in data.keys(): + if key in mappings: + # topic = "/".join([topicprefix,"devices",model,instance,key]) + topic = "/".join([base_topic, key]) + if publish_config(mqttc, topic, model, device_id, mappings[key], key): + published_keys.append(key) + else: + if key not in SKIP_KEYS: + skipped_keys.append(key) + + if "secret_knock" in data.keys(): + for m in secret_knock_mappings: + topic = "/".join([base_topic, "secret_knock"]) + if publish_config(mqttc, topic, model, device_id, m, "secret_knock"): + published_keys.append("secret_knock") + + if published_keys: + logging.info("Published %s: %s" % (device_id, ", ".join(published_keys))) + + if skipped_keys: + logging.info("Skipped %s: %s" % (device_id, ", ".join(skipped_keys))) + + +def rtl_433_bridge(): + """Run a MQTT Home Assistant auto discovery bridge for rtl_433.""" + + if hasattr(mqtt, 'CallbackAPIVersion'): # paho >= 2.0.0 + mqttc = mqtt.Client(callback_api_version=mqtt.CallbackAPIVersion.VERSION1) + else: + mqttc = mqtt.Client() + + if args.debug: + mqttc.enable_logger() + + if args.user is not None: + mqttc.username_pw_set(args.user, args.password) + + if args.ca_cert is not None: + mqttc.tls_set(ca_certs=args.ca_cert) + + mqttc.on_connect = mqtt_connect + mqttc.on_disconnect = mqtt_disconnect + mqttc.on_message = mqtt_message + mqttc.connect_async(args.host, args.port, 60) + logging.debug("MQTT Client: Starting Loop") + mqttc.loop_start() + + while True: + time.sleep(1) + + +def run(): + """Run main or daemon.""" + # with daemon.DaemonContext(files_preserve=[sock]): + # detach_process=True + # uid + # gid + # working_directory + rtl_433_bridge() + + +if __name__ == "__main__": + logging.basicConfig(format='[%(asctime)s] %(levelname)s:%(name)s:%(message)s',datefmt='%Y-%m-%dT%H:%M:%S%z') + logging.getLogger().setLevel(logging.INFO) + + parser = argparse.ArgumentParser(formatter_class=argparse.RawDescriptionHelpFormatter, + description=AP_DESCRIPTION, + epilog=AP_EPILOG) + + parser.add_argument("-d", "--debug", action="store_true") + parser.add_argument("-q", "--quiet", action="store_true") + parser.add_argument("-u", "--user", type=str, help="MQTT username") + parser.add_argument("-P", "--password", type=str, help="MQTT password") + parser.add_argument("-H", "--host", type=str, default="127.0.0.1", + help="MQTT hostname to connect to (default: %(default)s)") + parser.add_argument("-p", "--port", type=int, default=1883, + help="MQTT port (default: %(default)s)") + parser.add_argument("-c", "--ca_cert", type=str, help="MQTT TLS CA certificate path") + parser.add_argument("-r", "--retain", action="store_true") + parser.add_argument("-f", "--force_update", action="store_true", + help="Append 'force_update = true' to all configs.") + parser.add_argument("-R", "--rtl-topic", type=str, + default="rtl_433/+/events", + dest="rtl_topic", + help="rtl_433 MQTT event topic to subscribe to (default: %(default)s)") + parser.add_argument("-D", "--discovery-prefix", type=str, + dest="discovery_prefix", + default="homeassistant", + help="Home Assistant MQTT topic prefix (default: %(default)s)") + # This defaults to the rtl433 config default, so we assemble the same topic structure + parser.add_argument("-T", "--device-topic_suffix", type=str, + dest="device_topic_suffix", + default="devices[/type][/model][/subtype][/channel][/id]", + help="rtl_433 device topic suffix (default: %(default)s)") + parser.add_argument("-i", "--interval", type=int, + dest="discovery_interval", + default=600, + help="Interval to republish config topics in seconds (default: %(default)d)") + parser.add_argument("-x", "--expire-after", type=int, + dest="expire_after", + help="Number of seconds with no updates after which the sensor becomes unavailable") + parser.add_argument("-I", "--ids", type=int, nargs="+", + help="ID's of devices that will be discovered (omit for all)") + args = parser.parse_args() + + if args.debug and args.quiet: + logging.critical("Debug and quiet can not be specified at the same time") + exit(1) + + if args.debug: + logging.info("Enabling debug logging") + logging.getLogger().setLevel(logging.DEBUG) + if args.quiet: + logging.getLogger().setLevel(logging.ERROR) + + # allow setting MQTT username and password via environment variables + if not args.user and 'MQTT_USERNAME' in os.environ: + args.user = os.environ['MQTT_USERNAME'] + + if not args.password and 'MQTT_PASSWORD' in os.environ: + args.password = os.environ['MQTT_PASSWORD'] + + if not args.user or not args.password: + logging.warning("User or password is not set. Check credentials if subscriptions do not return messages.") + + if args.ids: + ids = ', '.join(str(id) for id in args.ids) + logging.info("Only discovering devices with ids: [%s]" % ids) + else: + logging.info("Discovering all devices") + + run() diff --git a/src/DLinkedList.h b/src/DLinkedList.h new file mode 100644 index 0000000..d1e4936 --- /dev/null +++ b/src/DLinkedList.h @@ -0,0 +1,345 @@ +/* + LinkedList.h - V1.1 - Generic LinkedList implementation + Works better with FIFO, because LIFO will need to + search the entire List to find the last one; + + For instructions, go to https://github.com/ivanseidel/LinkedList + + Created by Ivan Seidel Gomes, March, 2013. + Released into the public domain. +*/ + +#ifndef DLinkedList_h +#define DLinkedList_h + +#include + +template +struct DListNode +{ + T data; + DListNode *next; +}; + +template +class DLinkedList +{ + +protected: + int _size; + DListNode *root; + DListNode *last; + + // Helps "get" method, by saving last position + DListNode *lastNodeGot; + int lastIndexGot; + // isCached should be set to FALSE + // everytime the list suffer changes + bool isCached; + + DListNode *getNode(int index); + +public: + DLinkedList(); + ~DLinkedList(); + + /* + Returns current size of LinkedList + */ + virtual int size(); + /* + Adds a T object in the specified index; + Unlink and link the LinkedList correcly; + Increment _size + */ + virtual bool add(int index, T); + /* + Adds a T object in the end of the LinkedList; + Increment _size; + */ + virtual bool add(T); + /* + Adds a T object in the start of the LinkedList; + Increment _size; + */ + virtual bool unshift(T); + /* + Set the object at index, with T; + Increment _size; + */ + virtual bool set(int index, T); + /* + Remove object at index; + If index is not reachable, returns false; + else, decrement _size + */ + virtual T remove(int index); + /* + Remove last object; + */ + virtual T pop(); + /* + Remove first object; + */ + virtual T shift(); + /* + Get the index'th element on the list; + Return Element if accessible, + else, return false; + */ + virtual T get(int index); + + /* + Clear the entire array + */ + virtual void clear(); +}; + +// Initialize LinkedList with false values +template +DLinkedList::DLinkedList() +{ + root = NULL; + last = NULL; + _size = 0; + + lastNodeGot = root; + lastIndexGot = 0; + isCached = false; +} + +// Clear Nodes and free Memory +template +DLinkedList::~DLinkedList() +{ + DListNode *tmp; + while (root != NULL) + { + tmp = root; + root = root->next; + delete tmp; + } + last = NULL; + _size = 0; + isCached = false; +} + +/* + Actualy "logic" coding +*/ + +template +DListNode *DLinkedList::getNode(int index) +{ + + int _pos = 0; + DListNode *current = root; + + // Check if the node trying to get is + // immediatly AFTER the previous got one + if (isCached && lastIndexGot <= index) + { + _pos = lastIndexGot; + current = lastNodeGot; + } + + while (_pos < index && current) + { + current = current->next; + + _pos++; + } + + // Check if the object index got is the same as the required + if (_pos == index) + { + isCached = true; + lastIndexGot = index; + lastNodeGot = current; + + return current; + } + + return NULL; +} + +template +int DLinkedList::size() +{ + return _size; +} + +template +bool DLinkedList::add(int index, T _t) +{ + + if (index >= _size) + return add(_t); + + if (index == 0) + return unshift(_t); + + DListNode *tmp = new DListNode(), + *_prev = getNode(index - 1); + tmp->data = _t; + tmp->next = _prev->next; + _prev->next = tmp; + + _size++; + isCached = false; + + return true; +} + +template +bool DLinkedList::add(T _t) +{ + + DListNode *tmp = new DListNode(); + tmp->data = _t; + tmp->next = NULL; + + if (root) + { + // Already have elements inserted + last->next = tmp; + last = tmp; + } + else + { + // First element being inserted + root = tmp; + last = tmp; + } + + _size++; + isCached = false; + + return true; +} + +template +bool DLinkedList::unshift(T _t) +{ + + if (_size == 0) + return add(_t); + + DListNode *tmp = new DListNode(); + tmp->next = root; + tmp->data = _t; + root = tmp; + + _size++; + isCached = false; + + return true; +} + +template +bool DLinkedList::set(int index, T _t) +{ + // Check if index position is in bounds + if (index < 0 || index >= _size) + return false; + + getNode(index)->data = _t; + return true; +} + +template +T DLinkedList::pop() +{ + if (_size <= 0) + return T(); + + isCached = false; + + if (_size >= 2) + { + DListNode *tmp = getNode(_size - 2); + T ret = tmp->next->data; + delete (tmp->next); + tmp->next = NULL; + last = tmp; + _size--; + return ret; + } + else + { + // Only one element left on the list + T ret = root->data; + delete (root); + root = NULL; + last = NULL; + _size = 0; + return ret; + } +} + +template +T DLinkedList::shift() +{ + if (_size <= 0) + return T(); + + if (_size > 1) + { + DListNode *_next = root->next; + T ret = root->data; + delete (root); + root = _next; + _size--; + isCached = false; + + return ret; + } + else + { + // Only one left, then pop() + return pop(); + } +} + +template +T DLinkedList::remove(int index) +{ + if (index < 0 || index >= _size) + { + return T(); + } + + if (index == 0) + return shift(); + + if (index == _size - 1) + { + return pop(); + } + + DListNode *tmp = getNode(index - 1); + DListNode *toDelete = tmp->next; + T ret = toDelete->data; + tmp->next = tmp->next->next; + delete (toDelete); + _size--; + isCached = false; + return ret; +} + +template +T DLinkedList::get(int index) +{ + DListNode *tmp = getNode(index); + + return (tmp ? tmp->data : T()); +} + +template +void DLinkedList::clear() +{ + while (size() > 0) + shift(); +} + +#endif diff --git a/src/Dictionary.h b/src/Dictionary.h new file mode 100644 index 0000000..4014988 --- /dev/null +++ b/src/Dictionary.h @@ -0,0 +1,63 @@ +#include + +template +class Dictionary +{ +private: + DLinkedList KeyList = DLinkedList(); + DLinkedList ValList = DLinkedList(); + +public: + void set(T key, U val) + { + for (int i = 0; i < KeyList.size(); i++) + { + if (KeyList.get(i) == key) { + ValList.set(i, val); + return; + } + } + KeyList.add(key); + ValList.add(val); + } + + U get(T key) + { + for (int i = 0; i < KeyList.size(); i++) + { + if (KeyList.get(i) == key) + { + return ValList.get(i); + } + } + } + + T getKey(U val) + { + for (int i = 0; i < ValList.size(); i++) + { + if (ValList.get(i) == val) + { + return KeyList.get(i); + } + } + } + + int length() + { + return KeyList.size(); + } + + bool contains(T key) { + for (int x = 0; x < length(); x++) { + if(KeyList.get(x) == key) { + return true; + } + } + return false; + } + + T getKeyByIndex(int index) { + return KeyList.get(index); + } +}; diff --git a/src/main.ino b/src/main.ino new file mode 100644 index 0000000..bf1031c --- /dev/null +++ b/src/main.ino @@ -0,0 +1,132 @@ +#include +#include +#include +#include "Dictionary.h" + + +#define _DICT_PACK_STRUCTURES + +#define LED 8 + +CC1101 radiotx = RADIO_LIB_MODULE; +RCSwitch mySwitch = RCSwitch(); + +#ifndef RF_MODULE_FREQUENCY +# define RF_MODULE_FREQUENCY 433.92 +#endif + +#define JSON_MSG_BUFFER 512 +#define MINDELAY 10L + +Dictionary d1; + +char messageBuffer[JSON_MSG_BUFFER]; + +rtl_433_ESP rf; // use -1 to disable transmitter + +int count = 0; + +void rtl_433_Callback(char* message) { + JsonDocument jsonDocument; + deserializeJson(jsonDocument,message); + logJson(jsonDocument); + count++; +} + +void logJson(JsonDocument jsondata) { + if(jsondata["model"].is()) + { + String ref = jsondata["model"]; + if(jsondata["channel"].is()) { ref = ref + jsondata["channel"].as(); } + if(jsondata["id"].is()) { ref = ref + jsondata["id"].as(); } + 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 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 + d1.set(ref.c_str(), millis()); + ledblink(); + Serial.println(JSONmessageBuffer); + } + } +} + +void ledblink() { + digitalWrite(LED, LOW); + delay(150); + digitalWrite(LED, HIGH); +} + +void setup() { + 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 ( ; ; ) { + 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); + rf.setCallback(rtl_433_Callback, messageBuffer, JSON_MSG_BUFFER); + rf.enableReceiver(); + rf.getModuleStatus(); + + //d = new Dictionary(25); + //d("aa", "bb"); + //if(d1("aa")) { Serial.println("toto"); } + +} + +void loop() { + rf.loop(); + if (Serial.available()) + { + // Read the JSON document from the "link" serial port + JsonDocument doc; + DeserializationError err = deserializeJson(doc, Serial); + + if (err == DeserializationError::Ok) + { + Serial.print("cmd = "); + Serial.println(doc["cmd"].as()); + Serial.print("value = "); + Serial.println(doc["value"].as()); + 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); + mySwitch.send(doc["value"].as(), 24); + mySwitch.disableTransmit(); + radiotx.SPIsendCommand(RADIOLIB_CC1101_CMD_RX); + rf.enableReceiver(); + //rf.getModuleStatus(); + } + } + else + { + // Print error to the "debug" serial port + Serial.print("deserializeJson() returned "); + Serial.println(err.c_str()); + + // Flush all bytes in the "link" serial port buffer + while (Serial.available() > 0) + Serial.read(); + } + } +} diff --git a/src/pinsME_arduino.h b/src/pinsME_arduino.h new file mode 100644 index 0000000..10db8e8 --- /dev/null +++ b/src/pinsME_arduino.h @@ -0,0 +1,39 @@ +#ifndef PinsME_Arduino_h +#define PinsME_Arduino_h + +#include +#include "soc/soc_caps.h" + +#define EXTERNAL_NUM_INTERRUPTS 22 +#define NUM_DIGITAL_PINS 22 +#define NUM_ANALOG_INPUTS 6 + +static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT+8; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN +#define RGB_BUILTIN LED_BUILTIN +#define RGB_BRIGHTNESS 64 + +#define analogInputToDigitalPin(p) (((p)