UNPKG

@graphql-tools/utils

Version:

Common package containing utils and types for GraphQL tools

171 lines (170 loc) • 8.43 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.pruneSchema = pruneSchema; const graphql_1 = require("graphql"); const get_implementing_types_js_1 = require("./get-implementing-types.js"); const Interfaces_js_1 = require("./Interfaces.js"); const mapSchema_js_1 = require("./mapSchema.js"); const rootTypes_js_1 = require("./rootTypes.js"); /** * Prunes the provided schema, removing unused and empty types * @param schema The schema to prune * @param options Additional options for removing unused types from the schema */ function pruneSchema(schema, options = {}) { const { skipEmptyCompositeTypePruning, skipEmptyUnionPruning, skipPruning, skipUnimplementedInterfacesPruning, skipUnusedTypesPruning, } = options; let prunedTypes = []; // Pruned types during mapping let prunedSchema = schema; do { let visited = visitSchema(prunedSchema); // Custom pruning was defined, so we need to pre-emptively revisit the schema accounting for this if (skipPruning) { const revisit = []; for (const typeName in prunedSchema.getTypeMap()) { if (typeName.startsWith('__')) { continue; } const type = prunedSchema.getType(typeName); // if we want to skip pruning for this type, add it to the list of types to revisit if (type && skipPruning(type)) { revisit.push(typeName); } } visited = visitQueue(revisit, prunedSchema, visited); // visit again } prunedTypes = []; prunedSchema = (0, mapSchema_js_1.mapSchema)(prunedSchema, { [Interfaces_js_1.MapperKind.TYPE]: type => { if (!visited.has(type.name) && !(0, graphql_1.isSpecifiedScalarType)(type)) { if ((0, graphql_1.isUnionType)(type) || (0, graphql_1.isInputObjectType)(type) || (0, graphql_1.isInterfaceType)(type) || (0, graphql_1.isObjectType)(type) || (0, graphql_1.isScalarType)(type)) { // skipUnusedTypesPruning: skip pruning unused types if (skipUnusedTypesPruning) { return type; } // skipEmptyUnionPruning: skip pruning empty unions if ((0, graphql_1.isUnionType)(type) && skipEmptyUnionPruning && !Object.keys(type.getTypes()).length) { return type; } if ((0, graphql_1.isInputObjectType)(type) || (0, graphql_1.isInterfaceType)(type) || (0, graphql_1.isObjectType)(type)) { // skipEmptyCompositeTypePruning: skip pruning object types or interfaces with no fields if (skipEmptyCompositeTypePruning && !Object.keys(type.getFields()).length) { return type; } } // skipUnimplementedInterfacesPruning: skip pruning interfaces that are not implemented by any other types if ((0, graphql_1.isInterfaceType)(type) && skipUnimplementedInterfacesPruning) { return type; } } prunedTypes.push(type.name); visited.delete(type.name); return null; } return type; }, }); } while (prunedTypes.length); // Might have empty types and need to prune again return prunedSchema; } function visitSchema(schema) { const queue = []; // queue of nodes to visit // Grab the root types and start there for (const type of (0, rootTypes_js_1.getRootTypes)(schema)) { queue.push(type.name); } return visitQueue(queue, schema); } function visitQueue(queue, schema, visited = new Set()) { // Interfaces encountered that are field return types need to be revisited to add their implementations const revisit = new Map(); // Navigate all types starting with pre-queued types (root types) while (queue.length) { const typeName = queue.pop(); // Skip types we already visited unless it is an interface type that needs revisiting if (visited.has(typeName) && revisit[typeName] !== true) { continue; } const type = schema.getType(typeName); if (type) { // Get types for union if ((0, graphql_1.isUnionType)(type)) { queue.push(...type.getTypes().map(type => type.name)); } // If it is an interface and it is a returned type, grab all implementations so we can use proper __typename in fragments if ((0, graphql_1.isInterfaceType)(type) && revisit[typeName] === true) { queue.push(...(0, get_implementing_types_js_1.getImplementingTypes)(type.name, schema)); // No need to revisit this interface again revisit[typeName] = false; } if ((0, graphql_1.isEnumType)(type)) { // Visit enum values directives argument types queue.push(...type.getValues().flatMap(value => getDirectivesArgumentsTypeNames(schema, value))); } // Visit interfaces this type is implementing if they haven't been visited yet if ('getInterfaces' in type) { // Only pushes to queue to visit but not return types queue.push(...type.getInterfaces().map(iface => iface.name)); } // If the type has fields visit those field types if ('getFields' in type) { const fields = type.getFields(); const entries = Object.entries(fields); if (!entries.length) { continue; } for (const [, field] of entries) { if ((0, graphql_1.isObjectType)(type)) { // Visit arg types and arg directives arguments types queue.push(...field.args.flatMap(arg => { const typeNames = [(0, graphql_1.getNamedType)(arg.type).name]; typeNames.push(...getDirectivesArgumentsTypeNames(schema, arg)); return typeNames; })); } const namedType = (0, graphql_1.getNamedType)(field.type); queue.push(namedType.name); queue.push(...getDirectivesArgumentsTypeNames(schema, field)); // Interfaces returned on fields need to be revisited to add their implementations if ((0, graphql_1.isInterfaceType)(namedType) && !(namedType.name in revisit)) { revisit[namedType.name] = true; } } } queue.push(...getDirectivesArgumentsTypeNames(schema, type)); visited.add(typeName); // Mark as visited (and therefore it is used and should be kept) } } return visited; } function getDirectivesArgumentsTypeNames(schema, directableObj) { const argTypeNames = new Set(); if (directableObj.astNode?.directives) { for (const directiveNode of directableObj.astNode.directives) { const directive = schema.getDirective(directiveNode.name.value); if (directive?.args) { for (const arg of directive.args) { const argType = (0, graphql_1.getNamedType)(arg.type); argTypeNames.add(argType.name); } } } } if (directableObj.extensions?.['directives']) { for (const directiveName in directableObj.extensions['directives']) { const directive = schema.getDirective(directiveName); if (directive?.args) { for (const arg of directive.args) { const argType = (0, graphql_1.getNamedType)(arg.type); argTypeNames.add(argType.name); } } } } return [...argTypeNames]; }