Source code for molten.validation.forward

# This file is a part of molten.
#
# Copyright (C) 2018 CLEARTYPE SRL <[email protected]>
#
# molten is free software; you can redistribute it and/or modify it
# under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or (at
# your option) any later version.
#
# molten is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
# License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.

import inspect
from typing import Any, no_type_check


def is_forward_ref(annotation: Any) -> bool:
    """Returns True when the given annotation is a forward ref.
    """
    return issubclass(type(annotation), _forward_ref)


[docs]@no_type_check def forward_ref(name: str) -> Any: """Generate a proxy type for a schema that is going to be defined at some point after the current statement. Examples: Schema "A" includes a reference to a yet-to-be-defined schema "B":: @schema class A: b: forward_ref("B") @schema class B: x: int ``forward_ref`` can be used as a type parameter to other types as well:: @schema class A: b: List[forward_ref("B")] @schema class B: x: int Parameters: name: The name of the schema being referenced. This name must eventually be bound in the scope of the call site. """ try: caller_frame_info = inspect.stack()[1] return _forward_ref(f'_forward_ref("{name}")', (object,), { "_schema": None, "_name": name, "_namespace": caller_frame_info.frame.f_globals, }) finally: del caller_frame_info
class _forward_ref(type): @no_type_check def lookup(cls): schema = cls._schema if schema is None: schema = cls._schema = cls._namespace[cls._name] del cls._name del cls._namespace return schema