Source code for inmanta.execute.proxy

"""
    Copyright 2017 Inmanta

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    Contact: code@inmanta.com
"""

from collections.abc import Iterable, Mapping, Sequence
from copy import copy
from typing import Callable, Union

# Keep UnsetException, UnknownException and AttributeNotFound in place for backward compat with <iso8
from inmanta.ast import AttributeNotFound as AttributeNotFound
from inmanta.ast import NotFoundException, RuntimeException
from inmanta.ast import UnknownException as UnknownException
from inmanta.ast import UnsetException as UnsetException  # noqa F401
from inmanta.execute.util import NoneValue, Unknown
from inmanta.stable_api import stable_api
from inmanta.types import PrimitiveTypes
from inmanta.util import JSONSerializable

try:
    from typing import TYPE_CHECKING
except ImportError:
    TYPE_CHECKING = False

if TYPE_CHECKING:
    from inmanta.ast.entity import Entity
    from inmanta.execute.runtime import Instance


[docs] @stable_api class DynamicProxy: """ This class wraps an object and makes sure that a model is never modified by native code. """ def __init__(self, instance: "Instance") -> None: object.__setattr__(self, "__instance", instance) def _get_instance(self) -> "Instance": return object.__getattribute__(self, "__instance")
[docs] @classmethod def unwrap(cls, item: object) -> object: """ Converts a value from the plugin domain to the internal domain. """ if item is None: return NoneValue() if isinstance(item, DynamicProxy): return item._get_instance() if isinstance(item, list): return [cls.unwrap(x) for x in item] if isinstance(item, dict): def recurse_dict_item(key_value: tuple[object, object]) -> tuple[object, object]: (key, value) = key_value if not isinstance(key, str): raise RuntimeException( None, f"dict keys should be strings, got {key} of type {type(key)} with dict value {value}" ) return (key, cls.unwrap(value)) return dict(map(recurse_dict_item, item.items())) return item
[docs] @classmethod def return_value(cls, value: object) -> Union[None, str, tuple[object, ...], int, float, bool, "DynamicProxy"]: """ Converts a value from the internal domain to the plugin domain. """ if value is None: return None if isinstance(value, NoneValue): return None if isinstance(value, Unknown): raise UnknownException(value) if isinstance(value, (str, tuple, int, float, bool)): return copy(value) if isinstance(value, DynamicProxy): return value if isinstance(value, dict): return DictProxy(value) if hasattr(value, "__len__"): return SequenceProxy(value) if hasattr(value, "__call__"): return CallProxy(value) return DynamicProxy(value)
def __getattr__(self, attribute: str): instance = self._get_instance() try: value = instance.get_attribute(attribute).get_value() except NotFoundException as e: # allow for hasattr(proxy, "some_attr") raise AttributeNotFound(e.stmt, e.name) return DynamicProxy.return_value(value) def __setattr__(self, attribute: str, value: object) -> None: raise Exception("Readonly object") def _type(self) -> "Entity": """ Return the type of the proxied instance """ return self._get_instance().type def is_unknown(self) -> bool: """ Return true if this value is unknown and cannot be determined during this compilation run """ if isinstance(self._get_instance(), Unknown): return True return False def __hash__(self) -> int: return hash(self._get_instance()) def __eq__(self, other: object) -> bool: if hasattr(other, "_get_instance"): other = other._get_instance() return self._get_instance() == other def __lt__(self, other: object) -> bool: if hasattr(other, "_get_instance"): other = other._get_instance() return self._get_instance() < other def __repr__(self) -> str: return "@%s" % repr(self._get_instance())
class SequenceProxy(DynamicProxy, JSONSerializable): def __init__(self, iterator: Sequence) -> None: DynamicProxy.__init__(self, iterator) def __getitem__(self, key: str) -> object: instance = self._get_instance() if isinstance(key, str): raise RuntimeException(self, f"can not get a attribute {key}, {self._get_instance()} is a list") return DynamicProxy.return_value(instance[key]) def __len__(self) -> int: return len(self._get_instance()) def __iter__(self) -> Iterable: instance = self._get_instance() return IteratorProxy(instance.__iter__()) def json_serialization_step(self) -> list[PrimitiveTypes]: # Ensure proper unwrapping by using __getitem__ return [i for i in self] class DictProxy(DynamicProxy, Mapping, JSONSerializable): def __init__(self, mydict: dict[object, object]) -> None: DynamicProxy.__init__(self, mydict) def __getitem__(self, key): instance = self._get_instance() if not isinstance(key, str): raise RuntimeException(self, f"Expected string key, but got {key}, {self._get_instance()} is a dict") return DynamicProxy.return_value(instance[key]) def __len__(self) -> int: return len(self._get_instance()) def __iter__(self): instance = self._get_instance() return IteratorProxy(instance.__iter__()) def json_serialization_step(self) -> dict[str, PrimitiveTypes]: # Ensure proper unwrapping by using __getitem__ return {k: v for k, v in self.items()} class CallProxy(DynamicProxy): """ Proxy a value that implements a __call__ function """ def __init__(self, instance: Callable[..., object]) -> None: DynamicProxy.__init__(self, instance) def __call__(self, *args, **kwargs): instance = self._get_instance() return instance(*args, **kwargs) class IteratorProxy(DynamicProxy): """ Proxy an iterator call """ def __init__(self, iterator: Iterable[object]) -> None: DynamicProxy.__init__(self, iterator) def __iter__(self): return self def __next__(self): i = self._get_instance() return DynamicProxy.return_value(next(i))