Skip to content

Latest commit

 

History

History
950 lines (696 loc) · 20.9 KB

REFERENCE.md

File metadata and controls

950 lines (696 loc) · 20.9 KB

API Reference

ObjectType

New ObjectType is base class for Python classes representing GraphQL types (either type or extend type).

__schema__

ObjectType key attribute is __schema__ string that can define only one GraphQL type:

class QueryType(ObjectType):
    __schema__ = """
    type Query {
        year: Int!
    }
    """

ObjectType implements validation logic for __schema__. It verifies that its valid SDL string defining exactly one GraphQL type.

Resolvers

Resolvers are class methods or static methods named after schema's fields:

class QueryType(ObjectType):
    __schema__ = """
    type Query {
        year: Int!
    }
    """

    @staticmethod
    def resolve_year(_, info: GraphQLResolveInfo) -> int:
        return 2022

If resolver function is not present for field, default resolver implemented by graphql-core will be used in its place.

In situations when field's name should be resolved to different value, custom mappings can be defined via __aliases__ attribute:

class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        dateJoined: String!
    }
    """
    __aliases__ = {
        "dateJoined": "date_joined"
    }

Above code will result in Ariadne generating resolver resolving dateJoined field to date_joined attribute on resolved object.

If date_joined exists as resolve_date_joined callable on ObjectType, it will be used as resolver for dateJoined:

class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        dateJoined: String
    }
    """
    __aliases__ = {
        "dateJoined": "date_joined"
    }

    @staticmethod
    def resolve_date_joined(user, info) -> Optional[str]:
        if can_see_activity(info.context):
            return user.date_joined

        return None

__requires__

When GraphQL type requires on other GraphQL type (or scalar/directive etc. ect.) ObjectType will raise an error about missing dependency. This dependency can be provided through __requires__ attribute:

class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        dateJoined: String!
    }
    """


class UsersGroupType(ObjectType):
    __schema__ = """
    type UsersGroup {
        id: ID!
        users: [User!]!
    }
    """
    __requires__ = [UserType]

ObjectType verifies that types specified in __requires__ actually define required types. If __schema__ in UserType is not defining User, error will be raised about missing dependency.

In case of circular dependencies, special DeferredType can be used:

class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        dateJoined: String!
        group: UsersGroup
    }
    """
    __requires__ = [DeferredType("UsersGroup")]


class UsersGroupType(ObjectType):
    __schema__ = """
    type UsersGroup {
        id: ID!
        users: [User!]!
    }
    """
    __requires__ = [UserType]

DeferredType makes UserType happy about UsersGroup dependency, deferring dependency check to make_executable_schema. If "real" UsersGroup is not provided at that time, error will be raised about missing types required to create schema.

__fields_args__

Optional attribute that can be used to specify custom mappings between GraphQL args and Python kwargs:

from ariadne_graphql_modules import DeferredType, ObjectType, gql

from my_app.models import Article


class SearchQuery(ObjectType):
    __schema__ = gql(
        """
        type Query {
            search(query: String!, includeDrafts: Boolean): [Article!]!
        }
        """
    )
    __fields_args__ = {
        "includeDrafts": "with_drafts",
    }
    __requires__ = [DeferredType("Article")]

    @staticmethod
    async def resolve_search(*_, query: str, with_drafts: bool | None):
        articles = Article.query.search(query)
        if not with_drafts:
            articles = articles.filter(is_draft=False)
        return await articles.all()

MutationType

Convenience type for defining single mutation:

from ariadne_graphql_modules import MutationType, gql

from my_app import create_user


class UserRegisterMutation(MutationType):
    __schema__ = gql(
        """
        type Mutation {
            registerUser(username: String!, email: String!): Boolean!
        }
        """
    )

    @staticmethod
    async def resolve_mutation(*_, username: str, email: str):
        user = await create_user(
            full_name=username,
            email=email,
        )
        return bool(user)

Recommended use for this type is to create custom base class for your GraphQL API:

from ariadne_graphql_modules import MutationType, gql


class BaseMutation(MutationType):
    __abstract__ = True

    @classmethod
    async def resolve_mutation(cls, _, *args, **kwargs):
        try:
            return await cls.perform_mutation(cls, *args, **kwargs)
        except Exception as e:
            return {"errors": e}

    @classmethod
    def get_error_result(cls, error):
            return {"errors": [e]}

__args__

Optional attribute that can be used to specify custom mapping between GraphQL schema and Python:

from ariadne_graphql_modules import MutationType, gql

from my_app import create_user


