mirror of
https://github.com/JamesIves/github-pages-deploy-action.git
synced 2023-12-15 20:03:39 +08:00
176 lines
4.6 KiB
Plaintext
176 lines
4.6 KiB
Plaintext
|
// @flow strict
|
||
|
|
||
|
import objectValues from '../polyfills/objectValues';
|
||
|
|
||
|
import inspect from '../jsutils/inspect';
|
||
|
import invariant from '../jsutils/invariant';
|
||
|
import keyValMap from '../jsutils/keyValMap';
|
||
|
import { type ObjMap } from '../jsutils/ObjMap';
|
||
|
|
||
|
import { GraphQLSchema } from '../type/schema';
|
||
|
import { GraphQLDirective } from '../type/directives';
|
||
|
import { isIntrospectionType } from '../type/introspection';
|
||
|
import {
|
||
|
type GraphQLNamedType,
|
||
|
GraphQLObjectType,
|
||
|
GraphQLInterfaceType,
|
||
|
GraphQLUnionType,
|
||
|
GraphQLEnumType,
|
||
|
GraphQLInputObjectType,
|
||
|
GraphQLList,
|
||
|
GraphQLNonNull,
|
||
|
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(type) {
|
||
|
if (isListType(type)) {
|
||
|
return new GraphQLList(replaceType(type.ofType));
|
||
|
} else if (isNonNullType(type)) {
|
||
|
return new GraphQLNonNull(replaceType(type.ofType));
|
||
|
}
|
||
|
return replaceNamedType(type);
|
||
|
}
|
||
|
|
||
|
function replaceNamedType<T: GraphQLNamedType>(type: T): T {
|
||
|
return ((typeMap[type.name]: any): T);
|
||
|
}
|
||
|
|
||
|
function replaceMaybeType(maybeType) {
|
||
|
return maybeType && replaceNamedType(maybeType);
|
||
|
}
|
||
|
|
||
|
function sortDirective(directive) {
|
||
|
const config = directive.toConfig();
|
||
|
return new GraphQLDirective({
|
||
|
...config,
|
||
|
locations: sortBy(config.locations, x => x),
|
||
|
args: sortArgs(config.args),
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function sortArgs(args) {
|
||
|
return sortObjMap(args, arg => ({
|
||
|
...arg,
|
||
|
type: replaceType(arg.type),
|
||
|
}));
|
||
|
}
|
||
|
|
||
|
function sortFields(fieldsMap) {
|
||
|
return sortObjMap(fieldsMap, field => ({
|
||
|
...field,
|
||
|
type: replaceType(field.type),
|
||
|
args: sortArgs(field.args),
|
||
|
}));
|
||
|
}
|
||
|
|
||
|
function sortInputFields(fieldsMap) {
|
||
|
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) {
|
||
|
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),
|
||
|
});
|
||
|
}
|
||
|
if (isInputObjectType(type)) {
|
||
|
const config = type.toConfig();
|
||
|
return new GraphQLInputObjectType({
|
||
|
...config,
|
||
|
fields: () => sortInputFields(config.fields),
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// 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 key1.localeCompare(key2);
|
||
|
});
|
||
|
}
|