re-import sources as agreed with the maintainer
This commit is contained in:
parent
8f01d41977
commit
64fd060bac
12
.gitignore
vendored
12
.gitignore
vendored
@ -1,2 +1,12 @@
|
||||
SOURCES/python-linux-procfs-0.7.1.tar.xz
|
||||
*.swp
|
||||
python-linux-procfs-0.4.4.tar.bz2
|
||||
/python-linux-procfs-0.4.6.tar.bz2
|
||||
/python-linux-procfs-0.4.5.tar.bz2
|
||||
/python-linux-procfs-0.4.6.tar.xz
|
||||
/python-linux-procfs-0.4.10.tar.xz
|
||||
/python-linux-procfs-0.5.1.tar.xz
|
||||
/python-linux-procfs-0.6.tar.xz
|
||||
/python-linux-procfs-0.6.2.tar.xz
|
||||
/python-linux-procfs-0.6.3.tar.xz
|
||||
/python-linux-procfs-0.7.0.tar.xz
|
||||
/python-linux-procfs-0.7.1.tar.xz
|
||||
|
68
python-linux-procfs-Fix-UnicodeDecodeError.patch
Normal file
68
python-linux-procfs-Fix-UnicodeDecodeError.patch
Normal file
@ -0,0 +1,68 @@
|
||||
From a752e7995bfdabb08b5bb6bcd437b9a5d2e53a7a Mon Sep 17 00:00:00 2001
|
||||
From: John Kacur <jkacur@redhat.com>
|
||||
Date: Thu, 9 Dec 2021 10:11:07 -0500
|
||||
Subject: [PATCH 2/2] python-linux-procfs: Fix UnicodeDecodeError
|
||||
|
||||
Commit 7570fc0d6082 meant to solve the UnicodeDecodeError
|
||||
|
||||
Instead it actually increased the problem by reading lines as bytes
|
||||
and decoding them.
|
||||
|
||||
The original problem is hard to trigger and doesn't trigger consistently
|
||||
with reproducers. In addition there seems to be a difference in how this
|
||||
is handled between python-3.6 to python-3.9
|
||||
|
||||
For now, we should return the code to reading as utf-8 (the default)
|
||||
since that handles more cases than the current code.
|
||||
|
||||
We can catch the UnicodeDecodeError and ignore it for now. It is not
|
||||
ideal because we are not handling some pids that trigger the error.
|
||||
|
||||
This patch also includes a fix for a FileNotFoundError which can occur if
|
||||
a pid exits and disappears before we try to read it in the /proc file
|
||||
system.
|
||||
|
||||
Signed-off-by: John Kacur <jkacur@redhat.com>
|
||||
---
|
||||
procfs/procfs.py | 18 ++++++++++++++----
|
||||
1 file changed, 14 insertions(+), 4 deletions(-)
|
||||
|
||||
diff --git a/procfs/procfs.py b/procfs/procfs.py
|
||||
index a78bac5376e3..de55dfc1aef4 100755
|
||||
--- a/procfs/procfs.py
|
||||
+++ b/procfs/procfs.py
|
||||
@@ -44,7 +44,11 @@ def process_cmdline(pid_info):
|
||||
if pid_info["cmdline"]:
|
||||
return reduce(lambda a, b: a + " %s" % b, pid_info["cmdline"]).strip()
|
||||
|
||||
- return pid_info["stat"]["comm"]
|
||||
+ try:
|
||||
+ """ If a pid disappears before we query it, return None """
|
||||
+ return pid_info["stat"]["comm"]
|
||||
+ except:
|
||||
+ return None
|
||||
|
||||
|
||||
class pidstat:
|
||||
@@ -374,9 +378,15 @@ class process:
|
||||
return hasattr(self, attr)
|
||||
|
||||
def load_cmdline(self):
|
||||
- with open("/proc/%d/cmdline" % self.pid, mode='rb') as f:
|
||||
- cmdline = f.readline().decode(encoding='unicode_escape')
|
||||
- self.cmdline = cmdline.strip().split('\0')[:-1]
|
||||
+ try:
|
||||
+ with open("/proc/%d/cmdline" % self.pid) as f:
|
||||
+ self.cmdline = f.readline().strip().split('\0')[:-1]
|
||||
+ except FileNotFoundError:
|
||||
+ """ This can happen when a pid disappears """
|
||||
+ self.cmdline = None
|
||||
+ except UnicodeDecodeError:
|
||||
+ """ TODO - this shouldn't happen, needs to be investigated """
|
||||
+ self.cmdline = None
|
||||
|
||||
def load_threads(self):
|
||||
self.threads = pidstats("/proc/%d/task/" % self.pid)
|
||||
--
|
||||
2.31.1
|
||||
|
33
python-linux-procfs-Fix-traceback-with-non-utf8-char.patch
Normal file
33
python-linux-procfs-Fix-traceback-with-non-utf8-char.patch
Normal file
@ -0,0 +1,33 @@
|
||||
From 7570fc0d6082cb476c32233c2904214dd57737a8 Mon Sep 17 00:00:00 2001
|
||||
From: John Kacur <jkacur@redhat.com>
|
||||
Date: Fri, 19 Nov 2021 16:03:22 -0500
|
||||
Subject: [PATCH] python-linux-procfs: Fix traceback with non-utf8 chars in the
|
||||
/proc/PID/cmdline
|
||||
|
||||
Fix traceback if there are non-utf8 characters in the /proc/PID/cmdline
|
||||
|
||||
Signed-off-by: John Kacur <jkacur@redhat.com>
|
||||
---
|
||||
procfs/procfs.py | 6 +++---
|
||||
1 file changed, 3 insertions(+), 3 deletions(-)
|
||||
|
||||
diff --git a/procfs/procfs.py b/procfs/procfs.py
|
||||
index 3b7474cccb01..408b2bcd0a31 100755
|
||||
--- a/procfs/procfs.py
|
||||
+++ b/procfs/procfs.py
|
||||
@@ -357,9 +357,9 @@ class process:
|
||||
return hasattr(self, attr)
|
||||
|
||||
def load_cmdline(self):
|
||||
- f = open("/proc/%d/cmdline" % self.pid)
|
||||
- self.cmdline = f.readline().strip().split('\0')[:-1]
|
||||
- f.close()
|
||||
+ with open("/proc/%d/cmdline" % self.pid, mode='rb') as f:
|
||||
+ cmdline = f.readline().decode(encoding='unicode_escape')
|
||||
+ self.cmdline = cmdline.strip().split('\0')[:-1]
|
||||
|
||||
def load_threads(self):
|
||||
self.threads = pidstats("/proc/%d/task/" % self.pid)
|
||||
--
|
||||
2.31.1
|
||||
|
68
python-linux-procfs-Propagate-error-to-user-if-a-pid.patch
Normal file
68
python-linux-procfs-Propagate-error-to-user-if-a-pid.patch
Normal file
@ -0,0 +1,68 @@
|
||||
From b7ea06b21456d465f2d9d11358fb803eb277357f Mon Sep 17 00:00:00 2001
|
||||
From: John Kacur <jkacur@redhat.com>
|
||||
Date: Tue, 23 Nov 2021 09:58:58 -0500
|
||||
Subject: [PATCH 1/3] python-linux-procfs: Propagate error to user if a pid is
|
||||
completed
|
||||
|
||||
If a pid is completed and disappears a FileNotFoundError will occur
|
||||
because /proc/pid/stat will disappear too.
|
||||
|
||||
It is not possible to check for the file first because it could still
|
||||
disappear between the time of the check and the time of use.
|
||||
|
||||
Propagate this error to the user.
|
||||
The user should handle this with a try, except clause and ignore it if
|
||||
an exception occurs.
|
||||
|
||||
Signed-off-by: John Kacur <jkacur@redhat.com>
|
||||
---
|
||||
procfs/procfs.py | 19 ++++++++++++++++---
|
||||
1 file changed, 16 insertions(+), 3 deletions(-)
|
||||
|
||||
diff --git a/procfs/procfs.py b/procfs/procfs.py
|
||||
index 408b2bcd0a31..a0e9977214fe 100755
|
||||
--- a/procfs/procfs.py
|
||||
+++ b/procfs/procfs.py
|
||||
@@ -130,7 +130,12 @@ class pidstat:
|
||||
|
||||
def __init__(self, pid, basedir="/proc"):
|
||||
self.pid = pid
|
||||
- self.load(basedir)
|
||||
+ try:
|
||||
+ self.load(basedir)
|
||||
+ except FileNotFoundError:
|
||||
+ # The file representing the pid has disappeared
|
||||
+ # propagate the error to the user to handle
|
||||
+ raise
|
||||
|
||||
def __getitem__(self, fieldname):
|
||||
return self.fields[fieldname]
|
||||
@@ -151,7 +156,11 @@ class pidstat:
|
||||
return fieldname in self.fields
|
||||
|
||||
def load(self, basedir="/proc"):
|
||||
- f = open("%s/%d/stat" % (basedir, self.pid))
|
||||
+ try:
|
||||
+ f = open("%s/%d/stat" % (basedir, self.pid))
|
||||
+ except FileNotFoundError:
|
||||
+ # The pid has disappeared, propagate the error
|
||||
+ raise
|
||||
fields = f.readline().strip().split(') ')
|
||||
f.close()
|
||||
fields = fields[0].split(' (') + fields[1].split()
|
||||
@@ -338,7 +347,11 @@ class process:
|
||||
else:
|
||||
sclass = pidstatus
|
||||
|
||||
- setattr(self, attr, sclass(self.pid, self.basedir))
|
||||
+ try:
|
||||
+ setattr(self, attr, sclass(self.pid, self.basedir))
|
||||
+ except FileNotFoundError:
|
||||
+ # The pid has disappeared, progate the error
|
||||
+ raise
|
||||
elif attr == "cmdline":
|
||||
self.load_cmdline()
|
||||
elif attr == "threads":
|
||||
--
|
||||
2.31.1
|
||||
|
502
python-linux-procfs-Various-clean-ups.patch
Normal file
502
python-linux-procfs-Various-clean-ups.patch
Normal file
@ -0,0 +1,502 @@
|
||||
From ce8cacdd515bf7270daef62648d5f994f111cded Mon Sep 17 00:00:00 2001
|
||||
From: John Kacur <jkacur@redhat.com>
|
||||
Date: Thu, 2 Dec 2021 16:52:10 -0500
|
||||
Subject: [PATCH 1/2] python-linux-procfs: Various clean-ups
|
||||
|
||||
- Replace f=open with 'with' (context managers), except in try-except blocks
|
||||
- Reformat lines that are too long, especially in comments
|
||||
- Use min() instead of if construct
|
||||
- Use bool instead of complicated and True or False constructs
|
||||
- Reorder imports
|
||||
|
||||
Signed-off-by: John Kacur <jkacur@redhat.com>
|
||||
---
|
||||
procfs/procfs.py | 292 +++++++++++++++++++++++------------------------
|
||||
1 file changed, 142 insertions(+), 150 deletions(-)
|
||||
|
||||
diff --git a/procfs/procfs.py b/procfs/procfs.py
|
||||
index a0e9977214fe..a78bac5376e3 100755
|
||||
--- a/procfs/procfs.py
|
||||
+++ b/procfs/procfs.py
|
||||
@@ -19,10 +19,10 @@
|
||||
#
|
||||
|
||||
import os
|
||||
-import time
|
||||
-from functools import reduce
|
||||
import platform
|
||||
import re
|
||||
+import time
|
||||
+from functools import reduce
|
||||
from six.moves import range
|
||||
from procfs.utilist import bitmasklist
|
||||
|
||||
@@ -37,8 +37,9 @@ def is_s390():
|
||||
|
||||
def process_cmdline(pid_info):
|
||||
"""
|
||||
- Returns the process command line, if available in the given `process' class, if
|
||||
- not available, falls back to using the comm (short process name) in its pidstat key.
|
||||
+ Returns the process command line, if available in the given `process' class,
|
||||
+ if not available, falls back to using the comm (short process name) in its
|
||||
+ pidstat key.
|
||||
"""
|
||||
if pid_info["cmdline"]:
|
||||
return reduce(lambda a, b: a + " %s" % b, pid_info["cmdline"]).strip()
|
||||
@@ -47,10 +48,12 @@ def process_cmdline(pid_info):
|
||||
|
||||
|
||||
class pidstat:
|
||||
- """Provides a dictionary to access the fields in the per process /proc/PID/stat
|
||||
- files.
|
||||
+ """
|
||||
+ Provides a dictionary to access the fields in the
|
||||
+ per process /proc/PID/stat files.
|
||||
|
||||
- One can obtain the available fields asking for the keys of the dictionary, e.g.:
|
||||
+ One can obtain the available fields by asking for the keys of the
|
||||
+ dictionary, e.g.:
|
||||
|
||||
>>> p = procfs.pidstat(1)
|
||||
>>> print p.keys()
|
||||
@@ -182,8 +185,7 @@ class pidstat:
|
||||
Returns true if this process has a fixed smp affinity mask,
|
||||
not allowing it to be moved to a different set of CPUs.
|
||||
"""
|
||||
- return self.fields["flags"] & self.PF_THREAD_BOUND and \
|
||||
- True or False
|
||||
+ return bool(self.fields["flags"] & self.PF_THREAD_BOUND)
|
||||
|
||||
def process_flags(self):
|
||||
"""
|
||||
@@ -193,33 +195,34 @@ class pidstat:
|
||||
|
||||
As of v4.2-rc7 these include (from include/linux/sched.h comments):
|
||||
|
||||
- PF_EXITING Getting shut down
|
||||
- PF_EXITPIDONE Pi exit done on shut down
|
||||
+ PF_EXITING Getting shut down
|
||||
+ PF_EXITPIDONE Pi exit done on shut down
|
||||
PF_VCPU I'm a virtual CPU
|
||||
- PF_WQ_WORKER I'm a workqueue worker
|
||||
- PF_FORKNOEXEC Forked but didn't exec
|
||||
- PF_MCE_PROCESS Process policy on mce errors
|
||||
- PF_SUPERPRIV Used super-user privileges
|
||||
+ PF_WQ_WORKER I'm a workqueue worker
|
||||
+ PF_FORKNOEXEC Forked but didn't exec
|
||||
+ PF_MCE_PROCESS Process policy on mce errors
|
||||
+ PF_SUPERPRIV Used super-user privileges
|
||||
PF_DUMPCORE Dumped core
|
||||
PF_SIGNALED Killed by a signal
|
||||
PF_MEMALLOC Allocating memory
|
||||
- PF_NPROC_EXCEEDED Set_user noticed that RLIMIT_NPROC was exceeded
|
||||
- PF_USED_MATH If unset the fpu must be initialized before use
|
||||
- PF_USED_ASYNC Used async_schedule*(), used by module init
|
||||
+ PF_NPROC_EXCEEDED Set_user noticed that RLIMIT_NPROC was exceeded
|
||||
+ PF_USED_MATH If unset the fpu must be initialized before use
|
||||
+ PF_USED_ASYNC Used async_schedule*(), used by module init
|
||||
PF_NOFREEZE This thread should not be frozen
|
||||
- PF_FROZEN Frozen for system suspend
|
||||
- PF_FSTRANS Inside a filesystem transaction
|
||||
- PF_KSWAPD I am kswapd
|
||||
+ PF_FROZEN Frozen for system suspend
|
||||
+ PF_FSTRANS Inside a filesystem transaction
|
||||
+ PF_KSWAPD I am kswapd
|
||||
PF_MEMALLOC_NOIO Allocating memory without IO involved
|
||||
PF_LESS_THROTTLE Throttle me less: I clean memory
|
||||
- PF_KTHREAD I am a kernel thread
|
||||
+ PF_KTHREAD I am a kernel thread
|
||||
PF_RANDOMIZE Randomize virtual address space
|
||||
PF_SWAPWRITE Allowed to write to swap
|
||||
PF_NO_SETAFFINITY Userland is not allowed to meddle with cpus_allowed
|
||||
PF_MCE_EARLY Early kill for mce process policy
|
||||
- PF_MUTEX_TESTER Thread belongs to the rt mutex tester
|
||||
- PF_FREEZER_SKIP Freezer should not count it as freezable
|
||||
- PF_SUSPEND_TASK This thread called freeze_processes and should not be frozen
|
||||
+ PF_MUTEX_TESTER Thread belongs to the rt mutex tester
|
||||
+ PF_FREEZER_SKIP Freezer should not count it as freezable
|
||||
+ PF_SUSPEND_TASK This thread called freeze_processes and
|
||||
+ should not be frozen
|
||||
|
||||
"""
|
||||
sflags = []
|
||||
@@ -236,8 +239,8 @@ class pidstat:
|
||||
def cannot_set_affinity(self, pid):
|
||||
PF_NO_SETAFFINITY = 0x04000000
|
||||
try:
|
||||
- return int(self.processes[pid]["stat"]["flags"]) & \
|
||||
- PF_NO_SETAFFINITY and True or False
|
||||
+ return bool(int(self.processes[pid]["stat"]["flags"]) &
|
||||
+ PF_NO_SETAFFINITY)
|
||||
except:
|
||||
return True
|
||||
|
||||
@@ -245,19 +248,21 @@ def cannot_set_affinity(self, pid):
|
||||
def cannot_set_thread_affinity(self, pid, tid):
|
||||
PF_NO_SETAFFINITY = 0x04000000
|
||||
try:
|
||||
- return int(self.processes[pid].threads[tid]["stat"]["flags"]) & \
|
||||
- PF_NO_SETAFFINITY and True or False
|
||||
+ return bool(int(self.processes[pid].threads[tid]["stat"]["flags"]) &
|
||||
+ PF_NO_SETAFFINITY)
|
||||
except:
|
||||
return True
|
||||
|
||||
|
||||
class pidstatus:
|
||||
"""
|
||||
- Provides a dictionary to access the fields in the per process /proc/PID/status
|
||||
- files. This provides additional information about processes and threads to what
|
||||
- can be obtained with the procfs.pidstat() class.
|
||||
+ Provides a dictionary to access the fields
|
||||
+ in the per process /proc/PID/status files.
|
||||
+ This provides additional information about processes and threads to
|
||||
+ what can be obtained with the procfs.pidstat() class.
|
||||
|
||||
- One can obtain the available fields asking for the keys of the dictionary, e.g.:
|
||||
+ One can obtain the available fields by asking for the keys of the
|
||||
+ dictionary, e.g.:
|
||||
|
||||
>>> import procfs
|
||||
>>> p = procfs.pidstatus(1)
|
||||
@@ -312,19 +317,18 @@ class pidstatus:
|
||||
return fieldname in self.fields
|
||||
|
||||
def load(self, basedir="/proc"):
|
||||
- f = open("%s/%d/status" % (basedir, self.pid))
|
||||
self.fields = {}
|
||||
- for line in f.readlines():
|
||||
- fields = line.split(":")
|
||||
- if len(fields) != 2:
|
||||
- continue
|
||||
- name = fields[0]
|
||||
- value = fields[1].strip()
|
||||
- try:
|
||||
- self.fields[name] = int(value)
|
||||
- except:
|
||||
- self.fields[name] = value
|
||||
- f.close()
|
||||
+ with open("%s/%d/status" % (basedir, self.pid)) as f:
|
||||
+ for line in f.readlines():
|
||||
+ fields = line.split(":")
|
||||
+ if len(fields) != 2:
|
||||
+ continue
|
||||
+ name = fields[0]
|
||||
+ value = fields[1].strip()
|
||||
+ try:
|
||||
+ self.fields[name] = int(value)
|
||||
+ except:
|
||||
+ self.fields[name] = value
|
||||
|
||||
|
||||
class process:
|
||||
@@ -380,14 +384,13 @@ class process:
|
||||
del self.threads[self.pid]
|
||||
|
||||
def load_cgroups(self):
|
||||
- f = open("/proc/%d/cgroup" % self.pid)
|
||||
self.cgroups = ""
|
||||
- for line in reversed(f.readlines()):
|
||||
- if len(self.cgroups) != 0:
|
||||
- self.cgroups = self.cgroups + "," + line[:-1]
|
||||
- else:
|
||||
- self.cgroups = line[:-1]
|
||||
- f.close()
|
||||
+ with open("/proc/%d/cgroup" % self.pid) as f:
|
||||
+ for line in reversed(f.readlines()):
|
||||
+ if len(self.cgroups) != 0:
|
||||
+ self.cgroups = self.cgroups + "," + line[:-1]
|
||||
+ else:
|
||||
+ self.cgroups = line[:-1]
|
||||
|
||||
def load_environ(self):
|
||||
"""
|
||||
@@ -416,12 +419,11 @@ class process:
|
||||
>>>
|
||||
"""
|
||||
self.environ = {}
|
||||
- f = open("/proc/%d/environ" % self.pid)
|
||||
- for x in f.readline().split('\0'):
|
||||
- if len(x) > 0:
|
||||
- y = x.split('=')
|
||||
- self.environ[y[0]] = y[1]
|
||||
- f.close()
|
||||
+ with open("/proc/%d/environ" % self.pid) as f:
|
||||
+ for x in f.readline().split('\0'):
|
||||
+ if len(x) > 0:
|
||||
+ y = x.split('=')
|
||||
+ self.environ[y[0]] = y[1]
|
||||
|
||||
|
||||
class pidstats:
|
||||
@@ -465,18 +467,18 @@ class pidstats:
|
||||
|
||||
def reload(self):
|
||||
"""
|
||||
- This operation will throw away the current dictionary contents, if any, and
|
||||
- read all the pid files from /proc/, instantiating a 'process' instance for
|
||||
- each of them.
|
||||
+ This operation will throw away the current dictionary contents,
|
||||
+ if any, and read all the pid files from /proc/, instantiating a
|
||||
+ 'process' instance for each of them.
|
||||
|
||||
- This is a high overhead operation, and should be avoided if the perf python
|
||||
- binding can be used to detect when new threads appear and existing ones
|
||||
- terminate.
|
||||
+ This is a high overhead operation, and should be avoided if the
|
||||
+ perf python binding can be used to detect when new threads appear
|
||||
+ and existing ones terminate.
|
||||
|
||||
In RHEL it is found in the python-perf rpm package.
|
||||
|
||||
- More information about the perf facilities can be found in the 'perf_event_open'
|
||||
- man page.
|
||||
+ More information about the perf facilities can be found in the
|
||||
+ 'perf_event_open' man page.
|
||||
"""
|
||||
del self.processes
|
||||
self.processes = {}
|
||||
@@ -643,24 +645,21 @@ class interrupts:
|
||||
def reload(self):
|
||||
del self.interrupts
|
||||
self.interrupts = {}
|
||||
- f = open("/proc/interrupts")
|
||||
-
|
||||
- for line in f.readlines():
|
||||
- line = line.strip()
|
||||
- fields = line.split()
|
||||
- if fields[0][:3] == "CPU":
|
||||
- self.nr_cpus = len(fields)
|
||||
- continue
|
||||
- irq = fields[0].strip(":")
|
||||
- self.interrupts[irq] = {}
|
||||
- self.interrupts[irq] = self.parse_entry(fields[1:], line)
|
||||
- try:
|
||||
- nirq = int(irq)
|
||||
- except:
|
||||
- continue
|
||||
- self.interrupts[irq]["affinity"] = self.parse_affinity(nirq)
|
||||
-
|
||||
- f.close()
|
||||
+ with open("/proc/interrupts") as f:
|
||||
+ for line in f.readlines():
|
||||
+ line = line.strip()
|
||||
+ fields = line.split()
|
||||
+ if fields[0][:3] == "CPU":
|
||||
+ self.nr_cpus = len(fields)
|
||||
+ continue
|
||||
+ irq = fields[0].strip(":")
|
||||
+ self.interrupts[irq] = {}
|
||||
+ self.interrupts[irq] = self.parse_entry(fields[1:], line)
|
||||
+ try:
|
||||
+ nirq = int(irq)
|
||||
+ except:
|
||||
+ continue
|
||||
+ self.interrupts[irq]["affinity"] = self.parse_affinity(nirq)
|
||||
|
||||
def parse_entry(self, fields, line):
|
||||
dict = {}
|
||||
@@ -681,9 +680,8 @@ class interrupts:
|
||||
|
||||
def parse_affinity(self, irq):
|
||||
try:
|
||||
- f = open("/proc/irq/%s/smp_affinity" % irq)
|
||||
- line = f.readline()
|
||||
- f.close()
|
||||
+ with open("/proc/irq/%s/smp_affinity" % irq) as f:
|
||||
+ line = f.readline()
|
||||
return bitmasklist(line, self.nr_cpus)
|
||||
except IOError:
|
||||
return [0, ]
|
||||
@@ -741,11 +739,11 @@ class cmdline:
|
||||
"""
|
||||
Parses the kernel command line (/proc/cmdline), turning it into a dictionary."
|
||||
|
||||
- Useful to figure out if some kernel boolean knob has been turned on, as well
|
||||
- as to find the value associated to other kernel knobs.
|
||||
+ Useful to figure out if some kernel boolean knob has been turned on,
|
||||
+ as well as to find the value associated to other kernel knobs.
|
||||
|
||||
- It can also be used to find out about parameters passed to the init process,
|
||||
- such as 'BOOT_IMAGE', etc.
|
||||
+ It can also be used to find out about parameters passed to the
|
||||
+ init process, such as 'BOOT_IMAGE', etc.
|
||||
|
||||
E.g.:
|
||||
>>> import procfs
|
||||
@@ -762,15 +760,13 @@ class cmdline:
|
||||
self.parse()
|
||||
|
||||
def parse(self):
|
||||
- f = open("/proc/cmdline")
|
||||
- for option in f.readline().strip().split():
|
||||
- fields = option.split("=")
|
||||
- if len(fields) == 1:
|
||||
- self.options[fields[0]] = True
|
||||
- else:
|
||||
- self.options[fields[0]] = fields[1]
|
||||
-
|
||||
- f.close()
|
||||
+ with open("/proc/cmdline") as f:
|
||||
+ for option in f.readline().strip().split():
|
||||
+ fields = option.split("=")
|
||||
+ if len(fields) == 1:
|
||||
+ self.options[fields[0]] = True
|
||||
+ else:
|
||||
+ self.options[fields[0]] = fields[1]
|
||||
|
||||
def __getitem__(self, key):
|
||||
return self.options[key]
|
||||
@@ -790,9 +786,9 @@ class cpuinfo:
|
||||
Dictionary with information about CPUs in the system.
|
||||
|
||||
Please refer to 'man procfs(5)' for further information about the
|
||||
- '/proc/cpuinfo' file, that is the source of the information provided by this
|
||||
- class. The 'man lscpu(1)' also has information about a program that uses
|
||||
- the '/proc/cpuinfo' file.
|
||||
+ '/proc/cpuinfo' file, that is the source of the information provided
|
||||
+ by this class. The 'man lscpu(1)' also has information about a program that
|
||||
+ uses the '/proc/cpuinfo' file.
|
||||
|
||||
Using this class one can obtain the number of CPUs in a system:
|
||||
|
||||
@@ -801,14 +797,14 @@ class cpuinfo:
|
||||
4
|
||||
|
||||
It is also possible to figure out aspects of the CPU topology, such as
|
||||
- how many CPU physical sockets exists, i.e. groups of CPUs sharing components
|
||||
- such as CPU memory caches:
|
||||
+ how many CPU physical sockets exists, i.e. groups of CPUs sharing
|
||||
+ components such as CPU memory caches:
|
||||
|
||||
>>> print len(cpus.sockets)
|
||||
1
|
||||
|
||||
- Additionally dictionary with information common to all CPUs in the system is
|
||||
- available:
|
||||
+ Additionally dictionary with information common to all CPUs in the system
|
||||
+ is available:
|
||||
|
||||
>>> print cpus["model name"]
|
||||
Intel(R) Core(TM) i7-3667U CPU @ 2.00GHz
|
||||
@@ -836,28 +832,26 @@ class cpuinfo:
|
||||
return self.tags
|
||||
|
||||
def parse(self, filename):
|
||||
- f = open(filename)
|
||||
- for line in f.readlines():
|
||||
- line = line.strip()
|
||||
- if not line:
|
||||
- continue
|
||||
- fields = line.split(":")
|
||||
- tagname = fields[0].strip().lower()
|
||||
- if tagname == "processor":
|
||||
- self.nr_cpus += 1
|
||||
- continue
|
||||
- if is_s390() and tagname == "cpu number":
|
||||
- self.nr_cpus += 1
|
||||
- continue
|
||||
- if tagname == "core id":
|
||||
- continue
|
||||
- self.tags[tagname] = fields[1].strip()
|
||||
- if tagname == "physical id":
|
||||
- socket_id = self.tags[tagname]
|
||||
- if socket_id not in self.sockets:
|
||||
- self.sockets.append(socket_id)
|
||||
-
|
||||
- f.close()
|
||||
+ with open(filename) as f:
|
||||
+ for line in f.readlines():
|
||||
+ line = line.strip()
|
||||
+ if not line:
|
||||
+ continue
|
||||
+ fields = line.split(":")
|
||||
+ tagname = fields[0].strip().lower()
|
||||
+ if tagname == "processor":
|
||||
+ self.nr_cpus += 1
|
||||
+ continue
|
||||
+ if is_s390() and tagname == "cpu number":
|
||||
+ self.nr_cpus += 1
|
||||
+ continue
|
||||
+ if tagname == "core id":
|
||||
+ continue
|
||||
+ self.tags[tagname] = fields[1].strip()
|
||||
+ if tagname == "physical id":
|
||||
+ socket_id = self.tags[tagname]
|
||||
+ if socket_id not in self.sockets:
|
||||
+ self.sockets.append(socket_id)
|
||||
self.nr_sockets = self.sockets and len(self.sockets) or \
|
||||
(self.nr_cpus /
|
||||
("siblings" in self.tags and int(self.tags["siblings"]) or 1))
|
||||
@@ -870,7 +864,8 @@ class smaps_lib:
|
||||
Representation of an mmap in place for a process. Can be used to figure
|
||||
out which processes have an library mapped, etc.
|
||||
|
||||
- The 'perm' member can be used to figure out executable mmaps, i.e. libraries.
|
||||
+ The 'perm' member can be used to figure out executable mmaps,
|
||||
+ i.e. libraries.
|
||||
|
||||
The 'vm_start' and 'vm_end' in turn can be used when trying to resolve
|
||||
processor instruction pointer addresses to a symbol name in a library.
|
||||
@@ -967,13 +962,12 @@ class smaps:
|
||||
return self.entries[index]
|
||||
|
||||
def reload(self):
|
||||
- f = open("/proc/%d/smaps" % self.pid)
|
||||
line = None
|
||||
- while True:
|
||||
- line = self.parse_entry(f, line)
|
||||
- if not line:
|
||||
- break
|
||||
- f.close()
|
||||
+ with("/proc/%d/smaps" % self.pid) as f:
|
||||
+ while True:
|
||||
+ line = self.parse_entry(f, line)
|
||||
+ if not line:
|
||||
+ break
|
||||
self.nr_entries = len(self.entries)
|
||||
|
||||
def find_by_name_fragment(self, fragment):
|
||||
@@ -1055,18 +1049,17 @@ class cpusstats:
|
||||
def reload(self):
|
||||
last_entries = self.entries
|
||||
self.entries = {}
|
||||
- f = open(self.filename)
|
||||
- for line in f.readlines():
|
||||
- fields = line.strip().split()
|
||||
- if fields[0][:3].lower() != "cpu":
|
||||
- continue
|
||||
- c = cpustat(fields)
|
||||
- if c.name == "cpu":
|
||||
- idx = 0
|
||||
- else:
|
||||
- idx = int(c.name[3:]) + 1
|
||||
- self.entries[idx] = c
|
||||
- f.close()
|
||||
+ with open(self.filename) as f:
|
||||
+ for line in f.readlines():
|
||||
+ fields = line.strip().split()
|
||||
+ if fields[0][:3].lower() != "cpu":
|
||||
+ continue
|
||||
+ c = cpustat(fields)
|
||||
+ if c.name == "cpu":
|
||||
+ idx = 0
|
||||
+ else:
|
||||
+ idx = int(c.name[3:]) + 1
|
||||
+ self.entries[idx] = c
|
||||
last_time = self.time
|
||||
self.time = time.time()
|
||||
if last_entries:
|
||||
@@ -1082,8 +1075,7 @@ class cpusstats:
|
||||
(curr.nice - prev.nice) + \
|
||||
(curr.system - prev.system)
|
||||
curr.usage = (delta / interval_hz) * 100
|
||||
- if curr.usage > 100:
|
||||
- curr.usage = 100
|
||||
+ curr.usage = min(curr.usage, 100)
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
--
|
||||
2.31.1
|
||||
|
50
python-linux-procfs-pflags-Handle-pids-that-complete.patch
Normal file
50
python-linux-procfs-pflags-Handle-pids-that-complete.patch
Normal file
@ -0,0 +1,50 @@
|
||||
From eb984b30e325bbf27844bf9c1f90767504468db5 Mon Sep 17 00:00:00 2001
|
||||
From: John Kacur <jkacur@redhat.com>
|
||||
Date: Tue, 23 Nov 2021 13:01:05 -0500
|
||||
Subject: [PATCH 2/3] python-linux-procfs: pflags: Handle pids that completed
|
||||
|
||||
Sometimes pids disappear when they are completed.
|
||||
|
||||
Programs such as pflags that use procfs must account for that.
|
||||
The solution is to simply recognize this situation, and to continue.
|
||||
|
||||
Signed-off-by: John Kacur <jkacur@redhat.com>
|
||||
---
|
||||
pflags | 15 +++++++++++++--
|
||||
1 file changed, 13 insertions(+), 2 deletions(-)
|
||||
|
||||
diff --git a/pflags b/pflags
|
||||
index 3407b6f51c96..46d396c87c2b 100755
|
||||
--- a/pflags
|
||||
+++ b/pflags
|
||||
@@ -50,14 +50,25 @@ def main(argv):
|
||||
pids = list(ps.processes.keys())
|
||||
|
||||
pids.sort()
|
||||
- len_comms = [len(ps[pid]["stat"]["comm"]) for pid in pids if pid in ps]
|
||||
+ len_comms = []
|
||||
+ for pid in pids:
|
||||
+ if pid in ps:
|
||||
+ try:
|
||||
+ len(ps[pid]["stat"]["comm"])
|
||||
+ except (TypeError, FileNotFoundError):
|
||||
+ continue
|
||||
+ len_comms.append(len(ps[pid]["stat"]["comm"]))
|
||||
+
|
||||
max_comm_len = max(len_comms, default=0)
|
||||
del len_comms
|
||||
|
||||
for pid in pids:
|
||||
if pid not in ps:
|
||||
continue
|
||||
- flags = ps[pid].stat.process_flags()
|
||||
+ try:
|
||||
+ flags = ps[pid].stat.process_flags()
|
||||
+ except AttributeError:
|
||||
+ continue
|
||||
# Remove flags that were superseeded
|
||||
if "PF_THREAD_BOUND" in flags and "PF_NO_SETAFFINITY" in flags:
|
||||
flags.remove("PF_THREAD_BOUND")
|
||||
--
|
||||
2.31.1
|
||||
|
21
tests/scripts/run_tests.sh
Normal file
21
tests/scripts/run_tests.sh
Normal file
@ -0,0 +1,21 @@
|
||||
#!/usr/bin/bash
|
||||
|
||||
# make sure we have python-linux-procfs installed
|
||||
if rpm -q --quiet python3-linux-procfs; then
|
||||
:
|
||||
else
|
||||
sudo dnf install -y python3-linux-procfs
|
||||
if [[ $? != 0 ]]; then
|
||||
echo "install of python3-linux-procfs failed!"
|
||||
exit 1
|
||||
fi
|
||||
fi
|
||||
|
||||
# see if pflags is installed and executable
|
||||
pflags --help
|
||||
|
||||
if [[ $? != 0 ]]; then
|
||||
exit 2
|
||||
fi
|
||||
|
||||
exit 0
|
11
tests/tests.yml
Normal file
11
tests/tests.yml
Normal file
@ -0,0 +1,11 @@
|
||||
- hosts: localhost
|
||||
roles:
|
||||
- role: standard-test-basic
|
||||
tags:
|
||||
- classic
|
||||
tests:
|
||||
- simple:
|
||||
dir: .
|
||||
run: pflags --help
|
||||
required_packages:
|
||||
- python3-linux-procfs
|
Loading…
Reference in New Issue
Block a user