forked from graphql-python/graphql-server
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgraphqlview.py
171 lines (147 loc) · 5.31 KB
/
graphqlview.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
import copy
from collections.abc import MutableMapping
from functools import partial
from typing import List
from graphql.error import GraphQLError
from graphql.type.schema import GraphQLSchema
from webob import Response
from graphql_server import (
GraphQLParams,
HttpQueryError,
encode_execution_results,
format_error_default,
json_encode,
load_json_body,
run_http_query,
)
from graphql_server.render_graphiql import (
GraphiQLConfig,
GraphiQLData,
render_graphiql_sync,
)
class GraphQLView:
schema = None
request = None
root_value = None
context = None
pretty = False
graphiql = False
graphiql_version = None
graphiql_template = None
graphiql_html_title = None
middleware = None
batch = False
enable_async = False
subscriptions = None
headers = None
charset = "UTF-8"
format_error = staticmethod(format_error_default)
encode = staticmethod(json_encode)
def __init__(self, **kwargs):
super(GraphQLView, self).__init__()
for key, value in kwargs.items():
if hasattr(self, key):
setattr(self, key, value)
assert isinstance(
self.schema, GraphQLSchema
), "A Schema is required to be provided to GraphQLView."
def get_root_value(self):
return self.root_value
def get_context(self, request):
context = (
copy.copy(self.context)
if self.context and isinstance(self.context, MutableMapping)
else {}
)
if isinstance(context, MutableMapping) and "request" not in context:
context.update({"request": request})
return context
def get_middleware(self):
return self.middleware
def dispatch_request(self, request):
try:
request_method = request.method.lower()
data = self.parse_body(request)
show_graphiql = request_method == "get" and self.should_display_graphiql(
request
)
catch = show_graphiql
pretty = self.pretty or show_graphiql or request.params.get("pretty")
all_params: List[GraphQLParams]
execution_results, all_params = run_http_query(
self.schema,
request_method,
data,
query_data=request.params,
batch_enabled=self.batch,
catch=catch,
# Execute options
run_sync=not self.enable_async,
root_value=self.get_root_value(),
context_value=self.get_context(request),
middleware=self.get_middleware(),
)
result, status_code = encode_execution_results(
execution_results,
is_batch=isinstance(data, list),
format_error=self.format_error,
encode=partial(self.encode, pretty=pretty), # noqa
)
if show_graphiql:
graphiql_data = GraphiQLData(
result=result,
query=getattr(all_params[0], "query"),
variables=getattr(all_params[0], "variables"),
operation_name=getattr(all_params[0], "operation_name"),
subscription_url=self.subscriptions,
headers=self.headers,
)
graphiql_config = GraphiQLConfig(
graphiql_version=self.graphiql_version,
graphiql_template=self.graphiql_template,
graphiql_html_title=self.graphiql_html_title,
jinja_env=None,
)
return Response(
render_graphiql_sync(data=graphiql_data, config=graphiql_config),
charset=self.charset,
content_type="text/html",
)
return Response(
result,
status=status_code,
charset=self.charset,
content_type="application/json",
)
except HttpQueryError as e:
parsed_error = GraphQLError(e.message)
return Response(
self.encode(dict(errors=[self.format_error(parsed_error)])),
status=e.status_code,
charset=self.charset,
headers=e.headers or {},
content_type="application/json",
)
# WebOb
@staticmethod
def parse_body(request):
# We use mimetype here since we don't need the other
# information provided by content_type
content_type = request.content_type
if content_type == "application/graphql":
return {"query": request.body.decode("utf8")}
elif content_type == "application/json":
return load_json_body(request.body.decode("utf8"))
elif content_type in (
"application/x-www-form-urlencoded",
"multipart/form-data",
):
return request.params
return {}
def should_display_graphiql(self, request):
if not self.graphiql or "raw" in request.params:
return False
return self.request_wants_html()
def request_wants_html(self):
best = self.request.accept.best_match(["application/json", "text/html"])
return best == "text/html"