lsusers/lsusers.py

98 lines
3.6 KiB
Python
Executable File

#!/usr/bin/env python
# coding: utf-8
#
# Copyright © 2023 - Rennes Physics Institute
#
# This file is part of lsusers.
#
# lsusers is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
# lsusers is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
# You should have received a copy of the GNU General Public License
# along with lsusers. If not, see <http://www.gnu.org/licenses/>.
#
# Source file : <Will be updated during commit>
# Last modified: <Will be updated during commit>
# Committed by : <Will be updated during commit>
import os
from subprocess import Popen, PIPE, STDOUT
NCPUS = os.cpu_count()
TOTALMEM = os.sysconf("SC_PAGE_SIZE") * os.sysconf("SC_PHYS_PAGES") / (1024**3)
def get_users_list():
cmd = Popen(['loginctl', 'list-users', '--no-legend'], stdin=PIPE, stdout=PIPE, stderr=STDOUT)
output = cmd.stdout.read().decode('utf8').split()[1::2]
cmd = Popen(['users'], stdin=PIPE, stdout=PIPE, stderr=STDOUT)
users_output = cmd.stdout.read().decode('utf8').split()
for name in users_output:
if name not in output:
output.append(name)
return output
def get_resources_percent(username=None):
if username is None:
user_opt = ['-e']
else:
user_opt = ['-u', username]
cmd = Popen(['ps', '-o', 'pcpu', '--no-headers'] + user_opt, stdin=PIPE, stdout=PIPE, stderr=STDOUT)
output = list(map(float, cmd.stdout.read().decode('utf8').split()))
cpu_percent = sum(output) / NCPUS
cmd = Popen(['ps', '-o', 'pmem', '--no-headers'] + user_opt, stdin=PIPE, stdout=PIPE, stderr=STDOUT)
output = map(float, cmd.stdout.read().decode('utf8').split())
mem_percent = sum(output)
return cpu_percent, mem_percent
def get_free_resources():
cpu, mem = get_resources_percent()
free_cpu = (100 - cpu) * NCPUS / 100
free_mem = (100 - mem) * TOTALMEM / 100
return free_cpu, free_mem
def get_resources():
users = get_users_list()
resources = list(map(get_resources_percent, users))
cpu_percent, mem_percent = list(zip(*resources))
cpu_usage = [int(_ / 100 * NCPUS) + 1 for _ in cpu_percent]
mem_usage = [_ /100 * TOTALMEM for _ in mem_percent]
free_cpu, free_mem = get_free_resources()
return users, cpu_percent, cpu_usage, mem_percent, mem_usage, free_cpu, free_mem
def print_output(users, cpu_percent, cpu_usage, mem_percent, mem_usage, free_cpu, free_mem):
header = "| {:<20s} | {:<20s} | {:>23s} |".format("Name ({:d} users)".format(len(users)), "CPU".center(20), "Memory")
double_line = "+" + "="*(len(header) - 2) + "+"
simple_line = "|" + "-"*(len(header) - 2) + "|"
row_lines = []
for i, user in enumerate(users):
s = "| {:<20s} | {:5.1f}% [{:2d}/{:d} cores] | {:5.1f}% [{:7.3f}/{:d} GB] |".format(
user, cpu_percent[i], cpu_usage[i], NCPUS, mem_percent[i], mem_usage[i], int(TOTALMEM))
row_lines.append(s)
free_line = "| {:<20s} | {:13.1f} cores | {:19.0f} GB |".format("FREE", free_cpu, free_mem)
s = double_line + '\n'
s += header + '\n'
s += simple_line + '\n'
s += '\n'.join(row_lines) + '\n'
s += simple_line + '\n'
s += free_line + '\n'
s += double_line
print(s)
if __name__ == "__main__":
resources = get_resources()
print_output(*resources)