Mini Shell
"""
Helpful decorators for module writing
"""
import errno
import inspect
import logging
import subprocess
import sys
import time
from collections import defaultdict
from functools import wraps
import salt.utils.args
import salt.utils.data
import salt.utils.versions
from salt.exceptions import (
CommandExecutionError,
SaltConfigurationError,
SaltInvocationError,
)
IS_WINDOWS = False
if getattr(sys, "getwindowsversion", False):
IS_WINDOWS = True
log = logging.getLogger(__name__)
class Depends:
"""
This decorator will check the module when it is loaded and check that the
dependencies passed in are in the globals of the module. If not, it will
cause the function to be unloaded (or replaced).
"""
# kind -> Dependency -> list of things that depend on it
dependency_dict = defaultdict(lambda: defaultdict(dict))
def __init__(self, *dependencies, **kwargs):
"""
The decorator is instantiated with a list of dependencies (string of
global name)
An example use of this would be:
.. code-block:: python
@depends('modulename')
def test():
return 'foo'
OR
@depends('modulename', fallback_function=function)
def test():
return 'foo'
.. code-block:: python
This can also be done with the retcode of a command, using the
``retcode`` argument:
@depends('/opt/bin/check_cmd', retcode=0)
def test():
return 'foo'
It is also possible to check for any nonzero retcode using the
``nonzero_retcode`` argument:
@depends('/opt/bin/check_cmd', nonzero_retcode=True)
def test():
return 'foo'
.. note::
The command must be formatted as a string, not a list of args.
Additionally, I/O redirection and other shell-specific syntax are
not supported since this uses shell=False when calling
subprocess.Popen().
"""
log.trace(
"Depends decorator instantiated with dep list of %s and kwargs %s",
dependencies,
kwargs,
)
self.dependencies = dependencies
self.params = kwargs
def __call__(self, function):
"""
The decorator is "__call__"d with the function, we take that function
and determine which module and function name it is to store in the
class wide dependency_dict
"""
try:
# This inspect call may fail under certain conditions in the loader.
# Possibly related to a Python bug here:
# http://bugs.python.org/issue17735
frame = inspect.currentframe().f_back
# due to missing *.py files under esky we cannot use inspect.getmodule
# module name is something like salt.loaded.int.modules.test
_, kind, mod_name = frame.f_globals["__name__"].rsplit(".", 2)
fun_name = function.__name__
for dep in self.dependencies:
self.dependency_dict[kind][dep][(mod_name, fun_name)] = (
frame,
self.params,
)
except Exception as exc: # pylint: disable=broad-except
log.exception(
"Exception encountered when attempting to inspect frame in "
"dependency decorator"
)
return function
@staticmethod
def run_command(dependency, mod_name, func_name):
full_name = f"{mod_name}.{func_name}"
log.trace("Running '%s' for '%s'", dependency, full_name)
if IS_WINDOWS:
args = salt.utils.args.shlex_split(dependency, posix=False)
else:
args = salt.utils.args.shlex_split(dependency)
log.trace("Command after shlex_split: %s", args)
proc = subprocess.Popen(
args, shell=False, stdout=subprocess.PIPE, stderr=subprocess.STDOUT
)
output = proc.communicate()[0]
retcode = proc.returncode
log.trace("Output from '%s': %s", dependency, output)
log.trace("Retcode from '%s': %d", dependency, retcode)
return retcode
@classmethod
def enforce_dependencies(cls, functions, kind, tgt_mod):
"""
This is a class global method to enforce the dependencies that you
currently know about.
It will modify the "functions" dict and remove/replace modules that
are missing dependencies.
"""
for dependency, dependent_dict in cls.dependency_dict[kind].items():
for (mod_name, func_name), (frame, params) in dependent_dict.items():
if mod_name != tgt_mod:
continue
# Imports from local context take presedence over those from the global context.
dep_found = frame.f_locals.get(dependency) or frame.f_globals.get(
dependency
)
# Default to version ``None`` if not found, which will be less than anything.
dep_version = getattr(dep_found, "__version__", None)
if "retcode" in params or "nonzero_retcode" in params:
try:
retcode = cls.run_command(dependency, mod_name, func_name)
except OSError as exc:
if exc.errno == errno.ENOENT:
log.trace(
"Failed to run command %s, %s not found",
dependency,
exc.filename,
)
else:
log.trace("Failed to run command '%s': %s", dependency, exc)
retcode = -1
if "retcode" in params:
if params["retcode"] == retcode:
continue
elif "nonzero_retcode" in params:
if params["nonzero_retcode"]:
if retcode != 0:
continue
else:
if retcode == 0:
continue
# check if dependency is loaded
elif dependency is True:
log.trace(
"Dependency for %s.%s exists, not unloading",
mod_name,
func_name,
)
continue
# check if you have the dependency
elif dep_found:
if "version" in params:
if (
salt.utils.versions.version_cmp(
dep_version, params["version"]
)
>= 0
):
log.trace(
"Dependency (%s) already loaded inside %s with "
"version (%s), required (%s), skipping",
dependency,
mod_name,
dep_version,
params["version"],
)
continue
else:
log.trace(
"Dependency (%s) already loaded inside %s, skipping",
dependency,
mod_name,
)
continue
log.trace(
"Unloading %s.%s because dependency (%s%s) is not met",
mod_name,
func_name,
dependency,
(
" version {}".format(params["version"])
if "version" in params
else ""
),
)
# if not, unload the function
if frame:
try:
func_name = frame.f_globals["__func_alias__"][func_name]
except (AttributeError, KeyError):
pass
mod_key = f"{mod_name}.{func_name}"
# if we don't have this module loaded, skip it!
if mod_key not in functions:
continue
try:
fallback_function = params.get("fallback_function")
if fallback_function is not None:
functions[mod_key] = fallback_function
else:
del functions[mod_key]
except AttributeError:
# we already did???
log.trace("%s already removed, skipping", mod_key)
continue
depends = Depends
def timing(function):
"""
Decorator wrapper to log execution time, for profiling purposes
"""
@wraps(function)
def wrapped(*args, **kwargs):
start_time = time.time()
ret = function(*args, **salt.utils.args.clean_kwargs(**kwargs))
end_time = time.time()
if function.__module__.startswith("salt.loaded.int."):
mod_name = function.__module__[16:]
else:
mod_name = function.__module__
fstr = f"Function %s.%s took %.{sys.float_info.dig}f seconds to execute"
log.profile(fstr, mod_name, function.__name__, end_time - start_time)
return ret
return wrapped
def memoize(func):
"""
Memoize aka cache the return output of a function
given a specific set of arguments
.. versionedited:: 2016.3.4
Added **kwargs support.
"""
cache = {}
@wraps(func)
def _memoize(*args, **kwargs):
str_args = []
for arg in args:
if not isinstance(arg, str):
str_args.append(str(arg))
else:
str_args.append(arg)
args_ = ",".join(list(str_args) + [f"{k}={kwargs[k]}" for k in sorted(kwargs)])
if args_ not in cache:
cache[args_] = func(*args, **kwargs)
return cache[args_]
return _memoize
class _DeprecationDecorator:
"""
Base mix-in class for the deprecation decorator.
Takes care of a common functionality, used in its derivatives.
"""
OPT_IN = 1
OPT_OUT = 2
def __init__(self, globals, version):
"""
Constructor.
:param globals: Module globals. Important for finding out replacement functions
:param version: Expiration version
:return:
"""
from salt.version import SaltStackVersion, __saltstack_version__
self._globals = globals
self._exp_version_name = version
self._exp_version = SaltStackVersion.from_name(self._exp_version_name)
self._curr_version = __saltstack_version__.info
self._raise_later = None
self._function = None
self._orig_f_name = None
def _get_args(self, kwargs):
"""
Discard all keywords which aren't function-specific from the kwargs.
:param kwargs:
:return:
"""
_args = list()
_kwargs = salt.utils.args.clean_kwargs(**kwargs)
return _args, _kwargs
def _call_function(self, kwargs):
"""
Call target function that has been decorated.
:return:
"""
if self._raise_later:
raise self._raise_later # pylint: disable=E0702
if self._function:
args, kwargs = self._get_args(kwargs)
try:
return self._function(*args, **kwargs)
except TypeError as error:
error = str(error).replace(
self._function, self._orig_f_name
) # Hide hidden functions
log.error(
'Function "%s" was not properly called: %s',
self._orig_f_name,
error,
)
return self._function.__doc__
except Exception as error: # pylint: disable=broad-except
log.error(
'Unhandled exception occurred in function "%s: %s',
self._function.__name__,
error,
)
raise
else:
raise CommandExecutionError(
"Function is deprecated, but the successor function was not found."
)
def __call__(self, function):
"""
Callable method of the decorator object when
the decorated function is gets called.
:param function:
:return:
"""
self._function = function
self._orig_f_name = self._function.__name__
class _IsDeprecated(_DeprecationDecorator):
"""
This decorator should be used only with the deprecated functions
to mark them as deprecated and alter its behavior a corresponding way.
The usage is only suitable if deprecation process is renaming
the function from one to another. In case function name or even function
signature stays the same, please use 'with_deprecated' decorator instead.
It has the following functionality:
1. Put a warning level message to the log, informing that
the deprecated function has been in use.
2. Raise an exception, if deprecated function is being called,
but the lifetime of it already expired.
3. Point to the successor of the deprecated function in the
log messages as well during the blocking it, once expired.
Usage of this decorator as follows. In this example no successor
is mentioned, hence the function "foo()" will be logged with the
warning each time is called and blocked completely, once EOF of
it is reached:
from salt.util.decorators import is_deprecated
@is_deprecated(globals(), "Beryllium")
def foo():
pass
In the following example a successor function is mentioned, hence
every time the function "bar()" is called, message will suggest
to use function "baz()" instead. Once EOF is reached of the function
"bar()", an exception will ask to use function "baz()", in order
to continue:
from salt.util.decorators import is_deprecated
@is_deprecated(globals(), "Beryllium", with_successor="baz")
def bar():
pass
def baz():
pass
"""
def __init__(self, globals, version, with_successor=None):
"""
Constructor of the decorator 'is_deprecated'.
:param globals: Module globals
:param version: Version to be deprecated
:param with_successor: Successor function (optional)
:return:
"""
_DeprecationDecorator.__init__(self, globals, version)
self._successor = with_successor
def __call__(self, function):
"""
Callable method of the decorator object when
the decorated function is gets called.
:param function:
:return:
"""
_DeprecationDecorator.__call__(self, function)
@wraps(function)
def _decorate(*args, **kwargs):
"""
Decorator function.
:param args:
:param kwargs:
:return:
"""
if self._curr_version < self._exp_version:
msg = [
'The function "{f_name}" is deprecated and will '
'expire in version "{version_name}".'.format(
f_name=self._function.__name__,
version_name=self._exp_version_name,
)
]
if self._successor:
msg.append(
'Use successor "{successor}" instead.'.format(
successor=self._successor
)
)
log.warning(" ".join(msg))
else:
msg = [
'The lifetime of the function "{f_name}" expired.'.format(
f_name=self._function.__name__
)
]
if self._successor:
msg.append(
'Please use its successor "{successor}" instead.'.format(
successor=self._successor
)
)
log.warning(" ".join(msg))
raise CommandExecutionError(" ".join(msg))
return self._call_function(kwargs)
return _decorate
is_deprecated = _IsDeprecated
class _WithDeprecated(_DeprecationDecorator):
"""
This decorator should be used with the successor functions
to mark them as a new and alter its behavior in a corresponding way.
It is used alone if a function content or function signature
needs to be replaced, leaving the name of the function same.
In case function needs to be renamed or just dropped, it has
to be used in pair with 'is_deprecated' decorator.
It has the following functionality:
1. Put a warning level message to the log, in case a component
is using its deprecated version.
2. Switch between old and new function in case an older version
is configured for the desired use.
3. Raise an exception, if deprecated version reached EOL and
point out for the new version.
Usage of this decorator as follows. If 'with_name' is not specified,
then the name of the deprecated function is assumed with the "_" prefix.
In this case, in order to deprecate a function, it is required:
- Add a prefix "_" to an existing function. E.g.: "foo()" to "_foo()".
- Implement a new function with exactly the same name, just without
the prefix "_".
Example:
from salt.util.decorators import with_deprecated
@with_deprecated(globals(), "Beryllium")
def foo():
"This is a new function"
def _foo():
"This is a deprecated function"
In case there is a need to deprecate a function and rename it,
the decorator should be used with the 'with_name' parameter. This
parameter is pointing to the existing deprecated function. In this
case deprecation process as follows:
- Leave a deprecated function without changes, as is.
- Implement a new function and decorate it with this decorator.
- Set a parameter 'with_name' to the deprecated function.
- If a new function has a different name than a deprecated,
decorate a deprecated function with the 'is_deprecated' decorator
in order to let the function have a deprecated behavior.
Example:
from salt.util.decorators import with_deprecated
@with_deprecated(globals(), "Beryllium", with_name="an_old_function")
def a_new_function():
"This is a new function"
@is_deprecated(globals(), "Beryllium", with_successor="a_new_function")
def an_old_function():
"This is a deprecated function"
"""
MODULE_NAME = "__virtualname__"
CFG_USE_DEPRECATED = "use_deprecated"
CFG_USE_SUPERSEDED = "use_superseded"
def __init__(
self, globals, version, with_name=None, policy=_DeprecationDecorator.OPT_OUT
):
"""
Constructor of the decorator 'with_deprecated'
:param globals:
:param version:
:param with_name:
:param policy:
:return:
"""
_DeprecationDecorator.__init__(self, globals, version)
self._with_name = with_name
self._policy = policy
def _set_function(self, function):
"""
Based on the configuration, set to execute an old or a new function.
:return:
"""
full_name = "{m_name}.{f_name}".format(
m_name=self._globals.get(self.MODULE_NAME, "")
or self._globals["__name__"].split(".")[-1],
f_name=function.__name__,
)
if full_name.startswith("."):
self._raise_later = CommandExecutionError(
'Module not found for function "{f_name}"'.format(
f_name=function.__name__
)
)
opts = self._globals.get("__opts__", "{}")
pillar = self._globals.get("__pillar__", "{}")
use_deprecated = full_name in opts.get(
self.CFG_USE_DEPRECATED, list()
) or full_name in pillar.get(self.CFG_USE_DEPRECATED, list())
use_superseded = full_name in opts.get(
self.CFG_USE_SUPERSEDED, list()
) or full_name in pillar.get(self.CFG_USE_SUPERSEDED, list())
if use_deprecated and use_superseded:
raise SaltConfigurationError(
"Function '{}' is mentioned both in deprecated "
"and superseded sections. Please remove any of that.".format(full_name)
)
old_function = self._globals.get(self._with_name or f"_{function.__name__}")
if self._policy == self.OPT_IN:
self._function = function if use_superseded else old_function
else:
self._function = old_function if use_deprecated else function
def _is_used_deprecated(self):
"""
Returns True, if a component configuration explicitly is
asking to use an old version of the deprecated function.
:return:
"""
func_path = "{m_name}.{f_name}".format(
m_name=self._globals.get(self.MODULE_NAME, "")
or self._globals["__name__"].split(".")[-1],
f_name=self._orig_f_name,
)
return (
func_path
in self._globals.get("__opts__").get(self.CFG_USE_DEPRECATED, list())
or func_path
in self._globals.get("__pillar__").get(self.CFG_USE_DEPRECATED, list())
or (
self._policy == self.OPT_IN
and not (
func_path
in self._globals.get("__opts__", {}).get(
self.CFG_USE_SUPERSEDED, list()
)
)
and not (
func_path
in self._globals.get("__pillar__", {}).get(
self.CFG_USE_SUPERSEDED, list()
)
)
),
func_path,
)
def __call__(self, function):
"""
Callable method of the decorator object when
the decorated function is gets called.
:param function:
:return:
"""
_DeprecationDecorator.__call__(self, function)
@wraps(function)
def _decorate(*args, **kwargs):
"""
Decorator function.
:param args:
:param kwargs:
:return:
"""
self._set_function(function)
is_deprecated, func_path = self._is_used_deprecated()
if is_deprecated:
if self._curr_version < self._exp_version:
msg = list()
if self._with_name:
msg.append(
'The function "{f_name}" is deprecated and will '
'expire in version "{version_name}".'.format(
f_name=self._with_name.startswith("_")
and self._orig_f_name
or self._with_name,
version_name=self._exp_version_name,
)
)
msg.append(
'Use its successor "{successor}" instead.'.format(
successor=self._orig_f_name
)
)
else:
msg.append(
'The function "{f_name}" is using its deprecated version'
' and will expire in version "{version_name}".'.format(
f_name=func_path, version_name=self._exp_version_name
)
)
log.warning(" ".join(msg))
else:
msg_patt = 'The lifetime of the function "{f_name}" expired.'
if "_" + self._orig_f_name == self._function.__name__:
msg = [
msg_patt.format(f_name=self._orig_f_name),
"Please turn off its deprecated version in the"
" configuration",
]
else:
msg = [
'Although function "{f_name}" is called, an alias'
' "{f_alias}" is configured as its deprecated version.'.format(
f_name=self._orig_f_name,
f_alias=self._with_name or self._orig_f_name,
),
msg_patt.format(
f_name=self._with_name or self._orig_f_name
),
'Please use its successor "{successor}" instead.'.format(
successor=self._orig_f_name
),
]
log.error(" ".join(msg))
raise CommandExecutionError(" ".join(msg))
return self._call_function(kwargs)
_decorate.__doc__ = self._function.__doc__
_decorate.__wrapped__ = self._function
return _decorate
with_deprecated = _WithDeprecated
def require_one_of(*kwarg_names):
"""
Decorator to filter out exclusive arguments from the call.
kwarg_names:
Limit which combination of arguments may be passed to the call.
Example:
# Require one of the following arguments to be supplied to foo()
@require_one_of('arg1', 'arg2', 'arg3')
def foo(arg1, arg2, arg3):
"""
def wrapper(f):
@wraps(f)
def func(*args, **kwargs):
names = [key for key in kwargs if kwargs[key] and key in kwarg_names]
names.extend(
[
args[i]
for i, arg in enumerate(args)
if args[i] and f.__code__.co_varnames[i] in kwarg_names
]
)
if len(names) > 1:
raise SaltInvocationError(
"Only one of the following is allowed: {}".format(
", ".join(kwarg_names)
)
)
if not names:
raise SaltInvocationError(
"One of the following must be provided: {}".format(
", ".join(kwarg_names)
)
)
return f(*args, **kwargs)
return func
return wrapper
def allow_one_of(*kwarg_names):
"""
Decorator to filter out exclusive arguments from the call.
kwarg_names:
Limit which combination of arguments may be passed to the call.
Example:
# Allow only one of the following arguments to be supplied to foo()
@allow_one_of('arg1', 'arg2', 'arg3')
def foo(arg1, arg2, arg3):
"""
def wrapper(f):
@wraps(f)
def func(*args, **kwargs):
names = [key for key in kwargs if kwargs[key] and key in kwarg_names]
names.extend(
[
args[i]
for i, arg in enumerate(args)
if args[i] and f.__code__.co_varnames[i] in kwarg_names
]
)
if len(names) > 1:
raise SaltInvocationError(
"Only of the following is allowed: {}".format(
", ".join(kwarg_names)
)
)
return f(*args, **kwargs)
return func
return wrapper
def ignores_kwargs(*kwarg_names):
"""
Decorator to filter out unexpected keyword arguments from the call
kwarg_names:
List of argument names to ignore
"""
def _ignores_kwargs(fn):
@wraps(fn)
def __ignores_kwargs(*args, **kwargs):
kwargs_filtered = kwargs.copy()
for name in kwarg_names:
if name in kwargs_filtered:
del kwargs_filtered[name]
return fn(*args, **kwargs_filtered)
return __ignores_kwargs
return _ignores_kwargs
def ensure_unicode_args(function):
"""
Decodes all arguments passed to the wrapped function
"""
@wraps(function)
def wrapped(*args, **kwargs):
return function(*args, **kwargs)
return wrapped
Zerion Mini Shell 1.0