# -*- coding: utf-8 -*- # rpi-mqtt-monitor is a python script to monitor cpu load, temperature, frequency, free space etc. # on a Raspberry Pi or Ubuntu computer and publish the data to a MQTT server or Home Assistant API. from __future__ import division import subprocess import time from datetime import datetime import socket import paho.mqtt.client as paho import json import os import sys import argparse import threading import update import config import re import html import uuid import glob import requests import configparser import psutil #import external sensor lib only if one uses external sensors if config.ext_sensors: # append folder ext_sensor_lib sys.path.append(os.path.abspath(os.path.join(os.path.dirname(os.path.abspath(__file__)), '..', 'ext_sensor_lib'))) import ds18b20 from sht21 import SHT21 configlanguage = configparser.ConfigParser() configlanguage.read(os.path.join(os.path.dirname(os.path.abspath(__file__)), 'translations.ini')) def get_translation(key): """ get the correct translation""" return configlanguage.get(config.language, key, fallback=key) def check_wifi_signal(format): try: full_cmd = "ls /sys/class/ieee80211/*/device/net/" interface = subprocess.Popen(full_cmd, shell=True, stdout=subprocess.PIPE).communicate()[0].strip().decode("utf-8") full_cmd = "/sbin/iwconfig {} | grep -i quality".format(interface) wifi_signal = subprocess.Popen(full_cmd, shell=True, stdout=subprocess.PIPE).communicate()[0] if format == 'dbm': wifi_signal = wifi_signal.decode("utf-8").strip().split(' ')[4].split('=')[1] else: wifi_signal = wifi_signal.decode("utf-8").strip().split(' ')[1].split('=')[1].split('/')[0] wifi_signal = round((int(wifi_signal) / 70)* 100) except Exception: wifi_signal = None if config.use_availability else 0 return wifi_signal def check_used_space(path): st = os.statvfs(path) free_space = st.f_bavail * st.f_frsize total_space = st.f_blocks * st.f_frsize used_space = int(100 - ((free_space / total_space) * 100)) return used_space def check_cpu_load(): return psutil.cpu_percent(interval=1) def check_voltage(): try: full_cmd = "vcgencmd measure_volts | cut -f2 -d= | sed 's/000//'" voltage = subprocess.Popen(full_cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE).communicate()[0] voltage = voltage.strip()[:-1] except Exception: voltage = None if config.use_availability else 0 return voltage.decode('utf8') def check_swap(): full_cmd = "free | grep -i swap | awk 'NR == 1 {if($2 > 0) {print $3/$2*100} else {print 0}}'" swap = subprocess.Popen(full_cmd, shell=True, stdout=subprocess.PIPE).communicate()[0] swap = round(float(swap.decode("utf-8").replace(",", ".")), 1) return swap def check_memory(): full_cmd = 'free -b | awk \'NR==2 {printf "%.2f\\n", $3/$2 * 100}\'' memory = subprocess.Popen(full_cmd, shell=True, stdout=subprocess.PIPE).communicate()[0] if memory: memory = round(float(memory.decode("utf-8").replace(",", "."))) else: memory = 0 return memory def check_rpi_power_status(): try: full_cmd = "vcgencmd get_throttled | cut -d= -f2" throttled = subprocess.Popen(full_cmd, shell=True, stdout=subprocess.PIPE).communicate()[0] throttled = throttled.decode('utf-8').strip() if throttled == "0x0": return "OK" else: return "KO" except Exception as e: return "Error: " + str(e) def check_service_file_exists(): service_file_path = "/etc/systemd/system/rpi-mqtt-monitor.service" return os.path.exists(service_file_path) def check_crontab_entry(script_name="rpi-cpu1mqtt.py"): try: # Get the current user's crontab result = subprocess.run(['crontab', '-l'], stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True) # If the crontab command fails, it means there is no crontab for the user if result.returncode != 0: return False # Check if the script name is in the crontab output return script_name in result.stdout except Exception as e: print(f"Error checking crontab: {e}") return False def read_ext_sensors(): """ here we read the external sensors we create a list with the external sensors where we append the values """ # we copy the variable from the config file and replace the values by the real sensor values ext_sensors = config.ext_sensors # item[0] = name # item[1] = sensor_type # item[2] = ID # item[3] = value # now we iterate over the external sensors for item in config.ext_sensors: # if it is a DS18B20 sensor if item[1] == "ds18b20": # if sensor ID in unknown, then we try to get it # this only works for a single DS18B20 sensor if item[2]==0: item[2] = ds18b20.get_available_sensors()[0] temp = ds18b20.sensor_DS18B20(sensor_id=item[2]) item[3] = temp # in case that some error occurs during reading, we get -300 if temp==-300: print ("Error while reading sensor %s, %s" % (item[1], item[2])) if config.use_availability: item[3] = None if item[1] == "sht21": try: with SHT21(1) as sht21: temp = sht21.read_temperature() temp = '%2.1f' % temp hum = sht21.read_humidity() hum = '%2.1f' % hum item[3] = [temp, hum] # in case we have any problems to read the sensor, we continue and keep default values except Exception: print ("Error while reading sensor %s" % item[1]) if config.use_availability: item[3] = [None, None] #print (ext_sensors) return ext_sensors def check_cpu_temp(): try: temps = psutil.sensors_temperatures() if not temps: raise ValueError("No temperature sensors found.") if "cpu_thermal" in temps: cpu_temp = temps["cpu_thermal"][0].current elif "coretemp" in temps: cpu_temp = temps["coretemp"][0].current else: raise ValueError("CPU temperature sensor not found.") return round(cpu_temp, 2) except Exception as e: print(f"Error reading CPU temperature: {e}") return None if config.use_availability else 0 def check_sys_clock_speed(): full_cmd = "awk '{printf (\"%0.0f\",$1/1000); }' 0: for device, temp in drive_temps.items(): output += f" {device.capitalize()} Temp: {temp:.2f}°C\n" output += "\n:: Scheduling\n " if check_service_file_exists(): output += " Running as Service\n" elif check_crontab_entry(): output += " Running as Cron Job\n" output += """\n:: Installation directory :: {} :: Release notes {}: {}""".format(os.path.dirname(script_dir), remote_version, get_release_notes(remote_version)) print(output) def extract_text(html_string): html_string = html.unescape(html_string) text = re.sub('<[^<]+?>', '', html_string) return text def get_release_notes(version): url = "https://github.com/hjelev/rpi-mqtt-monitor/releases/tag/" + version try: response = subprocess.run(['curl', '-s', url], capture_output=True) release_notes = response.stdout.decode('utf-8').split("What's Changed")[1].split("")[0].replace("","").split("
")[0] except Exception: release_notes = None if config.use_availability else "No release notes available" lines = extract_text(release_notes).split('\n') lines = [" * "+ line for line in lines if line.strip() != ""] release_notes = '\n'.join(lines) if len(release_notes) > 255: release_notes = release_notes[:250] + " ..." return release_notes def config_json(what_config, device="0", hass_api=False): model_name = check_model_name() manufacturer = get_manufacturer() os = get_os() data = { "state_topic": "", "icon": "", "name": "", "unique_id": "", "device": { "identifiers": [hostname], "manufacturer": 'github.com/hjelev', "model": 'RPi MQTT Monitor ' + config.version, "name": hostname, "sw_version": os, "hw_version": model_name + " by " + manufacturer + " IP:" + get_network_ip(), "configuration_url": "https://github.com/hjelev/rpi-mqtt-monitor", "connections": [["mac", get_mac_address()]] } } data["state_topic"] = config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + what_config data["unique_id"] = hostname + "_" + what_config if what_config == "cpu_load": data["icon"] = "mdi:speedometer" data["name"] = get_translation("cpu_load") data["state_class"] = "measurement" data["unit_of_measurement"] = "%" elif what_config == "cpu_temp": data["icon"] = "hass:thermometer" data["name"] = get_translation("cpu_temperature") data["unit_of_measurement"] = "°C" data["device_class"] = "temperature" data["state_class"] = "measurement" elif what_config == "used_space": data["icon"] = "mdi:harddisk" data["name"] = get_translation("disk_usage") data["unit_of_measurement"] = "%" data["state_class"] = "measurement" elif what_config == "voltage": data["icon"] = "mdi:flash" data["name"] = get_translation("cpu_voltage") data["unit_of_measurement"] = "V" data["device_class"] = "voltage" data["state_class"] = "measurement" elif what_config == "swap": data["icon"] = "mdi:harddisk" data["name"] = get_translation("disk_swap") data["unit_of_measurement"] = "%" data["state_class"] = "measurement" elif what_config == "memory": data["icon"] = "mdi:memory" data["name"] = get_translation("memory_usage") data["unit_of_measurement"] = "%" data["state_class"] = "measurement" elif what_config == "sys_clock_speed": data["icon"] = "mdi:speedometer" data["name"] = get_translation("cpu_clock_speed") data["unit_of_measurement"] = "MHz" data["device_class"] = "frequency" data["state_class"] = "measurement" elif what_config == "uptime": data["icon"] = "mdi:calendar" data["name"] = get_translation("uptime") data["value_template"] = "{{ as_datetime(value) }}" data["device_class"] = "timestamp" elif what_config == "uptime_seconds": data["icon"] = "mdi:timer-outline" data["name"] = get_translation("uptime") data["unit_of_measurement"] = "s" data["device_class"] = "duration" data["state_class"] = "total_increasing" elif what_config == "wifi_signal": data["icon"] = "mdi:wifi" data["name"] = get_translation("wifi_signal") data["unit_of_measurement"] = "%" data["state_class"] = "measurement" data["device_class"] = "signal_strength" elif what_config == "wifi_signal_dbm": data["icon"] = "mdi:wifi" data["name"] = get_translation("wifi_signal_strength") data["unit_of_measurement"] = "dBm" data["device_class"] = "signal_strength" data["state_class"] = "measurement" elif what_config == "rpi5_fan_speed": data["icon"] = "mdi:fan" data["name"] = get_translation("fan_speed") data["unit_of_measurement"] = "RPM" data["state_class"] = "measurement" elif what_config == "status": data["icon"] = "mdi:lan-connect" data["name"] = get_translation("status") data["value_template"] = "{{ 'online' if value == '1' else 'offline' }}" elif what_config == "git_update": data["icon"] = "mdi:git" data["name"] = get_translation("rpi_mqtt_monitor") data["title"] = "Device Update" data["device_class"] = "update" data["state_class"] = "measurement" data["value_template"] = "{{ 'ON' if value_json.installed_ver != value_json.new_ver else 'OFF' }}" elif what_config == "update": version = update.check_git_version_remote(script_dir) data["icon"] = "mdi:update" data["name"] = get_translation("rpi_mqtt_monitor") data["title"] = "New Version" data["state_topic"] = config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + "git_update" data["value_template"] = "{{ {'installed_version': value_json.installed_ver, 'latest_version': value_json.new_ver } | to_json }}" data["device_class"] = "firmware" data["command_topic"] = config.mqtt_discovery_prefix + "/update/" + hostname + "/command" data["payload_install"] = "install" data['release_url'] = "https://github.com/hjelev/rpi-mqtt-monitor/releases/tag/" + version data['entity_picture'] = "https://raw.githubusercontent.com/hjelev/rpi-mqtt-monitor/refs/heads/master/images/update_icon.png" data['release_summary'] = get_release_notes(version) elif what_config == "restart_button": data["icon"] = "mdi:restart" data["name"] = get_translation("system_restart") data["command_topic"] = config.mqtt_discovery_prefix + "/update/" + hostname + "/command" data["payload_press"] = "restart" data["device_class"] = "restart" elif what_config == "shutdown_button": data["icon"] = "mdi:power" data["name"] = get_translation("system_shutdown") data["command_topic"] = config.mqtt_discovery_prefix + "/update/" + hostname + "/command" data["payload_press"] = "shutdown" data["device_class"] = "restart" elif what_config == "display_on": data["icon"] = "mdi:monitor" data["name"] = get_translation("monitor_on") data["command_topic"] = config.mqtt_discovery_prefix + "/update/" + hostname + "/command" data["payload_press"] = "display_on" data["device_class"] = "restart" elif what_config == "display_off": data["icon"] = "mdi:monitor" data["name"] = get_translation("monitor_off") data["command_topic"] = config.mqtt_discovery_prefix + "/update/" + hostname + "/command" data["payload_press"] = "display_off" data["device_class"] = "restart" elif what_config == device + "_temp": data["icon"] = "hass:thermometer" data["name"] = device + " " + get_translation("temperature") data["unit_of_measurement"] = "°C" data["device_class"] = "temperature" data["state_class"] = "measurement" elif what_config == "rpi_power_status": data["icon"] = "mdi:flash" data["name"] = get_translation("rpi_power_status") elif what_config == "apt_updates": data["icon"] = "mdi:update" data["name"] = get_translation("apt_updates") elif what_config == "ds18b20_status": data["icon"] = "hass:thermometer" data["name"] = device + " " + get_translation("temperature") data["unit_of_measurement"] = "°C" data["device_class"] = "temperature" data["state_class"] = "measurement" # we define again the state topic in order to get a unique state topic if we have two sensors of the same type data["state_topic"] = config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + what_config + "_" + device data["unique_id"] = hostname + "_" + what_config + "_" + device elif what_config == "sht21_temp_status": data["icon"] = "hass:thermometer" data["name"] = device + " " + get_translation("temperature") data["unit_of_measurement"] = "°C" data["device_class"] = "temperature" data["state_class"] = "measurement" # we define again the state topic in order to get a unique state topic if we have two sensors of the same type data["state_topic"] = config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + what_config + "_" + device data["unique_id"] = hostname + "_" + what_config + "_" + device elif what_config == "sht21_hum_status": data["icon"] = "mdi:water-percent" data["name"] = device + " " + get_translation("humidity") data["unit_of_measurement"] = "%" data["device_class"] = "temperature" data["state_class"] = "measurement" # we define again the state topic in order to get a unique state topic if we have two sensors of the same type data["state_topic"] = config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + what_config + "_" + device data["unique_id"] = hostname + "_" + what_config + "_" + device else: return "" # Return our built discovery config # If this is a "measurement" add expiry information and availability from config file # exclude "git_update", since that is running with a different update rate :( if "state_class" in data and data["state_class"] == "measurement" and what_config != "git_update": if config.expire_after_time: data["expire_after"] = config.expire_after_time if config.use_availability: data["availability_topic"] = data["state_topic"] + "_availability" if hass_api: result = { "name": data["name"], "icon": data["icon"], "state_class": data["state_class"], } if "unit_of_measurement" in data: result["unit_of_measurement"] = data["unit_of_measurement"] if "device_class" in data: result["device_class"] = data["device_class"] if "unique_id" in data: result["unique_id"] = data["unique_id"] if "value_template" in data: result["value_template"] = data["value_template"] return result return json.dumps(data) def create_mqtt_client(): def on_log(client, userdata, level, buf): if level == paho.MQTT_LOG_ERR: print("MQTT error: ", buf) def on_connect(client, userdata, flags, rc): if rc != 0: print("Error: Unable to connect to MQTT broker, return code:", rc) client = paho.Client(client_id="rpi-mqtt-monitor-" + hostname + str(int(time.time()))) client.username_pw_set(config.mqtt_user, config.mqtt_password) client.on_log = on_log client.on_connect = on_connect try: client.connect(config.mqtt_host, int(config.mqtt_port)) except Exception as e: print("Error connecting to MQTT broker:", e) return None return client def publish_update_status_to_mqtt(git_update, apt_updates): client = create_mqtt_client() if client is None: print("Error: Unable to connect to MQTT broker") return client.loop_start() if config.git_update: if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/binary_sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_git_update/config", config_json('git_update'), qos=config.qos) client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/git_update", git_update, qos=1, retain=config.retain) if config.update: if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/update/" + hostname + "/config", config_json('update'), qos=1) if config.apt_updates: if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_apt_updates/config", config_json('apt_updates'), qos=config.qos) client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/apt_updates", apt_updates, qos=config.qos, retain=config.retain) # Wait for all messages to be delivered while len(client._out_messages) > 0: time.sleep(0.1) client.loop() client.loop_stop() client.disconnect() def publish_to_hass_api(monitored_values): for param, value in monitored_values.items(): if value: if param == 'drive_temps' and isinstance(value, dict): for device, temp in value.items(): entity_id = f"sensor.{hostname.replace('-','_')}_{device}_temp" state = temp attributes = config_json(device + "_temp", device, True) send_sensor_data_to_home_assistant(entity_id, state, attributes) else: entity_id = f"sensor.{hostname.replace('-','_')}_{param}" state = value attributes = config_json(param, "0", True) send_sensor_data_to_home_assistant(entity_id, state, attributes) def send_sensor_data_to_home_assistant(entity_id, state, attributes): home_assistant_url = config.hass_host api_token = config.hass_token url = f"{home_assistant_url}/api/states/{entity_id}" headers = { "Authorization": f"Bearer {api_token}", "Content-Type": "application/json" } data = { "state": state, "attributes": attributes } response = requests.post(url, headers=headers, json=data) if response.status_code in [200, 201]: pass else: print(f"Failed to update {entity_id}: {response.status_code} - {response.text}") def publish_to_mqtt(monitored_values): client = create_mqtt_client() if client is None: return client.loop_start() non_standard_values = ['restart_button', 'shutdown_button', 'display_control', 'drive_temps', 'ext_sensors'] # Publish standard monitored values for key, value in monitored_values.items(): if key not in non_standard_values and key in config.__dict__ and config.__dict__[key]: if config.discovery_messages: client.publish(f"{config.mqtt_discovery_prefix}/sensor/{config.mqtt_topic_prefix}/{hostname}_{key}/config", config_json(key), qos=config.qos) if config.use_availability: client.publish(f"{config.mqtt_uns_structure}{config.mqtt_topic_prefix}/{hostname}/{key}_availability", 'offline' if value is None else 'online', qos=config.qos) client.publish(f"{config.mqtt_uns_structure}{config.mqtt_topic_prefix}/{hostname}/{key}", value, qos=config.qos, retain=config.retain) # Publish non standard values if config.restart_button: if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/button/" + config.mqtt_topic_prefix + "/" + hostname + "_restart/config", config_json('restart_button'), qos=config.qos) if config.shutdown_button: if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/button/" + config.mqtt_topic_prefix + "/" + hostname + "_shutdown/config", config_json('shutdown_button'), qos=config.qos) if config.display_control: if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/button/" + config.mqtt_topic_prefix + "/" + hostname + "_display_on/config", config_json('display_on'), qos=config.qos) client.publish(config.mqtt_discovery_prefix + "/button/" + config.mqtt_topic_prefix + "/" + hostname + "_display_off/config", config_json('display_off'), qos=config.qos) if config.drive_temps: for device, temp in monitored_values['drive_temps'].items(): if config.discovery_messages: client.publish(config.mqtt_discovery_prefix + "/sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_" + device + "_temp/config", config_json(device + "_temp", device), qos=config.qos) if config.use_availability: client.publish(f"{config.mqtt_uns_structure}{config.mqtt_topic_prefix}/{hostname}/{device}_temp_availability", 'offline' if temp is None else 'online', qos=config.qos) client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + device + "_temp", temp, qos=config.qos, retain=config.retain) if config.ext_sensors: # we loop through all sensors for item in monitored_values['ext_sensors']: # item[0] = name # item[1] = sensor_type # item[2] = ID # item[3] = value, like temperature or humidity if item[1] == "ds18b20": if config.discovery_messages: client.publish( config.mqtt_discovery_prefix + "/sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_" + item[0] + "_status/config", config_json('ds18b20_status', device=item[0]), qos=config.qos) if config.use_availability: client.publish(f"{config.mqtt_uns_structure}{config.mqtt_topic_prefix}/{hostname}/ds18b20_status_{item[0]}_availability", 'offline' if item[3] is None else 'online', qos=config.qos) client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + "ds18b20_status_" + item[0], item[3], qos=config.qos, retain=config.retain) if item[1] == "sht21": if config.discovery_messages: # temperature client.publish( config.mqtt_discovery_prefix + "/sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_" + item[0] + "_temp_status/config", config_json('sht21_temp_status', device=item[0]), qos=config.qos) # humidity client.publish( config.mqtt_discovery_prefix + "/sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_" + item[0] + "_hum_status/config", config_json('sht21_hum_status', device=item[0]), qos=config.qos) if config.use_availability: client.publish(f"{config.mqtt_uns_structure}{config.mqtt_topic_prefix}/{hostname}/sht21_temp_status_{item[0]}_availability", 'offline' if item[3][0] is None else 'online', qos=config.qos) client.publish(f"{config.mqtt_uns_structure}{config.mqtt_topic_prefix}/{hostname}/sht21_hum_status_{item[0]}_availability", 'offline' if item[3][1] is None else 'online', qos=config.qos) # temperature client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + "sht21_temp_status_" + item[0], item[3][0], qos=config.qos, retain=config.retain) # humidity client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/" + "sht21_hum_status_" + item[0], item[3][1], qos=config.qos, retain=config.retain) status_sensor_topic = config.mqtt_discovery_prefix + "/sensor/" + config.mqtt_topic_prefix + "/" + hostname + "_status/config" client.publish(status_sensor_topic, config_json('status'), qos=config.qos) client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/status", "1", qos=config.qos, retain=config.retain) while len(client._out_messages) > 0: time.sleep(0.1) client.loop() client.loop_stop() client.disconnect() def bulk_publish_to_mqtt(monitored_values): values = [monitored_values.get(key, 0) for key in [ 'cpu_load', 'cpu_temp', 'used_space', 'voltage', 'sys_clock_speed', 'swap', 'memory', 'uptime', 'uptime_seconds', 'wifi_signal', 'wifi_signal_dbm', 'rpi5_fan_speed', 'git_update', 'rpi_power_status' ]] ext_sensors = monitored_values.get('ext_sensors', []) values.extend(sensor[3] for sensor in ext_sensors) values_str = ', '.join(map(str, values)) client = create_mqtt_client() if client is None: return client.loop_start() client.publish(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname, values_str, qos=config.qos, retain=config.retain) while len(client._out_messages) > 0: time.sleep(0.1) client.loop() client.loop_stop() client.disconnect() def parse_arguments(): parser = argparse.ArgumentParser( prog='rpi-mqtt-monitor', description='Monitor CPU load, temperature, frequency, free space, etc., and publish the data to an MQTT server or Home Assistant API.' ) parser.add_argument('-H', '--hass_api', action='store_true', help='send readings via Home Assistant API (not via MQTT)', default=False) parser.add_argument('-d', '--display', action='store_true', help='display values on screen', default=False) parser.add_argument('-s', '--service', action='store_true', help='run script as a service, sleep interval is configurable in config.py', default=False) parser.add_argument('-v', '--version', action='store_true', help='display installed version and exit', default=False) parser.add_argument('-u', '--update', action='store_true', help='update script and config then exit', default=False) parser.add_argument('-w', '--hass_wake', action='store_true', help='display Home assistant wake on lan configuration', default=False) args = parser.parse_args() if args.update: version = update.check_git_version_remote(script_dir).strip() update.do_update(script_dir, version, True) exit() if args.version: installed_version = config.version latest_versino = update.check_git_version_remote(script_dir).strip() print("Installed version: " + installed_version) print("Latest version: " + latest_versino) if installed_version != latest_versino: print("Update available") else: print("No update available") exit() if args.hass_wake: hass_config = """Add this to your Home Assistant switches.yaml file: - platform: wake_on_lan mac: "{}" host: "{}" name: "{}-switch" turn_off: service: mqtt.publish data: topic: config.mqtt_discovery_prefix + "/update/{}/command" payload: "shutdown" """.format(get_mac_address(), get_network_ip(), hostname, hostname ) print(hass_config) exit() return args def collect_monitored_values(): monitored_values = {} if config.cpu_load: cpu_load = check_cpu_load() monitored_values["cpu_load"] = cpu_load if config.cpu_temp: cpu_temp = check_cpu_temp() monitored_values["cpu_temp"] = cpu_temp if config.used_space: used_space = check_used_space(config.used_space_path) monitored_values["used_space"] = used_space if config.voltage: voltage = check_voltage() monitored_values["voltage"] = voltage if config.sys_clock_speed: sys_clock_speed = check_sys_clock_speed() monitored_values["sys_clock_speed"] = sys_clock_speed if config.swap: swap = check_swap() monitored_values["swap"] = swap if config.memory: memory = check_memory() monitored_values["memory"] = memory if config.uptime: uptime = check_uptime('timestamp') monitored_values["uptime"] = uptime if config.uptime_seconds: uptime_seconds = check_uptime('') monitored_values["uptime_seconds"] = uptime_seconds if config.wifi_signal: wifi_signal = check_wifi_signal('') monitored_values["wifi_signal"] = wifi_signal if config.wifi_signal_dbm: wifi_signal_dbm = check_wifi_signal('dbm') monitored_values["wifi_signal_dbm"] = wifi_signal_dbm if config.rpi5_fan_speed: rpi5_fan_speed = check_rpi5_fan_speed() monitored_values["rpi5_fan_speed"] = rpi5_fan_speed if config.drive_temps: drive_temps = check_all_drive_temps() monitored_values["drive_temps"] = drive_temps if config.rpi_power_status: rpi_power_status = check_rpi_power_status() monitored_values["rpi_power_status"] = rpi_power_status if config.ext_sensors: ext_sensors = read_ext_sensors() monitored_values["ext_sensors"] = ext_sensors return monitored_values def gather_and_send_info(): while not stop_event.is_set(): monitored_values = collect_monitored_values() if hasattr(config, 'random_delay'): time.sleep(config.random_delay) if args.display: print_measured_values(monitored_values) # write some output to a file if config.output_filename: # the only options are "a" for append or "w" for (over)write # check if one of this options is defined if config.output_mode not in ["a", "w"]: print("Error, output_type not known. Default w is set.") config.output_type = "w" try: # open the text file output_file = open(config.output_filename, config.output_mode) # read what should be written into the textfile # we need to define this is a function, otherwise the values are not updated and default values are taken output_content = config.get_content_outputfile() output_file.write(output_content) output_file.close() except Exception as e: print("Error writing to output file:", e) if args.hass_api: publish_to_hass_api(monitored_values) else: if hasattr(config, 'group_messages') and config.group_messages: bulk_publish_to_mqtt(monitored_values) else: publish_to_mqtt(monitored_values) if not args.service: break # Break the sleep into 1-second intervals and check stop_event after each interval for _ in range(config.service_sleep_time): if stop_event.is_set(): break time.sleep(1) def update_status(): while not stop_event.is_set(): git_update = check_git_update(script_dir) apt_updates = get_apt_updates() publish_update_status_to_mqtt(git_update, apt_updates) stop_event.wait(config.update_check_interval) if stop_event.is_set(): break def on_message(client, userdata, msg): global exit_flag, thread1, thread2 print("Received message: ", msg.payload.decode()) if msg.payload.decode() == "install": def update_and_exit(): version = update.check_git_version_remote(script_dir).strip() update.do_update(script_dir, version, git_update=True, config_update=True) print("Update completed. Stopping MQTT client loop...") client.loop_stop() # Stop the MQTT client loop print("Setting exit flag...") exit_flag = True stop_event.set() # Signal the threads to stop if thread1 is not None: thread1.join() # Wait for thread1 to finish if thread2 is not None: thread2.join() # Wait for thread2 to finish os._exit(0) # Exit the script immediately update_thread = threading.Thread(target=update_and_exit) update_thread.start() elif msg.payload.decode() == "restart": print("Restarting the system...") os.system("sudo reboot") elif msg.payload.decode() == "shutdown": print("Shutting down the system...") os.system("sudo shutdown now") elif msg.payload.decode() == "display_off": print("Turn off display") os.system('su -l {} -c "xset -display :0 dpms force off"'.format(config.os_user)) elif msg.payload.decode() == "display_on": print("Turn on display") os.system('su -l {} -c "xset -display :0 dpms force on"'.format(config.os_user)) exit_flag = False stop_event = threading.Event() script_dir = os.path.dirname(os.path.realpath(__file__)) # get device host name - used in mqtt topic # and adhere to the allowed character set if hasattr(config, 'ha_device_name') and config.ha_device_name: hostname = re.sub(r'[^a-zA-Z0-9_-]', '_', config.ha_device_name) else: hostname = re.sub(r'[^a-zA-Z0-9_-]', '_', socket.gethostname()) if __name__ == '__main__': args = parse_arguments(); if args.service: if not args.hass_api: client = paho.Client() client.username_pw_set(config.mqtt_user, config.mqtt_password) client.on_message = on_message # set will_set to send a message when the client disconnects client.will_set(config.mqtt_uns_structure + config.mqtt_topic_prefix + "/" + hostname + "/status", "0", qos=config.qos, retain=config.retain) try: client.connect(config.mqtt_host, int(config.mqtt_port)) except Exception as e: print("Error connecting to MQTT broker:", e) sys.exit(1) client.subscribe(config.mqtt_discovery_prefix + "/update/" + hostname + "/command") print("Listening to topic : " + config.mqtt_discovery_prefix + "/update/" + hostname + "/command") client.loop_start() thread1 = threading.Thread(target=gather_and_send_info) thread1.daemon = True # Set thread1 as a daemon thread thread1.start() if not args.hass_api: if config.update: thread2 = threading.Thread(target=update_status) thread2.daemon = True # Set thread2 as a daemon thread thread2.start() try: while True: time.sleep(1) # Check the exit flag every second except KeyboardInterrupt: print(" Ctrl+C pressed. Setting exit flag...") client.loop_stop() exit_flag = True stop_event.set() # Signal the threads to stop sys.exit(0) # Exit the script else: gather_and_send_info()