class UserRegisterMutation(MutationType):
    __schema__ = gql(
        """
        type Mutation {
            registerUser(
                userName: String!,
                email: String!,
                admin: Boolean,
            ): Boolean!
        }
        """
    )
    __args__ = {"userName": "username", "admin": "is_admin"}

    @staticmethod
    async def resolve_mutation(*_, username: str, email: str, is_admin: bool | None):
        user = await create_user(
            full_name=username,
            email=email,
            is_admin=bool(is_admin),
        )
        return bool(user)

SubscriptionType

Specialized subclass of ObjectType that defines GraphQL subscription:

class ChatSubscriptions(SubscriptionType):
    __schema__ = """
    type Subscription {
        chat: Chat
    }
    """
    __requires__ = [ChatType]

    @staticmethod
    async def subscribe_chat(*_):
        async for event in subscribe("chats"):
            yield event["chat_id"]

    @staticmethod
    async def resolve_chat(chat_id, *_):
        # Optional
        return await get_chat_from_db(chat_id)

InputType

Defines GraphQL input:

class UserCreateInput(InputType):
    __schema__ = """
    input UserInput {
        name: String!
        email: String!
        fullName: String!
    }
    """
    __args__ = {
        "fullName": "full_name",
    }

__args__

Optional attribue __args__ is a Dict[str, str] used to override key names for dict representing input's data.

Following JSON:

{
    "name": "Alice",
    "email:" "[email protected]",
    "fullName": "Alice Chains"
}

Will be represented as following dict:

{
    "name": "Alice",
    "email": "[email protected]",
    "full_name": "Alice Chains",
}

ScalarType

Allows you to define custom scalar in your GraphQL schema.

class DateScalar(ScalarType):
    __schema__ = "scalar Datetime"

    @staticmethod
    def serialize(value) -> str:
        # Called by GraphQL to serialize Python value to
        # JSON-serializable format
        return value.strftime("%Y-%m-%d")

    @staticmethod
    def parse_value(value) -> str:
        # Called by GraphQL to parse JSON-serialized value to
        # Python type
        parsed_datetime = datetime.strptime(formatted_date, "%Y-%m-%d")
        return parsed_datetime.date()

Note that those methods are only required if Python type is not JSON serializable, or you want to customize its serialization process.

Additionally you may define third method called parse_literal that customizes value's deserialization from GraphQL query's AST, but this is only useful for complex types that represent objects:

from graphql import StringValueNode


class DateScalar(Scalar):
    __schema__ = "scalar Datetime"

    @staticmethod
    def def parse_literal(ast, variable_values: Optional[Dict[str, Any]] = None):
        if not isinstance(ast, StringValueNode):
            raise ValueError()

        parsed_datetime = datetime.strptime(ast.value, "%Y-%m-%d")
        return parsed_datetime.date()

If you won't define parse_literal, GraphQL will use custom logic that will unpack value from AST and then call parse_value on it.

EnumType

Defines enum in GraphQL schema:

class UserRoleEnum(EnumType):
    __schema__ = """
    enum UserRole {
        USER
        MOD
        ADMIN
    }
    """

__enum__ attribute allows you to specify Python enum to represent GraphQL enum in your Python logic:

class UserRole(IntEnum):
    USER = 0
    MOD = 1
    ADMIN = 1


class UserRoleEnum(EnumType):
    __schema__ = """
    enum UserRole {
        USER
        MOD
        ADMIN
    }
    """
    __enum__ = UserRole

You can also make __enum__ a dict to skip enum if you want:

class UserRoleEnum(EnumType):
    __schema__ = """
    enum UserRole {
        USER
        MOD
        ADMIN
    }
    """
    __enum__ = {
        "USER": 0,
        "MOD": 1,
        "ADMIN": 2,
    }

InterfaceType

Defines interface in GraphQL schema:

class SearchResultInterface(InterfaceType):
    __schema__ = """
    interface SearchResult {
        summary: String!
        score: Int!
    }
    """

    @staticmethod
    def resolve_type(obj, info):
        # Returns string with name of GraphQL type representing Python type
        # from your business logic
        if isinstance(obj, UserModel):
            return UserType.graphql_name

        if isinstance(obj, CommentModel):
            return CommentType.graphql_name

        return None

    @staticmethod
    def resolve_summary(obj, info):
        # Optional default resolver for summary field, used by types implementing
        # this interface when they don't implement their own

UnionType

Defines GraphQL union:

