From f8b2f22deba84205720aad9622b96eb8719c6d2c Mon Sep 17 00:00:00 2001 From: Daniel Schwab Date: Mon, 24 Jan 2022 11:16:41 +0100 Subject: [PATCH 1/3] chore: Fix typo on log message --- bin/p1decrypter.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bin/p1decrypter.py b/bin/p1decrypter.py index 7dca9b1..4a9c593 100644 --- a/bin/p1decrypter.py +++ b/bin/p1decrypter.py @@ -213,7 +213,7 @@ def process(self): hex_input = binascii.hexlify(self._connection.read()) if self._state == self.STATE_IGNORING: - logging.debug("STATE_IGNORING: Wait for start byte, get: ({0})".format(hex_input)) + logging.debug("STATE_IGNORING: Wait for start byte, got: ({0})".format(hex_input)) if hex_input == b'db': logging.debug("STATE_IGNORING: Start byte has been detected: ({0})".format(hex_input)) self._state = self.STATE_STARTED From b047e1c5fd23334863bdd0b5addabd5dbe6e8450 Mon Sep 17 00:00:00 2001 From: Daniel Schwab Date: Wed, 26 Jan 2022 18:42:28 +0100 Subject: [PATCH 2/3] fix: Change direct path to configure option --- cron/cron.05min | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/cron/cron.05min b/cron/cron.05min index 2252f16..f76fc27 100644 --- a/cron/cron.05min +++ b/cron/cron.05min @@ -9,9 +9,9 @@ # Will be executed as user "loxberry". -enabled=$(awk -F "=" '/ENABLED/ {print $2}' /opt/loxberry/config/plugins/p1decrypter/p1decrypter.cfg) +enabled=$(awk -F "=" '/ENABLED/ {print $2}' REPLACELBPCONFIGDIR/p1decrypter.cfg) -if ! pgrep -f p1decrypter.py >/dev/null && [[ $enabled == "1" || $1 == "1" ]] && [ -e /opt/loxberry/bin/plugins/p1decrypter/p1decrypter.py ] +if ! pgrep -f p1decrypter.py >/dev/null && [[ $enabled == "1" || $1 == "1" ]] && [ -e REPLACELBPBINDIR/p1decrypter.py ] then - /usr/bin/python3 /opt/loxberry/bin/plugins/p1decrypter/p1decrypter.py KEY --logfile=/opt/loxberry/log/plugins/p1decrypter/p1decrypter.log --configfile=/opt/loxberry/config/plugins/p1decrypter/p1decrypter.cfg >> /opt/loxberry/log/plugins/p1decrypter/p1decrypter.log 2>&1 + /usr/bin/python3 REPLACELBPBINDIR/p1decrypter.py KEY --logfile=REPLACELBPLOGDIR/p1decrypter.log --configfile=REPLACELBPCONFIGDIR/p1decrypter.cfg >> REPLACELBPLOGDIR/p1decrypter.log 2>&1 fi \ No newline at end of file From 378a1016973d05322b82728085c88993837de608 Mon Sep 17 00:00:00 2001 From: Daniel Schwab Date: Thu, 27 Jan 2022 15:51:05 +0100 Subject: [PATCH 3/3] feat: Add mqtt support --- bin/p1decrypter.py | 175 ++++++++++++++++++++++++++------- config/p1decrypter-default.cfg | 8 ++ config/p1decrypter.cfg | 8 ++ dpkg/apt | 3 +- plugin.cfg | 2 +- postupgrade.sh | 14 ++- prerelease.cfg | 4 +- templates/content.html | 149 ++++++++++++++++++++++------ webfrontend/htmlauth/index.cgi | 64 ++++++++++++ 9 files changed, 351 insertions(+), 76 deletions(-) diff --git a/bin/p1decrypter.py b/bin/p1decrypter.py index 4a9c593..6a35e39 100644 --- a/bin/p1decrypter.py +++ b/bin/p1decrypter.py @@ -9,6 +9,7 @@ import configparser import json import base64 +import paho.mqtt.client as mqtt from Cryptodome.Cipher import AES @@ -49,6 +50,9 @@ def __init__(self): self.LBSCONFIG = "" self.miniserver_id = "" self.general_json = {} + self.mqtt_use_gateway = False + self.mqtt_connected = False + self.mqtt_client = {} def main(self): self.args() @@ -74,10 +78,20 @@ def args(self): parser.add_argument('-a', '--aad', required=False, default="3000112233445566778899AABBCCDDEEFF", help="Additional authenticated data. Default: 3000112233445566778899AABBCCDDEEFF") - parser.add_argument('-u', '--send-to-udp', required=False, default=True, action='store_true', - help="Send data to UDP. Default: true") + parser.add_argument('-u', '--send-to-udp', required=False, default=False, action='store_true', + help="Send data over UDP. Default: false") parser.add_argument('-ui', '--udp-host', help="UDP IP / Host") - parser.add_argument('-up', '--udp-port', type=int, help="UDP port") + parser.add_argument('-up', '--udp-port', type=int, help="UDP port. Default: 54321") + + parser.add_argument('-q', '--send-mqtt', required=False, default=False, action='store_true', + help="Send data over MQTT. Default: false") + parser.add_argument('-qb', '--mqtt-broker', help="MQTT Broker") + parser.add_argument('-qu', '--mqtt-broker-username', help="MQTT Broker Username") + parser.add_argument('-qw', '--mqtt-broker-password', help="MQTT Broker Password") + parser.add_argument('-qp', '--mqtt-broker-port', default="1883", type=int, help="MQTT port. Default: 1883") + parser.add_argument('-qt', '--mqtt-topic-prefix', default="p1decrypter", + help="MQTT Topic prefix. Default: p1decrypter") + parser.add_argument('-qq', '--mqtt-topic-qos', type=int, default=1, help="MQTT QOS Default: 1") parser.add_argument('-s', '--send-to-serial-port', required=False, default=False, action='store_true', help="Send data to output serial port. Use socat to generate virtual port e.g.: socat -d -d pty,raw,echo=0,link=/dev/p1decrypterI pty,raw,echo=0,link=/dev/p1decrypterO") @@ -115,6 +129,8 @@ def config(self): datefmt='%Y-%m-%d %H:%M:%S', handlers=[logging.StreamHandler()]) + logging.info("Process arguments and config") + if self._args.configfile: logging.info("Read config file and overwrite arguments") if not os.path.exists(self._args.configfile): @@ -126,6 +142,7 @@ def config(self): self.LBSCONFIG = os.getenv("LBSCONFIG", os.getcwd()) self.miniserver_id = pluginconfig.get('P1DECRYPTER', 'MINISERVER_ID') + self.mqtt_use_gateway = bool(int(pluginconfig.get('P1DECRYPTER', 'MQTT_USE_GATEWAY'))) self._args.enabled = bool(int(pluginconfig.get('P1DECRYPTER', 'ENABLED'))) self._args.key = pluginconfig.get('P1DECRYPTER', 'KEY') @@ -137,14 +154,25 @@ def config(self): pluginconfig.get('P1DECRYPTER', 'MAPPING').replace('\\n', '').encode('ascii') ).decode('ascii') self._args.aad = pluginconfig.get('P1DECRYPTER', 'AAD') + self._args.send_to_udp = bool(int(pluginconfig.get('P1DECRYPTER', 'SEND_TO_UDP'))) self._args.udp_host = pluginconfig.get('P1DECRYPTER', 'UDP_HOST') self._args.udp_port = int(pluginconfig.get('P1DECRYPTER', 'UDP_PORT')) + + self._args.send_mqtt = bool(int(pluginconfig.get('P1DECRYPTER', 'SEND_MQTT'))) + self._args.mqtt_broker = pluginconfig.get('P1DECRYPTER', 'MQTT_BROKER') + self._args.mqtt_broker_username = pluginconfig.get('P1DECRYPTER', 'MQTT_BROKER_USERNAME') + self._args.mqtt_broker_password = pluginconfig.get('P1DECRYPTER', 'MQTT_BROKER_PASSWORD') + self._args.mqtt_broker_port = int(pluginconfig.get('P1DECRYPTER', 'MQTT_BROKER_PORT')) + self._args.mqtt_topic_prefix = pluginconfig.get('P1DECRYPTER', 'MQTT_TOPIC_PREFIX') + self._args.mqtt_topic_qos = int(pluginconfig.get('P1DECRYPTER', 'MQTT_TOPIC_QOS')) + self._args.send_to_serial_port = bool(int(pluginconfig.get('P1DECRYPTER', 'SEND_TO_SERIAL_PORT'))) self._args.serial_output_port = pluginconfig.get('P1DECRYPTER', 'SERIAL_OUTPUT_PORT') self._args.serial_output_baudrate = int(pluginconfig.get('P1DECRYPTER', 'SERIAL_OUTPUT_BAUDRATE')) self._args.serial_output_parity = pluginconfig.get('P1DECRYPTER', 'SERIAL_OUTPUT_PARITY') self._args.serial_output_stopbits = int(pluginconfig.get('P1DECRYPTER', 'SERIAL_OUTPUT_STOPBITS')) + self._args.raw = bool(int(pluginconfig.get('P1DECRYPTER', 'RAW'))) self._args.verbose = bool(int(pluginconfig.get('P1DECRYPTER', 'VERBOSE'))) else: @@ -152,44 +180,67 @@ def config(self): if self._args.verbose: logging.getLogger().setLevel(logging.DEBUG) - key = self._args.key - aad = self._args.aad - self._args.key = 'KEY_WILL_NOT_SHOWN_IN_LOGFILE' - self._args.aad = 'AAD_WILL_NOT_SHOWN_IN_LOGFILE' - logging.debug("Arguments: {0}".format(self._args)) - self._args.key = key - self._args.aad = aad - - logging.info("Arguments and config processed") if self._args.enabled == "0": - logging.warning("P1 Decrypter is not enabled in configuration file. exit") + logging.critical("P1 Decrypter is not enabled in configuration file. exit") sys.exit(-1) - self.miniserver() + if not self.mqtt_use_gateway and self._args.mqtt_broker == "" and self._args.send_mqtt: + logging.critical("No MQTT Broker defined. exit") + sys.exit(-1) - def miniserver(self): + self.loxberry() - if not self._args.udp_host and self._args.send_to_udp: - if not self.miniserver_id: - logging.error("No UDP Host or Miniserver ID is set.") - sys.exit(-1) + def loxberry(self): + + set_udp = not self._args.udp_host and self._args.send_to_udp + if set_udp or self.mqtt_use_gateway: config_path = os.path.join(self.LBSCONFIG, "general.json") logging.info("Try load Miniserver system configuration file {0}".format(config_path)) with open(config_path, "r") as config_path_handle: self.general_json = json.load(config_path_handle) + # get miniserver ip + if set_udp and not self.miniserver_id: + logging.error("No UDP Host or Miniserver ID is set.") + sys.exit(-1) + else: logging.info("Check if miniserver exists in {0}".format(config_path)) if not self.miniserver_id in self.general_json["Miniserver"].keys(): - logging.critical( - "Miniserver with id {0} is not configured in {1}".format(self.miniserver_id, config_path)) + logging.critical("Miniserver with id {0} is not configured in {1}. exit" + .format(self.miniserver_id, config_path)) sys.exit(-1) self._args.udp_host = self.general_json["Miniserver"][self.miniserver_id]["Ipaddress"] - logging.info("Miniserver ip address: {0}".format(self._args.udp_host)) + # get mqtt settings + if self.mqtt_use_gateway: + logging.info("MQTT Gateway settings enabled") + if not "Mqtt" in self.general_json: + logging.critical("MQTT Gateway settings not available. exit" + .format(self.miniserver_id, config_path)) + sys.exit(-1) + + self._args.mqtt_broker = self.general_json["Mqtt"]["Brokerhost"] + self._args.mqtt_broker_username = self.general_json["Mqtt"]["Brokeruser"] + self._args.mqtt_broker_password = self.general_json["Mqtt"]["Brokerpass"] + self._args.mqtt_broker_port = int(self.general_json["Mqtt"]["Brokerport"]) + + # debug log of config + if self._args.verbose: + key = self._args.key + aad = self._args.aad + mqtt_broker_password = self._args.mqtt_broker_password + self._args.key = 'KEY_WILL_NOT_SHOWN_IN_LOGFILE' + self._args.aad = 'AAD_WILL_NOT_SHOWN_IN_LOGFILE' + self._args.mqtt_broker_password = 'PASSWORD_WILL_NOT_SHOWN_IN_LOGFILE' + logging.debug("Config processed: {0}".format(self._args)) + self._args.key = key + self._args.aad = aad + self._args.mqtt_broker_password = mqtt_broker_password + self.connect() logging.info("Start processing incoming data.") while True: @@ -206,7 +257,7 @@ def connect(self): stopbits=self._args.serial_input_stopbits ) except Exception as e: - logging.error("Connection failed: {0}".format(e)) + logging.critical("Connection to serial input port failed: {0}. exit".format(e)) sys.exit(-1) def process(self): @@ -248,7 +299,7 @@ def process(self): self._state = self.STATE_HAS_SYSTEM_TITLE_SUFFIX # Ignore separator byte logging.debug("STATE_HAS_SYSTEM_TITLE: Additional byte after system title: ({0})".format(hex_input)) else: - logging.warning("Expected 0x82 separator byte not found, dropping frame:") + logging.warning("Expected 0x82 separator byte not found, dropping frame") logging.debug("Buffer ({0})".format(self._buffer)) self._state = self.STATE_IGNORING elif self._state == self.STATE_HAS_SYSTEM_TITLE_SUFFIX: @@ -256,7 +307,8 @@ def process(self): self._data_length_bytes += hex_input self._data_length = int(self._data_length_bytes, 16) self._state = self.STATE_HAS_DATA_LENGTH - logging.debug("STATE_HAS_SYSTEM_TITLE_SUFFIX: Length of remaining data: ({0})".format(self._data_length_bytes)) + logging.debug("STATE_HAS_SYSTEM_TITLE_SUFFIX: Length of remaining data: ({0})" + .format(self._data_length_bytes)) else: self._data_length_bytes += hex_input elif self._state == self.STATE_HAS_DATA_LENGTH: @@ -310,26 +362,34 @@ def decrypt(self): def mapping(self, decryption): logging.debug("Decryption done. Extract data by mapping configuration: {0}".format(decryption)) - output = "" + decryption_decoded = decryption.decode() + mapped_values_string = "" if self._args.raw: logging.debug("Raw output is enabled. Mapping extraction stopped. Send complete telegram") - output = decryption; + mapped_values_string = decryption_decoded + mapped_values_array = decryption_decoded else: input_multi_array = [] + mapped_values_array = [] for i in self._args.mapping.splitlines(): input_multi_array.append([i.split(',')[0].strip().strip("'"), i.split(',')[1].strip().strip("'")]) for i in input_multi_array: - output += i[0] + ":" + re.search(i[1], decryption.decode()).group(0) + "\n" + value = re.search(i[1], decryption_decoded).group(0) + mapped_values_string += i[0] + ":" + value + "\n" + mapped_values_array.append([i[0], value]) - output = output.encode() + mapped_values_string = mapped_values_string if self._args.send_to_udp: - self.send_to_udb(output) + self.send_to_udb(mapped_values_string) if self._args.send_to_serial_port: - self.send_to_serial_port(output) + self.send_to_serial_port(mapped_values_string) + + if self._args.send_mqtt: + self.send_mqtt(mapped_values_array) def send_to_serial_port(self, output): logging.debug("Send the decrypted data to output serial port: {0}".format(output.decode())) @@ -339,17 +399,60 @@ def send_to_serial_port(self, output): parity=self._args.serial_output_parity, stopbits=self._args.serial_output_stopbits ) - serial_port.write(output) + serial_port.write(output.encode()) serial_port.close() def send_to_udb(self, output): - logging.debug("Send the decrypted data over udp: {0}".format(output.decode())) + logging.debug("Send the decrypted data over udp: {0}".format(output)) connection = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) - res = connection.sendto(output, (self._args.udp_host, self._args.udp_port)) + res = connection.sendto(output.encode(), (self._args.udp_host, self._args.udp_port)) connection.close() - if res != output.__len__(): - logging.error("Sent bytes not matching. Expected {0} to be {1}".format(output.decode().__len__(), res)) + if res != output.encode().__len__(): + logging.error("Sent bytes not matching. Expected {0} to be {1}".format(output.__len__(), res)) + + def send_mqtt(self, output): + def get_mqtt_client(): + if not self.mqtt_connected: + logging.info("Try to connect to MQTT Broker: {0}".format(self._args.mqtt_broker)) + self.mqtt_connected = True + + def on_connect(client, userdata, flags, rc): + if rc == 0: + logging.info("Connected to MQTT Broker: {0}".format(self._args.mqtt_broker)) + else: + self.mqtt_connected = False + logging.error("MQTT connection failed {0}".format(rc)) + + def on_disconnect(): + self.mqtt_connected = False + logging.error("MQTT disconnected") + + self.mqtt_client = mqtt.Client() + if self._args.mqtt_broker_username or self._args.mqtt_broker_password: + self.mqtt_client.username_pw_set(self._args.mqtt_broker_username, self._args.mqtt_broker_password) + else: + logging.info("MQTT Broker username and password not set") + + self.mqtt_client.on_connect = on_connect + self.mqtt_client.on_disconnect = on_disconnect + + self.mqtt_client.connect(self._args.mqtt_broker, self._args.mqtt_broker_port) + return self.mqtt_client + else: + return self.mqtt_client + + mqtt_client = get_mqtt_client() + if self._args.raw: + logging.debug( + "Send decrypted data over mqtt: {0} -> {1}".format(self._args.mqtt_topic_prefix + "/raw", output)) + mqtt_client.publish(self._args.mqtt_topic_prefix + "/raw", output, qos=self._args.mqtt_topic_qos) + else: + for i in output: + logging.debug( + "Send decrypted data over mqtt: {0} -> {1}".format(self._args.mqtt_topic_prefix + "/" + i[0], i[1])) + mqtt_client.publish(self._args.mqtt_topic_prefix + "/" + i[0], i[1], qos=self._args.mqtt_topic_qos) + mqtt_client.loop() if __name__ == '__main__': diff --git a/config/p1decrypter-default.cfg b/config/p1decrypter-default.cfg index 240178c..36364f7 100644 --- a/config/p1decrypter-default.cfg +++ b/config/p1decrypter-default.cfg @@ -11,6 +11,14 @@ SEND_TO_UDP=1 MINISERVER_ID=1 UDP_HOST= UDP_PORT=54321 +SEND_MQTT=0 +MQTT_USE_GATEWAY=0 +MQTT_BROKER=localhost +MQTT_BROKER_USERNAME= +MQTT_BROKER_PASSWORD= +MQTT_BROKER_PORT=1883 +MQTT_TOPIC_PREFIX=p1decrypter +MQTT_TOPIC_QOS=1 SEND_TO_SERIAL_PORT=0 SERIAL_OUTPUT_PORT=/dev/p1decrypter SERIAL_OUTPUT_BAUDRATE=115200 diff --git a/config/p1decrypter.cfg b/config/p1decrypter.cfg index 240178c..36364f7 100644 --- a/config/p1decrypter.cfg +++ b/config/p1decrypter.cfg @@ -11,6 +11,14 @@ SEND_TO_UDP=1 MINISERVER_ID=1 UDP_HOST= UDP_PORT=54321 +SEND_MQTT=0 +MQTT_USE_GATEWAY=0 +MQTT_BROKER=localhost +MQTT_BROKER_USERNAME= +MQTT_BROKER_PASSWORD= +MQTT_BROKER_PORT=1883 +MQTT_TOPIC_PREFIX=p1decrypter +MQTT_TOPIC_QOS=1 SEND_TO_SERIAL_PORT=0 SERIAL_OUTPUT_PORT=/dev/p1decrypter SERIAL_OUTPUT_BAUDRATE=115200 diff --git a/dpkg/apt b/dpkg/apt index 81257d6..3ca8e59 100644 --- a/dpkg/apt +++ b/dpkg/apt @@ -3,4 +3,5 @@ # python3 python3-serial -python3-pycryptodome \ No newline at end of file +python3-pycryptodome +python3-paho-mqtt \ No newline at end of file diff --git a/plugin.cfg b/plugin.cfg index 0f46394..a5da6c5 100644 --- a/plugin.cfg +++ b/plugin.cfg @@ -11,7 +11,7 @@ EMAIL=p1decrypter@gmail.com # The version of your plugin - important if you would like to write an # upgrade script. Use a correct syntax, which is supported by LoxBerry: # More info: http://www.loxwiki.eu/x/LYG3AQ -VERSION=1.1.1 +VERSION=1.2.0 # Short name and prefered subfolder of your Plugin (do not use blanks - they # will be filtered - and use lowercase only)! Used for script names in diff --git a/postupgrade.sh b/postupgrade.sh index f50fc5a..d761287 100644 --- a/postupgrade.sh +++ b/postupgrade.sh @@ -63,13 +63,17 @@ PBIN=$LBPBIN/$PDIR echo " Copy back existing config files" cp -v /tmp/$PTEMPDIR/_upgrade/config/p1decrypter.cfg $PCONFIG/p1decrypter.cfg -# echo " Adding new config parameters" -# grep -q -F "VERBOSE=" $PCONFIG/p1decrypter.cfg || echo "VERBOSE=0" >> $PCONFIG/p1decrypter.cfg +echo " Adding new config parameters" +grep -q -F "SEND_MQTT=" $PCONFIG/p1decrypter.cfg || echo "SEND_MQTT=0" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_USE_GATEWAY=" $PCONFIG/p1decrypter.cfg || echo "MQTT_USE_GATEWAY=0" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_BROKER=" $PCONFIG/p1decrypter.cfg || echo "MQTT_BROKER=" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_BROKER_USERNAME=" $PCONFIG/p1decrypter.cfg || echo "MQTT_BROKER_USERNAME=" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_BROKER_PASSWORD=" $PCONFIG/p1decrypter.cfg || echo "MQTT_BROKER_PASSWORD=" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_BROKER_PORT=" $PCONFIG/p1decrypter.cfg || echo "MQTT_BROKER_PORT=1883" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_TOPIC_PREFIX=" $PCONFIG/p1decrypter.cfg || echo "MQTT_TOPIC_PREFIX=p1decrypter" >> $PCONFIG/p1decrypter.cfg +grep -q -F "MQTT_TOPIC_QOS=" $PCONFIG/p1decrypter.cfg || echo "MQTT_TOPIC_QOS=1" >> $PCONFIG/p1decrypter.cfg echo " Remove temporary folders" rm -rf /tmp/$PTEMPDIR/_upgrade -echo " Start P1 Decrypter" -/bin/bash $PDIR/system/cron/cron.05min/p1decrypter > /dev/null 2>&1 & - exit 0 diff --git a/prerelease.cfg b/prerelease.cfg index 0903a60..e3dab3d 100644 --- a/prerelease.cfg +++ b/prerelease.cfg @@ -5,10 +5,10 @@ # the donwload link here -VERSION=1.1.1 +VERSION=1.2.0 # Download URL of the ZIP Archive -ARCHIVEURL=https://github.com/metrophos/LoxBerry-Plugin-P1-Decrypter/archive/1.1.1.zip +ARCHIVEURL=https://github.com/metrophos/LoxBerry-Plugin-P1-Decrypter/archive/1.2.0.zip # URL for further information about this release INFOURL=https://github.com/metrophos/LoxBerry-Plugin-P1-Decrypter/releases diff --git a/templates/content.html b/templates/content.html index 603e924..53a383d 100644 --- a/templates/content.html +++ b/templates/content.html @@ -25,7 +25,7 @@ Value Mapping - Value mapping of Smart Meter output. Only one per line 'label','regex' + Value mapping of Smart Meter output. Only one per line: 'label','regex' @@ -35,20 +35,7 @@ - Miniserver UDP-Port - - - - - - SEND_TO_UDP - - - - - -

