Source code for cloup._util

"""Generic utilities."""
from typing import Any, Dict, Iterable, List, Optional, TypeVar

import click

click_version_tuple = click.__version__.split('.')

T = TypeVar('T')


[docs]def class_name(obj): return obj.__class__.__name__
[docs]def check_arg(condition: bool, msg: str = ''): if not condition: raise ValueError(msg)
[docs]def indent_lines(lines: Iterable[str], width=2) -> List[str]: spaces = ' ' * width return [spaces + line for line in lines]
[docs]def make_repr(obj, *args, _line_len: int = 60, _indent: int = 2, **kwargs) -> str: """ Generate repr(obj). :param obj: object to represent :param args: positional arguments in the repr :param _line_len: if the repr length exceeds this, arguments will be on their own line; if negative, the repr will be in a single line regardless of its length :param _indent: indentation width of arguments in case they are shown in their own line :param kwargs: keyword arguments in the repr :return: str """ cls_name = obj.__class__.__name__ arglist = [ *(repr(arg) for arg in args), *(f'{key}={value!r}' for key, value in kwargs.items()), ] len_arglist = sum(len(s) for s in arglist) total_len = len(cls_name) + len_arglist + 2 * len(arglist) if 0 <= _line_len < total_len: lines = indent_lines(arglist, width=_indent) args_text = ',\n'.join(lines) return f'{cls_name}(\n{args_text}\n)' else: args_text = ', '.join(arglist) return f'{cls_name}({args_text})'
[docs]def make_one_line_repr(obj, *args, **kwargs): return make_repr(obj, *args, _line_len=-1, **kwargs)
[docs]def pluralize( count: int, zero: str = '', one: str = '', many: str = '', ) -> str: if count == 0 and zero: return zero if count == 1 and one: return one return many.format(count=count)
[docs]def coalesce(*values: Optional[T], default=None) -> Optional[T]: """Returns the first value that is not None (or ``default`` if no such value exists). Inspired by the homonym SQL function.""" for val in values: if val is not None: return val return default
[docs]def listOfNotNone(iterable: Iterable[Optional[T]]) -> List[T]: return [x for x in iterable if x is not None]
[docs]def check_positive_int(value, arg_name): error_type = None if not isinstance(value, int): error_type = TypeError elif value <= 0: error_type = ValueError if error_type: raise error_type(f'{arg_name} should be a positive integer. It is: {value}.')
[docs]def identity(x: T) -> T: return x
[docs]class FrozenSpaceMeta(type): def __init__(cls, *args): d = {k: v for k, v in vars(cls).items() if not k.startswith('_')} type.__setattr__(cls, '_dict', d)
[docs] def __setattr__(cls, key, value): raise Exception("you can't set attributes on this class")
[docs] def asdict(cls) -> Dict[str, Any]: return cls._dict # type: ignore
[docs] def __contains__(cls, item: str) -> bool: return item in cls.asdict()
[docs] def __getitem__(cls, item): return cls._dict[item]
[docs]class FrozenSpace(metaclass=FrozenSpaceMeta): """A class used just as frozen namespace for constants.""" def __init__(self): raise Exception( "this class is just a namespace for constants, it's not instantiable.")
[docs]def pop_many(d: dict, *keys: str) -> dict: for key in keys: d.pop(key) return d