123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- // @flow strict
- import objectValues from '../polyfills/objectValues';
- import type { ObjMap } from '../jsutils/ObjMap';
- import inspect from '../jsutils/inspect';
- import invariant from '../jsutils/invariant';
- import keyValMap from '../jsutils/keyValMap';
- import naturalCompare from '../jsutils/naturalCompare';
- import type {
- GraphQLType,
- GraphQLNamedType,
- GraphQLFieldConfigMap,
- GraphQLFieldConfigArgumentMap,
- GraphQLInputFieldConfigMap,
- } from '../type/definition';
- import { GraphQLSchema } from '../type/schema';
- import { GraphQLDirective } from '../type/directives';
- import { isIntrospectionType } from '../type/introspection';
- import {
- GraphQLList,
- GraphQLNonNull,
- GraphQLObjectType,
- GraphQLInterfaceType,
- GraphQLUnionType,
- GraphQLEnumType,
- GraphQLInputObjectType,
- isListType,
- isNonNullType,
- isScalarType,
- isObjectType,
- isInterfaceType,
- isUnionType,
- isEnumType,
- isInputObjectType,
- } from '../type/definition';
- /**
- * Sort GraphQLSchema.
- *
- * This function returns a sorted copy of the given GraphQLSchema.
- */
- export function lexicographicSortSchema(schema: GraphQLSchema): GraphQLSchema {
- const schemaConfig = schema.toConfig();
- const typeMap = keyValMap(
- sortByName(schemaConfig.types),
- (type) => type.name,
- sortNamedType,
- );
- return new GraphQLSchema({
- ...schemaConfig,
- types: objectValues(typeMap),
- directives: sortByName(schemaConfig.directives).map(sortDirective),
- query: replaceMaybeType(schemaConfig.query),
- mutation: replaceMaybeType(schemaConfig.mutation),
- subscription: replaceMaybeType(schemaConfig.subscription),
- });
- function replaceType<T: GraphQLType>(type: T): T {
- if (isListType(type)) {
- // $FlowFixMe[incompatible-return]
- return new GraphQLList(replaceType(type.ofType));
- } else if (isNonNullType(type)) {
- // $FlowFixMe[incompatible-return]
- return new GraphQLNonNull(replaceType(type.ofType));
- }
- return replaceNamedType(type);
- }
- function replaceNamedType<T: GraphQLNamedType>(type: T): T {
- return ((typeMap[type.name]: any): T);
- }
- function replaceMaybeType<T: ?GraphQLNamedType>(maybeType: T): T {
- return maybeType && replaceNamedType(maybeType);
- }
- function sortDirective(directive: GraphQLDirective) {
- const config = directive.toConfig();
- return new GraphQLDirective({
- ...config,
- locations: sortBy(config.locations, (x) => x),
- args: sortArgs(config.args),
- });
- }
- function sortArgs(args: GraphQLFieldConfigArgumentMap) {
- return sortObjMap(args, (arg) => ({
- ...arg,
- type: replaceType(arg.type),
- }));
- }
- function sortFields(fieldsMap: GraphQLFieldConfigMap<mixed, mixed>) {
- return sortObjMap(fieldsMap, (field) => ({
- ...field,
- type: replaceType(field.type),
- args: sortArgs(field.args),
- }));
- }
- function sortInputFields(fieldsMap: GraphQLInputFieldConfigMap) {
- return sortObjMap(fieldsMap, (field) => ({
- ...field,
- type: replaceType(field.type),
- }));
- }
- function sortTypes<T: GraphQLNamedType>(arr: $ReadOnlyArray<T>): Array<T> {
- return sortByName(arr).map(replaceNamedType);
- }
- function sortNamedType(type: GraphQLNamedType): GraphQLNamedType {
- if (isScalarType(type) || isIntrospectionType(type)) {
- return type;
- }
- if (isObjectType(type)) {
- const config = type.toConfig();
- return new GraphQLObjectType({
- ...config,
- interfaces: () => sortTypes(config.interfaces),
- fields: () => sortFields(config.fields),
- });
- }
- if (isInterfaceType(type)) {
- const config = type.toConfig();
- return new GraphQLInterfaceType({
- ...config,
- interfaces: () => sortTypes(config.interfaces),
- fields: () => sortFields(config.fields),
- });
- }
- if (isUnionType(type)) {
- const config = type.toConfig();
- return new GraphQLUnionType({
- ...config,
- types: () => sortTypes(config.types),
- });
- }
- if (isEnumType(type)) {
- const config = type.toConfig();
- return new GraphQLEnumType({
- ...config,
- values: sortObjMap(config.values),
- });
- }
- // istanbul ignore else (See: 'https://github.com/graphql/graphql-js/issues/2618')
- if (isInputObjectType(type)) {
- const config = type.toConfig();
- return new GraphQLInputObjectType({
- ...config,
- fields: () => sortInputFields(config.fields),
- });
- }
- // istanbul ignore next (Not reachable. All possible types have been considered)
- invariant(false, 'Unexpected type: ' + inspect((type: empty)));
- }
- }
- function sortObjMap<T, R>(map: ObjMap<T>, sortValueFn?: (T) => R): ObjMap<R> {
- const sortedMap = Object.create(null);
- const sortedKeys = sortBy(Object.keys(map), (x) => x);
- for (const key of sortedKeys) {
- const value = map[key];
- sortedMap[key] = sortValueFn ? sortValueFn(value) : value;
- }
- return sortedMap;
- }
- function sortByName<T: { +name: string, ... }>(
- array: $ReadOnlyArray<T>,
- ): Array<T> {
- return sortBy(array, (obj) => obj.name);
- }
- function sortBy<T>(
- array: $ReadOnlyArray<T>,
- mapToKey: (T) => string,
- ): Array<T> {
- return array.slice().sort((obj1, obj2) => {
- const key1 = mapToKey(obj1);
- const key2 = mapToKey(obj2);
- return naturalCompare(key1, key2);
- });
- }
|