2016-09-20 08:33:46 +03:00
|
|
|
import inspect
|
2016-09-20 08:04:23 +03:00
|
|
|
import json
|
2016-09-20 08:15:10 +03:00
|
|
|
import re
|
2016-09-18 02:29:00 +03:00
|
|
|
|
2016-09-20 08:15:10 +03:00
|
|
|
import six
|
2016-09-20 08:04:23 +03:00
|
|
|
from django.http import HttpResponse, HttpResponseNotAllowed
|
|
|
|
from django.http.response import HttpResponseBadRequest
|
|
|
|
from django.shortcuts import render
|
2016-09-20 09:14:59 +03:00
|
|
|
from django.utils.decorators import method_decorator
|
2016-09-20 08:15:10 +03:00
|
|
|
from django.views.generic import View
|
2016-09-20 09:14:59 +03:00
|
|
|
from django.views.decorators.csrf import ensure_csrf_cookie
|
2016-09-18 02:29:00 +03:00
|
|
|
|
2016-09-20 08:15:10 +03:00
|
|
|
from graphql import Source, execute, parse, validate
|
|
|
|
from graphql.error import format_error as format_graphql_error
|
|
|
|
from graphql.error import GraphQLError
|
2016-09-20 08:04:23 +03:00
|
|
|
from graphql.execution import ExecutionResult
|
2016-09-20 08:15:10 +03:00
|
|
|
from graphql.type.schema import GraphQLSchema
|
2016-09-20 08:04:23 +03:00
|
|
|
from graphql.utils.get_operation_ast import get_operation_ast
|
2016-09-18 03:09:56 +03:00
|
|
|
|
2016-09-20 08:04:23 +03:00
|
|
|
from .settings import graphene_settings
|
|
|
|
|
|
|
|
|
|
|
|
class HttpError(Exception):
|
2016-09-20 08:15:10 +03:00
|
|
|
|
2016-09-20 08:04:23 +03:00
|
|
|
def __init__(self, response, message=None, *args, **kwargs):
|
|
|
|
self.response = response
|
|
|
|
self.message = message = message or response.content.decode()
|
|
|
|
super(HttpError, self).__init__(message, *args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
def get_accepted_content_types(request):
|
|
|
|
def qualify(x):
|
|
|
|
parts = x.split(';', 1)
|
|
|
|
if len(parts) == 2:
|
|
|
|
match = re.match(r'(^|;)q=(0(\.\d{,3})?|1(\.0{,3})?)(;|$)',
|
|
|
|
parts[1])
|
|
|
|
if match:
|
2017-12-12 05:08:42 +03:00
|
|
|
return parts[0].strip(), float(match.group(2))
|
|
|
|
return parts[0].strip(), 1
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
raw_content_types = request.META.get('HTTP_ACCEPT', '*/*').split(',')
|
|
|
|
qualified_content_types = map(qualify, raw_content_types)
|
|
|
|
return list(x[0] for x in sorted(qualified_content_types,
|
|
|
|
key=lambda x: x[1], reverse=True))
|
|
|
|
|
|
|
|
|
2016-09-20 08:33:46 +03:00
|
|
|
def instantiate_middleware(middlewares):
|
|
|
|
for middleware in middlewares:
|
|
|
|
if inspect.isclass(middleware):
|
|
|
|
yield middleware()
|
|
|
|
continue
|
|
|
|
yield middleware
|
|
|
|
|
|
|
|
|
2016-09-20 08:04:23 +03:00
|
|
|
class GraphQLView(View):
|
2017-12-28 11:55:03 +03:00
|
|
|
graphiql_version = '0.11.10'
|
2016-09-20 08:04:23 +03:00
|
|
|
graphiql_template = 'graphene/graphiql.html'
|
|
|
|
|
2016-09-20 08:50:51 +03:00
|
|
|
schema = None
|
2016-09-20 08:04:23 +03:00
|
|
|
graphiql = False
|
|
|
|
executor = None
|
|
|
|
middleware = None
|
|
|
|
root_value = None
|
|
|
|
pretty = False
|
2016-10-31 13:56:51 +03:00
|
|
|
batch = False
|
2016-09-20 08:04:23 +03:00
|
|
|
|
2016-10-31 13:56:51 +03:00
|
|
|
def __init__(self, schema=None, executor=None, middleware=None, root_value=None, graphiql=False, pretty=False,
|
|
|
|
batch=False):
|
2016-09-20 08:04:23 +03:00
|
|
|
if not schema:
|
|
|
|
schema = graphene_settings.SCHEMA
|
|
|
|
|
|
|
|
if middleware is None:
|
|
|
|
middleware = graphene_settings.MIDDLEWARE
|
|
|
|
|
2017-07-11 13:03:08 +03:00
|
|
|
self.schema = self.schema or schema
|
2016-09-20 08:33:46 +03:00
|
|
|
if middleware is not None:
|
2016-09-20 10:08:14 +03:00
|
|
|
self.middleware = list(instantiate_middleware(middleware))
|
2016-09-20 08:04:23 +03:00
|
|
|
self.executor = executor
|
|
|
|
self.root_value = root_value
|
2017-07-11 13:03:08 +03:00
|
|
|
self.pretty = self.pretty or pretty
|
|
|
|
self.graphiql = self.graphiql or graphiql
|
|
|
|
self.batch = self.batch or batch
|
2016-09-20 08:04:23 +03:00
|
|
|
|
2017-10-25 20:54:13 +03:00
|
|
|
assert isinstance(
|
|
|
|
self.schema, GraphQLSchema), 'A Schema is required to be provided to GraphQLView.'
|
|
|
|
assert not all((graphiql, batch)
|
|
|
|
), 'Use either graphiql or batch processing'
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
# noinspection PyUnusedLocal
|
|
|
|
def get_root_value(self, request):
|
|
|
|
return self.root_value
|
|
|
|
|
|
|
|
def get_middleware(self, request):
|
|
|
|
return self.middleware
|
|
|
|
|
|
|
|
def get_context(self, request):
|
|
|
|
return request
|
|
|
|
|
2016-09-20 09:14:59 +03:00
|
|
|
@method_decorator(ensure_csrf_cookie)
|
2016-09-20 08:04:23 +03:00
|
|
|
def dispatch(self, request, *args, **kwargs):
|
|
|
|
try:
|
|
|
|
if request.method.lower() not in ('get', 'post'):
|
2017-10-25 20:54:13 +03:00
|
|
|
raise HttpError(HttpResponseNotAllowed(
|
|
|
|
['GET', 'POST'], 'GraphQL only supports GET and POST requests.'))
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
data = self.parse_body(request)
|
2017-10-25 20:54:13 +03:00
|
|
|
show_graphiql = self.graphiql and self.can_display_graphiql(
|
|
|
|
request, data)
|
2016-09-20 08:04:23 +03:00
|
|
|
|
2016-10-31 13:56:51 +03:00
|
|
|
if self.batch:
|
|
|
|
responses = [self.get_response(request, entry) for entry in data]
|
|
|
|
result = '[{}]'.format(','.join([response[0] for response in responses]))
|
2017-09-27 18:46:32 +03:00
|
|
|
status_code = responses and max(responses, key=lambda response: response[1])[1] or 200
|
2016-09-20 08:04:23 +03:00
|
|
|
else:
|
2017-10-25 20:54:13 +03:00
|
|
|
result, status_code = self.get_response(
|
|
|
|
request, data, show_graphiql)
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
if show_graphiql:
|
2017-10-25 20:54:13 +03:00
|
|
|
query, variables, operation_name, id = self.get_graphql_params(
|
|
|
|
request, data)
|
2016-09-20 08:04:23 +03:00
|
|
|
return self.render_graphiql(
|
|
|
|
request,
|
|
|
|
graphiql_version=self.graphiql_version,
|
|
|
|
query=query or '',
|
2016-10-31 05:05:30 +03:00
|
|
|
variables=json.dumps(variables) or '',
|
2016-09-20 08:04:23 +03:00
|
|
|
operation_name=operation_name or '',
|
|
|
|
result=result or ''
|
|
|
|
)
|
|
|
|
|
|
|
|
return HttpResponse(
|
|
|
|
status=status_code,
|
|
|
|
content=result,
|
|
|
|
content_type='application/json'
|
|
|
|
)
|
|
|
|
|
|
|
|
except HttpError as e:
|
|
|
|
response = e.response
|
|
|
|
response['Content-Type'] = 'application/json'
|
|
|
|
response.content = self.json_encode(request, {
|
|
|
|
'errors': [self.format_error(e)]
|
|
|
|
})
|
|
|
|
return response
|
|
|
|
|
2016-10-31 13:56:51 +03:00
|
|
|
def get_response(self, request, data, show_graphiql=False):
|
2017-10-25 20:54:13 +03:00
|
|
|
query, variables, operation_name, id = self.get_graphql_params(
|
|
|
|
request, data)
|
2016-10-31 13:56:51 +03:00
|
|
|
|
|
|
|
execution_result = self.execute_graphql_request(
|
|
|
|
request,
|
|
|
|
data,
|
|
|
|
query,
|
|
|
|
variables,
|
|
|
|
operation_name,
|
|
|
|
show_graphiql
|
|
|
|
)
|
|
|
|
|
2016-10-31 14:16:58 +03:00
|
|
|
status_code = 200
|
2016-10-31 13:56:51 +03:00
|
|
|
if execution_result:
|
|
|
|
response = {}
|
|
|
|
|
|
|
|
if execution_result.errors:
|
2017-10-25 20:54:13 +03:00
|
|
|
response['errors'] = [self.format_error(
|
|
|
|
e) for e in execution_result.errors]
|
2016-10-31 13:56:51 +03:00
|
|
|
|
|
|
|
if execution_result.invalid:
|
|
|
|
status_code = 400
|
|
|
|
else:
|
|
|
|
response['data'] = execution_result.data
|
|
|
|
|
|
|
|
if self.batch:
|
2017-03-02 09:47:28 +03:00
|
|
|
response['id'] = id
|
|
|
|
response['status'] = status_code
|
2016-10-31 13:56:51 +03:00
|
|
|
|
|
|
|
result = self.json_encode(request, response, pretty=show_graphiql)
|
|
|
|
else:
|
|
|
|
result = None
|
|
|
|
|
|
|
|
return result, status_code
|
|
|
|
|
2016-09-20 08:04:23 +03:00
|
|
|
def render_graphiql(self, request, **data):
|
|
|
|
return render(request, self.graphiql_template, data)
|
|
|
|
|
|
|
|
def json_encode(self, request, d, pretty=False):
|
|
|
|
if not (self.pretty or pretty) and not request.GET.get('pretty'):
|
|
|
|
return json.dumps(d, separators=(',', ':'))
|
|
|
|
|
|
|
|
return json.dumps(d, sort_keys=True,
|
|
|
|
indent=2, separators=(',', ': '))
|
|
|
|
|
|
|
|
def parse_body(self, request):
|
|
|
|
content_type = self.get_content_type(request)
|
|
|
|
|
|
|
|
if content_type == 'application/graphql':
|
|
|
|
return {'query': request.body.decode()}
|
|
|
|
|
|
|
|
elif content_type == 'application/json':
|
2017-04-12 13:25:51 +03:00
|
|
|
# noinspection PyBroadException
|
2016-09-20 08:04:23 +03:00
|
|
|
try:
|
2017-04-12 13:25:51 +03:00
|
|
|
body = request.body.decode('utf-8')
|
|
|
|
except Exception as e:
|
|
|
|
raise HttpError(HttpResponseBadRequest(str(e)))
|
|
|
|
|
|
|
|
try:
|
|
|
|
request_json = json.loads(body)
|
2016-10-31 13:56:51 +03:00
|
|
|
if self.batch:
|
2017-02-20 12:08:42 +03:00
|
|
|
assert isinstance(request_json, list), (
|
|
|
|
'Batch requests should receive a list, but received {}.'
|
|
|
|
).format(repr(request_json))
|
|
|
|
assert len(request_json) > 0, (
|
|
|
|
'Received an empty list in the batch request.'
|
|
|
|
)
|
2016-10-31 13:56:51 +03:00
|
|
|
else:
|
2017-02-20 12:08:42 +03:00
|
|
|
assert isinstance(request_json, dict), (
|
|
|
|
'The received data is not a valid JSON query.'
|
|
|
|
)
|
2016-09-20 08:04:23 +03:00
|
|
|
return request_json
|
2017-02-20 12:08:42 +03:00
|
|
|
except AssertionError as e:
|
|
|
|
raise HttpError(HttpResponseBadRequest(str(e)))
|
2017-04-12 13:25:51 +03:00
|
|
|
except (TypeError, ValueError):
|
2017-10-25 20:54:13 +03:00
|
|
|
raise HttpError(HttpResponseBadRequest(
|
|
|
|
'POST body sent invalid JSON.'))
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
elif content_type in ['application/x-www-form-urlencoded', 'multipart/form-data']:
|
|
|
|
return request.POST
|
|
|
|
|
|
|
|
return {}
|
|
|
|
|
|
|
|
def execute(self, *args, **kwargs):
|
|
|
|
return execute(self.schema, *args, **kwargs)
|
|
|
|
|
|
|
|
def execute_graphql_request(self, request, data, query, variables, operation_name, show_graphiql=False):
|
|
|
|
if not query:
|
|
|
|
if show_graphiql:
|
|
|
|
return None
|
2017-10-25 20:54:13 +03:00
|
|
|
raise HttpError(HttpResponseBadRequest(
|
|
|
|
'Must provide query string.'))
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
source = Source(query, name='GraphQL request')
|
|
|
|
|
|
|
|
try:
|
|
|
|
document_ast = parse(source)
|
|
|
|
validation_errors = validate(self.schema, document_ast)
|
|
|
|
if validation_errors:
|
|
|
|
return ExecutionResult(
|
|
|
|
errors=validation_errors,
|
|
|
|
invalid=True,
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
return ExecutionResult(errors=[e], invalid=True)
|
|
|
|
|
|
|
|
if request.method.lower() == 'get':
|
|
|
|
operation_ast = get_operation_ast(document_ast, operation_name)
|
|
|
|
if operation_ast and operation_ast.operation != 'query':
|
|
|
|
if show_graphiql:
|
|
|
|
return None
|
2016-09-20 08:15:10 +03:00
|
|
|
|
2016-09-20 08:04:23 +03:00
|
|
|
raise HttpError(HttpResponseNotAllowed(
|
2017-10-25 20:54:13 +03:00
|
|
|
['POST'], 'Can only perform a {} operation from a POST request.'.format(
|
|
|
|
operation_ast.operation)
|
2016-09-20 08:04:23 +03:00
|
|
|
))
|
|
|
|
|
|
|
|
try:
|
|
|
|
return self.execute(
|
|
|
|
document_ast,
|
|
|
|
root_value=self.get_root_value(request),
|
|
|
|
variable_values=variables,
|
|
|
|
operation_name=operation_name,
|
|
|
|
context_value=self.get_context(request),
|
2016-09-20 10:08:14 +03:00
|
|
|
middleware=self.get_middleware(request),
|
2016-09-20 08:04:23 +03:00
|
|
|
executor=self.executor,
|
|
|
|
)
|
|
|
|
except Exception as e:
|
|
|
|
return ExecutionResult(errors=[e], invalid=True)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def can_display_graphiql(cls, request, data):
|
|
|
|
raw = 'raw' in request.GET or 'raw' in data
|
|
|
|
return not raw and cls.request_wants_html(request)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def request_wants_html(cls, request):
|
|
|
|
accepted = get_accepted_content_types(request)
|
2017-12-12 05:08:42 +03:00
|
|
|
accepted_length = len(accepted)
|
2017-12-18 19:40:19 +03:00
|
|
|
# the list will be ordered in preferred first - so we have to make
|
|
|
|
# sure the most preferred gets the highest number
|
2017-12-17 03:32:01 +03:00
|
|
|
html_priority = accepted_length - accepted.index('text/html') if 'text/html' in accepted else 0
|
|
|
|
json_priority = accepted_length - accepted.index('application/json') if 'application/json' in accepted else 0
|
2016-09-20 08:04:23 +03:00
|
|
|
|
2017-12-17 03:32:01 +03:00
|
|
|
return html_priority > json_priority
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_graphql_params(request, data):
|
|
|
|
query = request.GET.get('query') or data.get('query')
|
|
|
|
variables = request.GET.get('variables') or data.get('variables')
|
2016-10-31 13:56:51 +03:00
|
|
|
id = request.GET.get('id') or data.get('id')
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
if variables and isinstance(variables, six.text_type):
|
|
|
|
try:
|
|
|
|
variables = json.loads(variables)
|
2017-10-25 20:54:13 +03:00
|
|
|
except Exception:
|
|
|
|
raise HttpError(HttpResponseBadRequest(
|
|
|
|
'Variables are invalid JSON.'))
|
2016-09-20 08:04:23 +03:00
|
|
|
|
2017-10-25 20:54:13 +03:00
|
|
|
operation_name = request.GET.get(
|
|
|
|
'operationName') or data.get('operationName')
|
2017-04-13 20:11:10 +03:00
|
|
|
if operation_name == "null":
|
|
|
|
operation_name = None
|
2016-09-20 08:04:23 +03:00
|
|
|
|
2016-10-31 13:56:51 +03:00
|
|
|
return query, variables, operation_name, id
|
2016-09-20 08:04:23 +03:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def format_error(error):
|
|
|
|
if isinstance(error, GraphQLError):
|
|
|
|
return format_graphql_error(error)
|
|
|
|
|
|
|
|
return {'message': six.text_type(error)}
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_content_type(request):
|
|
|
|
meta = request.META
|
2017-10-25 20:54:13 +03:00
|
|
|
content_type = meta.get(
|
|
|
|
'CONTENT_TYPE', meta.get('HTTP_CONTENT_TYPE', ''))
|
2016-09-20 08:04:23 +03:00
|
|
|
return content_type.split(';', 1)[0].lower()
|