-
Notifications
You must be signed in to change notification settings - Fork 31
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fixes #36973 - Use dnf needs-restarting to collect tracer information
Co-Authored-by: Ewoud Kohl van Wijngaarden <[email protected]>
- Loading branch information
Showing
3 changed files
with
166 additions
and
15 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
# SystemdUnit.py | ||
# Module for getting data from Systemd about Units | ||
|
||
# Copyright (C) 2017 Sean O'Keeffe | ||
# | ||
# This copyrighted material is made available to anyone wishing to use, | ||
# modify, copy, or redistribute it subject to the terms and conditions of | ||
# the GNU General Public License v.2, or (at your option) any later version. | ||
# This program is distributed in the hope that it will be useful, but WITHOUT | ||
# ANY WARRANTY expressed or implied, including the implied warranties 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 this program; if not, write to the | ||
# Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA | ||
# 02110-1301, USA. | ||
# | ||
|
||
import dbus | ||
|
||
class SystemdDbus(object): | ||
def __init__(self): | ||
self.__systemd = dbus.SystemBus().get_object('org.freedesktop.systemd1','/org/freedesktop/systemd1') | ||
self.__manager = dbus.Interface(self.__systemd, dbus_interface='org.freedesktop.systemd1.Manager') | ||
|
||
def unit_path_from_pid(self, pid): | ||
try: | ||
return self.__manager.GetUnitByPID(pid) | ||
except dbus.exceptions.DBusException: | ||
return False | ||
|
||
def has_service_property_from_pid(self, pid, attr): | ||
try: | ||
unit = self.unit_path_from_pid(pid) | ||
if not unit: | ||
return False | ||
|
||
proxy = dbus.SystemBus().get_object('org.freedesktop.systemd1', unit) | ||
propty = proxy.Get('org.freedesktop.systemd1.Service', attr, dbus_interface='org.freedesktop.DBus.Properties') | ||
except dbus.exceptions.DBusException: | ||
return False | ||
return bool(propty) | ||
|
||
def get_unit_property_from_pid(self, pid, attr): | ||
unit_path = self.unit_path_from_pid(pid) | ||
if bool(unit_path): | ||
proxy = dbus.SystemBus().get_object('org.freedesktop.systemd1', self.unit_path_from_pid(pid)) | ||
return proxy.Get('org.freedesktop.systemd1.Unit', attr, dbus_interface='org.freedesktop.DBus.Properties') | ||
else: | ||
return False |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,103 @@ | ||
import os | ||
import re | ||
import subprocess | ||
import psutil | ||
from katello.tracer.SystemdDbus import SystemdDbus | ||
|
||
STATIC_SERVICES = [ | ||
"systemd", | ||
"dbus", | ||
] | ||
|
||
REBOOT_HELPER= 'You will have to reboot your computer' | ||
SESSION_HELPER = 'You will have to log out & log in again' | ||
|
||
class DnfTracerApp: | ||
def __init__(self, name, helper, app_type): | ||
self.name = name | ||
self.helper = helper | ||
self.type = app_type | ||
|
||
|
||
class Process: | ||
def __init__(self, bus, pid): | ||
self.bus = bus | ||
self.pid = int(pid) | ||
self.process = psutil.Process(self.pid) | ||
self.name = self.detect_name() | ||
|
||
# special handling for ssh sessions | ||
try: | ||
if self.name == 'sshd': | ||
exe = self.process.exe() | ||
cmdline = self.process.cmdline() | ||
if exe not in cmdline and len(cmdline) > 1: | ||
self.name = 'ssh-{0}-session'.format(re.split(' |@',' '.join(cmdline))[1]) | ||
except psutil.AccessDenied: | ||
pass | ||
|
||
def detect_name(self): | ||
if self.pid and self.bus.unit_path_from_pid(self.pid): | ||
if not self.bus.has_service_property_from_pid(self.pid, 'PAMName'): | ||
unit_id = self.bus.get_unit_property_from_pid(self.pid, 'Id') | ||
if unit_id and unit_id.endswith('.service'): | ||
return unit_id[:-8] | ||
return self.process.name() | ||
|
||
def is_session(self): | ||
if self.name.startswith("ssh-") and self.name.endswith("-session"): | ||
return True | ||
|
||
terminal = self.process.terminal() | ||
if terminal is not None: | ||
parent = self.process.parent() | ||
if parent is None or terminal != parent.terminal(): | ||
return True | ||
return False | ||
|
||
def is_reboot_required(self): | ||
return self.name in STATIC_SERVICES | ||
|
||
|
||
def collect_services_state(): | ||
env = dict(os.environ, LANG='C') | ||
process = subprocess.run(['dnf', 'needs-restarting'], env=env, stdout=subprocess.PIPE, stderr=subprocess.PIPE, universal_newlines=True) | ||
if process.returncode != 0: | ||
return [] | ||
|
||
lines = process.stdout.split('\n') | ||
pids = [line.split(' : ')[0].strip() for line in lines if ' : ' in line] | ||
|
||
apps = set() | ||
bus = SystemdDbus() | ||
for pid in pids: | ||
p = Process(bus, pid) | ||
|
||
app_type = 'daemon' | ||
helper = "systemctl restart " + p.name | ||
if p.is_session(): | ||
app_type = 'session' | ||
helper = SESSION_HELPER | ||
if p.is_reboot_required(): | ||
app_type = 'static' | ||
helper = REBOOT_HELPER | ||
|
||
if p.name is not None: | ||
app = DnfTracerApp(p.name, helper, app_type) | ||
apps.add(app) | ||
return list(apps) | ||
|
||
|
||
def collect_restart(): | ||
apps = [] | ||
process = subprocess.run(["dnf", "needs-restarting", "-r"], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL) | ||
if process.returncode == 1: | ||
app = DnfTracerApp("kernel", REBOOT_HELPER, "static") | ||
apps.append(app) | ||
return apps | ||
|
||
|
||
def collect_apps(plugin=None): | ||
apps = collect_services_state() | ||
reboot = collect_restart() | ||
return apps + reboot |