# Licensed under the LGPL: https://www.gnu.org/licenses/old-licenses/lgpl-2.1.en.html # For details: https://github.com/PyCQA/astroid/blob/main/LICENSE # Copyright (c) https://github.com/PyCQA/astroid/blob/main/CONTRIBUTORS.txt """Transform utilities (filters and decorator).""" from __future__ import annotations import typing from collections.abc import Iterator import wrapt from astroid.exceptions import InferenceOverwriteError, UseInferenceDefault from astroid.nodes import NodeNG from astroid.typing import InferenceResult, InferFn _cache: dict[tuple[InferFn, NodeNG], list[InferenceResult] | None] = {} def clear_inference_tip_cache() -> None: """Clear the inference tips cache.""" _cache.clear() @wrapt.decorator def _inference_tip_cached( func: InferFn, instance: None, args: typing.Any, kwargs: typing.Any ) -> Iterator[InferenceResult]: """Cache decorator used for inference tips.""" node = args[0] try: result = _cache[func, node] # If through recursion we end up trying to infer the same # func + node we raise here. if result is None: raise UseInferenceDefault() except KeyError: _cache[func, node] = None result = _cache[func, node] = list(func(*args, **kwargs)) assert result return iter(result) def inference_tip(infer_function: InferFn, raise_on_overwrite: bool = False) -> InferFn: """Given an instance specific inference function, return a function to be given to AstroidManager().register_transform to set this inference function. :param bool raise_on_overwrite: Raise an `InferenceOverwriteError` if the inference tip will overwrite another. Used for debugging Typical usage .. sourcecode:: python AstroidManager().register_transform(Call, inference_tip(infer_named_tuple), predicate) .. Note:: Using an inference tip will override any previously set inference tip for the given node. Use a predicate in the transform to prevent excess overwrites. """ def transform(node: NodeNG, infer_function: InferFn = infer_function) -> NodeNG: if ( raise_on_overwrite and node._explicit_inference is not None and node._explicit_inference is not infer_function ): raise InferenceOverwriteError( "Inference already set to {existing_inference}. " "Trying to overwrite with {new_inference} for {node}".format( existing_inference=infer_function, new_inference=node._explicit_inference, node=node, ) ) # pylint: disable=no-value-for-parameter node._explicit_inference = _inference_tip_cached(infer_function) return node return transform
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
__pycache__ | Folder | 0755 |
|
|
brain | Folder | 0755 |
|
|
interpreter | Folder | 0755 |
|
|
nodes | Folder | 0755 |
|
|
__init__.py | File | 4.98 KB | 0644 |
|
__pkginfo__.py | File | 274 B | 0644 |
|
_ast.py | File | 4.05 KB | 0644 |
|
_backport_stdlib_names.py | File | 6.85 KB | 0644 |
|
_cache.py | File | 786 B | 0644 |
|
arguments.py | File | 12.65 KB | 0644 |
|
astroid_manager.py | File | 572 B | 0644 |
|
bases.py | File | 24.99 KB | 0644 |
|
builder.py | File | 18.35 KB | 0644 |
|
const.py | File | 1.07 KB | 0644 |
|
constraint.py | File | 4.92 KB | 0644 |
|
context.py | File | 5.85 KB | 0644 |
|
decorators.py | File | 9.85 KB | 0644 |
|
exceptions.py | File | 12.78 KB | 0644 |
|
filter_statements.py | File | 9.42 KB | 0644 |
|
helpers.py | File | 11.07 KB | 0644 |
|
inference.py | File | 44.06 KB | 0644 |
|
inference_tip.py | File | 2.82 KB | 0644 |
|
manager.py | File | 17.54 KB | 0644 |
|
mixins.py | File | 1.15 KB | 0644 |
|
modutils.py | File | 22.96 KB | 0644 |
|
node_classes.py | File | 1.8 KB | 0644 |
|
objects.py | File | 12.46 KB | 0644 |
|
protocols.py | File | 32.2 KB | 0644 |
|
raw_building.py | File | 22.34 KB | 0644 |
|
rebuilder.py | File | 77.86 KB | 0644 |
|
scoped_nodes.py | File | 958 B | 0644 |
|
test_utils.py | File | 2.38 KB | 0644 |
|
transforms.py | File | 3.19 KB | 0644 |
|
typing.py | File | 1.94 KB | 0644 |
|
util.py | File | 4.62 KB | 0644 |
|