Skip to content

utils

Class info¶

Classes¶

Name Children Inherits
DataclassInstance
jinjarope.utils

    🛈 DocStrings¶

    fsspec_get cached ¶

    fsspec_get(path: str | PathLike[str]) -> str
    

    Fetch a file via fsspec and return file content as a string.

    Parameters:

    Name Type Description Default
    path str | PathLike[str]

    The path to fetch the file from

    required
    Source code in src/jinjarope/utils.py
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    @functools.cache
    def fsspec_get(path: str | os.PathLike[str]) -> str:
        """Fetch a file via fsspec and return file content as a string.
    
        Args:
            path: The path to fetch the file from
        """
        import fsspec
    
        with fsspec.open(path) as file:
            return file.read().decode()
    

    get_dataclass_nondefault_values ¶

    get_dataclass_nondefault_values(instance: DataclassInstance) -> dict[str, Any]
    

    Return dictionary with non-default key-value pairs of given dataclass.

    Parameters:

    Name Type Description Default
    instance DataclassInstance

    dataclass instance

    required
    Source code in src/jinjarope/utils.py
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    def get_dataclass_nondefault_values(
        instance: DataclassInstance,
    ) -> dict[str, Any]:
        """Return dictionary with non-default key-value pairs of given dataclass.
    
        Args:
            instance: dataclass instance
        """
        non_default_fields: dict[str, Any] = {}
    
        for field in dataclasses.fields(instance):
            value = getattr(instance, field.name)
    
            # Check if the field has a default value
            if field.default is not dataclasses.MISSING:
                default = field.default
            elif field.default_factory is not dataclasses.MISSING:
                default = field.default_factory()
            else:
                # If there's no default, we consider the current value as non-default
                non_default_fields[field.name] = value
                continue
    
            # Compare the current value with the default
            if value != default:
                non_default_fields[field.name] = value
        return non_default_fields
    

    get_hash ¶

    get_hash(obj: Any, hash_length: int | None = 7) -> str
    

    Get a Md5 hash for given object.

    Parameters:

    Name Type Description Default
    obj Any

    The object to get a hash for ()

    required
    hash_length int | None

    Optional cut-off value to limit length

    7
    Source code in src/jinjarope/utils.py
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    def get_hash(obj: Any, hash_length: int | None = 7) -> str:
        """Get a Md5 hash for given object.
    
        Args:
            obj: The object to get a hash for ()
            hash_length: Optional cut-off value to limit length
        """
        import hashlib
    
        hash_md5 = hashlib.md5(str(obj).encode("utf-8"))
        return hash_md5.hexdigest()[:hash_length]
    

    get_repr ¶

    get_repr(_obj: Any, *args: Any, **kwargs: Any) -> str
    

    Get a suitable repr string for an object.

    Parameters:

    Name Type Description Default
    _obj Any

    The object to get a repr for.

    required
    args Any

    Arguments for the repr

    ()
    kwargs Any

    Keyword arguments for the repr

    {}
    Source code in src/jinjarope/utils.py
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    def get_repr(_obj: Any, *args: Any, **kwargs: Any) -> str:
        """Get a suitable __repr__ string for an object.
    
        Args:
            _obj: The object to get a repr for.
            args: Arguments for the repr
            kwargs: Keyword arguments for the repr
        """
        classname = type(_obj).__name__
        parts = [repr(v) for v in args]
        kw_parts = [f"{k}={v!r}" for k, v in kwargs.items()]
        sig = ", ".join(parts + kw_parts)
        return f"{classname}({sig})"
    

    partial ¶

    partial(fn: Callable, *args: Any, **kwargs: Any)
    

    Create new function with partial application of given arguments / keywords.

    Parameters:

    Name Type Description Default
    fn Callable

    The function to generate a partial from

    required
    args Any

    patially applied arguments

    ()
    kwargs Any

    partially applied keywords

    {}
    Source code in src/jinjarope/utils.py
    26
    27
    28
    29
    30
    31
    32
    33
    34
    def partial(fn: Callable, *args: Any, **kwargs: Any):
        """Create new function with partial application of given arguments / keywords.
    
        Args:
            fn: The function to generate a partial from
            args: patially applied arguments
            kwargs: partially applied keywords
        """
        return functools.partial(fn, *args, **kwargs)
    

    resolve cached ¶

    resolve(name: str, module: str | None = None, *, py_type: None = None) -> Any
    
    resolve(name: str, module: str | None = None, *, py_type: type[T]) -> T
    
    resolve(
        name: str, module: str | None = None, py_type: type[T] | None = None
    ) -> T | Any
    

    Resolve name to a Python object via imports / attribute lookups.

    If module is None, name must be "absolute" (no leading dots).

    If module is not None, and name is "relative" (has leading dots), the object will be found by navigating relative to module.

    Parameters:

    Name Type Description Default
    name str

    The name to resolve

    required
    module str | None

    Optional base module for relative imports

    None
    py_type type[T] | None

    Optional type to validate the resolved object against

    None

    Returns:

    Type Description
    T | Any

    The resolved object

    Raises:

    Type Description
    ValueError

    If using a relative name without a base module

    TypeError

    If py_type is provided and the resolved object doesn't match that type

    Source code in src/jinjarope/utils.py
    159
    160
    161
    162
    163
    164
    165
    166
    167
    168
    169
    170
    171
    172
    173
    174
    175
    176
    177
    178
    179
    180
    181
    182
    183
    184
    185
    186
    187
    188
    189
    190
    191
    192
    193
    194
    195
    196
    197
    198
    199
    200
    201
    202
    203
    204
    205
    206
    207
    208
    209
    210
    211
    212
    213
    214
    215
    216
    217
    218
    219
    220
    221
    222
    223
    224
    @functools.cache
    def resolve[T](
        name: str,
        module: str | None = None,
        py_type: type[T] | None = None,
    ) -> T | Any:
        """Resolve ``name`` to a Python object via imports / attribute lookups.
    
        If ``module`` is None, ``name`` must be "absolute" (no leading dots).
    
        If ``module`` is not None, and ``name`` is "relative" (has leading dots),
        the object will be found by navigating relative to ``module``.
    
        Args:
            name: The name to resolve
            module: Optional base module for relative imports
            py_type: Optional type to validate the resolved object against
    
        Returns:
            The resolved object
    
        Raises:
            ValueError: If using a relative name without a base module
            TypeError: If py_type is provided and the resolved object
                      doesn't match that type
        """
        from jinjarope import envtests
    
        names = name.split(".")
        if not names[0]:
            if module is None:
                msg = "relative name without base module"
                raise ValueError(msg)
            modules = module.split(".")
            names.pop(0)
            while not name[0]:
                modules.pop()
                names.pop(0)
            names = modules + names
    
        used = names.pop(0)
        if envtests.is_python_builtin(used):
            import builtins
    
            found = getattr(builtins, used)
        else:
            found = importlib.import_module(used)
    
        for n in names:
            used += "." + n
            try:
                found = getattr(found, n)
            except AttributeError:
                try:
                    importlib.import_module(used)
                    found = getattr(found, n)
                except ModuleNotFoundError:
                    mod = ".".join(used.split(".")[:-1])
                    importlib.import_module(mod)
                    found = getattr(found, n)
    
        if py_type is not None and not isinstance(found, py_type):
            msg = f"Expected {py_type.__name__}, but {name} is {type(found).__name__}"
            raise TypeError(msg)
    
        return cast(T, found) if py_type is not None else found