Source code for atenvironment

# -*- coding: utf-8 -*-

"""Top-level package for @environment."""

__author__ = """Alexandr Mansurov"""
__email__ = 'alex@eghuro.cz'
__version__ = '0.2.1'

import os
import logging
from functools import wraps


[docs]class UnknownKeyword(BaseException): """Exception indicating unknown keyword was provided to @environment decorator in **kwargs. """ pass
[docs]class EnvironMiss(KeyError): """Error indicating key is not present in environment.""" pass
[docs]class DecoratorSyntaxError(BaseException): pass
def _missing(value): log = logging.getLogger(__name__) log.error("Missing environment variable: %s" % (value)) raise EnvironMiss(value) _allowed_keywords = ['onerror', 'in_self', 'default']
[docs]def environment(*value, **kwargs): """@environment decorator. Arguments: value -- one or more environment tokens requested onerror -- optional function to be called if any of the environment tokens in value is not present in environment. Such function must take one parameter what is a string value of a missing environment token. If onerror is not set, error is logged and EnvironMiss exception is raised in_self -- optional list of variable names in case instance property is to be initialized. If present, must be of same length as value. None in this list means particular element will be passed into the function and not into any instance property. default -- optional list of default values in case environment token in value is not present in environment. If present, must be of same length as value. onerror will not be called in such case. The decorator checks for presence of environment tokens and if successful reads their values to the function parameters of the decorated function after any called parameters provided. Eg. if calling function(a, b, c) that is decorated with @environment('X') the function must be defined as def function(a, b, c, x) and X from the environment is read as last parameter. When combining decorators or using multiple environment tokens in one @environment('X', 'Y', 'Z') the values are loaded from the left to the right, from the top to the bottom. If a function parameter for @environment is missing, when such function is called a TypeError is raised by the interpreter. """ err = _missing if kwargs is not None: for k in kwargs: if k not in _allowed_keywords: raise UnknownKeyword(k) if 'onerror' in kwargs: err = kwargs['onerror'] def environ_decorator(func): @wraps(func) def inner(*args): for v in value: if v not in os.environ: err(v) vals = [] if 'in_self' not in kwargs: in_self = [None] * len(value) else: in_self = kwargs['in_self'] if len(in_self) != len(value): raise DecoratorSyntaxError("Incorrect amount of in_self values") if 'default' in kwargs: default = kwargs['default'] if len(default) != len(value): raise DecoratorSyntaxError("Incorrect amount of default values") have_default = [True] * len(value) else: default = [None] * len(value) have_default = [False] * len(value) for v, s, have, defval in zip(value, in_self, have_default, default): if v not in os.environ: if have: val = defval else: err(v) else: val = os.environ[v] if s is not None: try: args[0].__dict__[s] = val except AttributeError as e: raise DecoratorSyntaxError("Instance property initialization failed: " + s) from e else: vals.append(val) return func(*(list(args) + vals)) return inner return environ_decorator