201 lines
7.5 KiB
Python
Executable File
201 lines
7.5 KiB
Python
Executable File
#!/bin/python3
|
|
# Importing the library
|
|
import psutil
|
|
import argparse
|
|
import time
|
|
import subprocess
|
|
import json
|
|
from pathlib import Path
|
|
from typing import Dict, List
|
|
from karanage import KaranageState, KaranageConnection, KaranageException, StateSystem
|
|
|
|
cpu_core_count = psutil.cpu_count(logical=False)
|
|
cpu_thread_count = psutil.cpu_count()
|
|
|
|
def get_mounts() -> Dict:
|
|
process = subprocess.Popen(
|
|
["mount", "-v"], stdout=subprocess.PIPE, stderr=subprocess.PIPE
|
|
)
|
|
stdout, stderr = process.communicate()
|
|
"""get data format:
|
|
$ mount -v
|
|
proc on /proc type proc (rw,nosuid,nodev,noexec,relatime)
|
|
sys on /sys type sysfs (rw,nosuid,nodev,noexec,relatime)
|
|
dev on /dev type devtmpfs (rw,nosuid,relatime,size=7049156k,nr_inodes=1762289,mode=755,inode64)
|
|
run on /run type tmpfs (rw,nosuid,nodev,relatime,mode=755,inode64)
|
|
efivarfs on /sys/firmware/efi/efivars type efivarfs (rw,nosuid,nodev,noexec,relatime)
|
|
/dev/nvme0n1p5 on / type ext4 (rw,relatime,stripe=32)
|
|
"""
|
|
out = {}
|
|
lines = stdout.decode("utf-8").split("\n")
|
|
for line in lines:
|
|
tmp = line.split(" ")
|
|
if len(tmp) <= 2:
|
|
continue
|
|
if tmp[0].startswith('/dev/'):
|
|
out[tmp[2]] = tmp[0][5:]
|
|
return out
|
|
|
|
mounting_points = get_mounts();
|
|
|
|
def filter(data: Dict, filter_list: List[str]) -> Dict:
|
|
out = {}
|
|
#print(f"Request filter {data.keys()} with filter {filter_list}")
|
|
for key in data:
|
|
if key in filter_list:
|
|
out[key] = data[key]
|
|
return out
|
|
|
|
def need_process(data: Dict) -> Dict:
|
|
return configuration["cpu"] == "auto" or "include" in configuration["cpu"]
|
|
|
|
|
|
def create_cpu_data() -> Dict:
|
|
# scpufreq(current=1605.5205625000003, min=400.0, max=4372.0)
|
|
cpu_frequency = psutil.cpu_freq(percpu=False)
|
|
cpu_percent_use = psutil.cpu_percent()
|
|
return {
|
|
"core": cpu_core_count,
|
|
"thread": cpu_thread_count,
|
|
"frequency": cpu_frequency.current,
|
|
"use": cpu_percent_use,
|
|
"max": cpu_frequency.max,
|
|
}
|
|
|
|
def create_memory_data() -> Dict:
|
|
# svmem(total=14473519104, available=8289726464, percent=42.7, used=5380497408, free=3276263424, active=1775763456, inactive=8335540224, buffers=361771008, cached=5454987264, shared=243720192, slab=544526336)
|
|
mem = psutil.virtual_memory()
|
|
return {
|
|
"used": mem.used,
|
|
"total": mem.total,
|
|
}
|
|
|
|
def create_swap_data() -> Dict:
|
|
# sswap(total=17179865088, used=262144, free=17179602944, percent=0.0, sin=0, sout=0)
|
|
swap = psutil.swap_memory()
|
|
return {
|
|
"used": swap.used,
|
|
"total": swap.total,
|
|
}
|
|
def create_drive_data() -> Dict:
|
|
tmp_elem = {}
|
|
# nvme0n1p1 => sdiskio(read_count=187, write_count=3, read_bytes=6002176, write_bytes=5120, read_time=36, write_time=7, read_merged_count=504, write_merged_count=0, busy_time=67)
|
|
drive_access = psutil.disk_io_counters(perdisk=True)
|
|
for elem in mounting_points:
|
|
# sdiskusage(total=255162540032, used=112077000704, free=130049380352, percent=46.3)
|
|
drive_usage = psutil.disk_usage(elem)
|
|
# print(f"plop {mounting_points[elem]} ==> {drive_access.keys()}")
|
|
if mounting_points[elem] in drive_access:
|
|
tmp_elem[elem] = {
|
|
"read_bytes": drive_access[mounting_points[elem]].read_bytes,
|
|
"write_bytes": drive_access[mounting_points[elem]].write_bytes,
|
|
"used": drive_usage.used,
|
|
"total": drive_usage.total,
|
|
}
|
|
else:
|
|
tmp_elem[elem] = {
|
|
"used": drive_usage.used,
|
|
"total": drive_usage.total,
|
|
}
|
|
return tmp_elem
|
|
|
|
def create_network_data() -> Dict:
|
|
# eth0 => snetio(bytes_sent=0, bytes_recv=0, packets_sent=0, packets_recv=0, errin=0, errout=0, dropin=0, dropout=0)
|
|
data = psutil.net_io_counters(pernic=True)
|
|
tmp_elem = {}
|
|
for elem in data:
|
|
tmp_elem[elem] = {
|
|
"bytes_sent": data[elem].bytes_sent,
|
|
"bytes_recv": data[elem].bytes_recv,
|
|
}
|
|
return tmp_elem
|
|
|
|
def agglutinate(configuration, name, data):
|
|
if configuration[name] == "auto":
|
|
return data
|
|
elif "include" in configuration[name]:
|
|
return filter(data, configuration[name]["include"])
|
|
return none
|
|
|
|
|
|
if __name__ == '__main__':
|
|
# Load arguments:
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("-c", "--config", type=str, default="/etc/karanage/system.json", help="json configuration file")
|
|
parser.add_argument("-C", "--connection", type=str, default=None, help="json configuration file")
|
|
|
|
# This element are read from the connection file:
|
|
parser.add_argument("-u", "--url", type=str, default=None, help="Base URL of the web service")
|
|
parser.add_argument("-g", "--group", type=str, default=None, help="Group the the message")
|
|
parser.add_argument("-T", "--token", type=str, default=None, help="Token to access to the server")
|
|
|
|
# This element are read from the configuration file:
|
|
parser.add_argument("-t", "--topic", type=str, default="PC/system", help="Topic of the message")
|
|
parser.add_argument("-s", "--sleep", type=int, default=30, help="Periodicity of the messages")
|
|
parser.add_argument("-d", "--display", help="Display the current state", action='store_true')
|
|
|
|
args = parser.parse_args()
|
|
|
|
if Path(args.config).exists():
|
|
f = open(args.config, "r")
|
|
configuration = json.loads(f.read())
|
|
f.close()
|
|
else:
|
|
configuration = {
|
|
"cpu": "auto",
|
|
"memory": "auto",
|
|
"swap": "auto",
|
|
"drive": "auto",
|
|
"network": "auto",
|
|
}
|
|
# manage the configuration model
|
|
if "config" not in configuration:
|
|
configuration["config"] = {}
|
|
if "display" not in configuration["config"]:
|
|
configuration["config"]["display"] = args.display
|
|
if "sleep" not in configuration["config"]:
|
|
configuration["config"]["sleep"] = args.sleep
|
|
if "topic" not in configuration["config"]:
|
|
configuration["config"]["topic"] = args.topic
|
|
|
|
connection = KaranageConnection(
|
|
url = args.url,
|
|
group = args.group,
|
|
token = args.token,
|
|
config_file = args.connection
|
|
)
|
|
|
|
# create the rest interface of karanage
|
|
stateInterface = KaranageState(connection)
|
|
|
|
while True:
|
|
out = {}
|
|
if need_process(configuration["cpu"]):
|
|
base_elem = create_cpu_data()
|
|
out["cpu"] = agglutinate(configuration, "cpu", base_elem)
|
|
|
|
if need_process(configuration["memory"]):
|
|
base_elem = create_memory_data()
|
|
out["memory"] = agglutinate(configuration, "memory", base_elem)
|
|
|
|
if need_process(configuration["swap"]):
|
|
base_elem = create_swap_data()
|
|
out["swap"] = agglutinate(configuration, "swap", base_elem)
|
|
|
|
if need_process(configuration["drive"]):
|
|
base_elem = create_drive_data()
|
|
out["drive"] = agglutinate(configuration, "drive", base_elem)
|
|
|
|
if configuration["network"] == "auto" or "include" in configuration["network"]:
|
|
base_elem = create_network_data()
|
|
out["network"] = agglutinate(configuration, "network", base_elem)
|
|
# display of needed:
|
|
if configuration["config"]["display"]:
|
|
print(json.dumps(out, indent=4))
|
|
# send message to the server:
|
|
try:
|
|
stateInterface.send(configuration["config"]["topic"], out, state = StateSystem.OK)
|
|
except KaranageException as ex:
|
|
print(f"Can not send to the server: {ex}")
|
|
time.sleep(configuration["config"]["sleep"])
|