WIP - add extractor, generate snippet_data
This commit is contained in:
395
node_modules/graphql/utilities/extendSchema.mjs
generated
vendored
Normal file
395
node_modules/graphql/utilities/extendSchema.mjs
generated
vendored
Normal file
@ -0,0 +1,395 @@
|
||||
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
|
||||
|
||||
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
||||
|
||||
import flatMap from '../polyfills/flatMap';
|
||||
import objectValues from '../polyfills/objectValues';
|
||||
import inspect from '../jsutils/inspect';
|
||||
import invariant from '../jsutils/invariant';
|
||||
import mapValue from '../jsutils/mapValue';
|
||||
import keyValMap from '../jsutils/keyValMap';
|
||||
import { ASTDefinitionBuilder } from './buildASTSchema';
|
||||
import { assertValidSDLExtension } from '../validation/validate';
|
||||
import { assertSchema, GraphQLSchema } from '../type/schema';
|
||||
import { isIntrospectionType } from '../type/introspection';
|
||||
import { isSpecifiedScalarType } from '../type/scalars';
|
||||
import { isScalarType, isObjectType, isInterfaceType, isUnionType, isListType, isNonNullType, isEnumType, isInputObjectType, GraphQLList, GraphQLNonNull, GraphQLScalarType, GraphQLObjectType, GraphQLInterfaceType, GraphQLUnionType, GraphQLEnumType, GraphQLInputObjectType } from '../type/definition';
|
||||
import { GraphQLDirective } from '../type/directives';
|
||||
import { Kind } from '../language/kinds';
|
||||
import { isTypeDefinitionNode, isTypeExtensionNode } from '../language/predicates';
|
||||
|
||||
/**
|
||||
* Produces a new schema given an existing schema and a document which may
|
||||
* contain GraphQL type extensions and definitions. The original schema will
|
||||
* remain unaltered.
|
||||
*
|
||||
* Because a schema represents a graph of references, a schema cannot be
|
||||
* extended without effectively making an entire copy. We do not know until it's
|
||||
* too late if subgraphs remain unchanged.
|
||||
*
|
||||
* This algorithm copies the provided schema, applying extensions while
|
||||
* producing the copy. The original schema remains unaltered.
|
||||
*
|
||||
* Accepts options as a third argument:
|
||||
*
|
||||
* - commentDescriptions:
|
||||
* Provide true to use preceding comments as the description.
|
||||
*
|
||||
*/
|
||||
export function extendSchema(schema, documentAST, options) {
|
||||
assertSchema(schema);
|
||||
!(documentAST && documentAST.kind === Kind.DOCUMENT) ? invariant(0, 'Must provide valid Document AST') : void 0;
|
||||
|
||||
if (!options || !(options.assumeValid || options.assumeValidSDL)) {
|
||||
assertValidSDLExtension(documentAST, schema);
|
||||
} // Collect the type definitions and extensions found in the document.
|
||||
|
||||
|
||||
var typeDefs = [];
|
||||
var typeExtsMap = Object.create(null); // New directives and types are separate because a directives and types can
|
||||
// have the same name. For example, a type named "skip".
|
||||
|
||||
var directiveDefs = [];
|
||||
var schemaDef; // Schema extensions are collected which may add additional operation types.
|
||||
|
||||
var schemaExts = [];
|
||||
var _iteratorNormalCompletion = true;
|
||||
var _didIteratorError = false;
|
||||
var _iteratorError = undefined;
|
||||
|
||||
try {
|
||||
for (var _iterator = documentAST.definitions[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
||||
var def = _step.value;
|
||||
|
||||
if (def.kind === Kind.SCHEMA_DEFINITION) {
|
||||
schemaDef = def;
|
||||
} else if (def.kind === Kind.SCHEMA_EXTENSION) {
|
||||
schemaExts.push(def);
|
||||
} else if (isTypeDefinitionNode(def)) {
|
||||
typeDefs.push(def);
|
||||
} else if (isTypeExtensionNode(def)) {
|
||||
var extendedTypeName = def.name.value;
|
||||
var existingTypeExts = typeExtsMap[extendedTypeName];
|
||||
typeExtsMap[extendedTypeName] = existingTypeExts ? existingTypeExts.concat([def]) : [def];
|
||||
} else if (def.kind === Kind.DIRECTIVE_DEFINITION) {
|
||||
directiveDefs.push(def);
|
||||
}
|
||||
} // If this document contains no new types, extensions, or directives then
|
||||
// return the same unmodified GraphQLSchema instance.
|
||||
|
||||
} catch (err) {
|
||||
_didIteratorError = true;
|
||||
_iteratorError = err;
|
||||
} finally {
|
||||
try {
|
||||
if (!_iteratorNormalCompletion && _iterator.return != null) {
|
||||
_iterator.return();
|
||||
}
|
||||
} finally {
|
||||
if (_didIteratorError) {
|
||||
throw _iteratorError;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (Object.keys(typeExtsMap).length === 0 && typeDefs.length === 0 && directiveDefs.length === 0 && schemaExts.length === 0 && !schemaDef) {
|
||||
return schema;
|
||||
}
|
||||
|
||||
var schemaConfig = schema.toConfig();
|
||||
var astBuilder = new ASTDefinitionBuilder(options, function (typeName) {
|
||||
var type = typeMap[typeName];
|
||||
!type ? invariant(0, "Unknown type: \"".concat(typeName, "\".")) : void 0;
|
||||
return type;
|
||||
});
|
||||
var typeMap = keyValMap(typeDefs, function (node) {
|
||||
return node.name.value;
|
||||
}, function (node) {
|
||||
return astBuilder.buildType(node);
|
||||
});
|
||||
var _iteratorNormalCompletion2 = true;
|
||||
var _didIteratorError2 = false;
|
||||
var _iteratorError2 = undefined;
|
||||
|
||||
try {
|
||||
for (var _iterator2 = schemaConfig.types[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
|
||||
var existingType = _step2.value;
|
||||
typeMap[existingType.name] = extendNamedType(existingType);
|
||||
} // Get the extended root operation types.
|
||||
|
||||
} catch (err) {
|
||||
_didIteratorError2 = true;
|
||||
_iteratorError2 = err;
|
||||
} finally {
|
||||
try {
|
||||
if (!_iteratorNormalCompletion2 && _iterator2.return != null) {
|
||||
_iterator2.return();
|
||||
}
|
||||
} finally {
|
||||
if (_didIteratorError2) {
|
||||
throw _iteratorError2;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
var operationTypes = {
|
||||
query: schemaConfig.query && schemaConfig.query.name,
|
||||
mutation: schemaConfig.mutation && schemaConfig.mutation.name,
|
||||
subscription: schemaConfig.subscription && schemaConfig.subscription.name
|
||||
};
|
||||
|
||||
if (schemaDef) {
|
||||
var _iteratorNormalCompletion3 = true;
|
||||
var _didIteratorError3 = false;
|
||||
var _iteratorError3 = undefined;
|
||||
|
||||
try {
|
||||
for (var _iterator3 = schemaDef.operationTypes[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
|
||||
var _ref2 = _step3.value;
|
||||
var operation = _ref2.operation;
|
||||
var type = _ref2.type;
|
||||
operationTypes[operation] = type.name.value;
|
||||
}
|
||||
} catch (err) {
|
||||
_didIteratorError3 = true;
|
||||
_iteratorError3 = err;
|
||||
} finally {
|
||||
try {
|
||||
if (!_iteratorNormalCompletion3 && _iterator3.return != null) {
|
||||
_iterator3.return();
|
||||
}
|
||||
} finally {
|
||||
if (_didIteratorError3) {
|
||||
throw _iteratorError3;
|
||||
}
|
||||
}
|
||||
}
|
||||
} // Then, incorporate schema definition and all schema extensions.
|
||||
|
||||
|
||||
for (var _i = 0, _schemaExts = schemaExts; _i < _schemaExts.length; _i++) {
|
||||
var schemaExt = _schemaExts[_i];
|
||||
|
||||
if (schemaExt.operationTypes) {
|
||||
var _iteratorNormalCompletion4 = true;
|
||||
var _didIteratorError4 = false;
|
||||
var _iteratorError4 = undefined;
|
||||
|
||||
try {
|
||||
for (var _iterator4 = schemaExt.operationTypes[Symbol.iterator](), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) {
|
||||
var _ref4 = _step4.value;
|
||||
var _operation = _ref4.operation;
|
||||
var _type = _ref4.type;
|
||||
operationTypes[_operation] = _type.name.value;
|
||||
}
|
||||
} catch (err) {
|
||||
_didIteratorError4 = true;
|
||||
_iteratorError4 = err;
|
||||
} finally {
|
||||
try {
|
||||
if (!_iteratorNormalCompletion4 && _iterator4.return != null) {
|
||||
_iterator4.return();
|
||||
}
|
||||
} finally {
|
||||
if (_didIteratorError4) {
|
||||
throw _iteratorError4;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} // Support both original legacy names and extended legacy names.
|
||||
|
||||
|
||||
var allowedLegacyNames = schemaConfig.allowedLegacyNames.concat(options && options.allowedLegacyNames || []); // Then produce and return a Schema with these types.
|
||||
|
||||
return new GraphQLSchema({
|
||||
// Note: While this could make early assertions to get the correctly
|
||||
// typed values, that would throw immediately while type system
|
||||
// validation with validateSchema() will produce more actionable results.
|
||||
query: getMaybeTypeByName(operationTypes.query),
|
||||
mutation: getMaybeTypeByName(operationTypes.mutation),
|
||||
subscription: getMaybeTypeByName(operationTypes.subscription),
|
||||
types: objectValues(typeMap),
|
||||
directives: getMergedDirectives(),
|
||||
astNode: schemaDef || schemaConfig.astNode,
|
||||
extensionASTNodes: schemaConfig.extensionASTNodes.concat(schemaExts),
|
||||
allowedLegacyNames: allowedLegacyNames
|
||||
}); // Below are functions used for producing this schema that have closed over
|
||||
// this scope and have access to the schema, cache, and newly defined types.
|
||||
|
||||
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(type) {
|
||||
return typeMap[type.name];
|
||||
}
|
||||
|
||||
function getMaybeTypeByName(typeName) {
|
||||
return typeName ? typeMap[typeName] : null;
|
||||
}
|
||||
|
||||
function getMergedDirectives() {
|
||||
var existingDirectives = schema.getDirectives().map(extendDirective);
|
||||
!existingDirectives ? invariant(0, 'schema must have default directives') : void 0;
|
||||
return existingDirectives.concat(directiveDefs.map(function (node) {
|
||||
return astBuilder.buildDirective(node);
|
||||
}));
|
||||
}
|
||||
|
||||
function extendNamedType(type) {
|
||||
if (isIntrospectionType(type) || isSpecifiedScalarType(type)) {
|
||||
// Builtin types are not extended.
|
||||
return type;
|
||||
} else if (isScalarType(type)) {
|
||||
return extendScalarType(type);
|
||||
} else if (isObjectType(type)) {
|
||||
return extendObjectType(type);
|
||||
} else if (isInterfaceType(type)) {
|
||||
return extendInterfaceType(type);
|
||||
} else if (isUnionType(type)) {
|
||||
return extendUnionType(type);
|
||||
} else if (isEnumType(type)) {
|
||||
return extendEnumType(type);
|
||||
} else if (isInputObjectType(type)) {
|
||||
return extendInputObjectType(type);
|
||||
} // Not reachable. All possible types have been considered.
|
||||
|
||||
/* istanbul ignore next */
|
||||
|
||||
|
||||
throw new Error("Unexpected type: \"".concat(inspect(type), "\"."));
|
||||
}
|
||||
|
||||
function extendDirective(directive) {
|
||||
var config = directive.toConfig();
|
||||
return new GraphQLDirective(_objectSpread({}, config, {
|
||||
args: mapValue(config.args, extendArg)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendInputObjectType(type) {
|
||||
var config = type.toConfig();
|
||||
var extensions = typeExtsMap[config.name] || [];
|
||||
var fieldNodes = flatMap(extensions, function (node) {
|
||||
return node.fields || [];
|
||||
});
|
||||
return new GraphQLInputObjectType(_objectSpread({}, config, {
|
||||
fields: function fields() {
|
||||
return _objectSpread({}, mapValue(config.fields, function (field) {
|
||||
return _objectSpread({}, field, {
|
||||
type: replaceType(field.type)
|
||||
});
|
||||
}), keyValMap(fieldNodes, function (field) {
|
||||
return field.name.value;
|
||||
}, function (field) {
|
||||
return astBuilder.buildInputField(field);
|
||||
}));
|
||||
},
|
||||
extensionASTNodes: config.extensionASTNodes.concat(extensions)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendEnumType(type) {
|
||||
var config = type.toConfig();
|
||||
var extensions = typeExtsMap[type.name] || [];
|
||||
var valueNodes = flatMap(extensions, function (node) {
|
||||
return node.values || [];
|
||||
});
|
||||
return new GraphQLEnumType(_objectSpread({}, config, {
|
||||
values: _objectSpread({}, config.values, keyValMap(valueNodes, function (value) {
|
||||
return value.name.value;
|
||||
}, function (value) {
|
||||
return astBuilder.buildEnumValue(value);
|
||||
})),
|
||||
extensionASTNodes: config.extensionASTNodes.concat(extensions)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendScalarType(type) {
|
||||
var config = type.toConfig();
|
||||
var extensions = typeExtsMap[config.name] || [];
|
||||
return new GraphQLScalarType(_objectSpread({}, config, {
|
||||
extensionASTNodes: config.extensionASTNodes.concat(extensions)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendObjectType(type) {
|
||||
var config = type.toConfig();
|
||||
var extensions = typeExtsMap[config.name] || [];
|
||||
var interfaceNodes = flatMap(extensions, function (node) {
|
||||
return node.interfaces || [];
|
||||
});
|
||||
var fieldNodes = flatMap(extensions, function (node) {
|
||||
return node.fields || [];
|
||||
});
|
||||
return new GraphQLObjectType(_objectSpread({}, config, {
|
||||
interfaces: function interfaces() {
|
||||
return [].concat(type.getInterfaces().map(replaceNamedType), interfaceNodes.map(function (node) {
|
||||
return astBuilder.getNamedType(node);
|
||||
}));
|
||||
},
|
||||
fields: function fields() {
|
||||
return _objectSpread({}, mapValue(config.fields, extendField), keyValMap(fieldNodes, function (node) {
|
||||
return node.name.value;
|
||||
}, function (node) {
|
||||
return astBuilder.buildField(node);
|
||||
}));
|
||||
},
|
||||
extensionASTNodes: config.extensionASTNodes.concat(extensions)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendInterfaceType(type) {
|
||||
var config = type.toConfig();
|
||||
var extensions = typeExtsMap[config.name] || [];
|
||||
var fieldNodes = flatMap(extensions, function (node) {
|
||||
return node.fields || [];
|
||||
});
|
||||
return new GraphQLInterfaceType(_objectSpread({}, config, {
|
||||
fields: function fields() {
|
||||
return _objectSpread({}, mapValue(config.fields, extendField), keyValMap(fieldNodes, function (node) {
|
||||
return node.name.value;
|
||||
}, function (node) {
|
||||
return astBuilder.buildField(node);
|
||||
}));
|
||||
},
|
||||
extensionASTNodes: config.extensionASTNodes.concat(extensions)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendUnionType(type) {
|
||||
var config = type.toConfig();
|
||||
var extensions = typeExtsMap[config.name] || [];
|
||||
var typeNodes = flatMap(extensions, function (node) {
|
||||
return node.types || [];
|
||||
});
|
||||
return new GraphQLUnionType(_objectSpread({}, config, {
|
||||
types: function types() {
|
||||
return [].concat(type.getTypes().map(replaceNamedType), typeNodes.map(function (node) {
|
||||
return astBuilder.getNamedType(node);
|
||||
}));
|
||||
},
|
||||
extensionASTNodes: config.extensionASTNodes.concat(extensions)
|
||||
}));
|
||||
}
|
||||
|
||||
function extendField(field) {
|
||||
return _objectSpread({}, field, {
|
||||
type: replaceType(field.type),
|
||||
args: mapValue(field.args, extendArg)
|
||||
});
|
||||
}
|
||||
|
||||
function extendArg(arg) {
|
||||
return _objectSpread({}, arg, {
|
||||
type: replaceType(arg.type)
|
||||
});
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user