This updates Python2 print statements to Python3 print functions, and makes lists out of some things that are iterators in Python3. The latter we could not bother with as some code is fine with iterators, but it does keep the script behaving exactly as it was in case anyone does try to use this. (and it's clear it was purely 2to3 changes, no hand editing)
277 lines
10 KiB
Python
Executable File
277 lines
10 KiB
Python
Executable File
#!/usr/bin/env python
|
|
|
|
import os
|
|
import re
|
|
import sys
|
|
|
|
def _write_message(kind, message):
|
|
import inspect, os, sys
|
|
|
|
# Get the file/line where this message was generated.
|
|
f = inspect.currentframe()
|
|
# Step out of _write_message, and then out of wrapper.
|
|
f = f.f_back.f_back
|
|
file,line,_,_,_ = inspect.getframeinfo(f)
|
|
location = '%s:%d' % (os.path.basename(file), line)
|
|
|
|
print('%s: %s: %s' % (location, kind, message), file=sys.stderr)
|
|
|
|
note = lambda message: _write_message('note', message)
|
|
warning = lambda message: _write_message('warning', message)
|
|
error = lambda message: (_write_message('error', message), sys.exit(1))
|
|
|
|
def re_full_match(pattern, str):
|
|
m = re.match(pattern, str)
|
|
if m and m.end() != len(str):
|
|
m = None
|
|
return m
|
|
|
|
def parse_time(value):
|
|
minutes,value = value.split(':',1)
|
|
if '.' in value:
|
|
seconds,fseconds = value.split('.',1)
|
|
else:
|
|
seconds = value
|
|
return int(minutes) * 60 + int(seconds) + float('.'+fseconds)
|
|
|
|
def extractExecutable(command):
|
|
"""extractExecutable - Given a string representing a command line, attempt
|
|
to extract the executable path, even if it includes spaces."""
|
|
|
|
# Split into potential arguments.
|
|
args = command.split(' ')
|
|
|
|
# Scanning from the beginning, try to see if the first N args, when joined,
|
|
# exist. If so that's probably the executable.
|
|
for i in range(1,len(args)):
|
|
cmd = ' '.join(args[:i])
|
|
if os.path.exists(cmd):
|
|
return cmd
|
|
|
|
# Otherwise give up and return the first "argument".
|
|
return args[0]
|
|
|
|
class Struct:
|
|
def __init__(self, **kwargs):
|
|
self.fields = list(kwargs.keys())
|
|
self.__dict__.update(kwargs)
|
|
|
|
def __repr__(self):
|
|
return 'Struct(%s)' % ', '.join(['%s=%r' % (k,getattr(self,k))
|
|
for k in self.fields])
|
|
|
|
kExpectedPSFields = [('PID', int, 'pid'),
|
|
('USER', str, 'user'),
|
|
('COMMAND', str, 'command'),
|
|
('%CPU', float, 'cpu_percent'),
|
|
('TIME', parse_time, 'cpu_time'),
|
|
('VSZ', int, 'vmem_size'),
|
|
('RSS', int, 'rss')]
|
|
def getProcessTable():
|
|
import subprocess
|
|
p = subprocess.Popen(['ps', 'aux'], stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE)
|
|
out,err = p.communicate()
|
|
res = p.wait()
|
|
if p.wait():
|
|
error('unable to get process table')
|
|
elif err.strip():
|
|
error('unable to get process table: %s' % err)
|
|
|
|
lns = out.split('\n')
|
|
it = iter(lns)
|
|
header = it.next().split()
|
|
numRows = len(header)
|
|
|
|
# Make sure we have the expected fields.
|
|
indexes = []
|
|
for field in kExpectedPSFields:
|
|
try:
|
|
indexes.append(header.index(field[0]))
|
|
except:
|
|
if opts.debug:
|
|
raise
|
|
error('unable to get process table, no %r field.' % field[0])
|
|
|
|
table = []
|
|
for i,ln in enumerate(it):
|
|
if not ln.strip():
|
|
continue
|
|
|
|
fields = ln.split(None, numRows - 1)
|
|
if len(fields) != numRows:
|
|
warning('unable to process row: %r' % ln)
|
|
continue
|
|
|
|
record = {}
|
|
for field,idx in zip(kExpectedPSFields, indexes):
|
|
value = fields[idx]
|
|
try:
|
|
record[field[2]] = field[1](value)
|
|
except:
|
|
if opts.debug:
|
|
raise
|
|
warning('unable to process %r in row: %r' % (field[0], ln))
|
|
break
|
|
else:
|
|
# Add our best guess at the executable.
|
|
record['executable'] = extractExecutable(record['command'])
|
|
table.append(Struct(**record))
|
|
|
|
return table
|
|
|
|
def getSignalValue(name):
|
|
import signal
|
|
if name.startswith('SIG'):
|
|
value = getattr(signal, name)
|
|
if value and isinstance(value, int):
|
|
return value
|
|
error('unknown signal: %r' % name)
|
|
|
|
import signal
|
|
kSignals = {}
|
|
for name in dir(signal):
|
|
if name.startswith('SIG') and name == name.upper() and name.isalpha():
|
|
kSignals[name[3:]] = getattr(signal, name)
|
|
|
|
def main():
|
|
global opts
|
|
from optparse import OptionParser, OptionGroup
|
|
parser = OptionParser("usage: %prog [options] {pid}*")
|
|
|
|
# FIXME: Add -NNN and -SIGNAME options.
|
|
|
|
parser.add_option("-s", "", dest="signalName",
|
|
help="Name of the signal to use (default=%default)",
|
|
action="store", default='INT',
|
|
choices=list(kSignals.keys()))
|
|
parser.add_option("-l", "", dest="listSignals",
|
|
help="List known signal names",
|
|
action="store_true", default=False)
|
|
|
|
parser.add_option("-n", "--dry-run", dest="dryRun",
|
|
help="Only print the actions that would be taken",
|
|
action="store_true", default=False)
|
|
parser.add_option("-v", "--verbose", dest="verbose",
|
|
help="Print more verbose output",
|
|
action="store_true", default=False)
|
|
parser.add_option("", "--debug", dest="debug",
|
|
help="Enable debugging output",
|
|
action="store_true", default=False)
|
|
parser.add_option("", "--force", dest="force",
|
|
help="Perform the specified commands, even if it seems like a bad idea",
|
|
action="store_true", default=False)
|
|
|
|
inf = float('inf')
|
|
group = OptionGroup(parser, "Process Filters")
|
|
group.add_option("", "--name", dest="execName", metavar="REGEX",
|
|
help="Kill processes whose name matches the given regexp",
|
|
action="store", default=None)
|
|
group.add_option("", "--exec", dest="execPath", metavar="REGEX",
|
|
help="Kill processes whose executable matches the given regexp",
|
|
action="store", default=None)
|
|
group.add_option("", "--user", dest="userName", metavar="REGEX",
|
|
help="Kill processes whose user matches the given regexp",
|
|
action="store", default=None)
|
|
group.add_option("", "--min-cpu", dest="minCPU", metavar="PCT",
|
|
help="Kill processes with CPU usage >= PCT",
|
|
action="store", type=float, default=None)
|
|
group.add_option("", "--max-cpu", dest="maxCPU", metavar="PCT",
|
|
help="Kill processes with CPU usage <= PCT",
|
|
action="store", type=float, default=inf)
|
|
group.add_option("", "--min-mem", dest="minMem", metavar="N",
|
|
help="Kill processes with virtual size >= N (MB)",
|
|
action="store", type=float, default=None)
|
|
group.add_option("", "--max-mem", dest="maxMem", metavar="N",
|
|
help="Kill processes with virtual size <= N (MB)",
|
|
action="store", type=float, default=inf)
|
|
group.add_option("", "--min-rss", dest="minRSS", metavar="N",
|
|
help="Kill processes with RSS >= N",
|
|
action="store", type=float, default=None)
|
|
group.add_option("", "--max-rss", dest="maxRSS", metavar="N",
|
|
help="Kill processes with RSS <= N",
|
|
action="store", type=float, default=inf)
|
|
group.add_option("", "--min-time", dest="minTime", metavar="N",
|
|
help="Kill processes with CPU time >= N (seconds)",
|
|
action="store", type=float, default=None)
|
|
group.add_option("", "--max-time", dest="maxTime", metavar="N",
|
|
help="Kill processes with CPU time <= N (seconds)",
|
|
action="store", type=float, default=inf)
|
|
parser.add_option_group(group)
|
|
|
|
(opts, args) = parser.parse_args()
|
|
|
|
if opts.listSignals:
|
|
items = [(v,k) for k,v in list(kSignals.items())]
|
|
items.sort()
|
|
for i in range(0, len(items), 4):
|
|
print('\t'.join(['%2d) SIG%s' % (k,v)
|
|
for k,v in items[i:i+4]]))
|
|
sys.exit(0)
|
|
|
|
# Figure out the signal to use.
|
|
signal = kSignals[opts.signalName]
|
|
signalValueName = str(signal)
|
|
if opts.verbose:
|
|
name = dict((v,k) for k,v in list(kSignals.items())).get(signal,None)
|
|
if name:
|
|
signalValueName = name
|
|
note('using signal %d (SIG%s)' % (signal, name))
|
|
else:
|
|
note('using signal %d' % signal)
|
|
|
|
# Get the pid list to consider.
|
|
pids = set()
|
|
for arg in args:
|
|
try:
|
|
pids.add(int(arg))
|
|
except:
|
|
parser.error('invalid positional argument: %r' % arg)
|
|
|
|
filtered = ps = getProcessTable()
|
|
|
|
# Apply filters.
|
|
if pids:
|
|
filtered = [p for p in filtered
|
|
if p.pid in pids]
|
|
if opts.execName is not None:
|
|
filtered = [p for p in filtered
|
|
if re_full_match(opts.execName,
|
|
os.path.basename(p.executable))]
|
|
if opts.execPath is not None:
|
|
filtered = [p for p in filtered
|
|
if re_full_match(opts.execPath, p.executable)]
|
|
if opts.userName is not None:
|
|
filtered = [p for p in filtered
|
|
if re_full_match(opts.userName, p.user)]
|
|
filtered = [p for p in filtered
|
|
if opts.minCPU <= p.cpu_percent <= opts.maxCPU]
|
|
filtered = [p for p in filtered
|
|
if opts.minMem <= float(p.vmem_size) / (1<<20) <= opts.maxMem]
|
|
filtered = [p for p in filtered
|
|
if opts.minRSS <= p.rss <= opts.maxRSS]
|
|
filtered = [p for p in filtered
|
|
if opts.minTime <= p.cpu_time <= opts.maxTime]
|
|
|
|
if len(filtered) == len(ps):
|
|
if not opts.force and not opts.dryRun:
|
|
error('refusing to kill all processes without --force')
|
|
|
|
if not filtered:
|
|
warning('no processes selected')
|
|
|
|
for p in filtered:
|
|
if opts.verbose:
|
|
note('kill(%r, %s) # (user=%r, executable=%r, CPU=%2.2f%%, time=%r, vmem=%r, rss=%r)' %
|
|
(p.pid, signalValueName, p.user, p.executable, p.cpu_percent, p.cpu_time, p.vmem_size, p.rss))
|
|
if not opts.dryRun:
|
|
try:
|
|
os.kill(p.pid, signal)
|
|
except OSError:
|
|
if opts.debug:
|
|
raise
|
|
warning('unable to kill PID: %r' % p.pid)
|
|
|
|
if __name__ == '__main__':
|
|
main()
|