From 17515638e6d0358fa505c1df99a2449522c370dc Mon Sep 17 00:00:00 2001 From: Ted Zadniprovskyi Date: Tue, 20 Feb 2024 09:34:58 -0800 Subject: [PATCH] Allow user to pass in a custom resolve info context type (#213) --- src/graphql/type/definition.py | 65 +++++++++++++++++++++++----------- 1 file changed, 45 insertions(+), 20 deletions(-) diff --git a/src/graphql/type/definition.py b/src/graphql/type/definition.py index 9bea7eed..c04828a9 100644 --- a/src/graphql/type/definition.py +++ b/src/graphql/type/definition.py @@ -2,6 +2,7 @@ from __future__ import annotations # Python < 3.10 +import sys from enum import Enum from typing import ( TYPE_CHECKING, @@ -554,30 +555,54 @@ def to_kwargs(self) -> GraphQLFieldKwargs: def __copy__(self) -> GraphQLField: # pragma: no cover return self.__class__(**self.to_kwargs()) +if sys.version_info.minor == 9 or sys.version_info.minor == 10: + class GraphQLResolveInfo(NamedTuple): + """Collection of information passed to the resolvers. -class GraphQLResolveInfo(NamedTuple): - """Collection of information passed to the resolvers. + This is always passed as the first argument to the resolvers. - This is always passed as the first argument to the resolvers. - - Note that contrary to the JavaScript implementation, the context (commonly used to - represent an authenticated user, or request-specific caches) is included here and - not passed as an additional argument. - """ + Note that contrary to the JavaScript implementation, the context (commonly used to + represent an authenticated user, or request-specific caches) is included here and + not passed as an additional argument. + """ - field_name: str - field_nodes: List[FieldNode] - return_type: GraphQLOutputType - parent_type: GraphQLObjectType - path: Path - schema: GraphQLSchema - fragments: Dict[str, FragmentDefinitionNode] - root_value: Any - operation: OperationDefinitionNode - variable_values: Dict[str, Any] - context: Any - is_awaitable: Callable[[Any], bool] + field_name: str + field_nodes: List[FieldNode] + return_type: GraphQLOutputType + parent_type: GraphQLObjectType + path: Path + schema: GraphQLSchema + fragments: Dict[str, FragmentDefinitionNode] + root_value: Any + operation: OperationDefinitionNode + variable_values: Dict[str, Any] + context: Any + is_awaitable: Callable[[Any], bool] +else: + TContext = TypeVar("TContext") + + class GraphQLResolveInfo(NamedTuple, Generic[TContext]): + """Collection of information passed to the resolvers. + + This is always passed as the first argument to the resolvers. + + Note that contrary to the JavaScript implementation, the context (commonly used to + represent an authenticated user, or request-specific caches) is included here and + not passed as an additional argument. + """ + field_name: str + field_nodes: List[FieldNode] + return_type: GraphQLOutputType + parent_type: GraphQLObjectType + path: Path + schema: GraphQLSchema + fragments: Dict[str, FragmentDefinitionNode] + root_value: Any + operation: OperationDefinitionNode + variable_values: Dict[str, Any] + context: TContext + is_awaitable: Callable[[Any], bool] # Note: Contrary to the Javascript implementation of GraphQLFieldResolver, # the context is passed as part of the GraphQLResolveInfo and any arguments