Source code for lena.flow.selectors

"""Select specific items."""
from __future__ import print_function

import inspect

import lena.context
import lena.core
import lena.flow


# may be made public if there appears a reason
# to prohibit nested ()s and []s and force user
# to define type explicitly.
class _SelectorOr(object):
    def __init__(self, *args):
        self._selectors = []
        for arg in args:
            if isinstance(arg, Selector):
                self._selectors.append(arg)
            else:
                # may raise
                self._selectors.append(Selector(arg))

    def __call__(self, val):
        return any((f(val) for f in self._selectors))


class _SelectorAnd(object):
    def __init__(self, *args):
        self._selectors = []
        for arg in args:
            if isinstance(arg, Selector):
                self._selectors.append(arg)
            else:
                # may raise
                self._selectors.append(Selector(arg))

    def __call__(self, val):
        return all((f(val) for f in self._selectors))


[docs]class Selector(object): """Determine whether an item should be selected. Generally, *selected* means the result is convertible to ``True``, but other values can be used as well. """ def __init__(self, selector): """The usage of *selector* depends on its type. If *selector* is a class, :meth:`__call__` checks that data part of the value is subclassed from that. A callable is used as is. A string means that value's context must conform to that (as in :func:`lena.context.check_context_str`). *selector* can be a container. In this case its items are converted to selectors. If *selector* is a *list*, the result is *or* applied to results of each item. If it is a *tuple*, boolean *and* is applied to the results. If incorrect arguments are provided, :exc:`~lena.core.LenaTypeError` is raised. """ # Callable classes are treated as classes, not callables if inspect.isclass(selector): self._selector = lambda val: isinstance( lena.flow.get_data(val), selector ) elif callable(selector): self._selector = selector elif isinstance(selector, str): self._selector = lambda val: lena.context.check_context_str( lena.flow.get_context(val), selector ) elif isinstance(selector, list): try: self._selector = _SelectorOr(*selector) except lena.core.LenaTypeError as err: raise err elif isinstance(selector, tuple): try: self._selector = _SelectorAnd(*selector) except lena.core.LenaTypeError as err: raise err else: raise lena.core.LenaTypeError( "Selector must be initialized with a callable, list or tuple, " "{} provided".format(selector) )
[docs] def __call__(self, value): """Check whether *value* is selected. If an exception occurs, the result is False. It is safe to use non-existing attributes, etc. """ try: sel = self._selector(value) except Exception: # pylint: disable=broad-except # it can be really any exception: AttributeError, etc. return False else: return sel