- 
                Notifications
    You must be signed in to change notification settings 
- Fork 819
v3 release notes
Welcome to Graphene v3!
This is our biggest ever release and brings Graphene up to date with the latest changes in the GraphQL spec as well as delivering some new features and bug fixes. Since this is a major version release there are some backwards incompatible changes you’ll want to be aware of when upgrading from Graphene 2.1 or earlier.
Graphene 3.0 drops support for Python 2.x and now supports Python 3.6, 3.7, and 3.8.
Graphene builds on top of the excellent GraphQL-core library which is a direct port of the reference GraphQL.js library. Graphene v3 now uses GraphQL-core v3 which is a significant update from v2 bringing lots of new language features and other improvements.
Graphene v3 introduces better Enum support which should improve the developer experience of working with them in Graphene. Previously when resolving an Enum you had to return the Enum value from the field resolver. In v3 you can now return the Enum member directly:
from graphene import Enum, ObjectType, Schema
class Color(Enum):
    RED = 1
    GREEN = 2
    BLUE = 3
class Query(ObjectType):
    color = Color(required=True)
    def resolve_color(root, info):
        return Color.RED
schema = Schema(query=Query)
result = schema.execute("query { color }")
assert result.data["color"] == "RED"(This change is completely backwards compatible so any resolvers that return the member value will still work)
Also when Enum's are used as an input to a field the resolver now receives the Enum member directly rather than the value:
from graphene import Enum, ObjectType, Schema
class Color(Enum):
    RED = 1
    GREEN = 2
    BLUE = 3
class Query(ObjectType):
    color = Color(required=True, color_input=Color(required=True))
    def resolve_color(root, info, color_input):
        assert color_input is Color.RED
        return color_input
schema = Schema(query=Query)
result = schema.execute("query { color(colorInput: RED) }")
assert result.data["color"] == "RED"Reference: https://github.com/graphql-python/graphene/pull/1153
TODO
In Graphene v3 optimises the ObjectType initialization resulting in a x3 speed up! It does this by leveraging the same strategy that dataclasses introduced (dynamic creation of optimal __init__ functions based on eval).
More info here: https://github.com/graphql-python/graphene/pull/1157
- Add a new Base64 scalar type (https://github.com/graphql-python/graphene/pull/1221)
- Better error messages when relay global_idfails to parse (https://github.com/graphql-python/graphene/pull/1074)
- Graphene now accepts plain GraphQL types as well as Graphene types (https://github.com/graphql-python/graphene/pull/1224)
GraphQL-core v3 removed support for the backends feature that was available in v2 and so it is no longer possible to replace the GraphQL-core backend for another one. We are looking for alternatives to support the same features that backends did but at the moment there is no ETA on when it might be available.
As outlined above: if a field takes an Enum as an input the resolver will now get passed the Enum member rather than the member value.
For example, given the following schema:
from graphene import Enum, ObjectType, String, Schema
class Color(Enum):
    RED = 1
    GREEN = 2
    BLUE = 3
class Query(ObjectType):
    color = String(color_input=Color(required=True))
    def resolve_color(root, info, color_input):
        return color_input
schema = Schema(query=Query)Before:
result = schema.execute("query { color(colorInput: RED) }")
assert result.data["color"] == "1"After:
result = schema.execute("query { color(colorInput: RED) }")
assert result.data["color"] == "EnumMeta.RED"Previously you ObjectType's could be initialized with private variables by prepending the variable name with an _. This is no longer possible in v3 without defining a custom __init__ function on the ObjectType.
Before:
from graphene import ObjectType, String
class User(ObjectType):
    _private_state = None
    name = String()
user = User(name="Leia", _private_state="Extra info")
assert user._private_state == "Extra info"After:
from graphene import ObjectType, String
class User(ObjectType):
    def __init__(self, _private_state=None, **kwargs):
        self._private_state = _private_state
        super().__init__(**kwargs)
    _private_state = None
    name = String()
user = User(name="Leia", _private_state="Extra info")
assert user._private_state == "Extra info"The Graphene Schema type no longer subclasses the GraphQLSchema type but instead references it through the graphql_schema attribute.
Before:
from graphql import GraphQLSchema
from graphene import Schema
schema = Schema(query=Query)
assert isinstance(schema, GraphQLSchema)After:
from graphql import GraphQLSchema
from graphene import Schema
schema = Schema(query=Query)
assert isinstance(schema.graphql_schema, GraphQLSchema)A huge thanks to everyone involved in bringing this release together!
Full changelog https://github.com/graphql-python/graphene/compare/v2.1.8...v3.0.0