2019-08-05 20:51:20 +02:00
|
|
|
import logging
|
2021-08-12 18:09:15 +02:00
|
|
|
import os
|
2021-09-21 16:20:44 +02:00
|
|
|
import shutil
|
2024-01-12 17:57:34 +01:00
|
|
|
import pkgutil
|
2021-09-21 16:20:44 +02:00
|
|
|
import psutil
|
|
|
|
import pathlib
|
2021-12-27 18:11:40 +01:00
|
|
|
import json
|
2019-08-05 20:51:20 +02:00
|
|
|
import aiohttp
|
2021-09-21 16:20:44 +02:00
|
|
|
from voluptuous.schema_builder import message
|
|
|
|
from cbpi.api.dataclasses import NotificationAction, NotificationType
|
|
|
|
from cbpi.api.base import CBPiBase
|
|
|
|
from cbpi.api.config import ConfigType
|
|
|
|
from cbpi.api import *
|
|
|
|
import zipfile
|
|
|
|
import socket
|
2024-01-12 17:57:34 +01:00
|
|
|
import importlib
|
|
|
|
from tabulate import tabulate
|
2024-05-10 21:29:53 +02:00
|
|
|
from datetime import datetime, timedelta, date
|
2024-05-10 22:16:18 +02:00
|
|
|
import glob
|
2024-05-01 19:25:56 +02:00
|
|
|
|
|
|
|
try:
|
|
|
|
from systemd import journal
|
|
|
|
systemd_available=True
|
|
|
|
except Exception:
|
2024-05-02 17:39:43 +02:00
|
|
|
logging.warning("Failed to load systemd library. logfile download not available")
|
2024-05-01 19:25:56 +02:00
|
|
|
systemd_available=False
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2019-08-05 20:51:20 +02:00
|
|
|
class SystemController:
|
2018-11-01 19:50:04 +01:00
|
|
|
|
2018-11-01 21:25:42 +01:00
|
|
|
def __init__(self, cbpi):
|
|
|
|
self.cbpi = cbpi
|
|
|
|
self.service = cbpi.actor
|
2019-08-05 20:51:20 +02:00
|
|
|
self.logger = logging.getLogger(__name__)
|
2024-06-01 19:14:56 +02:00
|
|
|
self.logsFolderPath = self.cbpi.config_folder.logsFolderPath
|
2019-07-31 07:58:54 +02:00
|
|
|
|
|
|
|
self.cbpi.app.on_startup.append(self.check_for_update)
|
|
|
|
|
|
|
|
|
|
|
|
async def check_for_update(self, app):
|
2021-02-16 20:37:51 +01:00
|
|
|
pass
|
2019-01-28 22:21:31 +01:00
|
|
|
|
2021-08-12 18:09:15 +02:00
|
|
|
async def restart(self):
|
|
|
|
logging.info("RESTART")
|
|
|
|
os.system('systemctl reboot')
|
|
|
|
pass
|
2019-01-28 22:21:31 +01:00
|
|
|
|
2021-08-12 18:09:15 +02:00
|
|
|
async def shutdown(self):
|
|
|
|
logging.info("SHUTDOWN")
|
|
|
|
os.system('systemctl poweroff')
|
|
|
|
pass
|
2021-09-21 16:20:44 +02:00
|
|
|
|
|
|
|
async def backupConfig(self):
|
2024-05-10 22:16:18 +02:00
|
|
|
files=glob.glob('*cbpi4_config*.zip')
|
|
|
|
for f in files:
|
|
|
|
try:
|
|
|
|
os.remove(f)
|
|
|
|
except Exception as e:
|
|
|
|
logging.error("Cannot remove old config backup: {}".format(e))
|
|
|
|
|
2024-05-10 21:29:53 +02:00
|
|
|
try:
|
|
|
|
current_date = date.today()
|
|
|
|
current_date=str(current_date).replace("-","_")
|
|
|
|
output_filename = current_date+"_cbpi4_config"
|
|
|
|
except:
|
|
|
|
output_filename = "cbpi4_config"
|
2022-02-19 11:33:11 +00:00
|
|
|
dir_name = pathlib.Path(self.cbpi.config_folder.get_file_path(''))
|
2021-09-21 16:20:44 +02:00
|
|
|
shutil.make_archive(output_filename, 'zip', dir_name)
|
2024-05-10 21:29:53 +02:00
|
|
|
return output_filename+".zip"
|
2021-09-21 16:20:44 +02:00
|
|
|
|
2024-01-12 17:57:34 +01:00
|
|
|
async def plugins_list(self):
|
|
|
|
result = []
|
|
|
|
discovered_plugins = {
|
|
|
|
name: importlib.import_module(name)
|
|
|
|
for finder, name, ispkg
|
|
|
|
in pkgutil.iter_modules()
|
|
|
|
if name.startswith('cbpi') and len(name) > 4
|
|
|
|
}
|
|
|
|
for key, module in discovered_plugins.items():
|
|
|
|
from importlib.metadata import version
|
|
|
|
try:
|
|
|
|
from importlib.metadata import (distribution, metadata,
|
|
|
|
version)
|
|
|
|
meta = metadata(key)
|
|
|
|
result.append(dict(Name=meta["Name"], Version=meta["Version"], Author=meta["Author"], Homepage=meta["Home-page"], Summary=meta["Summary"]))
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
return tabulate(result, headers="keys")
|
|
|
|
|
2024-05-01 19:25:56 +02:00
|
|
|
async def downloadlog(self, logtime):
|
|
|
|
filename = "cbpi4.log"
|
|
|
|
fullname = pathlib.Path(os.path.join(".",filename))
|
|
|
|
pluginname = "cbpi4_plugins.txt"
|
|
|
|
fullpluginname = pathlib.Path(os.path.join(".",pluginname))
|
|
|
|
actorname = "cbpi4_actors.txt"
|
|
|
|
fullactorname = pathlib.Path(os.path.join(".",actorname))
|
|
|
|
sensorname = "cbpi4_sensors.txt"
|
|
|
|
fullsensorname = pathlib.Path(os.path.join(".",sensorname))
|
|
|
|
kettlename = "cbpi4_kettles.txt"
|
|
|
|
fullkettlename = pathlib.Path(os.path.join(".",kettlename))
|
|
|
|
|
|
|
|
output_filename="cbpi4_log.zip"
|
2024-05-02 12:45:44 +02:00
|
|
|
result=[]
|
|
|
|
if systemd_available:
|
|
|
|
j = journal.Reader()
|
|
|
|
if logtime == "b":
|
|
|
|
j.this_boot()
|
|
|
|
else:
|
2024-05-01 19:25:56 +02:00
|
|
|
since = datetime.now() - timedelta(hours=int(logtime))
|
|
|
|
j.seek_realtime(since)
|
2024-05-02 12:45:44 +02:00
|
|
|
j.add_match(_SYSTEMD_UNIT="craftbeerpi.service")
|
|
|
|
|
|
|
|
for entry in j:
|
|
|
|
result.append(entry['MESSAGE'])
|
2024-06-01 19:14:56 +02:00
|
|
|
else:
|
|
|
|
try:
|
|
|
|
logfilename=pathlib.Path(self.logsFolderPath+"/"+"cbpi.log")
|
|
|
|
with open(logfilename) as f:
|
|
|
|
for line in f:
|
|
|
|
result.append(line.rstrip('\n'))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2024-05-02 12:45:44 +02:00
|
|
|
try:
|
|
|
|
with open(fullname, 'w') as f:
|
|
|
|
for line in result:
|
|
|
|
f.write(f"{line}\n")
|
|
|
|
except Exception as e:
|
|
|
|
logging.error(e)
|
2024-05-01 19:25:56 +02:00
|
|
|
|
|
|
|
plugins = await self.plugins_list()
|
|
|
|
with open(fullpluginname, 'w') as f:
|
|
|
|
f.write(plugins)
|
|
|
|
|
|
|
|
try:
|
|
|
|
actors = self.cbpi.actor.get_state()
|
|
|
|
json.dump(actors['data'],open(fullactorname,'w'),indent=4, sort_keys=True)
|
|
|
|
sensors = self.cbpi.sensor.get_state()
|
|
|
|
json.dump(sensors['data'],open(fullsensorname,'w'),indent=4, sort_keys=True)
|
|
|
|
kettles = self.cbpi.kettle.get_state()
|
2021-12-27 18:11:40 +01:00
|
|
|
json.dump(kettles['data'],open(fullkettlename,'w'),indent=4, sort_keys=True)
|
|
|
|
except Exception as e:
|
|
|
|
logging.info(e)
|
|
|
|
self.cbpi.notify("Error", "Creation of files failed: {}".format(e), NotificationType.ERROR)
|
|
|
|
|
|
|
|
try:
|
|
|
|
zipObj=zipfile.ZipFile(output_filename , 'w', zipfile.ZIP_DEFLATED)
|
|
|
|
zipObj.write(fullname)
|
|
|
|
zipObj.write(fullpluginname)
|
|
|
|
zipObj.write(fullactorname)
|
|
|
|
zipObj.write(fullsensorname)
|
|
|
|
zipObj.write(fullkettlename)
|
|
|
|
zipObj.close()
|
|
|
|
except Exception as e:
|
|
|
|
logging.info(e)
|
|
|
|
self.cbpi.notify("Error", "Zip creation failed: {}".format(e), NotificationType.ERROR)
|
|
|
|
|
|
|
|
try:
|
|
|
|
os.remove(fullname)
|
|
|
|
os.remove(fullpluginname)
|
|
|
|
os.remove(fullactorname)
|
|
|
|
os.remove(fullsensorname)
|
|
|
|
os.remove(fullkettlename)
|
|
|
|
except Exception as e:
|
|
|
|
logging.info(e)
|
|
|
|
self.cbpi.notify("Error", "Removal of original files failed: {}".format(e), NotificationType.ERROR)
|
2021-12-19 18:00:15 +01:00
|
|
|
|
|
|
|
|
2021-09-21 16:20:44 +02:00
|
|
|
def allowed_file(self, filename, extension):
|
|
|
|
return '.' in filename and filename.rsplit('.', 1)[1] in set([extension])
|
|
|
|
|
|
|
|
def recursive_chown(self, path, owner, group):
|
|
|
|
for dirpath, dirnames, filenames in os.walk(path):
|
|
|
|
shutil.chown(dirpath, owner, group)
|
|
|
|
for filename in filenames:
|
|
|
|
shutil.chown(os.path.join(dirpath, filename), owner, group)
|
|
|
|
|
|
|
|
async def restoreConfig(self, data):
|
|
|
|
fileData = data['File']
|
|
|
|
filename = fileData.filename
|
|
|
|
backup_file = fileData.file
|
|
|
|
content_type = fileData.content_type
|
|
|
|
required_content=['dashboard/', 'recipes/', 'upload/', 'config.json', 'config.yaml']
|
|
|
|
|
|
|
|
if content_type == 'application/x-zip-compressed':
|
|
|
|
try:
|
|
|
|
content = backup_file.read()
|
|
|
|
if backup_file and self.allowed_file(filename, 'zip'):
|
2022-09-12 21:54:51 +02:00
|
|
|
self.path = os.path.join(self.cbpi.config_folder.configFolderPath, "restored_config.zip")
|
2022-09-03 13:43:17 +02:00
|
|
|
|
2021-09-21 16:20:44 +02:00
|
|
|
f=open(self.path, "wb")
|
|
|
|
f.write(content)
|
|
|
|
f.close()
|
|
|
|
zip=zipfile.ZipFile(self.path)
|
|
|
|
zip_content_list = zip.namelist()
|
|
|
|
zip_content = True
|
|
|
|
for content in required_content:
|
|
|
|
try:
|
|
|
|
check = zip_content_list.index(content)
|
|
|
|
except:
|
|
|
|
zip_content = False
|
|
|
|
if zip_content == True:
|
|
|
|
self.cbpi.notify("Success", "Config backup has been uploaded", NotificationType.SUCCESS)
|
2021-09-22 12:14:49 +02:00
|
|
|
self.cbpi.notify("Action Required!", "Please restart the server", NotificationType.WARNING)
|
2021-09-21 16:20:44 +02:00
|
|
|
else:
|
|
|
|
self.cbpi.notify("Error", "Wrong content type. Upload failed", NotificationType.ERROR)
|
2021-09-22 12:14:49 +02:00
|
|
|
os.remove(self.path)
|
2021-09-21 16:20:44 +02:00
|
|
|
except:
|
|
|
|
self.cbpi.notify("Error", "Config backup upload failed", NotificationType.ERROR)
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
self.cbpi.notify("Error", "Wrong content type. Upload failed", NotificationType.ERROR)
|
|
|
|
|
2021-11-09 18:32:00 +01:00
|
|
|
async def uploadSVG(self, data):
|
|
|
|
fileData = data['File']
|
|
|
|
filename = fileData.filename
|
|
|
|
svg_file = fileData.file
|
|
|
|
content_type = fileData.content_type
|
|
|
|
|
|
|
|
logging.info(content_type)
|
|
|
|
|
|
|
|
if content_type == 'image/svg+xml':
|
|
|
|
try:
|
|
|
|
content = svg_file.read().decode('utf-8','replace')
|
|
|
|
if svg_file and self.allowed_file(filename, 'svg'):
|
2022-02-19 11:33:11 +00:00
|
|
|
self.path = os.path.join(self.cbpi.config_folder.get_file_path("dashboard"),"widgets", filename)
|
2021-11-09 18:32:00 +01:00
|
|
|
logging.info(self.path)
|
|
|
|
|
|
|
|
f=open(self.path, "w")
|
|
|
|
f.write(content)
|
|
|
|
f.close()
|
2021-11-11 11:22:13 +01:00
|
|
|
self.cbpi.notify("Success", "SVG file ({}) has been uploaded.".format(filename), NotificationType.SUCCESS)
|
2021-11-09 18:32:00 +01:00
|
|
|
except:
|
|
|
|
self.cbpi.notify("Error", "SVG upload failed", NotificationType.ERROR)
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
self.cbpi.notify("Error", "Wrong content type. Upload failed", NotificationType.ERROR)
|
|
|
|
|
2021-09-21 16:20:44 +02:00
|
|
|
async def systeminfo(self):
|
|
|
|
logging.info("SYSTEMINFO")
|
|
|
|
system = ""
|
|
|
|
temp = 0
|
|
|
|
cpuload = 0
|
|
|
|
cpucount = 0
|
|
|
|
cpufreq = 0
|
|
|
|
totalmem = 0
|
|
|
|
availmem = 0
|
|
|
|
mempercent = 0
|
|
|
|
eth0IP = "N/A"
|
|
|
|
wlan0IP = "N/A"
|
2022-03-08 14:25:00 +01:00
|
|
|
eth0speed = "N/A"
|
|
|
|
wlan0speed = "N/A"
|
2021-09-21 16:20:44 +02:00
|
|
|
|
|
|
|
TEMP_UNIT=self.cbpi.config.get("TEMP_UNIT", "C")
|
|
|
|
FAHRENHEIT = False if TEMP_UNIT == "C" else True
|
|
|
|
|
|
|
|
af_map = { socket.AF_INET: 'IPv4',
|
|
|
|
socket.AF_INET6: 'IPv6',
|
|
|
|
}
|
|
|
|
|
|
|
|
try:
|
|
|
|
if psutil.LINUX == True:
|
|
|
|
system = "Linux"
|
|
|
|
elif psutil.WINDOWS == True:
|
|
|
|
system = "Windows"
|
|
|
|
elif psutil.MACOS == True:
|
|
|
|
system = "MacOS"
|
|
|
|
cpuload = round(psutil.cpu_percent(interval=None),1)
|
|
|
|
cpucount = psutil.cpu_count(logical=False)
|
|
|
|
cpufreq = psutil.cpu_freq()
|
|
|
|
mem = psutil.virtual_memory()
|
|
|
|
availmem = round((int(mem.available) / (1024*1024)),1)
|
|
|
|
mempercent = round(float(mem.percent),1)
|
|
|
|
totalmem = round((int(mem.total) / (1024*1024)),1)
|
|
|
|
if system == "Linux":
|
|
|
|
try:
|
|
|
|
temps = psutil.sensors_temperatures(fahrenheit=FAHRENHEIT)
|
|
|
|
for name, entries in temps.items():
|
|
|
|
for entry in entries:
|
|
|
|
if name == "cpu_thermal":
|
|
|
|
temp = round(float(entry.current),1)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
temp = "N/A"
|
|
|
|
if system == "Linux":
|
|
|
|
try:
|
|
|
|
ethernet = psutil.net_if_addrs()
|
|
|
|
for nic, addrs in ethernet.items():
|
|
|
|
if nic == "eth0":
|
|
|
|
for addr in addrs:
|
2023-11-03 06:55:42 +01:00
|
|
|
if str(addr.family) == "AddressFamily.AF_INET" or str(addr.family) == "2":
|
2021-09-21 16:20:44 +02:00
|
|
|
if addr.address:
|
|
|
|
eth0IP = addr.address
|
|
|
|
if nic == "wlan0":
|
|
|
|
for addr in addrs:
|
2023-11-03 06:55:42 +01:00
|
|
|
if str(addr.family) == "AddressFamily.AF_INET" or str(addr.family) == "2":
|
2021-09-21 16:20:44 +02:00
|
|
|
if addr.address:
|
|
|
|
wlan0IP = addr.address
|
2022-03-08 14:25:00 +01:00
|
|
|
info = psutil.net_if_stats()
|
|
|
|
try:
|
|
|
|
for nic in info:
|
|
|
|
if nic == 'eth0':
|
2022-03-08 19:43:37 +01:00
|
|
|
if info[nic].isup == True:
|
|
|
|
if info[nic].speed:
|
|
|
|
eth0speed = info[nic].speed
|
|
|
|
else:
|
|
|
|
eth0speed = "down"
|
2022-03-08 14:25:00 +01:00
|
|
|
if nic == 'wlan0':
|
2022-03-09 08:25:17 +01:00
|
|
|
if info[nic].isup == True:
|
|
|
|
ratestring = os.popen('iwlist wlan0 rate | grep Rate').read()
|
|
|
|
start = ratestring.find("=") + 1
|
|
|
|
end = ratestring.find(" Mb/s")
|
|
|
|
wlan0speed = ratestring[start:end]
|
2022-03-08 19:43:37 +01:00
|
|
|
else:
|
|
|
|
wlan0speed = "down"
|
2022-03-08 14:25:00 +01:00
|
|
|
except Exception as e:
|
|
|
|
logging.info(e)
|
2021-09-21 16:20:44 +02:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2021-11-09 18:32:00 +01:00
|
|
|
if system == "Windows":
|
|
|
|
try:
|
2022-03-08 14:25:00 +01:00
|
|
|
ethernet = psutil.net_if_addrs()
|
2021-11-09 18:32:00 +01:00
|
|
|
for nic, addrs in ethernet.items():
|
|
|
|
if nic == "Ethernet":
|
|
|
|
for addr in addrs:
|
|
|
|
if str(addr.family) == "AddressFamily.AF_INET":
|
|
|
|
if addr.address:
|
|
|
|
eth0IP = addr.address
|
|
|
|
if nic == "WLAN":
|
|
|
|
for addr in addrs:
|
|
|
|
if str(addr.family) == "AddressFamily.AF_INET":
|
|
|
|
if addr.address:
|
|
|
|
wlan0IP = addr.address
|
2022-03-08 14:25:00 +01:00
|
|
|
info = psutil.net_if_stats()
|
|
|
|
try:
|
|
|
|
for nic in info:
|
|
|
|
if nic == 'Ethernet':
|
2022-03-08 19:43:37 +01:00
|
|
|
if info[nic].isup == True:
|
|
|
|
if info[nic].speed:
|
|
|
|
eth0speed = info[nic].speed
|
|
|
|
else:
|
|
|
|
eth0speed = "down"
|
2022-03-08 14:25:00 +01:00
|
|
|
if nic == 'WLAN':
|
2022-03-08 19:43:37 +01:00
|
|
|
if info[nic].isup == True:
|
|
|
|
if info[nic].speed:
|
|
|
|
wlan0speed = info[nic].speed
|
|
|
|
else:
|
|
|
|
wlan0speed = "down"
|
2022-03-08 14:25:00 +01:00
|
|
|
except Exception as e:
|
|
|
|
logging.info(e)
|
2021-11-09 18:32:00 +01:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2021-09-21 16:20:44 +02:00
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
systeminfo = {'system': system,
|
|
|
|
'cpuload': cpuload,
|
|
|
|
'cpucount': cpucount,
|
|
|
|
'cpufreq': cpufreq.current,
|
|
|
|
'totalmem': totalmem,
|
|
|
|
'availmem': availmem,
|
|
|
|
'mempercent': mempercent,
|
|
|
|
'temp': temp,
|
|
|
|
'temp_unit': TEMP_UNIT,
|
|
|
|
'eth0': eth0IP,
|
2022-03-08 14:25:00 +01:00
|
|
|
'wlan0': wlan0IP,
|
|
|
|
'eth0speed': eth0speed,
|
|
|
|
'wlan0speed': wlan0speed}
|
2021-09-21 16:20:44 +02:00
|
|
|
return systeminfo
|
|
|
|
|
|
|
|
|