Files
entropy/lib/entropy/const.py

1685 lines
55 KiB
Python

# -*- coding: utf-8 -*-
"""
@author: Fabio Erculiani <lxnay@sabayon.org>
@contact: lxnay@sabayon.org
@copyright: Fabio Erculiani
@license: GPL-2
B{Entropy Framework constants module}.
This module contains all the Entropy constants used all around
the "entropy" package.
Some of the constants in this module are used as "default" for
the SystemSettings interface. So, make sure to read the documentation
of SystemSettings in the "entropy.core" module.
Even if possible, etpConst, etpUi, and etpSys objects
*SHOULD* be I{never ever modified manually}. This freedom could change
in future, so, if you want to produce a stable code, DON'T do that at all!
Basic Entropy constants handling functions are available in this module
and are all prefixed with "I{const_*}" or "I{initconfig_*}".
If you are writing a third party application, you should always try
to avoid to deal directly with functions here unless specified otherwise.
In fact, usually these here are wrapper in upper-level modules
(entropy.client, entropy.server, entropy.services).
"""
import sys
import os
import time
import codecs
import stat
import errno
import signal
import gzip
import bz2
import grp
import pwd
import tempfile
import traceback
import threading
try:
import thread
except ImportError:
# python 3.x
import _thread as thread
from entropy.i18n import _, ENCODING, RAW_ENCODING
# Setup debugger hook on SIGUSR1
def debug_signal(signum, frame):
import pdb
pdb.set_trace()
if os.getuid() == 0:
signal.signal(signal.SIGUSR1, debug_signal)
# Setup thread dump hook on SIGQUIT
def dump_signal(signum, frame, extended=True, stderr=sys.stderr):
def _std_print_err(msg):
stderr.write(msg + '\n')
stderr.flush()
_std_print_err("")
_std_print_err("")
_std_print_err("---- DUMP START [cut here] ----")
thread_count = 0
threads_map = dict((x.ident, x) for x in threading.enumerate())
for thread_id, stack in sys._current_frames().items():
thread_count += 1
thread_obj = threads_map.get(thread_id, "N/A")
_std_print_err("Thread: %s, object: %s" % (thread_id, thread_obj))
stack_list = []
_stack = stack
while True:
stack_list.append(_stack)
_stack = _stack.f_back
if _stack is None:
break
for filename, lineno, name, line in traceback.extract_stack(stack):
_std_print_err("File: '%s', line %d, in %s'" % (
filename, lineno, name,))
if line:
_std_print_err(" %s" % (line.rstrip(),))
else:
_std_print_err(" ???")
if not extended:
continue
try:
_stack = stack_list.pop()
except IndexError:
_stack = None
if _stack is None:
continue
for key, value in _stack.f_locals.items():
cur_str = "\t%20s = " % key
try:
cur_str += repr(value)
except (AttributeError, NameError, TypeError):
cur_str += "<ERROR WHILE PRINTING VALUE>"
_std_print_err(cur_str)
_std_print_err("--")
_std_print_err("")
_std_print_err("[thread count: %d]" % (thread_count,))
_std_print_err("---- DUMP END [cut here] ----")
_std_print_err("")
_installed_sigquit = False
if os.getuid() == 0:
_installed_sigquit = True
signal.signal(signal.SIGQUIT, dump_signal)
_uname_m = os.uname()[4]
_rootdir = os.getenv("ETP_ROOT", "").rstrip("/")
_arch_override_file = os.path.join("/", _rootdir, "etc/entropy/.arch")
ETP_ARCH_CONST = None
if os.path.isfile(_arch_override_file):
try:
with codecs.open(_arch_override_file, "r", encoding=ENCODING) \
as arch_f:
_arch_const = arch_f.readline().strip()
if _arch_const:
ETP_ARCH_CONST = _arch_const
except (IOError, OSError) as err:
const_debug_write("_init_", repr(err))
# ETP_ARCH_CONST setup
# add more arches here
ETP_ARCH_MAP = {
("i386", "i486", "i586", "i686",): "x86",
("x86_64",): "amd64",
("mips", "mips64",): "mips",
}
if ETP_ARCH_CONST is None:
for arches, arch in ETP_ARCH_MAP.items():
if _uname_m in arches:
ETP_ARCH_CONST = arch
break
_more_keywords = None
if _uname_m.startswith("arm"):
# ARM is "special", multiple subarches
# ahead, better use the full uname value
# and account "arm" to etpSys['keywords']
if ETP_ARCH_CONST is None:
ETP_ARCH_CONST = _uname_m
_more_keywords = set(["arm", "~arm"])
elif ETP_ARCH_CONST is None:
ETP_ARCH_CONST = "UNKNOWN"
etpSys = {
'archs': ['alpha', 'amd64', 'amd64-fbsd', 'arm', 'hppa', 'ia64', 'm68k',
'mips', 'ppc', 'ppc64', 's390', 'sh', 'sparc', 'sparc-fbsd', 'x86',
'x86-fbsd'],
'keywords': set([ETP_ARCH_CONST, "~"+ETP_ARCH_CONST]),
'api': '3',
'arch': ETP_ARCH_CONST,
'rootdir': _rootdir,
'serverside': False,
'unittest': False,
}
if _more_keywords is not None:
etpSys['keywords'] |= _more_keywords
# debug mode flag, will be triggered by ETP_DEBUG env var.
_DEBUG = os.getenv("ETP_DEBUG") is not None
if _DEBUG and not _installed_sigquit:
# install the dump signal function at
# SIGQUIT anyway if --debug is enabled
signal.signal(signal.SIGQUIT, dump_signal)
etpConst = {}
def initconfig_entropy_constants(rootdir):
"""
Main constants configurators, this is the only function that you should
call from the outside, anytime you want. it will reset all the variables
excluding those backed up previously.
@param rootdir: current root directory, if any, or ""
@type rootdir: string
@rtype: None
@return: None
@raise AttributeError: when specified rootdir is not a directory
"""
if rootdir and not os.path.isdir(rootdir):
raise AttributeError("not a valid chroot.")
# set env ROOT
# this way it doesn't need to be set around the code
os.environ['ROOT'] = rootdir + os.path.sep
# save backed up settings
if 'backed_up' in etpConst:
backed_up_settings = etpConst.pop('backed_up')
else:
backed_up_settings = {}
const_default_settings(rootdir)
const_read_entropy_release()
const_create_working_dirs()
# reflow back settings
etpConst.update(backed_up_settings)
etpConst['backed_up'] = backed_up_settings.copy()
# try to set proper permissions for /etc/entropy (at least group)
# /etc/entropy should be always writeable by "entropy" group !
# DO NOT FRIGGIN REMOVE
const_setup_perms(etpConst['confdir'], etpConst['entropygid'],
recursion = False)
# also setup /var/tmp/entropy if it doesn't exist.
# /var/tmp can be mounted on tmpfs
if not os.path.isdir(etpConst['entropyunpackdir']):
try:
const_setup_directory(etpConst['entropyunpackdir'])
except (OSError, IOError) as err:
sys.stderr.write("WARNING: cannot create %s: %s\n" % (
etpConst['entropyunpackdir'], repr(err),))
if sys.excepthook is sys.__excepthook__:
sys.excepthook = __const_handle_exception
def const_default_settings(rootdir):
"""
Initialization of all the Entropy base settings.
@param rootdir: current root directory, if any, or ""
@type rootdir: string
@rtype: None
@return: None
"""
original_rootdir = rootdir
if not rootdir.strip():
rootdir = os.path.sep
default_etp_dir = os.getenv(
'DEV_ETP_VAR_DIR',
os.path.join(rootdir, "var/lib/entropy"))
default_etp_dbdir_name = "database"
default_etp_run_dir = os.getenv(
'DEV_ETP_RUN_DIR',
os.path.join(rootdir, "run/entropy"))
default_etp_dbdir = os.path.join(
default_etp_dbdir_name, ETP_ARCH_CONST)
default_etp_dbfile = "packages.db"
default_etp_dbclientfile = "equo.db"
default_etp_client_repodir = "client"
default_etp_cachesdir = "caches"
default_etp_securitydir = "glsa"
default_etp_logdir = "logs"
default_etp_confdir = os.getenv(
'DEV_ETP_ETC_DIR',
os.path.join(rootdir, "etc/entropy"))
default_etp_syslogdir = os.getenv(
'DEV_ETP_LOG_DIR',
os.path.join(rootdir, "var/log/entropy"))
default_etp_vardir = os.getenv(
'DEV_ETP_TMP_DIR',
os.path.join(rootdir, "var/tmp/entropy"))
default_etp_tmpcache_dir = os.getenv('DEV_ETP_CACHE_DIR',
os.path.join(default_etp_dir, default_etp_cachesdir))
etpConst.clear()
my_const = {
'logging': {
'normal_loglevel_id': 1,
'verbose_loglevel_id': 2,
},
'backed_up': {},
# entropy default installation directory
'installdir': '/usr/lib/entropy',
# directory where entropy stores its configuration
'confdir': default_etp_confdir,
# name of the package sets directory
'confsetsdirname': "sets",
# used by entropy.spm to build pkgs relative URL metadata ("download",
# returned by EntropyRepository.retrieveDownloadURL())
'packagesrelativepath_basedir': "packages",
'packagesrelativepath_basedir_nonfree': "packages-nonfree",
'packagesrelativepath_basedir_restricted': "packages-restricted",
'packagesrelativepaths': ("packages", "packages-nonfree",
"packages-restricted"),
'packagesrelativepath_basename': ETP_ARCH_CONST,
'databaserelativepath_basedir': default_etp_dbdir_name,
'entropyrundir': default_etp_run_dir, # /run/entropy
'entropyworkdir': default_etp_dir, # Entropy workdir
# new (since 0.99.48) Entropy downloaded packages location
# equals to /var/lib/entropy/client/packages containing packages/,
# packages-nonfree/, packages-restricted/ etc
'entropypackagesworkdir': os.path.join(default_etp_dir,
default_etp_client_repodir, "packages"),
# Entropy unpack directory
'entropyunpackdir': default_etp_vardir,
# Entropy packages image directory
'entropyimagerelativepath': "image",
# entropy repository database upload timestamp
'etpdatabasetimestampfile': default_etp_dbfile+".timestamp",
# entropy repository database owned (in repo) package files
'etpdatabasepkglist': default_etp_dbfile+".pkglist",
# same for extra_download metadata
'etpdatabaseextrapkglist': default_etp_dbfile+".extra_pkglist",
# file containing a list of packages that are strictly
# required by the repository, thus forced
'etpdatabasesytemmaskfile': default_etp_dbfile+".system_mask",
'etpdatabasemaskfile': default_etp_dbfile+".mask",
'etpdatabasekeywordsfile': default_etp_dbfile+".keywords",
'etpdatabaseupdatefile': default_etp_dbfile+".repo_updates",
'etpdatabaselicwhitelistfile': default_etp_dbfile+".lic_whitelist",
'etpdatabasecriticalfile': default_etp_dbfile+".critical",
'etpdatabasemirrorsfile': default_etp_dbfile+".mirrors",
'etpdatabasefallbackmirrorsfile': default_etp_dbfile+".fallback_mirrors",
'etpdatabasewebservicesfile': default_etp_dbfile+".webservices",
# per-repository configuration file to list legally sensible pkgs
'etpdatabaserestrictedfile': default_etp_dbfile+".restricted",
# the local/remote database revision file
'etpdatabaserevisionfile': default_etp_dbfile+".revision",
# missing dependencies black list file
'etpdatabasemissingdepsblfile': default_etp_dbfile + \
".missing_deps_blacklist",
# compressed file that contains all the "meta"
# files in a repository dir
'etpdatabasemetafilesfile': default_etp_dbfile+".meta",
# file that contains a list of the "meta"
# files not available in the repository
'etpdatabasemetafilesnotfound': default_etp_dbfile+".meta_notfound",
# database file checksum
'etpdatabasehashfile': default_etp_dbfile+".md5",
# the remote database lock file
'etpdatabaselockfile': default_etp_dbfile+".lock",
# the remote database download lock file
'etpdatabasedownloadlockfile': default_etp_dbfile+".download.lock",
# eapi3 "there are updates" signal file
# used to let EAPI3 remote service daemon know about repository updates
'etpdatabaseeapi3updates': default_etp_dbfile+".eapi3_updates",
# "there are updates" signal file for webinstall packages
# can be used to trigger the generation of new webinstall files
'etpdatabasewebinstallupdates': default_etp_dbfile+".webinst_updates",
# repository GPG public key file
'etpdatabasegpgfile': "signature.asc",
'etpgpgextension': ".asc",
# Entropy Client GPG repositories keyring path
'etpclientgpgdir': default_etp_confdir+"/client-gpg-keys",
# when this file exists, the database is not synced
# anymore with the online one
'etpdatabasetaintfile': default_etp_dbfile+".tainted",
# Entropy sqlite database file default_etp_dir + \
# default_etp_dbdir+"/packages.db"
'etpdatabasefile': default_etp_dbfile,
# Entropy sqlite database file (gzipped)
'etpdatabasefilegzip': default_etp_dbfile+".gz",
'etpdatabasefilegziphash': default_etp_dbfile+".gz.md5",
# Entropy sqlite database file (bzipped2)
'etpdatabasefilebzip2': default_etp_dbfile+".bz2",
'etpdatabasefilebzip2hash': default_etp_dbfile+".bz2.md5",
# Entropy sqlite database file (gzipped)
'etpdatabasefilegziplight': default_etp_dbfile+".light.gz",
'etpdatabasefilehashgziplight': default_etp_dbfile+".light.gz.md5",
# Entropy sqlite database file (bzipped2)
'etpdatabasefilebzip2light': default_etp_dbfile+".light.bz2",
'etpdatabasefilehashbzip2light': default_etp_dbfile+".light.bz2.md5",
# Entropy sqlite database dump file (bzipped2)
'etpdatabasedumpbzip2': default_etp_dbfile+".dump.bz2",
'etpdatabasedumphashfilebz2': default_etp_dbfile+".dump.bz2.md5",
# Entropy sqlite database dump file (gzipped)
'etpdatabasedumpgzip': default_etp_dbfile+".dump.gz",
'etpdatabasedumphashfilegzip': default_etp_dbfile+".dump.gz.md5",
# Entropy sqlite database dump file
'etpdatabasedump': default_etp_dbfile+".dump",
# Entropy sqlite database dump file (bzipped2) light ver
'etpdatabasedumplightbzip2': default_etp_dbfile+".dumplight.bz2",
# Entropy sqlite database dump file (gzipped) light ver
'etpdatabasedumplightgzip': default_etp_dbfile+".dumplight.gz",
# Entropy sqlite database dump file, light ver (no content)
'etpdatabasedumplighthashfilebz2': default_etp_dbfile+".dumplight.bz2.md5",
'etpdatabasedumplighthashfilegzip': default_etp_dbfile+".dumplight.gz.md5",
'etpdatabasedumplight': default_etp_dbfile+".dumplight",
# expiration based server-side packages removal
'etpdatabaseexpbasedpkgsrm': default_etp_dbfile+".fatscope",
# Entropy default compressed database format
'etpdatabasefileformat': "bz2",
# Entropy compressed databases format support
'etpdatabasesupportedcformats': ["bz2", "gz"],
'etpdatabasecompressclasses': {
"bz2": (bz2.BZ2File, "unpack_bzip2", "etpdatabasefilebzip2",
"etpdatabasedumpbzip2", "etpdatabasedumphashfilebz2",
"etpdatabasedumplightbzip2", "etpdatabasedumplighthashfilebz2",
"etpdatabasefilebzip2light", "etpdatabasefilehashbzip2light",
"etpdatabasefilebzip2hash",),
"gz": (gzip.GzipFile, "unpack_gzip", "etpdatabasefilegzip",
"etpdatabasedumpgzip", "etpdatabasedumphashfilegzip",
"etpdatabasedumplightgzip", "etpdatabasedumplighthashfilegzip",
"etpdatabasefilegziplight", "etpdatabasefilehashgziplight",
"etpdatabasefilegziphash",)
},
# Distribution website URL
'distro_website_url': "http://www.sabayon.org",
'packages_website_url': "https://packages.sabayon.org",
'changelog_filename': "ChangeLog",
'changelog_filename_compressed': "ChangeLog.bz2",
'changelog_date_format': "%a, %d %b %Y %X +0000",
# enable/disable packages RSS feed feature
'rss-feed': True,
# default name of the RSS feed
'rss-name': "packages.rss",
# light version of rss-name
'rss-light-name': "updates.rss",
# default URL to the entropy web interface
# (overridden in reagent.conf)
'rss-base-url': "http://packages.sabayon.org/",
# default URL to the Operating System website
# (overridden in reagent.conf)
'rss-website-url': "http://www.sabayon.org/",
# xml file where will be dumped ServerInterface.rssMessages dictionary
'rss-dump-name': "rss_database_actions",
'rss-max-entries': 1000, # maximum rss entries
'rss-light-max-entries': 300, # max entries for the light version
'rss-managing-editor': "lxnay@sabayon.org", # updates submitter
# repository RSS-based notice board content
'rss-notice-board': "notice.rss",
# File containing user data related to repository notice board
'rss-notice-board-userdata': "notice.rss.userdata",
# default Entropy Client GPG support bit
'client_gpg': True,
# "or" dependencies support
# app-foo/foo;app-foo/abc?
'entropyordepsep': ";",
'entropyordepquestion': "?",
'entropyslotprefix': ":",
'entropytagprefix': "#",
'packagesetprefix': "@",
'entropyrepoprefix': "@",
'entropyrepoprefix_alt': "::",
'entropyrevisionprefix': "~",
'userpackagesetsid': "__user__",
'cachedumpext': ".dmp",
'packagesext': ".tbz2",
# extra download package file extension (mandatory)
'packagesextraext': ".tar.bz2",
'packagesdebugext': ".debug.tar.bz2", # .tar.bz2
'packagesext_webinstall': ".etp",
# entropy package files binary delta extension
'packagesdeltaext': ".edelta",
# entropy package files binary delta subdir
'packagesdeltasubdir': "deltas",
# Extension of the file that contains the checksum
# of its releated package file
'packagesmd5fileext': ".md5",
'packagessha512fileext': ".sha512",
'packagessha256fileext': ".sha256",
'packagessha1fileext': ".sha1",
# Supported Entropy Client package hashes encodings
'packagehashes': ("sha1", "sha256", "sha512", "gpg"),
# Used by Entropy client to override some digest checks
'packagemtimefileext': ".mtime",
# Extension of the file that "contains" expiration mtime
'packagesexpirationfileext': ".expired",
# Extension of the file that contains package file
# information, replacing the real tarballs to save space.
'packagesweakfileext': ".weak",
# number of days after a package will be removed from mirrors
'packagesexpirationdays': 15,
# name of the trigger file that would be executed
# by equo inside triggerTools
'triggername': "trigger",
'trigger_sh_interpreter': rootdir+"/usr/sbin/entropy.sh",
# entropy hardware hash generator executable
'etp_hw_hash_gen': rootdir+"/usr/bin/entropy_hwgen.sh",
# entropy client post valid branch migration (equo hop) script name
'etp_post_branch_hop_script': default_etp_dbfile+".post_branch.sh",
# entropy client post branch upgrade script
'etp_post_branch_upgrade_script': default_etp_dbfile+".post_upgrade.sh",
# previous branch file container
'etp_previous_branch_file': default_etp_confdir+"/.previous_branch",
'etp_in_branch_upgrade_file': default_etp_confdir+"/.in_branch_upgrade",
# entropy client post repository update script (this is executed
# every time)
'etp_post_repo_update_script': default_etp_dbfile+".post_update.sh",
# proxy configuration constants, used system wide
'proxy': {
'ftp': os.getenv("FTP_PROXY"),
'http': os.getenv("HTTP_PROXY"),
'rsync': os.getenv("RSYNC_PROXY"),
'username': None,
'password': None
},
# Entropy log level (default: 1 - see entropy.conf for more info)
'entropyloglevel': 1,
# Entropy Socket Interface log level
'socketloglevel': 2,
# Log dir where ebuilds store their stuff
'logdir': os.path.join(default_etp_dir, default_etp_logdir),
# Entropy system tools log directory
'syslogdir': default_etp_syslogdir,
'entropylogfile': os.path.join(
default_etp_syslogdir, "entropy.log"),
'securitylogfile': os.path.join(
default_etp_syslogdir, "security.log"),
'etpdatabaseclientdir': os.path.join(
default_etp_dir, default_etp_client_repodir,
default_etp_dbdir),
# path to equo.db - client side database file
'etpdatabaseclientfilepath': os.path.join(
default_etp_dir, default_etp_client_repodir,
default_etp_dbdir, default_etp_dbclientfile),
# prefix of database backups
'dbbackupprefix': 'entropy_backup_',
# Entropy database API revision
'etpapi': etpSys['api'],
# Entropy database API currently supported
'supportedapis': (1, 2, 3),
# contains the current running architecture
'currentarch': etpSys['arch'],
# Entropy supported Archs
'supportedarchs': etpSys['archs'],
# default choosen branch (overridden by setting in repositories.conf)
'branch': "5",
# default allowed package keywords
'keywords': etpSys['keywords'].copy(),
# allow multiple packages in single scope server-side?
# this makes possible to have multiple versions of packages
# and handle the removal through expiration (using creation date)
'expiration_based_scope': False,
# our official repository name
'defaultserverrepositoryid': None,
'officialrepositoryid': "sabayonlinux.org",
# tag to append to .tbz2 file before entropy database (must be 32bytes)
'databasestarttag': "|ENTROPY:PROJECT:DB:MAGIC:START|",
# option to keep a backup of config files after
# being overwritten by equo conf update
'filesbackup': True,
# option to enable Entropy Client splitdebug support
'splitdebug': False,
# directories where debug symbols are stored
'splitdebug_dirs': ("/usr/lib/debug",),
# option to enable forced installation of critical updates
'forcedupdates': True,
# collision protection option, see client.conf for more info
'collisionprotect': 1,
# this will be used to show the number of updated
# files at the end of the processes
'configprotectcounter': 0,
# default Entropy release version
'entropyversion': "1.0",
# default system name (overidden by entropy.conf settings)
'systemname': "Sabayon Linux",
# Product identificator (standard, professional...)
'product': "standard",
'systemroot': original_rootdir, # default system root
'uid': os.getuid(), # current running UID
'entropygid': None,
'entropygid_nopriv': None,
'sysgroup': "entropy",
'sysgroup_nopriv': "entropy-nopriv",
'sysuser_nopriv': "entropy-nopriv",
'sysuser_nopriv_fallback': "nobody",
'defaultumask': 0o22,
'storeumask': 0o02,
'gentle_nice': 15,
'current_nice': 0,
'default_nice': 0,
# Default download socket timeout for Entropy Client transceivers
'default_download_timeout': 30,
# Entropy package dependencies type identifiers
'dependency_type_ids': {
'rdepend_id': 0, # runtime dependencies
'pdepend_id': 1, # post dependencies
'mdepend_id': 2, # actually, this is entropy-only
'bdepend_id': 3, # build dependencies
},
'dependency_type_ids_desc': {
'rdepend_id': _("Runtime dependency"),
'pdepend_id': _("Post dependency"),
'mdepend_id': _('Manually added (by staff) dependency'),
'bdepend_id': _('Build dependency'),
},
# entropy client packages download speed limit (in kb/sec)
'downloadspeedlimit': None,
# data storage directory, useful to speed up
# entropy client across multiple issued commands
'dumpstoragedir': default_etp_tmpcache_dir,
# where GLSAs are stored
'securitydir': os.path.join(
default_etp_dir, default_etp_securitydir),
'securityurl': "http://community.sabayon.org/security"
"/security-advisories.tar.bz2",
'safemodeerrors': {
'clientdb': 1,
},
'safemodereasons': {
0: _("All fine"),
1: _("Corrupted Client Repository. Please restore a backup."),
},
'misc_counters': {
'forced_atoms_update_ids': {
'__idtype__': 1,
'kde': 1,
},
},
'system_settings_plugins_ids': {
'client_plugin': "client_plugin",
'server_plugin': "server_plugin",
'server_plugin_fatscope': "server_plugin_fatscope",
'server_plugin_fake_client': "server_plugin_fake_client",
},
# TODO: remove when Sulfur and PackageKit stopped using it
# keep in sync with InstalledPackagesRepository.NAME
'clientdbid': "__system__",
'spmdbid': "spm-db",
'spmetprev': 9999,
'systemreleasefile': "/etc/sabayon-release",
'install_sources': {
'unknown': 0,
'user': 1,
'automatic_dependency': 2,
},
'pkg_masking_reasons': {
0: _('reason not available'),
1: _('user package.mask'),
2: _('system keywords'),
3: _('user package.unmask'),
4: _('user repo package.keywords (all packages)'),
5: _('user repo package.keywords'),
6: _('user package.keywords'),
7: _('completely masked (by keyword?)'),
8: _('repository general packages.db.mask'),
9: _('repository general packages.db.keywords'),
10: _('user license.mask'),
11: _('user live unmask'),
12: _('user live mask'),
},
'pkg_masking_reference': {
'reason_not_avail': 0,
'user_package_mask': 1,
'system_keyword': 2,
'user_package_unmask': 3,
'user_repo_package_keywords_all': 4,
'user_repo_package_keywords': 5,
'user_package_keywords': 6,
'completely_masked': 7,
'repository_packages_db_mask': 8,
'repository_packages_db_keywords': 9,
'user_license_mask': 10,
'user_live_unmask': 11,
'user_live_mask': 12,
},
# default entropy configuration files encoding
'conf_encoding': ENCODING,
'conf_raw_encoding': RAW_ENCODING,
}
# set current nice level
try:
my_const['current_nice'] = os.nice(0)
except OSError:
pass
etpConst.update(my_const)
def const_is_python3():
"""
Return whether Python3 is interpreting this code.
"""
return sys.hexversion >= 0x3000000
def const_set_nice_level(nice_level = 0):
"""
Change current process scheduler "nice" level.
@param nice_level: new valid nice level
@type nice_level: int
@rtype: int
@return: current_nice new nice level
"""
default_nice = etpConst['default_nice']
current_nice = etpConst['current_nice']
delta = current_nice - default_nice
try:
etpConst['current_nice'] = os.nice(delta*-1+nice_level)
except OSError:
pass
return current_nice
def const_read_entropy_release():
"""
Read Entropy release file content and fill etpConst['entropyversion']
@rtype: None
@return: None
"""
# handle Entropy Version
revision_file = "../lib/entropy/revision"
if not os.path.isfile(revision_file):
revision_file = os.path.join(etpConst['installdir'],
'lib/entropy/revision')
try:
with codecs.open(revision_file, "r", encoding=ENCODING) as rev_f:
myrev = rev_f.readline().strip()
etpConst['entropyversion'] = myrev
except (OSError, IOError) as err:
if err.errno != errno.ENOENT:
raise
def const_pid_exists(pid):
"""
Determine whether given pid exists.
@param pid: process id
@type pid: int
@return: pid exists? 1; pid does not exist? 0
@rtype: int
"""
try:
os.kill(pid, signal.SIG_DFL)
return 1
except OverflowError:
# pid is invalid int, signed integer is greater than maximum
return 0
except OSError as err:
return err.errno == errno.EPERM
def const_secure_config_file(config_file):
"""
Setup entropy file needing strict permissions, no world readable.
@param config_file: valid config file path
@type config_file: string
@rtype: None
@return: None
"""
try:
mygid = const_get_entropy_gid()
except KeyError:
mygid = 0
try:
const_setup_file(config_file, mygid, 0o660)
except (OSError, IOError,):
pass
def const_drop_privileges(unpriv_uid = None, unpriv_gid = None):
"""
This function does its best to drop process privileges. If it fails, an
exception is raised. You can consider this function security-safe.
@param unpriv_uid: override default unprivileged uid
@type unpriv_uid: int
@param unpriv_gid: override default unprivileged gid
@type unpriv_gid: int
@raise entropy.exceptions.SecurityError: if unprivileged uid/gid cannot
be retrieived.
@raise ValueError: if program is already running as unprivileged user,
but this differs from the usual entropy unprivileged user.
@raise OSError: if privileges can't be dropped, the underlying syscall
fails.
@todo: when Python 2.7, see os.setresuid()
"""
cur_uid = os.getuid()
if unpriv_uid is None:
unpriv_uid = const_get_lazy_nopriv_uid()
if unpriv_gid is None:
unpriv_gid = const_get_lazy_nopriv_gid()
if cur_uid in (unpriv_uid, etpConst['sysuser_nopriv_fallback']):
# already unprivileged
return
elif cur_uid != 0:
raise ValueError("already running as another unprivileged user")
# privileges can be dropped
os.setregid(unpriv_gid, 0)
os.setreuid(unpriv_uid, 0) # real uid, effective uid
# make sure
if os.getuid() != unpriv_uid:
raise OSError("privileges (uid) have not been dropped")
if os.getgid() != unpriv_gid:
raise OSError("privileges (gid) have not been dropped")
etpConst['uid'] = unpriv_uid
def const_regain_privileges():
"""
This function should be called if, and only if, a previous
const_drop_privileges() has been called. It makes the program able to
get back privileges that were dropped previously.
@raise entropy.exceptions.SecurityError: if unprivileged uid/gid cannot
be retrieived.
@todo: when Python 2.7, see os.getresuid()
"""
cur_uid = os.getuid()
if cur_uid == 0:
# already running privileged
return
# privileges can be dropped
# set like this, otherwise we won't get back all our privs!
os.setreuid(0, 0) # real uid, effective uid
os.setregid(0, 0)
# make sure
if os.getuid() != 0:
raise OSError("privileges (uid) have not been dropped")
if os.getgid() != 0:
raise OSError("privileges (gid) have not been dropped")
etpConst['uid'] = 0
def const_setup_run_directory():
"""
Setup the Entropy /run directory with appropriate permissions.
"""
const_setup_directory(etpConst['entropyrundir'])
def const_create_working_dirs():
"""
Setup Entropy directory structure, as much automagically as possible.
@rtype: None
@return: None
"""
# create group if it doesn't exist
gid = None
try:
gid = const_get_entropy_gid()
except KeyError:
if etpConst['uid'] == 0:
_const_add_entropy_group(etpConst['sysgroup'])
try:
gid = const_get_entropy_gid()
except KeyError:
pass
# create unprivileged entropy-nopriv group
nopriv_gid = None
try:
nopriv_gid = const_get_entropy_nopriv_gid()
except KeyError:
if etpConst['uid'] == 0:
_const_add_entropy_group(etpConst['sysgroup_nopriv'])
try:
nopriv_gid = const_get_entropy_nopriv_gid()
except KeyError:
pass
if gid is not None:
etpConst['entropygid'] = gid
if nopriv_gid is not None:
etpConst['entropygid_nopriv'] = nopriv_gid
try:
const_setup_run_directory()
except (OSError, IOError):
# best effort
pass
def const_convert_log_level(entropy_log_level):
"""
Converts Entropy log levels (0, 1, 2) to logging.ERROR, logging.INFO,
logging.DEBUG.
@param entropy_log_level: entropy log level id (0, 1, 2), bogus values are
return logging.DEBUG
@type entropy_log_level: int
@return: logging.{ERROR,INFO,DEBUG} value
@rtype: int
"""
import logging
log_map = {
0: logging.ERROR,
1: logging.INFO,
2: logging.DEBUG
}
return log_map.get(entropy_log_level, logging.INFO)
def const_setup_perms(mydir, gid, f_perms = None, recursion = True, uid = -1):
"""
Setup permissions and group id (GID) to a directory, recursively.
@param mydir: valid file path
@type mydir: string
@param gid: valid group id (GID)
@type gid: int
@keyword f_perms: file permissions in octal type
@type f_perms: octal
@keyword recursion: set permissions recursively?
@type recursion: bool
@keyword uid: usually this argument shouldn't be used, but in cae
it sets the uid to the file
@type uid: int
@rtype: None
@return: None
"""
if gid == None:
return
if f_perms is None:
f_perms = 0o664
def do_setup_dir(currentdir):
try:
cur_gid = os.stat(currentdir)[stat.ST_GID]
if cur_gid != gid:
os.chown(currentdir, uid, gid)
cur_mod = const_get_chmod(currentdir)
if cur_mod != oct(0o775):
os.chmod(currentdir, 0o775)
except OSError:
pass
do_setup_dir(mydir)
if recursion:
for currentdir, subdirs, files in os.walk(mydir):
do_setup_dir(currentdir)
for item in files:
item = os.path.join(currentdir, item)
try:
const_setup_file(item, gid, f_perms, uid = uid)
except OSError:
pass
def const_setup_file(myfile, gid, chmod, uid = -1):
"""
Setup file permissions and group id (GID).
@param myfile: valid file path
@type myfile: string
@param gid: valid group id (GID)
@type gid: int
@param chmod: permissions
@type chmod: integer representing an octal
@keyword uid: usually this argument shouldn't be used, but in cae
it sets the uid to the file
@type uid: int
"""
cur_gid = os.stat(myfile)[stat.ST_GID]
if cur_gid != gid:
os.chown(myfile, uid, gid)
const_set_chmod(myfile, chmod)
def const_setup_directory(dirpath):
"""
Setup Entropy directory, creating it if required, changing
ownership and permissions as well.
@param dirpath: path to entropy directory
@type dirpath: string
@raise OSError: if permissions are fucked up
"""
try:
os.makedirs(dirpath)
except OSError as err:
if err.errno != errno.EEXIST:
raise
const_setup_perms(dirpath, etpConst['entropygid'],
recursion=False)
def const_mkdtemp(dir=None, prefix=None, suffix=None):
"""
mkdtemp() wrapper that creates a temporary directory inside
etpConst['entropyunpackdir'] if dir is None.
@keyword dir: TMPDIR
@type dir: string
@keyword prefix: the mkdtemp prefix argument
@type prefix: string
@keyword suffix: the mkdtemp suffix argument
@type suffix: string
@rtype: string
@return: the temporary directory path
"""
if dir is None:
dir = etpConst['entropyunpackdir']
# try creating it only if it's our provided dir
try:
os.makedirs(dir)
except OSError as err:
if err.errno != errno.EEXIST:
dir = os.getenv("TMPDIR", "/var/tmp")
if prefix is None:
prefix = ""
if suffix is None:
suffix = ""
return tempfile.mkdtemp(dir=dir, prefix=prefix, suffix=suffix)
def const_mkstemp(dir=None, prefix=None, suffix=None):
"""
mkstemp() wrapper that creates a temporary file inside
etpConst['entropyunpackdir'] if dir is None.
@keyword dir: TMPDIR
@type dir: string
@keyword prefix: the mkdtemp prefix argument
@type prefix: string
@keyword suffix: the mkdtemp suffix argument
@type suffix: string
@rtype: tuple
@return: the temporary file fd and path tuple
"""
if dir is None:
dir = etpConst['entropyunpackdir']
# try creating it only if it's our provided dir
try:
os.makedirs(dir)
except OSError as err:
if err.errno != errno.EEXIST:
dir = os.getenv("TMPDIR", "/var/tmp")
if prefix is None:
prefix = ""
if suffix is None:
suffix = ""
return tempfile.mkstemp(dir=dir, prefix=prefix, suffix=suffix)
def const_get_chmod(myfile):
"""
This function get the current permissions of the specified
file. If you want to use the returning value with const_set_chmod
you need to convert it back to int.
@param myfile: valid file path
@type myfile: string
@rtype: integer(8) (octal)
@return: octal representing permissions
"""
myst = os.stat(myfile)[stat.ST_MODE]
return oct(myst & 0o777)
def const_set_chmod(myfile, chmod):
"""
This function sets specified permissions to a file.
If they differ from the current ones.
@param myfile: valid file path
@type myfile: string
@param chmod: permissions
@type chmod: integer representing an octal
@rtype: None
@return: None
"""
cur_mod = const_get_chmod(myfile)
if cur_mod != oct(chmod):
os.chmod(myfile, chmod)
def _const_file_something(path, flags):
"""
Return whether file can be opened and is a regular one.
@param path: path to a file
@type path: string
@param flags: os.open() flags
@type flags: int
@return: True, if file exists and is readable
@rtype: bool
"""
fd = None
try:
fd = os.open(path, flags)
mode = os.fstat(fd).st_mode
if stat.S_ISREG(mode):
# this also handles symlinks to files
# because we don't use O_NOFOLLOW
return True
return False
except (OSError, IOError):
return False
finally:
if fd is not None:
os.close(fd)
def const_file_readable(path):
"""
Return whether path points to a readable file.
@param path: path to a file
@type path: string
@return: True, if file exists and is readable
@rtype: bool
"""
return _const_file_something(path, os.O_RDONLY)
def const_file_writable(path):
"""
Return whether path points to a writable file.
@param path: path to a file
@type path: string
@return: True, if file exists and is writable
@rtype: bool
"""
return _const_file_something(path, os.O_APPEND)
def _const_dir_something(path, flags):
"""
Return whether dir can be opened and is a directory.
@param path: path to a directory
@type path: string
@param flags: os.open() flags
@type flags: int
@return: True, if directory exists and can be opened with
the given flags
@rtype: bool
"""
fd = None
try:
fd = os.open(path, flags)
mode = os.fstat(fd).st_mode
if stat.S_ISDIR(mode):
# this also handles symlinks to files
# because we don't use O_NOFOLLOW
return True
return False
except (OSError, IOError):
return False
finally:
if fd is not None:
os.close(fd)
def const_dir_readable(dir_path):
"""
Return whether path points to a readable directory.
Readable directory is one that you can read the content.
@param path: path to a directory
@type path: string
@return: True, if directory exists and is readable
@rtype: bool
"""
return _const_dir_something(dir_path, os.O_RDONLY)
def const_dir_writable(dir_path):
"""
Return whether path points to a writable directory.
@param path: path to a directory
@type path: string
@return: True, if directory exists and is writable
@rtype: bool
"""
return _const_dir_something(dir_path, os.O_APPEND)
def const_get_entropy_gid():
"""
This function tries to retrieve the "entropy" user group
GID.
@rtype: int
@return: entropy group id
@raise KeyError: when "entropy" system GID is not available
"""
return int(grp.getgrnam(etpConst['sysgroup']).gr_gid)
def const_get_entropy_nopriv_gid():
"""
This function tries to retrieve the "entropy-nopriv" user group
GID. This is the unprivileged entropy users group.
@rtype: int
@return: entropy-nopriv group id
@raise KeyError: when "entropy-nopriv" system GID is not available
"""
return int(grp.getgrnam(etpConst['sysgroup_nopriv']).gr_gid)
def const_get_entropy_nopriv_uid():
"""
This function tries to retrieve the "entropy-nopriv" user id (uid).
@rtype: int
@return: entropy-nopriv user id
@raise KeyError: when "entropy-nopriv" system UID is not available
"""
return int(pwd.getpwnam(etpConst['sysuser_nopriv']).pw_uid)
def const_get_fallback_nopriv_uid():
"""
Fallback function that tries to retrieve the "nobody" user id (uid).
It is used when const_get_entropy_nopriv_uid() fails.
@rtype: int
@return: nobody user id
@raise KeyError: when "nobody" system UID is not available
"""
return int(pwd.getpwnam("nobody").pw_uid)
def const_get_lazy_nopriv_uid():
"""
This function returns an unprivileged uid by first trying to call
const_get_entropy_nopriv_uid() and then const_get_fallback_nopriv_uid()
@return: uid
@rtype: int
@raise entropy.exceptions.SecurityError: if unprivileged user id is not
available.
"""
unpriv_uid = None
try:
unpriv_uid = const_get_entropy_nopriv_uid()
except KeyError:
# fallback to "nobody"
try:
unpriv_uid = const_get_fallback_nopriv_uid()
except KeyError:
from entropy.exceptions import SecurityError
raise SecurityError("cannot find unprivileged user")
return unpriv_uid
def const_get_lazy_nopriv_gid():
"""
This function returns an unprivileged gid by first trying to call
const_get_entropy_nopriv_gid() and then const_get_fallback_nopriv_gid()
@return: uid
@rtype: int
@raise entropy.exceptions.SecurityError: if unprivileged group id is not
available.
"""
unpriv_gid = None
try:
unpriv_gid = const_get_entropy_nopriv_gid()
except KeyError:
try:
unpriv_gid = const_get_fallback_nopriv_gid()
except KeyError:
from entropy.exceptions import SecurityError
raise SecurityError("cannot find unprivileged group")
return unpriv_gid
def const_get_fallback_nopriv_gid():
"""
Fallback function that tries to retrieve the "nogroup" group id (gid).
It is used when const_get_entropy_nopriv_gid() fails.
@rtype: int
@return: nogroup user id
@raise KeyError: when "nogroup" system GID is not available
"""
return grp.getgrnam("nogroup").gr_gid
def _const_add_entropy_group(group_name):
"""
This function looks for an "entropy" user group.
If not available, it tries to create one.
@rtype: None
@return: None
@raise KeyError: if ${ROOT}/etc/group is not found
"""
group_file = etpConst['systemroot']+'/etc/group'
if not os.path.isfile(group_file):
raise KeyError
ids = set()
with codecs.open(group_file, "r", encoding=ENCODING) \
as group_f:
for line in group_f.readlines():
if line and line.split(":"):
try:
myid = int(line.split(":")[2])
except ValueError:
pass
ids.add(myid)
if ids:
# starting from 1000, get the first free
new_id = 1000
while True:
new_id += 1
if new_id not in ids:
break
else:
new_id = 10000
with codecs.open(group_file, "a", encoding=ENCODING) \
as group_fw:
group_fw.seek(0, 2)
app_line = "%s:x:%s:\n" % (group_name, new_id,)
group_fw.write(app_line)
def const_get_stringtype():
"""
Return generic string type for usage in isinstance().
On Python 2.x, it returns basestring while on Python 3.x it returns
(str, bytes,)
"""
if const_is_python3():
return (str, bytes,)
else:
return (basestring,)
def const_isstring(obj):
"""
Return whether obj is a string (unicode or raw).
@param obj: Python object
@type obj: Python object
@return: True, if object is string
@rtype: bool
"""
if const_is_python3():
return isinstance(obj, (str, bytes))
else:
return isinstance(obj, basestring)
def const_isunicode(obj):
"""
Return whether obj is a unicode.
@param obj: Python object
@type obj: Python object
@return: True, if object is unicode
@rtype: bool
"""
if const_is_python3():
return isinstance(obj, str)
else:
return isinstance(obj, unicode)
def const_israwstring(obj):
if const_is_python3():
return isinstance(obj, bytes)
else:
return isinstance(obj, str)
def const_convert_to_unicode(obj, enctype = RAW_ENCODING):
"""
Convert generic string to unicode format, this function supports both
Python 2.x and Python 3.x unicode bullshit.
@param obj: generic string object
@type obj: string
@return: unicode string object
@rtype: unicode object
"""
# None support
if obj is None:
if const_is_python3():
return "None"
else:
return unicode("None")
if const_isnumber(obj) or isinstance(obj, float):
if const_is_python3():
return str(obj)
else:
return unicode(obj)
if isinstance(obj, const_get_buffer()):
if const_is_python3():
return str(obj.tobytes(), enctype)
else:
return unicode(obj, enctype)
if const_isunicode(obj):
return obj
if hasattr(obj, 'decode'):
return obj.decode(enctype)
if const_is_python3():
return str(obj, enctype)
else:
return unicode(obj, enctype)
def const_convert_to_rawstring(obj, from_enctype = RAW_ENCODING):
"""
Convert generic string to raw string (str for Python 2.x or bytes for
Python 3.x).
@param obj: input string
@type obj: string object
@keyword from_enctype: encoding which string is using
@type from_enctype: string
@return: raw string
@rtype: bytes
"""
if obj is None:
return const_convert_to_rawstring("None")
if const_isnumber(obj) or isinstance(obj, float):
if const_is_python3():
return bytes(str(obj), from_enctype)
return str(obj)
if isinstance(obj, const_get_buffer()):
if const_is_python3():
return obj.tobytes()
return str(obj)
if not const_isunicode(obj):
return obj
return obj.encode(from_enctype)
def const_get_buffer():
"""
Return generic buffer object (supporting both Python 2.x and Python 3.x)
"""
if const_is_python3():
return memoryview
else:
return buffer
def const_get_int():
"""
Return generic int object (supporting both Python 2.x and Python 3.x).
For Python 2.x a (long, int) tuple is returned.
For Python 3.x a (int,) tuple is returned.
"""
if const_is_python3():
return (int,)
else:
return (long, int,)
def const_isfileobj(obj):
"""
Return whether obj is a file object
"""
if const_is_python3():
import io
return isinstance(obj, io.IOBase)
else:
return isinstance(obj, file)
def const_isnumber(obj):
"""
Return whether obj is an int, long object.
"""
if const_is_python3():
return isinstance(obj, int)
else:
return isinstance(obj, (int, long,))
def const_cmp(a, b):
"""
cmp() is gone in Python 3.x provide our own implementation.
"""
return (a > b) - (a < b)
_CMDLINE = None
def const_islive():
"""
Live environments (Operating System running off a CD/DVD)
must feature the "cdroot" parameter in kernel /proc/cmdline
Sample code:
>>> from entropy.const import const_islive
>>> const_islive()
False
@rtype: bool
@return: determine wether this is a Live system or not
"""
global _CMDLINE
if _CMDLINE is None:
try:
with codecs.open("/proc/cmdline", "r", encoding=ENCODING) \
as cmdline_f:
_CMDLINE = cmdline_f.readline().strip().split()
except IOError as err:
if err.errno not in (errno.EPERM, errno.ENOENT):
raise
_CMDLINE = []
return "cdroot" in _CMDLINE
def const_kill_threads(wait_seconds = 120.0):
"""
Entropy threads killer. Even if Python threads cannot
be stopped or killed, TimeScheduled ones can, exporting
the kill() method.
Sample code:
>>> from entropy.const import const_kill_threads
>>> const_kill_threads()
@param wait_seconds: number of seconds thread.join() should wait
@type wait_seconds: int
@rtype: None
@return: None
"""
threads = threading.enumerate()
for running_t in threads:
# do not join current thread
if running_t.getName() == 'MainThread':
continue
if hasattr(running_t, 'kill'):
running_t.kill()
if thread.get_ident() == running_t.ident:
# do not try to kill myself
continue
if running_t.daemon:
# will be killed by the VM
continue
running_t.join(wait_seconds) # wait n seconds?
def __const_handle_exception(etype, value, t_back):
"""
Our default Python exception handler. It kills
all the threads generated by Entropy before
raising exceptions. Overloads sys.excepthook,
internal function !!
@param etype: exception type
@type etype: exception type
@param value: exception data
@type value: string
@param t_back: traceback object?
@type t_back: Python traceback object
@rtype: default Python exceptions hook
@return: sys.__excepthook__
"""
try:
const_kill_threads()
except (AttributeError, ImportError, TypeError,):
pass
return sys.__excepthook__(etype, value, t_back)
def const_debug_enabled():
"""
Return whether debug is enabled.
@return: True, if debug is enabled
@rtype: bool
"""
return _DEBUG
_DEBUG_W_LOCK = threading.Lock()
def const_debug_write(identifier, msg, force = False, stdout=None):
"""
Entropy debugging output write functions.
@param identifier: debug identifier
@type identifier: string
@param msg: debugging message
@type msg: string
@keyword force: force print even if debug mode is off
@type force: bool
@keyword stdout: provide an alternative stdout file object
@type stdout: file object or None
@rtype: None
@return: None
"""
if const_debug_enabled() or force:
if stdout is None:
stdout = sys.stdout
# XXX: hierarchy violation, but hey, we're debugging shit
from entropy.output import brown, purple, teal, darkgreen, darkred
current_thread = threading.current_thread()
th_identifier = "[id:%s, name:%s, daemon:%s, ts:%s] %s" % (
brown(repr(current_thread.ident)),
purple(repr(current_thread.name)),
teal(repr(current_thread.daemon)), darkgreen(repr(time.time())),
darkred(identifier),)
with _DEBUG_W_LOCK:
if const_is_python3():
stdout.buffer.write(
const_convert_to_rawstring(th_identifier) + \
b" " + const_convert_to_rawstring(msg) + b"\n")
else:
stdout.write("%s: %s" % (th_identifier, msg + "\n"))
stdout.flush()
def const_get_caller():
"""
When called inside a function, return the caller function name.
@return: caller function name
@rtype: string
"""
import inspect
stack = inspect.stack()
try:
func = stack[3][3]
if func != "<module>":
return func
raise IndexError("module wtf?")
except IndexError:
return stack[2][3]
def const_get_stack():
"""
Return current function stack in form of list of tuples
@return: current function stack
@rtype: list
"""
import inspect
return inspect.stack()
def const_get_cpus():
"""
Return the number of CPUs/Cores the Operating system exposes
@return: number of CPUs/Cores available
@rtype: int
"""
import multiprocessing
return multiprocessing.cpu_count()
# load config
initconfig_entropy_constants(etpSys['rootdir'])
# Debug Watchdog support. If enabled, a thread dump
# will be pushed to stderr every ETP_DEBUG_WATCHDOG_INTERVAL
# seconds (or 60 seconds if unset).
_debug_watchdog = os.getenv("ETP_DEBUG_WATCHDOG")
if _debug_watchdog is not None:
from threading import Timer
_default_debug_watchdog_interval = 60
_debug_watchdog_interval = os.getenv(
"ETP_DEBUG_WATCHDOG_INTERVAL",
_default_debug_watchdog_interval)
try:
_debug_watchdog_interval = int(_debug_watchdog_interval)
except (ValueError, TypeError):
_debug_watchdog_interval = _default_debug_watchdog_interval
const_debug_write(
__name__,
"DebugWatchdogTimer enabled, interval: %d" % (
_debug_watchdog_interval,))
def _dumper():
dump_signal(None, None)
_setup_timer()
def _setup_timer():
_timer = Timer(_debug_watchdog_interval, _dumper)
_timer.name = "DebugWatchdogTimer"
_timer.daemon = True
_timer.start()
_setup_timer()