from __future__ import (
    annotations,
)
from collections.abc import (
    Callable,
)
from functools import (
    wraps,
)
from inspect import (
    isawaitable,
)
from typing import (
    Any,
    Awaitable,
    Optional,
    Union,
)
from graphql import (
    GraphQLArgument,
    GraphQLField,
    GraphQLObjectType,
    GraphQLSchema,
    GraphQLString,
)
from minos.networks import (
    EnrouteDecoratorKind,
    InMemoryRequest,
    Request,
    Response,
)
from ..decorators import (
    GraphQlEnrouteDecorator,
)
[docs]class GraphQLSchemaBuilder:
    """GraphQL Schema Builder class."""
[docs]    def __init__(self, *args, **kwargs):
        self.schema = GraphQLSchema(**kwargs) 
[docs]    @classmethod
    def build(cls, routes: dict[GraphQlEnrouteDecorator, Callable]) -> GraphQLSchema:
        """Build a new schema from routes.
        :param routes: The routes to build the schema.
        :return: A ``GraphQLSchema`` instance.
        """
        schema_args = cls._build(routes)
        return cls(**schema_args).schema 
    @classmethod
    def _build(cls, routes: dict[GraphQlEnrouteDecorator, Callable]) -> dict[str, Optional[GraphQLObjectType]]:
        query = cls._build_queries(routes)
        mutation = cls._build_mutations(routes)
        return {"query": query, "mutation": mutation}
[docs]    @staticmethod
    def adapt_callback(
        callback: Callable[[Request], Union[Optional[Response], Awaitable[Optional[Response]]]]
    ) -> Callable[[Any, Any, Any], Awaitable[Any]]:
        """Adapt a callback from framework's Request-Response to GraphQl structure.
        :param callback: The callback to be adapted.
        :return: The adapted callback.
        """
        @wraps(callback)
        async def _wrapper(_source, _info, request: Any = None):
            request = InMemoryRequest(request)
            response = callback(request)
            if isawaitable(response):
                response = await response
            return await response.content()
        return _wrapper 
    @classmethod
    def _build_queries(cls, routes: dict[GraphQlEnrouteDecorator, Callable]) -> GraphQLObjectType:
        fields = dict()
        for route, callback in routes.items():
            callback = cls.adapt_callback(callback)
            if route.KIND == EnrouteDecoratorKind.Query:
                fields[route.name] = cls._build_field(route, callback)
        if not len(fields):
            fields["dummy"] = GraphQLField(
                type_=GraphQLString,
                description="Dummy query added to surpass the 'Type Query must define at least one field' constraint.",
            )
        return GraphQLObjectType("Query", fields=fields)
    @classmethod
    def _build_mutations(cls, routes: dict[GraphQlEnrouteDecorator, Callable]) -> Optional[GraphQLObjectType]:
        fields = dict()
        for route, callback in routes.items():
            callback = cls.adapt_callback(callback)
            if route.KIND == EnrouteDecoratorKind.Command:
                fields[route.name] = cls._build_field(route, callback)
        if not len(fields):
            return None
        return GraphQLObjectType("Mutation", fields=fields)
    @staticmethod
    def _build_field(item: GraphQlEnrouteDecorator, callback: Callable) -> GraphQLField:
        argument = item.argument
        output = item.output
        args = None
        if argument is not None:
            args = {"request": GraphQLArgument(argument)}
        return GraphQLField(output, args=args, resolve=callback)