ptomulik/scons-tool-util

View on GitHub
lib/sconstool/util/misc_.py

Summary

Maintainability
A
0 mins
Test Coverage
# -*- coding: utf-8 -*-
"""Miscellaneous utilities.
"""

__all__ = ('add_ro_dict_property',
           'ensure_kwarg_in',
           'ensure_kwarg_not_in',
           'check_kwarg',
           'check_kwargs')


def _dict_property_doc(locs, kw):
    try:
        tdoc = kw['doc']
    except KeyError:
        doc = None
    else:
        doc = tdoc % dict(locs, **kw)
    return doc


def _dict_property_fget(dictattr, key, default):
    if callable(default):
        return lambda obj, k=key: getattr(obj, dictattr).get(k, default(obj))
    else:
        return lambda obj, k=key: getattr(obj, dictattr).get(k, default)


def add_ro_dict_property(cls, dictattr, attr, default=None, **kw):
    """Add to class **cls** a read-only property returning a predefined entry
       from a dict.

       :param type cls:
            the target class to be modified,
       :param str dictattr:
            name of the attribute of **cls** being the source dict,
       :param str,tuple attr:
            name of the generated attribute, or a tuple ``(attr, key)``,
            where ``attr`` is the generated attribute name and ``key`` is
            the corresponding key in the dictionary,
       :param default:
            default value returned by the property, if the dictionary
            has no requested key, if **default** is callable, it shall
            be a function ``default(obj)``, where an instance of **cls**
            will be passed as ``obj``,
       :keyword str doc:
            optional documentation string, the string may use substitutions,
            such as ``"%(cls)s"``, ``"%(dictattr)s"``, ``"%(attr)s"``,
            ``"%(key)s"``, ``"%(default)s"``. Extra variables for necessary
            substitutions may be provided via keyword arguments ``**kw``.
    """
    if isinstance(attr, str):
        key = attr
    else:
        (attr, key) = attr  # assumed it's a tuple or alike

    fget = _dict_property_fget(dictattr, key, default)
    locs = {k: v for k, v in locals().items() if k not in ('kw', 'fget')}
    doc = _dict_property_doc(locs, kw)
    setattr(cls, attr, property(fget, doc=doc))


def ensure_kwarg_in(caller, key, allowed):
    """Checks a single **key** from keyword arguments against allowed keys.

       If **key** is not in **allowed** keys, throws :exc:`TypeError`.

       :param str caller: name of the caller,
       :param str key: the key to be examined,
       :param allowed: sequence of allowed keys.

       :return: True
    """
    if allowed is not None and key not in allowed:
        msg = "%s got an unexpected keyword argument %r" % \
                (str(caller), key)
        raise TypeError(msg)
    return True


def ensure_kwarg_not_in(caller, key, forbidden):
    """Checks a single **key** from keyword arguments against forbidden keys.

       If **key** is in **forbidden** keys, throws :exc:`TypeError`.

       :param str caller: name of the caller,
       :param str key: the key to be examined,
       :param forbidden: sequence of forbidden keys.

       :return: True
    """
    if forbidden is not None and key in forbidden:
        msg = "%s got a forbidden keyword argument %r" % \
                (str(caller), key)
        raise TypeError(msg)
    return True


def check_kwarg(caller, key, allowed=None, forbidden=None):
    """Checks a single **key** from keyword arguments against allowed and
       forbidden keys.

       If **key** is not in **allowed** keys or it is in **forbidden** keys,
       throws :exc:`TypeError`.

       :param str caller: name of the caller,
       :param str key: the key to be examined,
       :param allowed: sequence of allowed keys.
       :param forbidden: sequence of forbidden keys.

       :return: True
    """
    ensure_kwarg_in(caller, key, allowed)
    ensure_kwarg_not_in(caller, key, forbidden)
    return True


def check_kwargs(caller, kw, allowed=None, forbidden=None):
    """Checks all keys from keyword arguments **kw** against allowed and
       forbidden keys.

       If **key** is not in **allowed** keys or it is in **forbidden** keys,
       throws :exc:`TypeError`.

       :param str caller: name of the caller,
       :param str key: the key to be examined,
       :param allowed: sequence of allowed keys.
       :param forbidden: sequence of forbidden keys.

       :return: True
    """
    for key in kw:
        check_kwarg(caller, key, allowed, forbidden)
    return True


# Local Variables:
# tab-width:4
# indent-tabs-mode:nil
# End:
# vim: set ft=python et ts=4 sw=4: