Source code for parametrize_from_file.namespace

from collections.abc import Mapping, Iterable
from functools import partial
from unittest.mock import Mock

SENTINEL = object()

class Namespace(Mapping):
    """\
    Evaluate and/or execute snippets of python code, with powerful control over the 
    names available to those snippets.

    .. note::
    
        It is conventional to:

        - Name your namespaces: ``with_{short description}``

        - Create any namespaces you will need in a single helper file (e.g.  
          ``param_helpers.py``) to be imported in test scripts as necessary.  
          Note that namespaces are immutable, so it's safe for them to be 
          global variables.

    Examples:

        The first step when using a namespace is to specify which names it 
        should include.   This can be done using...

        ...strings (which will be `python:exec`'d):
            
            >>> with_math = Namespace('import math')
            >>> with_math.eval('math.sqrt(4)')
            2.0

        ...dictionaries:

            >>> import math
            >>> with_math = Namespace(globals())
            >>> with_math.eval('math.sqrt(4)')
            2.0

        ...modules:
            
            >>> with_math = Namespace(math)
            >>> with_math.eval('math.sqrt(4)')
            2.0

        ...other namespaces (via the constructor):

            >>> with_np = Namespace(with_math, 'import numpy as np')
            >>> with_np.eval('np.arange(3) * math.sqrt(4)')
            array([0., 2., 4.])

        ...other namespaces (via the `fork` method):

            >>> with_np = with_math.fork('import numpy as np')
            >>> with_np.eval('np.arange(3) * math.sqrt(4)')
            array([0., 2., 4.])

        ...keyword arguments:

            >>> with_math = Namespace(math=math)
            >>> with_math.eval('math.sqrt(4)')
            2.0

        ...the `star` function:

            >>> with_math = Namespace(star(math))
            >>> with_math.eval('sqrt(4)')
            2.0

        Once you have an initialized namespace, you can use it to...

        ...evaluate expressions:

            >>> with_math.eval('sqrt(4)')
            2.0
            
        ...execute blocks of code:

            >>> ns = with_math.exec('''
            ... a = sqrt(4)
            ... b = sqrt(9)
            ... ''')
            >>> ns['a']
            2.0
            >>> ns['b']
            3.0
            >>> ns.eval('a + b')
            5.0

        ...make error-detecting context managers:

            >>> class MyError(Exception):
            ...     pass
            ...
            >>> with_err = Namespace(MyError=MyError)
            >>> err_cm = with_err.error({'type': 'MyError', 'pattern': r'\\d+'})
            >>> with err_cm:
            ...    raise MyError('404')
    """

