#
# Copyright (c), 2016-2020, SISSA (International School for Advanced Studies).
# All rights reserved.
# This file is distributed under the terms of the MIT License.
# See the file 'LICENSE' in the root directory of the present
# distribution, or http://opensource.org/licenses/MIT.
#
# @author Davide Brunato <brunato@sissa.it>
#
# type: ignore
"""
Observers for testing XMLSchema classes.
"""
from functools import wraps

from ..names import XSD_NAMESPACE, XSD_ANY_TYPE
from ..validators import XMLSchema10, XMLSchema11, XsdGroup, \
    XsdAttributeGroup, XsdComplexType, XsdComponent


class SchemaObserver:
    """
    Observer that registers created components. Run the 'clear' method after each usage.
    """
    components = []
    dummy_components = []

    @classmethod
    def observed_builder(cls, builder):
        if isinstance(builder, type):
            class BuilderProxy(builder):
                def __init__(self, *args, **kwargs):
                    super(BuilderProxy, self).__init__(*args, **kwargs)
                    assert isinstance(self, XsdComponent)

                    if not cls.is_dummy_component(self):
                        cls.components.append(self)
                    else:
                        cls.dummy_components.append(self)

            BuilderProxy.__name__ = builder.__name__
            return BuilderProxy

        elif callable(builder):
            @wraps(builder)
            def builder_proxy(*args, **kwargs):
                obj = builder(*args, **kwargs)
                assert isinstance(obj, XsdComponent)

                if not cls.is_dummy_component(obj):
                    cls.components.append(obj)
                else:
                    cls.dummy_components.append(obj)
                return obj

            return builder_proxy

    @classmethod
    def clear(cls) -> None:
        del cls.components[:]
        del cls.dummy_components[:]

    @classmethod
    def is_dummy_component(cls, component) -> bool:
        # Dummy components are empty attribute groups and xs:anyType
        # definitions not related to XSD namespace.
        if component.parent in cls.dummy_components:
            return True
        elif isinstance(component, XsdAttributeGroup):
            return not component
        elif isinstance(component, XsdComplexType):
            return component.name == XSD_ANY_TYPE and \
                component.target_namespace != XSD_NAMESPACE
        elif isinstance(component, XsdGroup) and component.parent is not None:
            return component.parent.name == XSD_ANY_TYPE and \
                component.target_namespace != XSD_NAMESPACE
        return False


observed_builder = SchemaObserver.observed_builder


class ObservedXMLSchema10(XMLSchema10):
    xsd_notation_class = observed_builder(XMLSchema10.xsd_notation_class)
    xsd_complex_type_class = observed_builder(XMLSchema10.xsd_complex_type_class)
    xsd_attribute_class = observed_builder(XMLSchema10.xsd_attribute_class)
    xsd_any_attribute_class = observed_builder(XMLSchema10.xsd_any_attribute_class)
    xsd_attribute_group_class = observed_builder(XMLSchema10.xsd_attribute_group_class)
    xsd_group_class = observed_builder(XMLSchema10.xsd_group_class)
    xsd_element_class = observed_builder(XMLSchema10.xsd_element_class)
    xsd_any_class = observed_builder(XMLSchema10.xsd_any_class)
    xsd_atomic_restriction_class = observed_builder(XMLSchema10.xsd_atomic_restriction_class)
    xsd_list_class = observed_builder(XMLSchema10.xsd_list_class)
    xsd_union_class = observed_builder(XMLSchema10.xsd_union_class)
    xsd_key_class = observed_builder(XMLSchema10.xsd_key_class)
    xsd_keyref_class = observed_builder(XMLSchema10.xsd_keyref_class)
    xsd_unique_class = observed_builder(XMLSchema10.xsd_unique_class)


class ObservedXMLSchema11(XMLSchema11):
    xsd_notation_class = observed_builder(XMLSchema11.xsd_notation_class)
    xsd_complex_type_class = observed_builder(XMLSchema11.xsd_complex_type_class)
    xsd_attribute_class = observed_builder(XMLSchema11.xsd_attribute_class)
    xsd_any_attribute_class = observed_builder(XMLSchema11.xsd_any_attribute_class)
    xsd_attribute_group_class = observed_builder(XMLSchema11.xsd_attribute_group_class)
    xsd_group_class = observed_builder(XMLSchema11.xsd_group_class)
    xsd_element_class = observed_builder(XMLSchema11.xsd_element_class)
    xsd_any_class = observed_builder(XMLSchema11.xsd_any_class)
    xsd_atomic_restriction_class = observed_builder(XMLSchema11.xsd_atomic_restriction_class)
    xsd_list_class = observed_builder(XMLSchema11.xsd_list_class)
    xsd_union_class = observed_builder(XMLSchema11.xsd_union_class)
    xsd_key_class = observed_builder(XMLSchema11.xsd_key_class)
    xsd_keyref_class = observed_builder(XMLSchema11.xsd_keyref_class)
    xsd_unique_class = observed_builder(XMLSchema11.xsd_unique_class)