class SearchResultUnion(UnionType):
    __schema__ = "union SearchResult = User | Post | Thread"
    __requires__ = [UserType, PostType, ThreadType]

    @staticmethod
    def resolve_type(obj, info):
        # Returns string with name of GraphQL type representing Python type
        # from your business logic
        if isinstance(obj, UserModel):
            return UserType.graphql_name

        if isinstance(obj, PostModel):
            return PostType.graphql_name

        if isinstance(obj, ThreadModel):
            return ThreadType.graphql_name

        return None

DirectiveType

Defines new GraphQL directive in your schema and specifies SchemaDirectiveVisitor for it:

from ariadne import SchemaDirectiveVisitor
from graphql import default_field_resolver


class PrefixStringSchemaVisitor(SchemaDirectiveVisitor):
    def visit_field_definition(self, field, object_type):
        original_resolver = field.resolve or default_field_resolver

        def resolve_prefixed_value(obj, info, **kwargs):
            result = original_resolver(obj, info, **kwargs)
            if result:
                return f"PREFIX: {result}"
            return result

        field.resolve = resolve_prefixed_value
        return field


class PrefixStringDirective(DirectiveType):
    __schema__ = "directive @example on FIELD_DEFINITION"
    __visitor__ = PrefixStringSchemaVisitor

make_executable_schema

New make_executable_schema takes list of Ariadne's types and constructs executable schema from them, performing last-stage validation for types consistency:

class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        username: String!
    }
    """


class QueryType(ObjectType):
    __schema__ = """
    type Query {
        user: User
    }
    """
    __requires__ = [UserType]

    @staticmethod
    def user(*_):
        return {
            "id": 1,
            "username": "Alice",
        }


schema = make_executable_schema(QueryType)

Automatic merging of roots

Passing multiple Query, Mutation or Subscription definitions to make_executable_schema by default results in schema defining single types containing sum of all fields defined on those types, ordered alphabetically by field name.

class UserQueriesType(ObjectType):
    __schema__ = """
    type Query {
        user(id: ID!): User
    }
    """
    ...


class ProductsQueriesType(ObjectType):
    __schema__ = """
    type Query {
        product(id: ID!): Product
    }
    """
    ...

schema = make_executable_schema(UserQueriesType, ProductsQueriesType)

Above schema will have single Query type looking like this:

type Query {
    product(id: ID!): Product
    user(id: ID!): User
}

To opt out of this behavior use merge_roots=False option:

schema = make_executable_schema(
    UserQueriesType,
    ProductsQueriesType,
    merge_roots=False,
)

DeferredType

DeferredType names required GraphQL type as provided at later time:

  • Via make_executable_schema call
  • Or via other type's __requires__

It's mostly used to define lazy relationships in reusable modules and to break circular relationships.

Type with DeferredType and other type passed to make_executable_schema:

class QueryType(ObjectType):
    __schema__ = """
    type Query {
        id: ID!
        users: [User!]!
    }
    """
    __requires__ = [DeferredType("User")]


class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        dateJoined: String!
    }
    """


schema = make_excutable_schema(QueryType, UserType)

Type with DeferredType and other type passed as dependency of third type:

class UsersGroupType(ObjectType):
    __schema__ = """
    type UsersGroup {
        id: ID!
        users: [User!]!
    }
    """
    __requires__ = [UserType]


class UserType(ObjectType):
    __schema__ = """
    type User {
        id: ID!
        dateJoined: String!
    }
    """


class QueryType(ObjectType):
    __schema__ = """
    type Query {
        id: ID!
        users: [User!]!
        groups: [UsersGroup!]!
    }
    """
    __requires__ = [DeferredType("User"), UsersGroupType]


schema = make_excutable_schema(QueryType)

CollectionType

Collection is an utility type that gathers multiple types into single object:

class UserMutations(CollectionType):
    __types__ = [
        BanUserMutation,
        UnbanUserMutation,
        CreateUserMutation,
        UpdateUserMutation,
        DeleteUserMutation,
    ]


schema = make_excutable_schema(UserMutations)

BaseType

Base type that all other types extend. You can use it to create custom types:

from typing import Dict

from ariadne_graphql_modules import BaseType
from django.db.models import Model
from graphql import GraphQLFieldResolver

class MyType(BaseType)
    __abstract__ = True

    @classmethod
    def __get_types__(cls) -> List[Type["BaseType"]]:
        # Called by make_executable_schema to get list of types
        # to build GraphQL schema from.
        return []

DefinitionType

Base for types that define __schema__:

class MyType(DefinitionType)
    __abstract__: bool = True
    __schema__: str
    __requires__: List[Union[Type["DefinitionType"], DeferredType]] = []

    graphql_name: str
    graphql_type: Type[DefinitionNode]

