Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Ability to have one class per query instead of one method per query #714

Closed
un33k opened this issue Apr 16, 2018 · 8 comments
Closed

Ability to have one class per query instead of one method per query #714

un33k opened this issue Apr 16, 2018 · 8 comments
Labels

Comments

@un33k
Copy link

un33k commented Apr 16, 2018

[x] Feature request
[x] Or - Docs request
graphene==2.1.0

Is it possible to have a similar pattern for queries, similar to mutation.
This is to avoid long classes with multiple resolvers.

It would be great if this would be possible.

# mutation -- working example
class JwtRefreshMutation(graphene.Mutation):
    class Arguments:
        token=graphene.String(required=True)

    token = graphene.String()

    def mutate(self, info, **args):
        token = refresh_token(args['token'])
        return JwtRefreshMutation(token=token)

class Mutations(graphene.ObjectType):
    jwt_refresh = JwtRefreshMutation.Field()
# query -- desired example
class JwtRefreshQuery(graphene.Query):
    class Arguments:
        token=graphene.String(required=True)

    token = graphene.String()

    def resolve(self, info, **args):
        token = refresh_token(args['token'])
        return JwtRefreshQuery(token=token)

class Query(graphene.ObjectType):
    jwt_refresh = JwtRefreshQuery.Field()
@HeyHugo
Copy link

HeyHugo commented May 1, 2018

Hi,

you can do like this:

class JwtRefreshQuery(object):
    token = graphene.String(graphene.String, token=graphene.String(required=True))

    def resolve_token(self, info, token):
        return refresh_token(token)


class Query(JwtRefreshQuery, SomeOtherQuery, graphene.ObjectType):
    pass

I agree intuitively I think it would make more sense if the API for mutations and queries were more similar. In any case yes the docs/examles are a bit lacking in showing how you can do "merge multiple query classes"

@jkimbo
Copy link
Member

jkimbo commented May 6, 2018

So I've been thinking about this a lot and I think that being able to define a class to resolve an ObjectType would be quite useful. I've got some working code based on the Mutation class @un33k if you want to include it in your application and see if it works:

from collections import OrderedDict

from graphene.utils.get_unbound_function import get_unbound_function
from graphene.utils.props import props
from graphene.types.field import Field
from graphene.types.objecttype import ObjectType, ObjectTypeOptions
from graphene.types.utils import yank_fields_from_attrs
from graphene.utils.deprecated import warn_deprecation


class FieldResolverOptions(ObjectTypeOptions):
    arguments = None  # type: Dict[str, Argument]
    output = None  # type: Type[ObjectType]
    resolver = None  # type: Callable


class FieldResolver(ObjectType):
    @classmethod
    def __init_subclass_with_meta__(cls, resolver=None, output=None, arguments=None,
                                    _meta=None, **options):
        if not _meta:
            _meta = FieldResolverOptions(cls)

        output = output or getattr(cls, 'Output', None)
        fields = {}
        if not output:
            # If output is defined, we don't need to get the fields
            fields = OrderedDict()
            for base in reversed(cls.__mro__):
                fields.update(
                    yank_fields_from_attrs(base.__dict__, _as=Field)
                )
            output = cls

        if not arguments:
            input_class = getattr(cls, 'Arguments', None)
            if input_class:
                arguments = props(input_class)
            else:
                arguments = {}

        if not resolver:
            _resolver = getattr(cls, 'resolve', None)
            assert _resolver, 'All field resolvers must define a resolve method'
            resolver = get_unbound_function(_resolver)

        if _meta.fields:
            _meta.fields.update(fields)
        else:
            _meta.fields = fields

        _meta.output = output
        _meta.resolver = resolver
        _meta.arguments = arguments

        super(FieldResolver, cls).__init_subclass_with_meta__(
            _meta=_meta, **options)

    @classmethod
    def Field(cls, name=None, description=None, deprecation_reason=None, required=False):
        return Field(
            cls._meta.output,
            args=cls._meta.arguments,
            resolver=cls._meta.resolver,
            name=name,
            description=description,
            deprecation_reason=deprecation_reason,
            required=required,
        )

You use it in pretty much the way you want to in your example:

class JwtRefreshQuery(FieldResolver):
    class Arguments:
        token = graphene.String(required=True)

    token = graphene.String()

    def resolve(self, info, **args):
        token = refresh_token(args['token'])
        return JwtRefreshQuery(token=token)

class Query(graphene.ObjectType):
    jwt_refresh = JwtRefreshQuery.Field()

I'm not sure what to call it at the moment or where it might fit in in the project. It might even be that the ObjectType class should be extended to allow this kind of use. Any thoughts @syrusakbary ?

@HeyHugo
Copy link

HeyHugo commented May 8, 2018

nice @jkimbo I like it!

@ProjectCheshire
Copy link
Member

ProjectCheshire commented May 23, 2018

@HeyHugo I have a fractal style schema where one module per entity and a mutation / type / model / query / subscription files within that.

I also have my resolvers separated out between query/mute and field resolvers just to keep the logic semi organized and then all my fields have resolver= in the types.

What is magic is that I auto generate my schema by some old fashioned python and inheriting from common classes by root type. I forget from whom I sourced this from, but it's worked really well for me for the past year ish.

It's a workaround, and I'd love to see the query / mutation homogenity too :)
https://gist.github.com/ProjectCheshire/277c5ff1468460a05c2a44556e270eff

@cmmartti
Copy link

cmmartti commented Jun 1, 2018

@ProjectCheshire How do you handle circular imports? I can use lambdas if I put everything in one file, but I'd like to use yours and ahopkins' approach, which otherwise works great.

@ProjectCheshire
Copy link
Member

ProjectCheshire commented Jun 1, 2018

@cmmartti similar to how graphene does with lazy import. https://pypi.org/project/lazy-import/

There's probably a github issue around here that mentions it specifically :)

For functions I often use lazy callable if there is some shared logic I need*. (such as inflating models via neomodel) Since I'm using a graph database, I get a lot of circular models.

*I'm not a guru, so if you delve deep and find the way to pass the type hints (eg your typical ide sugars) along when calling a lazy callable, I'm all ears.
(geography is just my mixin for wkt and geojson fields and some common funcs I use when using the spatial plugins on neo4j)

from graphene import String, Int, ObjectType, Float, Field, lazy_import
from api.models.global_resolvers import country_
from ..fields import GeographyFields


class CityType(ObjectType, GeographyFields):
    """A City represents authoritative data on a location as defined in Geonames.org"""
    name = String()
    country = Field(lazy_import('api.models.country.type.CountryType'), resolver=country_)
    timezone = String()
    latitude = Float()
    longitude = Float()
    population = Int()
    featureCode = String()
    featureClass = String()
    geonameid = Int()

@cmmartti
Copy link

cmmartti commented Jun 1, 2018

@ProjectCheshire Oh wow, yet another thing that should be in the docs but isn't. That solves my problem perfectly :)

Found an issue that mentions it. Thought I'd been thorough with my searches, but apparently not.

I put together a small working example of this "fractal-style schema" approach here.

@stale
Copy link

stale bot commented Jul 29, 2019

This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions.

@stale stale bot added the wontfix label Jul 29, 2019
@stale stale bot closed this as completed Aug 5, 2019
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

5 participants