[docs] def __init__(self, *args, **kwargs): """ Construct a namespace containing the given names. Arguments: args (str,dict,types.ModuleType): If string: The string will be executed as python code and any names that are defined by that code will be added to the namespace. Any names added by previous arguments will be available to the code. If dict: The items in the dictionary will the directly added to the namespace. All of the dictionary keys must be strings. If module: The module will be added to the namespace, with its name as the key. The *args* are processed in order, so later *args* may overwrite earlier *args*. kwargs: Each key/value pair will be added to the namespace. The *kwargs* are processed after the *args*, so if the same key is defined by both, the *kwargs* definition will be used. """ self._dict = {} _update_namespace(self._dict, *args, **kwargs)
[docs] def __repr__(self): return f'{self.__class__.__name__}({self._dict.__repr__()})'
[docs] def __getitem__(self, key): return self._dict.__getitem__(key)
[docs] def __iter__(self): return self._dict.__iter__()
[docs] def __len__(self): return self._dict.__len__()
[docs] def copy(self): """ Create a shallow copy of this namespace. This is an alias for `fork()` with no arguments, for compatibility with the `dict` API. """ return self.fork()
[docs] def fork(self, *args, **kwargs): """ Create a shallow copy of this namespace. The arguments allow new names to be added to the new namespace. All arguments have the same meaning as in the constructor. """ return self.__class__(self, *args, **kwargs)
[docs] def eval(self, *src, keys=False, defer=False): """ Evaluate the given expression within this namespace. Arguments: src (str,list,dict): The expression (or expressions) to evaluate. Strings are directly evaluated. List items and dict values are recursively evaluated. Dict keys are not evaluated unless *keys* is true. This allows you to switch freely between structured text and python syntax, depending on which makes most sense for each particular input. keys (bool): If true, evaluate dictionary keys. Disabled by default because most dictionary keys are strings, and it's annoying to have to quote them. defer (bool): If true, instead of actually evaluating the given expressions, return a no-argument callable that will evaluate them when called. The purpose of this is to make it possible to specify that a test parameter should be evaluated in a schema, but to defer actually evaluating it until inside the test function. Returns: Any: If no *src* expressions were given: A `functools.partial` version of this function that will remember any specified keyword arguments. If any *src* expressions were given and the *defer* argument was True: A `functools.partial` version of this function that will remember the expressions and keyword arguments given to it. The return value will also have an `eval` attribute that aliases the deferred function. If any *src* expressions were given and the *defer* argument was False: The result(s) of evaluating the given expression(s). Examples: Basic usage:: >>> with_a = Namespace(a=1) >>> with_a.eval('a + 1') 2 Control whether or not dictionary keys are evaluated:: >>> with_a.eval({'a': '2'}) {'a': 2} >>> with_a.eval({'a': '2'}, keys=True) {1: 2} Use a schema to specify that a test parameter should be evaluated, but defer that evaluation until within the test:: >>> @parametrize_from_file( # doctest: +SKIP ... schema=cast( ... expected=with_a.eval(defer=True), ... ), ... ) ... def test_snippet(expected) ... expected = expected.eval() Details: It's sometimes convenient to call this method on values that have been processed by `error_or`. Such inputs may contain instances of two types that cannot normally be evaluated: `unittest.mock.Mock` and the internal context manager type returned by `error`. To allow for this convenience, both of these types are treated specially by this method and passed through unchanged. """ from .schema import ExpectSuccess, ExpectError if not src: return partial(self.eval, keys=keys, defer=defer) if defer: f = f.eval = partial(self.eval, *src, keys=keys) return f src = src[0] if len(src) == 1 else list(src) recurse = partial(self.eval, keys=keys) if type(src) is list: return [recurse(x) for x in src] elif type(src) is dict: f = recurse if keys else lambda x: x return {f(k): recurse(v) for k, v in src.items()} elif isinstance(src, (Mock, ExpectSuccess, ExpectError)): return src else: return eval(src, self._dict.copy())
[docs] def exec(self, src=SENTINEL, get=SENTINEL, defer=False): """ Execute the given python snippet within this namespace. Arguments: src (str): A snippet of python code to execute. get (str, collections.abc.Iterable, collections.abc.Callable): If string: The name of a variable defined in the given snippet to look up and pass on to the caller. If iterable: The names of variables defined in the given snippet to look up and return to the caller. The return value will be a tuple with the same length as the given iterable. If callable: The given function will be passed a read-only dictionary containing all the names defined in the given snippet. Whatever that function returns will be passed on to the caller. defer (bool): If true, instead of actually evaluating the given snippet, return a no-argument callable that will evaluate it when called. The purpose of this is to make it possible to specify that a test parameter should be executed in a schema, but to defer actually executing it until inside the test function. Returns: Any: If *src* was not specified: A `functools.partial` version of this function that will remember any specified keyword arguments. If *src* was specified and the *defer* argument was True: A `functools.partial` version of this function that will remember the snippet and keyword arguments given to it. The return value will also have an `exec` attribute that aliases the deferred function. If *src* was specified and *get* was not: A new `Namespace` containing all of the variables defined in the snippet. If *src* and *get* were specified: The value(s) of the indicated variable(s) defined the snippet. Examples: Basic usage:: >>> with_a = Namespace(a=1) >>> with_b = with_a.exec('b = a + 1') >>> with_a Namespace({'a': 1}) >>> with_b Namespace({'a': 1, 'b': 2}) Get a specific value from the executed snippet:: >>> with_a.exec('b = a + 1', get='b') 2 Use a schema to specify that a test parameter should be executed, but defer that execution until within the test:: >>> @parametrize_from_file( # doctest: +SKIP ... schema={ ... 'snippet': with_a.exec(defer=True), ... }, ... ) ... def test_snippet(snippet) ... snippet = snippet.exec() Details: It's sometimes convenient to call this method on values that have been processed by `error_or`. Such inputs may contain instances of two types that cannot normally be evaluated: `unittest.mock.Mock` and the internal context manager type returned by `error`. To allow for this convenience, both of these types are treated specially by this method and passed through unchanged. """ from .schema import ExpectSuccess, ExpectError if src is SENTINEL: return partial(self.exec, get=get, defer=defer) if defer: f = f.exec = partial(self.exec, src, get=get) return f if isinstance(src, (Mock, ExpectSuccess, ExpectError)): return src fork = self.fork() exec(src, fork._dict) if get is SENTINEL: return fork elif callable(get): return get(fork) elif isinstance(get, Iterable) and not isinstance(get, str): return tuple(fork[x] for x in get) else: return fork[get]
[docs] def error(self, exc_spec): """\ Make an exception-checking context manager in the context of this namespace. See the `error` documentation for more information. This method simply calls `error` with the *globals* argument set to this namespace. """ from .schema import error return error(exc_spec, globals=self)
[docs] def error_or(self, *expected, **kwargs): """ Make an exception-checking schema function in the context of this namespace. See the `error_or` documentation for more information. This method simply calls `error_or` with the *globals* argument set to this namespace. """ from .schema import error_or return error_or(*expected, **kwargs, globals=self)
def star(module): """ Return a dictionary containing all public attributes exposed by the given module. This function follows the same rules as ``from <module> import *``: - If the given module defines ``__all__``, only those names will be used. - Otherwise, all names without leading underscores will be used. The dictionary returned by this function is meant to be used as input to :py:class:`Namespace.__init__()`. """ try: keys = module.__all__ except AttributeError: keys = (k for k in module.__dict__ if not k.startswith('_')) return { k: module.__dict__[k] for k in keys } def _update_namespace(ns_dict, *args, **kwargs): from inspect import ismodule for arg in args: if ismodule(arg): ns_dict[arg.__name__] = arg elif isinstance(arg, str): exec(arg, ns_dict) else: ns_dict.update(arg) ns_dict.update(kwargs)