Extends BaseType.

BindableType

Base for types that define __bind_to_schema__ class method:

class MyType(BindableType)
    __abstract__: bool = True
    __schema__: str
    __requires__: List[Union[Type["DefinitionType"], DeferredType]] = []

    graphql_name: str
    graphql_type: Type[DefinitionNode]

    @classmethod
    def __bind_to_schema__(cls, schema: GraphQLSchema):
        pass  # Bind python logic to GraphQL schema here

Extends DefinitionType.

make_executable_schema

def make_executable_schema(
    *args: Union[Type[BaseType], SchemaBindable, str],
    merge_roots: bool = True,
    extra_sdl: Optional[Union[str, Sequence[str]]] = None,
    extra_bindables: Optional[Sequence[SchemaBindable]] = None,
    extra_directives: Optional[Dict[str, Type[SchemaDirectiveVisitor]]] = None,
) -> GraphQLSchema:
    ...

Utility function that takes args with types definitions and creates executable schema.

merge_roots: bool = True

If set to true (default), make_executable_schema will automatically merge multiple Query, Mutation and Subscription types instead of raising error.

Final merged types fields will be ordered alphabetically:

class YearType(ObjectType):
    __schema__ = """
        type Query {
            year: Int!
        }
    """

    @staticmethod
    def resolve_year(*_):
        return 2022


class MonthType(ObjectType):
    __schema__ = """
        type Query {
            month: Int!
        }
    """

    @staticmethod
    def resolve_month(*_):
        return 10


schema = make_executable_schema(YearType, MonthType)

assert print_schema(schema) == """
type Query {
    month: Int!
    year: Int!
}
"""

When merge_roots=False is explicitly set, make_executable_schema will raise an GraphQL error that type Query is defined more than once.

extra_directives

Dict of Ariadne's directives names and implementation. Optional.

Used when moving schema definition to Ariadne GraphQL Modules approach from existing schema definition.

See moving guide for examples and details.

convert_case

Utility function that can be used to automatically setup case conversion rules for types.

Resolving fields values

Use __aliases__ = convert_case to automatically set aliases for fields that convert case

from ariadne_graphql_modules import ObjectType, convert_case, gql


class UserType(ObjectType):
    __schema__ = gql(
        """
        type User {
            id: ID!
            fullName: String!
        }
        """
    )
    __aliases__ = convert_case

Converting fields arguments

Use __fields_args__ = convert_case on type to automatically convert field arguments to python case in resolver kwargs:

from ariadne_graphql_modules import DeferredType, ObjectType, convert_case, gql

from my_app.models import Article


class SearchQuery(ObjectType):
    __schema__ = gql(
        """
        type Query {
            search(query: String!, includeDrafts: Boolean): [Article!]!
        }
        """
    )
    __fields_args__ = convert_case
    __requires__ = [DeferredType("Article")]

    @staticmethod
    async def resolve_search(*_, query: str, include_drafts: bool | None):
        articles = Article.query.search(query)
        if not include_drafts:
            articles = articles.filter(is_draft=False)
        return await articles.all()

Converting mutation arguments

Use __args__ = convert_case on MutationType to automatically convert input fields to python case in resolver kwargs:

from ariadne_graphql_modules import MutationType, convert_case, gql

from my_app import create_user


class UserRegisterMutation(MutationType):
    __schema__ = gql(
        """
        type Mutation {
            registerUser(fullName: String!, email: String!): Boolean!
        }
        """
    )
    __args__ = convert_case

    @staticmethod
    async def resolve_mutation(*_, full_name: str, email: str):
        user = await create_user(
            full_name=full_name,
            email=email,
        )
        return bool(user)

Converting inputs fields

Use __args__ = convert_case on InputType to automatically convert input fields to python case in resolver kwargs:

from ariadne_graphql_modules import InputType, MutationType, convert_case, gql

from my_app import create_user


class UserRegisterInput(InputType):
    __schema__ = gql(
        """
        input UserRegisterInput {
            fullName: String!
            email: String!
        }
        """
    )
    __args__ = convert_case


class UserRegisterMutation(MutationType):
    __schema__ = gql(
        """
        type Mutation {
            registerUser(input: UserRegisterInput!): Boolean!
        }
        """
    )
    __requires__ = [UserRegisterInput]

    @staticmethod
    async def resolve_mutation(*_, input: dict):
        user = await create_user(
            full_name=input["full_name"],
            email=input["email"],
        )
        return bool(user)