Advanced Options

- +

Advanced Options

Serial Input Port @@ -80,6 +67,39 @@ + + Raw + + +

+ Disable mapping and send encrypted Smart Meter output directly +

+ + + + Verbose + + +

+ Advanced debug log +

+ + + +

UDP Options

+ + + Send over UDP + + + + + + Miniserver UDP-Port + + + + UDP Host @@ -87,12 +107,73 @@ If set, this option overrules the Miniserver as host + +

MQTT Options

+ + + Send over MQTT + + +

+ Send data as MQTT Messages +

+ + + + Use MQTT Gateway + + +

+ Use MQTT Gateway Plugin configuration +

+ + + + MQTT Broker + + + + + + MQTT Broker Username + + + + + + MQTT Broker Password + + + + + + MQTT Port + + + + + + MQTT Topic Prefix + + +

+ + + + MQTT Topic QOS + + + + + +

Serial Output Options

+ Send To Serial Port

- P1 Decrypter will send data to specified output serial port. + Send data to specified output serial port. Generating virtual serial port example: socat -d -d pty,raw,echo=0,link=/dev/p1decrypter pty,raw,echo=0,link=/dev/smartmeter/p1decrypter

@@ -121,21 +202,6 @@ - - Raw - - -

- Disable mapping and send encrypted Smart Meter output directly -

