graphene-django/graphene_django/management/commands/graphql_schema.py

116 lines
3.6 KiB
Python
Raw Normal View History

import os
import importlib
import json
import functools
from django.core.management.base import BaseCommand, CommandError
from django.utils import autoreload
from graphql import print_schema
from graphene_django.settings import graphene_settings
2017-12-18 20:02:04 +03:00
2017-12-10 08:53:13 +03:00
class CommandArguments(BaseCommand):
def add_arguments(self, parser):
parser.add_argument(
2018-07-20 02:51:33 +03:00
"--schema",
2017-12-10 08:53:13 +03:00
type=str,
2018-07-20 02:51:33 +03:00
dest="schema",
2017-12-10 08:53:13 +03:00
default=graphene_settings.SCHEMA,
2018-07-20 02:51:33 +03:00
help="Django app containing schema to dump, e.g. myproject.core.schema.schema",
)
2017-12-10 08:53:13 +03:00
parser.add_argument(
2018-07-20 02:51:33 +03:00
"--out",
2017-12-10 08:53:13 +03:00
type=str,
2018-07-20 02:51:33 +03:00
dest="out",
2017-12-10 08:53:13 +03:00
default=graphene_settings.SCHEMA_OUTPUT,
help="Output file, --out=- prints to stdout (default: schema.json)",
2018-07-20 02:51:33 +03:00
)
2017-12-10 08:53:13 +03:00
parser.add_argument(
2018-07-20 02:51:33 +03:00
"--indent",
2017-12-10 08:53:13 +03:00
type=int,
2018-07-20 02:51:33 +03:00
dest="indent",
2017-12-10 08:53:13 +03:00
default=graphene_settings.SCHEMA_INDENT,
2018-07-20 02:51:33 +03:00
help="Output file indent (default: None)",
)
parser.add_argument(
"--watch",
dest="watch",
default=False,
action="store_true",
help="Updates the schema on file changes (default: False)",
)
class Command(CommandArguments):
help = "Dump Graphene schema as a JSON or GraphQL file"
can_import_settings = True
requires_system_checks = []
def save_json_file(self, out, schema_dict, indent):
2018-07-20 02:51:33 +03:00
with open(out, "w") as outfile:
json.dump(schema_dict, outfile, indent=indent, sort_keys=True)
def save_graphql_file(self, out, schema):
with open(out, "w") as outfile:
outfile.write(print_schema(schema))
def get_schema(self, schema, out, indent):
schema_dict = {"data": schema.introspect()}
if out == "-" or out == "-.json":
self.stdout.write(json.dumps(schema_dict, indent=indent, sort_keys=True))
elif out == "-.graphql":
self.stdout.write(print_schema(schema))
else:
# Determine format
_, file_extension = os.path.splitext(out)
if file_extension == ".graphql":
self.save_graphql_file(out, schema)
elif file_extension == ".json":
self.save_json_file(out, schema_dict, indent)
else:
raise CommandError(
'Unrecognised file format "{}"'.format(file_extension)
)
style = getattr(self, "style", None)
success = getattr(style, "SUCCESS", lambda x: x)
self.stdout.write(
success("Successfully dumped GraphQL schema to {}".format(out))
)
def handle(self, *args, **options):
2018-07-20 02:51:33 +03:00
options_schema = options.get("schema")
if options_schema and type(options_schema) is str:
2018-07-20 02:51:33 +03:00
module_str, schema_name = options_schema.rsplit(".", 1)
mod = importlib.import_module(module_str)
schema = getattr(mod, schema_name)
elif options_schema:
schema = options_schema
else:
schema = graphene_settings.SCHEMA
2018-07-20 02:51:33 +03:00
out = options.get("out") or graphene_settings.SCHEMA_OUTPUT
if not schema:
2018-07-20 02:51:33 +03:00
raise CommandError(
"Specify schema on GRAPHENE.SCHEMA setting or by using --schema"
)
2018-07-20 02:51:33 +03:00
indent = options.get("indent")
watch = options.get("watch")
if watch:
autoreload.run_with_reloader(
functools.partial(self.get_schema, schema, out, indent)
)
else:
self.get_schema(schema, out, indent)