from __future__ import absolute_import
from six.moves import range
# Authors: John Dennis <jdennis@redhat.com>
#
# Copyright (C) 2006,2007,2008 Red Hat, Inc.
#
# This program 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 2 of the License, or
# (at your option) any later version.
#
# This program 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 this program; if not, write to the Free Software
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
__all__ = [
'audit_msg_decode',
'merge_lists',
'preextend_list',
'fmt_obj',
'format_elapsed_time',
'format_2_column_name_value',
'wrap_text',
'format_msg',
'remove_linebreaks',
'default_text',
'default_date_text',
'get_standard_directories',
'get_rpm_nvr_from_header',
'get_package_nvr_by_name',
'get_package_nvr_by_file_path',
'get_rpm_nvr_by_type',
'get_rpm_nvr_by_scontext',
'get_rpm_source_package',
'is_hex',
'split_rpm_nvr',
'file_types',
'get_user_home_dir',
'get_plugin_names',
'load_plugins',
'get_os_environment',
'find_program',
'get_identity',
'get_hostname',
'make_database_filepath',
'valid_email_address',
'launch_web_browser_on_url',
'abstract',
'log_debug',
'get_error_from_socket_exception',
'assure_file_ownership_permissions',
'parse_datetime_offset',
'DATABASE_MAJOR_VERSION',
'DATABASE_MINOR_VERSION',
'database_version_compatible',
'syslog_trace',
'TimeStamp',
'Retry',
'PACKAGE_MANAGER',
]
import bz2
import six
import datetime
from pydbus import SystemBus
import glob
from gi.repository import GObject
import os
import pwd
import re
import shutil
import selinux
import subprocess
import sys
import textwrap
import time
from types import *
import syslog
from functools import cmp_to_key
from types import FunctionType, MethodType
from setroubleshoot.config import get_config
from setroubleshoot.errcode import *
cmp = lambda x, y: (x > y) - (x < y)
def is_type(obj):
try:
return isinstance(obj, TypeType)
except NameError:
return isinstance(obj, type)
DATABASE_MAJOR_VERSION = 3
DATABASE_MINOR_VERSION = 0
PACKAGE_MANAGER = None
if shutil.which('rpm'):
PACKAGE_MANAGER = 'rpm'
elif shutil.which('dpkg'):
PACKAGE_MANAGER = 'deb'
redhat_release_path = '/etc/redhat-release'
text_wrapper = textwrap.TextWrapper()
fix_newline_re = re.compile(r"\s*\n+\s*")
hex_re = re.compile('^[A-Fa-f0-9]+$')
href_re = re.compile(r'<a\s*href="([^"]+)"[^<]*</a>')
name_at_domain_re = re.compile(r'^([^\s@]+)@([^\s@]+)$')
audit_decode_re = re.compile(r'^\s*"([^"]+)"\s*$')
# regexp matching lines containing type definitions, eg. (type lib_t)
# contains only 1 group that matches the type name
typedef_regexp = re.compile(r"\s*\(\s*type\s+([\w-]+)\s*\)\s*")
#Dictionary with all types defined in the module store as keys
#and corresponding module paths as values. Used by get_package_nvr_by_name
module_type_cache = None
log_level = syslog.LOG_WARNING
log_levels = {
'CRITICAL': syslog.LOG_CRIT,
'ERROR': syslog.LOG_ERR,
'WARNING': syslog.LOG_WARNING,
'INFO': syslog.LOG_INFO,
'DEBUG': syslog.LOG_DEBUG,
}
def log_init(section):
global log_level
log_level = log_levels.get(get_config(section, 'level').upper())
if log_level is None:
log_level = syslog.LOG_WARNING
def log_debug(msg):
global log_level
if log_level >= syslog.LOG_DEBUG:
syslog.syslog(syslog.LOG_DEBUG, msg)
def syslog_trace(trace):
log_lines = trace.split('\n')
for line in log_lines:
if len(line):
syslog.syslog(line)
def database_version_compatible(version):
major = minor = None
components = version.split('.')
if len(components) >= 1:
major = int(components[0])
if len(components) >= 2:
minor = int(components[1])
if major < DATABASE_MAJOR_VERSION:
log_debug("database version %s not compatible with current %d.%d version" % (version, DATABASE_MAJOR_VERSION, DATABASE_MINOR_VERSION))
return False
else:
log_debug("database version %s compatible with current %d.%d version" % (version, DATABASE_MAJOR_VERSION, DATABASE_MINOR_VERSION))
return True
def format_elapsed_time(elapsed_time):
if elapsed_time is None:
return None
import math
fraction, whole = math.modf(elapsed_time)
whole = int(whole)
days = whole / 86400
whole = whole - days * 86400
hours = whole / 3600
whole = whole - hours * 3600
minutes = whole / 60
seconds = whole - minutes * 60
if days:
return "%dd:%dh:%dm:%.3fs" % (days, hours, minutes, seconds + fraction)
if hours:
return "%dh:%dm:%.3fs" % (hours, minutes, seconds + fraction)
if minutes:
return "%dm:%.3fs" % (minutes, seconds + fraction)
return "%.3fs" % (seconds + fraction)
def is_hex(str):
if hex_re.match(str):
return True
else:
return False
def audit_msg_decode(msg):
if msg is None:
return None
match = audit_decode_re.search(msg)
if match:
decoded = match.group(1)
else:
try:
if sys.version_info[0] < 3:
# Produces normal string instead of unicode string which is not
# accepted by libselinux functions.
# This means that len(path) will return inaccurate results when
# the string contains special characters. Also individual bytes
# of path may not be printable.
decoded = msg.decode('hex')
else:
# produces str in python 3 and unicode string in python 2
decoded = bytearray.fromhex(msg).decode('utf-8')
except:
decoded = msg
return decoded
def merge_lists(a, b):
'return a list containing the unique members of a+b'
if not b:
return a
if not a:
return b
d = {}
for i in a:
d[i] = None
for i in b:
d[i] = None
m = list(d.keys())
return m
def preextend_list(requested_length, _list=None, default=None):
if _list is None:
_list = []
cur_length = len(_list)
delta = requested_length - cur_length
if delta > 0:
if is_type(default):
_list.extend([default() for x in range(delta)])
else:
_list.extend([default] * delta)
return _list
def fmt_obj(obj):
if isinstance(obj, six.string_types):
return obj
elif isinstance(obj, (list, tuple)):
return "[" + " ".join(["%s" % fmt_obj(x) for x in obj]) + "]"
elif isinstance(obj, dict):
keys = list(obj.keys())
keys.sort()
return "{" + " ".join(["%s=%s" % (fmt_obj(key), fmt_obj(obj[key])) for key in keys]) + "}"
else:
return str(obj)
def format_2_column_name_value(name, value, value_indent=30, page_width=80):
if len(name) >= value_indent:
initial_indent = name[0:value_indent - 1] + ' '
else:
initial_indent = name + ' ' * (value_indent - len(name))
if not value or value.isspace():
return initial_indent + value + '\n'
else:
text_wrapper.initial_indent = initial_indent
text_wrapper.subsequent_indent = ' ' * value_indent
text_wrapper.width = page_width
return text_wrapper.fill(value) + '\n'
def wrap_text(s, width=80, indent=0):
prefix = ' ' * indent
text_wrapper.initial_indent = prefix
text_wrapper.subsequent_indent = prefix
text_wrapper.width = width
return text_wrapper.fill(s) + '\n'
def format_msg(title, msg, indent=4):
if msg is None:
msg = ''
msg = msg.strip()
indentString = " " * indent
text_wrapper.initial_indent = indentString
text_wrapper.subsequent_indent = indentString
text_wrapper.width = 80
return title + "\n" + text_wrapper.fill(msg) + "\n\n"
def remove_linebreaks(str):
new_str = fix_newline_re.sub(" ", str).strip()
if new_str is None:
return ""
else:
return new_str
def default_text(val):
if val is None:
val = '<' + _('Unknown') + '>'
return str(val)
def default_date_text(date):
if date is None:
return default_text(date)
return date.format()
def get_standard_directories():
"""
>>> get_standard_directories()
[...'/bin'...]
"""
lst = []
try:
if PACKAGE_MANAGER == "rpm":
lst = subprocess.check_output(["rpm", "-ql", "filesystem"], universal_newlines=True).rstrip().split("\n")
if PACKAGE_MANAGER == "deb":
lst = subprocess.check_output(["dpkg", "-L", "base-files"], universal_newlines=True).rstrip().split("\n")
except:
syslog.syslog(syslog.LOG_ERR, "failed to get list from {}".format(PACKAGE_MANAGER))
return lst
def get_rpm_nvr_from_header(hdr):
'Given an RPM header return the package NVR as a string'
name = hdr['name']
version = hdr['version']
release = hdr['release']
return "%s-%s-%s" % (name, version, release)
def get_package_nvr_by_name(name):
"""
>>> get_package_nvr_by_name("coreutils")
'coreutils-8.30-3+b1:amd64'
"""
if name is None:
return None
nvr = None
try:
if PACKAGE_MANAGER == 'rpm':
nvr = subprocess.check_output(["rpm", "-q", name], universal_newlines=True).rstrip()
if PACKAGE_MANAGER == 'deb':
# dpkg-query -f='${Package}_${Version}:${Architecture}\n' -W
nvr = subprocess.check_output(
["dpkg-query", "-f=${Package}-${Version}:${Architecture}", "-W", name], universal_newlines=True
).rstrip()
except:
syslog.syslog(syslog.LOG_ERR, "failed to retrieve rpm info for %s" % name)
return nvr
def get_package_nvr_by_file_path(name):
"""
>>> get_package_nvr_by_file_path("/bin/ls")
'coreutils-8.30-3+b1:amd64'
"""
if name is None:
return None
name = os.path.abspath(name)
if not os.path.exists(name):
return None
nvr = None
try:
if PACKAGE_MANAGER == 'rpm':
nvr = subprocess.check_output(["rpm", "-qf", name], universal_newlines=True).rstrip()
if PACKAGE_MANAGER == 'deb':
# dpkg -S foo |cut -d: -1f
# dpkg-query -f='${Package}_${Version}:${Architecture}\n' -W
package_name = subprocess.check_output(["dpkg", "-S", name]).decode().split(": ")[0]
nvr = subprocess.check_output(
["dpkg-query", "-f=${Package}-${Version}:${Architecture}", "-W", package_name], universal_newlines=True
).rstrip()
except:
syslog.syslog(syslog.LOG_ERR, "failed to retrieve rpm info for %s" % name)
return nvr
###
try:
from sepolicy import get_all_file_types
file_types = get_all_file_types()
except:
file_types = []
def split_rpm_nvr(nvr):
components = nvr.split('-')
release = components[-1]
version = components[-2]
name = '-'.join(components[:-2])
return (name, version, release)
def get_rpm_nvr_by_type(selinux_type):
"""
Finds an SELinux module which defines given SELinux type
##### arguments
* `selinux_type(s)`: an SELinux type
##### return values
* `nvr(s)`: nvr of rpm which ships module where `selinux_type` is defined
##### usage
>>> get_rpm_nvr_by_type("sshd_t")
'selinux-policy-...
>>> get_rpm_nvr_by_type("mysqld_log_t")
'mysql-selinux-...
"""
if module_type_cache is None:
build_module_type_cache()
if module_type_cache is None:
return None
path = module_type_cache.get(selinux_type, None)
return get_package_nvr_by_file_path(path)
# check if given string represents an integer
def __str_is_int(str):
try:
int(str)
return True
except:
return False
def build_module_type_cache():
"""
Creates a dictionary with all types defined in the module store as keys
and corresponding module paths as values.
The dictionary is stored in "module_type_cache" to be used by
"get_rpm_nvr_by_type"
"""
retval, policytype = selinux.selinux_getpolicytype()
if retval != 0:
return
module_type_dict = dict()
priorities = []
# get list of module priorities, present in the module store, sorted by integer value
with os.scandir("/var/lib/selinux/{}/active/modules".format(policytype)) as module_store:
priorities = sorted([x.name for x in module_store if x.is_dir() and __str_is_int(x.name)], key = lambda x: int(x))
for dir in priorities:
# find individual modules in each priority and identify type definitions
for (dirpath, dirnames, filenames) in os.walk("/var/lib/selinux/{}/active/modules/{}".format(policytype,dir)):
if "cil" in filenames:
try:
try:
# cil files are bzip2'ed by default
f = bz2.open("{}/cil".format(dirpath), mode = 'rt')
except:
# maybe cil file is not bzip2'ed, try plain text
f = open("{}/cil".format(dirpath))
for line in f:
result = typedef_regexp.match(line)
if result:
module_type_dict[result.group(1)] = dirpath
f.close()
except:
# something's wrong, move on
# FIXME: log a problem?
pass
global module_type_cache
module_type_cache = module_type_dict
def get_rpm_nvr_by_scontext(scontext, use_dbus=False):
"""
Finds an SELinux module which defines given SELinux context
##### arguments
* `scontext(s)`: an SELinux context
##### return values
* `nvr(s)`: nvr of rpm which ships module where SELinux type used in `scontext` is defined
##### usage
>>> get_rpm_nvr_by_scontext("system_u:system_r:syslogd_t:s0")
'selinux-policy-...
>>> get_rpm_nvr_by_scontext("system_u:system_r:mysqld_log_t:s0")
'mysql-selinux-...
>>> get_rpm_nvr_by_scontext("system_u:system_r:timedatex_t:s0", use_dbus=True)
'selinux-policy-...
"""
if use_dbus:
bus = SystemBus()
remote_object = bus.get("org.fedoraproject.SetroubleshootPrivileged")
return str(remote_object.get_rpm_nvr_by_scontext(str(scontext)))
else:
context = selinux.context_new(str(scontext))
return get_rpm_nvr_by_type(str(selinux.context_type_get(context)))
def get_rpm_source_package(name):
"""
Find a source package for `name` rpm
>>> get_rpm_source_package("policycoreutils-python-utils")
'policycoreutils'
>>> get_rpm_source_package("selinux-policy-targeted")
'selinux-policy'
"""
if name is None:
return None
src = None
try:
import subprocess
src = subprocess.check_output(["rpm", "-q", "--qf", "%{SOURCERPM}", name], universal_newlines=True).rsplit('-',2)[0]
except:
syslog.syslog(syslog.LOG_ERR, "failed to retrieve rpm info for %s" % name)
return src
def get_user_home_dir():
uid = os.getuid()
try:
pw = pwd.getpwuid(uid)
except KeyError:
return None
home_dir = pw.pw_dir
return home_dir
def valid_email_address(address):
match = name_at_domain_re.search(address)
if match:
return True
else:
return False
def launch_web_browser_on_url(url):
web_browser_launcher = get_config('helper_apps', 'web_browser_launcher')
os.spawnl(os.P_NOWAIT, web_browser_launcher, web_browser_launcher, url)
def get_error_from_socket_exception(e):
args = getattr(e, 'args', None)
if args:
errno = args[0]
strerror = args[1]
else:
errno = ERR_SOCKET_ERROR
strerror = get_strerror(errno)
return errno, strerror
def assure_file_ownership_permissions(filepath, mode, owner, group=None):
result = True
if not os.path.exists(filepath):
try:
f = open(filepath, "w")
f.close()
except Exception as e:
result = False
syslog.syslog(syslog.LOG_ERR, "cannot create file %s [%s]" % (filepath, e.strerror))
try:
os.chmod(filepath, mode)
except OSError as e:
result = False
syslog.syslog(syslog.LOG_ERR, "cannot chmod %s to %o [%s]" % (filepath, mode, e.strerror))
try:
if isinstance(owner, int):
uid = owner
else:
uid = pwd.getpwnam(owner)[2]
if group is None:
group = owner
if isinstance(group, int):
gid = group
else:
import grp
gid = grp.getgrnam(group)[2]
os.chown(filepath, uid, gid)
except OSError as e:
result = False
import grp
syslog.syslog(syslog.LOG_ERR, "cannot chown %s to %s:%s [%s]" % (filepath, pwd.getpwuid(uid)[0], grp.getgrgid(gid)[0], e.strerror))
return result
def abstract(obj):
import inspect
method = inspect.getouterframes(inspect.currentframe())[1][3]
subclass = obj.__class__.__name__
raise NotImplementedError('%s must be implemented in subclass %s or ancestor class of %s' %
(method, subclass, subclass))
#-----------------------------------------------------------------------------
def get_plugin_names(filter_glob=None):
if filter_glob is None:
filter_glob = '*'
else:
filter_glob = re.sub('.py$', '', filter_glob)
plugin_dir = get_config('plugins', 'plugin_dir')
plugin_names = []
for p in glob.glob(os.path.join(plugin_dir, filter_glob + ".py")):
p = os.path.basename(p)
if p in ['__init__.py']:
continue
plugin_name = os.path.splitext(os.path.basename(p))[0]
plugin_names.append(plugin_name)
return plugin_names
def sort_plugins(x, y):
return x.get_priority() - y.get_priority()
def load_plugins(filter_glob=None):
plugin_dir = get_config('plugins', 'plugin_dir')
plugin_base = os.path.basename(plugin_dir)
plugins = []
plugin_names = get_plugin_names(filter_glob)
log_debug("load_plugins() names=%s" % plugin_names)
# load the parent (e.g. the package containing the submodules), required for python 2.5 and above
module_name = plugin_base
plugin_name = '__init__'
if module_name not in sys.modules:
try:
import imp
mod_fp, mod_path, mod_description = imp.find_module(plugin_name, [plugin_dir])
mod = imp.load_module(module_name, mod_fp, mod_path, mod_description)
except Exception:
syslog.syslog(syslog.LOG_ERR, "failed to initialize plugins in %s" % plugin_dir)
return []
if mod_fp:
mod_fp.close()
for plugin_name in plugin_names:
module_name = "%s.%s" % (plugin_base, plugin_name)
mod = sys.modules.get(module_name)
if mod is not None:
log_debug("load_plugins() %s previously imported" % module_name)
plugins.append(mod.plugin())
continue
try:
import imp
mod_fp, mod_path, mod_description = imp.find_module(plugin_name, [plugin_dir])
mod = imp.load_module(module_name, mod_fp, mod_path, mod_description)
plugins.append(mod.plugin())
except Exception:
syslog.syslog(syslog.LOG_ERR, "failed to load %s plugin" % plugin_name)
if mod_fp:
mod_fp.close()
plugins.sort(key=cmp_to_key(sort_plugins))
return plugins
def get_os_environment():
try:
myplatform = open(redhat_release_path).readlines()[0].strip()
except:
try:
import distro
myplatform = ' '.join(distro.linux_distribution())
except:
myplatform = "unknown"
# uname returns (sysname, nodename, release, version, machine)
uname = os.uname()
kernel_release = uname[2]
cpu = uname[4]
os_desc = "%s %s" % (kernel_release, cpu)
return (myplatform, os_desc)
def get_identity(uid=None):
if uid is None:
uid = os.getuid()
try:
pwd_entry = pwd.getpwuid(uid)
except KeyError:
return None
username = pwd_entry[0]
return username
def get_hostname():
try:
import socket as Socket
hostname = Socket.gethostname()
return hostname
except Exception as e:
syslog.syslog(syslog.LOG_ERR, "cannot lookup hostname: %s" % e)
return None
def find_program(prog):
if os.path.isabs(prog):
return prog
basename = os.path.basename(prog)
search_path = get_config('fix_command', 'prog_search_path').split(':')
for d in search_path:
path = os.path.join(d, basename)
if os.path.exists(path):
return path
return None
def make_database_filepath(name):
database_dir = get_config('database', 'database_dir')
# strip off extension if one was provided
name = re.sub('\\.xml$', '', name)
filename = name + '_database.xml'
filepath = os.path.join(database_dir, filename)
return filepath
def parse_datetime_offset(text):
'''The time offset may be specified as a sequence of integer unit pairs.
Units may be one of year,month,week,day,hour,minute,second and may optionally be plural.
Example: '2 weeks 1 day' sets the threshold at 15 days.
'''
# Note, this regexp anything to follow the unit except an integer
# thus plural 's', commas, whitespace
datetime_offset_re = re.compile(r'(\d+)\s*(year|month|week|day|hour|minute|second)')
found = False
days = 0
hours = 0
minutes = 0
seconds = 0
text = text.lower()
for match in datetime_offset_re.finditer(text):
if match:
found = True
quantity = int(match.group(1))
unit = match.group(2)
if unit is not None:
if unit == 'year':
days += quantity * 365
if unit == 'month':
days += quantity * 31
if unit == 'week':
days += quantity * 7
if unit == 'day':
days += quantity
if unit == 'hour':
hours += quantity
if unit == 'minute':
minutes += quantity
if unit == 'second':
seconds += quantity
if found:
td = datetime.timedelta(days=days, hours=hours, minutes=minutes, seconds=seconds)
log_debug("parse_datetime_offset(%s) = time delta %s" % (text, td))
return td
else:
syslog.syslog(syslog.LOG_ERR, "could not parse datetime offset (%s)" % text)
return None
#------------------------------------------------------------------------------
STDOFFSET = datetime.timedelta(seconds=-time.timezone)
if time.daylight:
DSTOFFSET = datetime.timedelta(seconds=-time.altzone)
else:
DSTOFFSET = STDOFFSET
DSTDIFF = DSTOFFSET - STDOFFSET
ZERO = datetime.timedelta(0)
HOUR = datetime.timedelta(hours=1)
# A class capturing the platform's idea of local time.
class LocalTimezone(datetime.tzinfo):
def utcoffset(self, dt):
if self._isdst(dt):
return DSTOFFSET
else:
return STDOFFSET
def dst(self, dt):
if self._isdst(dt):
return DSTDIFF
else:
return ZERO
def tzname(self, dt):
return time.tzname[self._isdst(dt)]
def _isdst(self, dt):
tt = (dt.year, dt.month, dt.day,
dt.hour, dt.minute, dt.second,
dt.weekday(), 0, -1)
stamp = time.mktime(tt)
tt = time.localtime(stamp)
return tt.tm_isdst > 0
class UTC(datetime.tzinfo):
"""UTC"""
def utcoffset(self, dt):
return datetime.timedelta(0)
def tzname(self, dt):
return "UTC"
def dst(self, dt):
return datetime.timedelta(0)
class TimeStamp:
# class variables
utc_tz = UTC()
local_tz = LocalTimezone()
iso8601_fmt = '%Y-%m-%dT%H:%M:%SZ'
locale_fmt = '%c'
def __init__(self, t=None):
if t is None:
self._dt = self.now(local=False)
elif isinstance(t, six.string_types):
self.parse(t)
elif isinstance(t, float):
self._dt = datetime.datetime.fromtimestamp(t, self.utc_tz)
elif isinstance(t, datetime.datetime):
self._dt = t
elif isinstance(t, TimeStamp):
self._dt = t._dt
else:
raise TypeError("must be string, float, datetime, or TimeStamp")
def __lt__(self, other):
if isinstance(other, TimeStamp):
return self._dt < other._dt
else:
return self._dt < other
def __add__(self, other):
if isinstance(other, TimeStamp):
return self._dt + other._dt
else:
return self._dt + other
def __iadd__(self, other):
if isinstance(other, TimeStamp):
self._dt += other._dt
else:
self._dt += other
return self
def __sub__(self, other):
if isinstance(other, TimeStamp):
return self._dt - other._dt
else:
return self._dt - other
def __isub__(self, other):
if isinstance(other, TimeStamp):
self._dt -= other._dt
else:
self._dt -= other
return self
def now(self, local=False):
if local:
return datetime.datetime.now(self.local_tz)
else:
return datetime.datetime.now(self.utc_tz)
def local(self):
return self._dt.astimezone(self.local_tz)
def __str__(self):
return self.format(self.iso8601_fmt, local=False)
def parse(self, str):
(year, month, day, hour, minute, second, weekday, yearday, dst) = \
time.strptime(str, self.iso8601_fmt)
self._dt = datetime.datetime(year, month, day, hour, minute, second,
0, self.utc_tz)
return self._dt
def add(self, days=0, hours=0, minutes=0, seconds=0):
self._dt += datetime.timedelta(days=days, hours=hours,
minutes=minutes, seconds=seconds)
def in_future(self):
now = self.now()
if now < self._dt:
return True
else:
return False
def in_past(self):
now = self.now()
if now >= self._dt:
return True
else:
return False
def format(self, fmt=None, local=True):
if fmt is None:
fmt = self.locale_fmt
if local:
return self.local().strftime(fmt)
else:
return self._dt.strftime(fmt)
#------------------------------------------------------------------------------
class Retry(GObject.GObject):
'''
A class which schedules attempts until one succeeds.
Intervals are expressed as floating point seconds.
The retry attempt will be scheduled in the future based on the
retry_interval which may be either a number of seconds or a
callable object returning the number of seconds. The callable
form of the retry_interval is useful when the interval should be
adjusted based on prior history or other external factors,
e.g. backing off the frequency of the retry attempts if initial
attempts fail.
The retry callback should return False if the attempt fails, in
which case it will be scheduled again in the future based on the
current value obtained from the retry_interval. If the retry
callback returns True it indicates the retry attempt succeeded and
no more attempts will be made.
Retry's are started with the start() method and continues until
the retry callback returns True or the stop() method is called. It
is always safe to call stop() even if a retry is not pending.
The retry callback, user_data and notify_interval may be specified
in either the class init() or in the start() method for convenience.
If notify_interval is set a 'pending_retry' signal will be emitted
every time the notification interval elapses, this provides a
countdown till the next retry attempt.
The signature of the retry callback is: callback(retry_obj, user_data)
The signature of the pending_retry signal handler is: callback(retry_obj, seconds_pending, user_data)
The signature of the retry interval function is: interval(retry_obj, user_data)
'''
__gsignals__ = {
'pending_retry': # callback(retry_object, seconds_pending, user_data)
(GObject.SignalFlags.RUN_LAST, None, (GObject.TYPE_FLOAT, GObject.TYPE_PYOBJECT,)),
}
def __init__(self, callback, retry_interval, user_data=None, notify_interval=None):
GObject.GObject.__init__(self)
self.callback = callback
self.retry_interval = retry_interval
self.user_data = user_data
self.failed_attempts = 0 # how many times retry has been attempted but failed
self.notify_interval = notify_interval # how often pending_retry signal is emitted
self.trigger_time = None # time in future when retry is attempted
self.timeout_id = None # alarm timeout id
def stop(self):
if self.timeout_id is not None:
GLib.source_remove(self.timeout_id)
self.timeout_id = None
def start(self, retry_interval=None, user_data=None, notify_interval=None):
if retry_interval is not None:
self.retry_interval = retry_interval
if user_data is not None:
self.user_data = user_data
if notify_interval is not None:
self.notify_interval = notify_interval
self.stop()
self.failed_attempts = 0
self._schedule_alarm(True)
def _schedule_alarm(self, new_retry=False):
now = time.time()
if new_retry:
self.trigger_time = now + self._get_retry_interval()
seconds_pending = self.trigger_time - now
if self.notify_interval:
self.emit('pending_retry', seconds_pending, self.user_data)
alarm_time = min(self.notify_interval, seconds_pending)
else:
alarm_time = seconds_pending
self.timeout_id = GObject.timeout_add(int(alarm_time * 1000), self._alarm_callback)
def _alarm_callback(self):
self.timeout_id = None
now = time.time()
seconds_pending = self.trigger_time - now
# If seconds_pending is less than 0 we've gone past the
# trigger point so attempt a retry because its overdue. If
# seconds_pending is 0 we've exactly hit the trigger point
# (not likely). If seconds_pending is positive the trigger
# point is in the future, however, due to (minor) scheduling
# inaccuracies if seconds_pending is a small positive number
# we assume this alarm is triggering the retry attempt even
# though it is slightly in the future.
if seconds_pending <= 0.005:
self._attempt_retry()
else:
self._schedule_alarm()
return False
def _attempt_retry(self):
if self.callback(self, self.user_data):
self.stop()
else:
self.failed_attempts += 1
self._schedule_alarm(True)
def _get_retry_interval(self):
if isinstance(interval_type, (MethodType, FunctionType)):
return self.retry_interval(self, self.user_data)
return self.retry_interval
GObject.type_register(Retry)
#-----------------------------------------------------------------------------