- - - - Verbose - - - -   @@ -155,6 +221,27 @@ $('select[name=send_to_serial_port]').attr('data-role', 'flipswitch'); $('select[name=raw]').attr('data-role', 'flipswitch'); $('select[name=verbose]').attr('data-role', 'flipswitch'); + $('select[name=send_mqtt]').attr('data-role', 'flipswitch'); + $('select[name=mqtt_use_gateway]').attr('data-role', 'flipswitch'); + + if($('select[name=mqtt_use_gateway]').val() === '0') { + $('.mqtt').show(); + $('.mqtt_topic_prefix').hide(); + } else { + $('.mqtt').hide(); + $('.mqtt_topic_prefix').text(`Please don't forget to add MQTT Subscriptions: ${$('input[name=mqtt_topic_prefix]').val()}/#`); + $('.mqtt_topic_prefix').show(); + } + $('select[name=mqtt_use_gateway]').on('change', function() { + if($('select[name=mqtt_use_gateway]').val() === '0') { + $('.mqtt').show(); + $('.mqtt_topic_prefix').hide(); + } else { + $('.mqtt').hide(); + $('.mqtt_topic_prefix').text(`Please don't forget to add MQTT Subscriptions: ${$('input[name=mqtt_topic_prefix]').val()}/#`); + $('.mqtt_topic_prefix').show(); + } + }); // Reset to default let reset = false; diff --git a/webfrontend/htmlauth/index.cgi b/webfrontend/htmlauth/index.cgi index 0d14b7e..1ec1ff7 100644 --- a/webfrontend/htmlauth/index.cgi +++ b/webfrontend/htmlauth/index.cgi @@ -40,6 +40,16 @@ if ($R::action eq "Save & Restart") { $p1decrypterCfg->param('P1DECRYPTER.AAD', $R::aad); $p1decrypterCfg->param('P1DECRYPTER.SEND_TO_UDP', $R::send_to_udp); $p1decrypterCfg->param('P1DECRYPTER.UDP_HOST', $R::udp_host); + + $p1decrypterCfg->param('P1DECRYPTER.SEND_MQTT', $R::send_mqtt); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_USE_GATEWAY', $R::mqtt_use_gateway); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER', $R::mqtt_broker); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_USERNAME', $R::mqtt_broker_username); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_PASSWORD', $R::mqtt_broker_password); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_PORT', $R::mqtt_broker_port); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_TOPIC_PREFIX', $R::mqtt_topic_prefix); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_TOPIC_QOS', $R::mqtt_topic_qos); + $p1decrypterCfg->param('P1DECRYPTER.SEND_TO_SERIAL_PORT', $R::send_to_serial_port); $p1decrypterCfg->param('P1DECRYPTER.SERIAL_OUTPUT_PORT', $R::serial_output_port); $p1decrypterCfg->param('P1DECRYPTER.SERIAL_OUTPUT_BAUDRATE', $R::serial_output_baudrate); @@ -66,6 +76,16 @@ if ($R::action eq "Reset default configruation") { $p1decrypterCfg->param('P1DECRYPTER.AAD', $p1decrypterDefaultCfg->param('P1DECRYPTER.AAD')); $p1decrypterCfg->param('P1DECRYPTER.SEND_TO_UDP', $p1decrypterDefaultCfg->param('P1DECRYPTER.SEND_TO_UDP')); $p1decrypterCfg->param('P1DECRYPTER.UDP_HOST', $p1decrypterDefaultCfg->param('P1DECRYPTER.UDP_HOST')); + + $p1decrypterCfg->param('P1DECRYPTER.SEND_MQTT', $p1decrypterDefaultCfg->param('P1DECRYPTER.SEND_MQTT')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_USE_GATEWAY', $p1decrypterDefaultCfg->param('P1DECRYPTER.SEND_MQTT')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER', $p1decrypterDefaultCfg->param('P1DECRYPTER.MQTT_BROKER')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_USERNAME', $p1decrypterDefaultCfg->param('P1DECRYPTER.MQTT_BROKER_USERNAME')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_PASSWORD', $p1decrypterDefaultCfg->param('P1DECRYPTER.MQTT_BROKER_PASSWORD')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_PORT', $p1decrypterDefaultCfg->param('P1DECRYPTER.MQTT_BROKER_PORT')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_TOPIC_PREFIX', $p1decrypterDefaultCfg->param('P1DECRYPTER.MQTT_TOPIC_PREFIX')); + $p1decrypterCfg->param('P1DECRYPTER.MQTT_TOPIC_QOS', $p1decrypterDefaultCfg->param('P1DECRYPTER.MQTT_TOPIC_QOS')); + $p1decrypterCfg->param('P1DECRYPTER.SEND_TO_SERIAL_PORT', $p1decrypterDefaultCfg->param('P1DECRYPTER.SEND_TO_SERIAL_PORT')); $p1decrypterCfg->param('P1DECRYPTER.SERIAL_OUTPUT_PORT', $p1decrypterDefaultCfg->param('P1DECRYPTER.SERIAL_OUTPUT_PORT')); $p1decrypterCfg->param('P1DECRYPTER.SERIAL_OUTPUT_BAUDRATE', $p1decrypterDefaultCfg->param('P1DECRYPTER.SERIAL_OUTPUT_BAUDRATE')); @@ -92,6 +112,16 @@ $R::serial_input_stopbits = $p1decrypterCfg->param('P1DECRYPTER.SERIAL_INPUT_STO $R::aad = $p1decrypterCfg->param('P1DECRYPTER.AAD'); $R::send_to_udp = $p1decrypterCfg->param('P1DECRYPTER.SEND_TO_UDP'); $R::udp_host = $p1decrypterCfg->param('P1DECRYPTER.UDP_HOST'); + +$R::send_mqtt = $p1decrypterCfg->param('P1DECRYPTER.SEND_MQTT'); +$R::mqtt_use_gateway = $p1decrypterCfg->param('P1DECRYPTER.MQTT_USE_GATEWAY'); +$R::mqtt_broker = $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER'); +$R::mqtt_broker_username = $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_USERNAME'); +$R::mqtt_broker_password = $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_PASSWORD'); +$R::mqtt_broker_port = $p1decrypterCfg->param('P1DECRYPTER.MQTT_BROKER_PORT'); +$R::mqtt_topic_prefix = $p1decrypterCfg->param('P1DECRYPTER.MQTT_TOPIC_PREFIX'); +$R::mqtt_topic_qos = $p1decrypterCfg->param('P1DECRYPTER.MQTT_TOPIC_QOS'); + $R::send_to_serial_port = $p1decrypterCfg->param('P1DECRYPTER.SEND_TO_SERIAL_PORT'); $R::serial_output_port = $p1decrypterCfg->param('P1DECRYPTER.SERIAL_OUTPUT_PORT'); $R::serial_output_baudrate = $p1decrypterCfg->param('P1DECRYPTER.SERIAL_OUTPUT_BAUDRATE'); @@ -178,6 +208,40 @@ $template->param(SEND_TO_UDP => $cgi->popup_menu( ## UDP_HOST textfield $template->param(UDP_HOST => $cgi->textfield(-name => 'udp_host', -default => $R::udp_host)); +## SEND_MQTT switch +$template->param(SEND_MQTT => $cgi->popup_menu( + -name => 'send_mqtt', + -values => \@switch, + -labels => \%switchLabels, + -default => $R::send_mqtt +)); + +## MQTT_USE_GATEWAY switch +$template->param(MQTT_USE_GATEWAY => $cgi->popup_menu( + -name => 'mqtt_use_gateway', + -values => \@switch, + -labels => \%switchLabels, + -default => $R::mqtt_use_gateway +)); + +## MQTT_BROKER textfield +$template->param(MQTT_BROKER => $cgi->textfield(-name => 'mqtt_broker', -default => $R::mqtt_broker)); + +## MQTT_BROKER_USERNAME textfield +$template->param(MQTT_BROKER_USERNAME => $cgi->textfield(-name => 'mqtt_broker_username', -default => $R::mqtt_broker_username)); + +## MQTT_BROKER_PASSWORD password field +$template->param(MQTT_BROKER_PASSWORD => $cgi->password_field(-name => 'mqtt_broker_password', -default => $R::mqtt_broker_password)); + +## MQTT_BROKER_PORT textfield +$template->param(MQTT_BROKER_PORT => $cgi->textfield(-name => 'mqtt_broker_port', -default => $R::mqtt_broker_port)); + +## MQTT_TOPIC_PREFIX textfield +$template->param(MQTT_TOPIC_PREFIX => $cgi->textfield(-name => 'mqtt_topic_prefix', -default => $R::mqtt_topic_prefix)); + +## MQTT_TOPIC_QOS textfield +$template->param(MQTT_TOPIC_QOS => $cgi->textfield(-name => 'mqtt_topic_qos', -default => $R::mqtt_topic_qos)); + ## SEND_TO_SERIAL_PORT switch $template->param(SEND_TO_SERIAL_PORT => $cgi->popup_menu( -name => 'send_to_serial_port',