UNPKG

@graphql-mesh/serve-runtime

Version:
774 lines (772 loc) 34.4 kB
import { buildSchema, isSchema, parse } from 'graphql'; import { createYoga, isAsyncIterable, mergeSchemas, useExecutionCancellation, useReadinessCheck, } from 'graphql-yoga'; import { useDisableIntrospection } from '@envelop/disable-introspection'; import { useGenericAuth } from '@envelop/generic-auth'; import { createSchemaFetcher, createSupergraphSDLFetcher } from '@graphql-hive/core'; import { process } from '@graphql-mesh/cross-helpers'; import { getOnSubgraphExecute, getStitchingDirectivesTransformerForSubschema, handleFederationSubschema, restoreExtraDirectives, UnifiedGraphManager, } from '@graphql-mesh/fusion-runtime'; import { useHmacUpstreamSignature } from '@graphql-mesh/hmac-upstream-signature'; import useMeshHive from '@graphql-mesh/plugin-hive'; import useMeshResponseCache from '@graphql-mesh/plugin-response-cache'; import { DefaultLogger, getHeadersObj, isDisposable, isUrl, LogLevel, makeAsyncDisposable, mapMaybePromise, wrapFetchWithHooks, } from '@graphql-mesh/utils'; import { batchDelegateToSchema } from '@graphql-tools/batch-delegate'; import { delegateToSchema } from '@graphql-tools/delegate'; import { fetchSupergraphSdlFromManagedFederation } from '@graphql-tools/federation'; import { getDirectiveExtensions, isDocumentNode, isValidPath, mergeDeep, parseSelectionSet, } from '@graphql-tools/utils'; import { schemaFromExecutor, wrapSchema } from '@graphql-tools/wrap'; import { useCSRFPrevention } from '@graphql-yoga/plugin-csrf-prevention'; import { useDeferStream } from '@graphql-yoga/plugin-defer-stream'; import { usePersistedOperations } from '@graphql-yoga/plugin-persisted-operations'; import { AsyncDisposableStack } from '@whatwg-node/disposablestack'; import { getProxyExecutor } from './getProxyExecutor.js'; import { getReportingPlugin } from './getReportingPlugin.js'; import { getUnifiedGraphSDL, handleUnifiedGraphConfig } from './handleUnifiedGraphConfig.js'; import landingPageHtml from './landing-page-html.js'; import { useChangingSchema } from './plugins/useChangingSchema.js'; import { useCompleteSubscriptionsOnDispose } from './plugins/useCompleteSubscriptionsOnDispose.js'; import { useCompleteSubscriptionsOnSchemaChange } from './plugins/useCompleteSubscriptionsOnSchemaChange.js'; import { useContentEncoding } from './plugins/useContentEncoding.js'; import { useCustomAgent } from './plugins/useCustomAgent.js'; import { useFetchDebug } from './plugins/useFetchDebug.js'; import { usePropagateHeaders } from './plugins/usePropagateHeaders.js'; import { useRequestId } from './plugins/useRequestId.js'; import { useSubgraphExecuteDebug } from './plugins/useSubgraphExecuteDebug.js'; import { useUpstreamCancel } from './plugins/useUpstreamCancel.js'; import { useWebhooks } from './plugins/useWebhooks.js'; import { defaultProductLogo } from './productLogo.js'; import { checkIfDataSatisfiesSelectionSet, defaultQueryText } from './utils.js'; export function createGatewayRuntime(config) { let fetchAPI = config.fetchAPI; let logger; if (config.logging == null) { logger = new DefaultLogger(); } else if (typeof config.logging === 'boolean') { logger = config.logging ? new DefaultLogger() : new DefaultLogger('', LogLevel.silent); } if (typeof config.logging === 'number') { logger = new DefaultLogger(undefined, config.logging); } else if (typeof config.logging === 'object') { logger = config.logging; } const onFetchHooks = []; const wrappedFetchFn = wrapFetchWithHooks(onFetchHooks); const configContext = { fetch: wrappedFetchFn, logger, cwd: 'cwd' in config ? config.cwd : process.cwd?.(), cache: 'cache' in config ? config.cache : undefined, pubsub: 'pubsub' in config ? config.pubsub : undefined, }; let unifiedGraphPlugin; const readinessCheckEndpoint = config.readinessCheckEndpoint || '/readiness'; const onSubgraphExecuteHooks = []; // TODO: Will be deleted after v0 const onDelegateHooks = []; let unifiedGraph; let schemaInvalidator; let getSchema = () => unifiedGraph; let setSchema = schema => { unifiedGraph = schema; }; let contextBuilder; let readinessChecker; const { name: reportingTarget, plugin: registryPlugin } = getReportingPlugin(config, configContext); let persistedDocumentsPlugin = {}; if (config.reporting?.type !== 'hive' && config.persistedDocuments && 'type' in config.persistedDocuments && config.persistedDocuments?.type === 'hive') { persistedDocumentsPlugin = useMeshHive({ ...configContext, logger: configContext.logger.child('Hive'), experimental__persistedDocuments: { cdn: { endpoint: config.persistedDocuments.endpoint, accessToken: config.persistedDocuments.token, }, allowArbitraryDocuments: config.persistedDocuments.allowArbitraryDocuments, }, }); } else if (config.persistedDocuments && 'getPersistedOperation' in config.persistedDocuments) { persistedDocumentsPlugin = usePersistedOperations({ ...configContext, ...config.persistedDocuments, }); } let subgraphInformationHTMLRenderer = () => ''; const disposableStack = new AsyncDisposableStack(); if ('proxy' in config) { const proxyExecutor = getProxyExecutor({ config, configContext, getSchema() { return unifiedGraph; }, onSubgraphExecuteHooks, disposableStack, }); function createExecuteFnFromExecutor(executor) { return function executeFn(args) { return executor({ document: args.document, variables: args.variableValues, operationName: args.operationName, rootValue: args.rootValue, context: args.contextValue, }); }; } const executeFn = createExecuteFnFromExecutor(proxyExecutor); let currentTimeout; const pollingInterval = config.pollingInterval; function continuePolling() { if (currentTimeout) { clearTimeout(currentTimeout); } if (pollingInterval) { currentTimeout = setTimeout(schemaFetcher, pollingInterval); } } function pausePolling() { if (currentTimeout) { clearTimeout(currentTimeout); } } let lastFetchedSdl; let initialFetch$; let schemaFetcher; if (config.schema && typeof config.schema === 'object' && 'type' in config.schema) { // hive cdn const { endpoint, key } = config.schema; const fetcher = createSchemaFetcher({ endpoint, key, logger: configContext.logger.child('Hive CDN'), }); schemaFetcher = function fetchSchemaFromCDN() { pausePolling(); initialFetch$ = mapMaybePromise(fetcher(), ({ sdl }) => { if (lastFetchedSdl == null || lastFetchedSdl !== sdl) { unifiedGraph = buildSchema(sdl, { assumeValid: true, assumeValidSDL: true, }); setSchema(unifiedGraph); } continuePolling(); return true; }); return initialFetch$; }; } else if (config.schema) { // local or remote if (!isDynamicUnifiedGraphSchema(config.schema)) { // no polling for static schemas delete config.pollingInterval; } schemaFetcher = function fetchSchema() { pausePolling(); initialFetch$ = mapMaybePromise(handleUnifiedGraphConfig( // @ts-expect-error TODO: what's up with type narrowing config.schema, configContext), schema => { setSchema(schema); continuePolling(); return true; }); return initialFetch$; }; } else { // introspect endpoint schemaFetcher = function fetchSchemaWithExecutor() { pausePolling(); return mapMaybePromise(schemaFromExecutor(proxyExecutor, configContext, { assumeValid: true, }), schema => { unifiedGraph = schema; setSchema(schema); continuePolling(); return true; }, err => { configContext.logger.warn(`Failed to introspect schema`, err); return true; }); }; } getSchema = () => { if (unifiedGraph != null) { return unifiedGraph; } if (initialFetch$ != null) { return mapMaybePromise(initialFetch$, () => unifiedGraph); } if (!initialFetch$) { return mapMaybePromise(schemaFetcher(), () => unifiedGraph); } }; disposableStack.defer(pausePolling); const shouldSkipValidation = 'skipValidation' in config ? config.skipValidation : false; const executorPlugin = { onExecute({ setExecuteFn }) { setExecuteFn(executeFn); }, onSubscribe({ setSubscribeFn }) { setSubscribeFn(executeFn); }, onValidate({ params, setResult }) { if (shouldSkipValidation || !params.schema) { setResult([]); } }, }; unifiedGraphPlugin = executorPlugin; readinessChecker = () => { const res$ = proxyExecutor({ document: parse(`query ReadinessCheck { __typename }`), }); return mapMaybePromise(res$, res => !isAsyncIterable(res) && !!res.data?.__typename); }; schemaInvalidator = () => { unifiedGraph = undefined; initialFetch$ = schemaFetcher(); }; subgraphInformationHTMLRenderer = () => { const endpoint = config.proxy.endpoint; const htmlParts = []; htmlParts.push(`<section class="supergraph-information">`); htmlParts.push(`<h3>Proxy: <a href="${endpoint}">${endpoint}</a></h3>`); if (config.schema) { if (typeof config.schema === 'object' && 'type' in config.schema) { htmlParts.push(`<p><strong>Source: </strong> <i>${config.schema.type === 'hive' ? 'Hive' : 'Unknown'} CDN</i></p>`); } else if (isValidPath(config.schema) || isUrl(String(config.schema))) { htmlParts.push(`<p><strong>Source: </strong> <i>${config.schema}</i></p>`); } else { htmlParts.push(`<p><strong>Source: </strong> <i>GraphQL schema in config</i></p>`); } } if (reportingTarget) { htmlParts.push(`<p><strong>Usage Reporting: </strong> <i>${reportingTarget}</i></p>`); } htmlParts.push(`</section>`); return htmlParts.join(''); }; } else if ('subgraph' in config) { const subgraphInConfig = config.subgraph; let getSubschemaConfig$; let subschemaConfig; function getSubschemaConfig() { if (getSubschemaConfig$) { return getSubschemaConfig$; } return mapMaybePromise(handleUnifiedGraphConfig(subgraphInConfig, configContext), newUnifiedGraph => { unifiedGraph = newUnifiedGraph; unifiedGraph = restoreExtraDirectives(unifiedGraph); subschemaConfig = { name: getDirectiveExtensions(unifiedGraph)?.transport?.[0]?.subgraph, schema: unifiedGraph, }; const transportEntryMap = {}; const additionalTypeDefs = []; const additionalResolvers = []; const stitchingDirectivesTransformer = getStitchingDirectivesTransformerForSubschema(); const onSubgraphExecute = getOnSubgraphExecute({ onSubgraphExecuteHooks, transports: config.transports, transportContext: configContext, transportEntryMap, getSubgraphSchema() { return unifiedGraph; }, transportExecutorStack: disposableStack, }); subschemaConfig = handleFederationSubschema({ subschemaConfig, transportEntryMap, additionalTypeDefs, stitchingDirectivesTransformer, onSubgraphExecute, }); // TODO: Find better alternative later unifiedGraph = wrapSchema(subschemaConfig); unifiedGraph = mergeSchemas({ assumeValid: true, assumeValidSDL: true, schemas: [unifiedGraph], typeDefs: [ parse(/* GraphQL */ ` type Query { _entities(representations: [_Any!]!): [_Entity]! _service: _Service! } scalar _Any union _Entity = ${Object.keys(subschemaConfig.merge || {}).join(' | ')} type _Service { sdl: String } `), ], resolvers: { Query: { _entities(_root, args, context, info) { if (Array.isArray(args.representations)) { return args.representations.map(representation => { const typeName = representation.__typename; const mergeConfig = subschemaConfig.merge[typeName]; const entryPoints = mergeConfig?.entryPoints || [mergeConfig]; const satisfiedEntryPoint = entryPoints.find(entryPoint => { if (entryPoint.selectionSet) { const selectionSet = parseSelectionSet(entryPoint.selectionSet, { noLocation: true, }); return checkIfDataSatisfiesSelectionSet(selectionSet, representation); } return true; }); if (satisfiedEntryPoint) { if (satisfiedEntryPoint.key) { return mapMaybePromise(batchDelegateToSchema({ schema: subschemaConfig, fieldName: satisfiedEntryPoint.fieldName, key: satisfiedEntryPoint.key(representation), argsFromKeys: satisfiedEntryPoint.argsFromKeys, valuesFromResults: satisfiedEntryPoint.valuesFromResults, context, info, }), res => mergeDeep([representation, res])); } if (satisfiedEntryPoint.args) { return mapMaybePromise(delegateToSchema({ schema: subschemaConfig, fieldName: satisfiedEntryPoint.fieldName, args: satisfiedEntryPoint.args(representation), context, info, }), res => mergeDeep([representation, res])); } } return representation; }); } return []; }, _service() { return { sdl() { return getUnifiedGraphSDL(newUnifiedGraph); }, }; }, }, }, }); return true; }); } getSchema = () => mapMaybePromise(getSubschemaConfig(), () => unifiedGraph); } /** 'supergraph' in config */ else { let unifiedGraphFetcher; let supergraphLoadedPlace; if (typeof config.supergraph === 'object' && 'type' in config.supergraph) { if (config.supergraph.type === 'hive') { // hive cdn const { endpoint, key } = config.supergraph; supergraphLoadedPlace = 'Hive CDN <br>' + endpoint; const fetcher = createSupergraphSDLFetcher({ endpoint, key, logger: configContext.logger.child('Hive CDN'), }); unifiedGraphFetcher = () => fetcher().then(({ supergraphSdl }) => supergraphSdl); } else if (config.supergraph.type === 'graphos') { const opts = config.supergraph; supergraphLoadedPlace = 'GraphOS Managed Federation <br>' + opts.graphRef || ''; let lastSeenId; let lastSupergraphSdl; let minDelayMS = config.pollingInterval || 0; unifiedGraphFetcher = () => mapMaybePromise(fetchSupergraphSdlFromManagedFederation({ graphRef: opts.graphRef, apiKey: opts.apiKey, upLink: opts.upLink, lastSeenId, // @ts-expect-error TODO: what's up with type narrowing fetch: configContext.fetch, loggerByMessageLevel: { ERROR(message) { configContext.logger.child('GraphOS').error(message); }, INFO(message) { configContext.logger.child('GraphOS').info(message); }, WARN(message) { configContext.logger.child('GraphOS').warn(message); }, }, }), async (result) => { if (minDelayMS) { await new Promise(resolve => setTimeout(resolve, minDelayMS)); } if (result.minDelaySeconds && result.minDelaySeconds > minDelayMS) { minDelayMS = result.minDelaySeconds; } if ('error' in result) { configContext.logger.child('GraphOS').error(result.error.message); return lastSupergraphSdl; } if ('id' in result) { lastSeenId = result.id; } if ('supergraphSdl' in result) { lastSupergraphSdl = result.supergraphSdl; return result.supergraphSdl; } if (lastSupergraphSdl) { throw new Error('Failed to fetch supergraph SDL'); } return lastSupergraphSdl; }); } else { configContext.logger.error(`Unknown supergraph configuration: `, config.supergraph); } } else { // local or remote if (!isDynamicUnifiedGraphSchema(config.supergraph)) { // no polling for static schemas delete config.pollingInterval; } unifiedGraphFetcher = () => handleUnifiedGraphConfig( // @ts-expect-error TODO: what's up with type narrowing config.supergraph, configContext); if (typeof config.supergraph === 'function') { const fnName = config.supergraph.name || ''; supergraphLoadedPlace = `a custom loader ${fnName}`; } else if (typeof config.supergraph === 'string') { supergraphLoadedPlace = config.supergraph; } } const unifiedGraphManager = new UnifiedGraphManager({ getUnifiedGraph: unifiedGraphFetcher, onSchemaChange(unifiedGraph) { setSchema(unifiedGraph); }, transports: config.transports, transportEntryAdditions: config.transportEntries, pollingInterval: config.pollingInterval, additionalResolvers: config.additionalResolvers, transportContext: configContext, onDelegateHooks, onSubgraphExecuteHooks, }); getSchema = () => unifiedGraphManager.getUnifiedGraph(); readinessChecker = () => mapMaybePromise(unifiedGraphManager.getUnifiedGraph(), schema => { if (!schema) { logger.debug(`Readiness check failed: Supergraph cannot be loaded`); return false; } logger.debug(`Readiness check passed: Supergraph loaded`); return true; }, err => { logger.debug(`Readiness check failed due to errors on loading supergraph:\n${err.stack || err.message}`); logger.error(err); return false; }); schemaInvalidator = () => unifiedGraphManager.invalidateUnifiedGraph(); contextBuilder = base => unifiedGraphManager.getContext(base); disposableStack.use(unifiedGraphManager); subgraphInformationHTMLRenderer = async () => { const htmlParts = []; let loaded = false; let loadError; let transportEntryMap; try { transportEntryMap = await unifiedGraphManager.getTransportEntryMap(); loaded = true; } catch (e) { loaded = false; loadError = e; } if (loaded) { htmlParts.push(`<h3>Supergraph Status: Loaded ✅</h3>`); if (supergraphLoadedPlace) { htmlParts.push(`<p><strong>Source: </strong> <i>${supergraphLoadedPlace}</i></p>`); if (reportingTarget) { htmlParts.push(`<p><strong>Usage Reporting: </strong> <i>${reportingTarget}</i></p>`); } } htmlParts.push(`<table>`); htmlParts.push(`<tr><th>Subgraph</th><th>Transport</th><th>Location</th></tr>`); for (const subgraphName in transportEntryMap) { const transportEntry = transportEntryMap[subgraphName]; htmlParts.push(`<tr>`); htmlParts.push(`<td>${subgraphName}</td>`); htmlParts.push(`<td>${transportEntry.kind}</td>`); htmlParts.push(`<td><a href="${transportEntry.location}">${transportEntry.location}</a></td>`); htmlParts.push(`</tr>`); } htmlParts.push(`</table>`); } else if (loadError) { htmlParts.push(`<h3>Status: Failed ❌</h3>`); if (supergraphLoadedPlace) { htmlParts.push(`<p><strong>Source: </strong> <i>${supergraphLoadedPlace}</i></p>`); } htmlParts.push(`<h3>Error:</h3>`); htmlParts.push(`<pre>${loadError.stack}</pre>`); } else { htmlParts.push(`<h3>Status: Unknown</h3>`); if (supergraphLoadedPlace) { htmlParts.push(`<p><strong>Source: </strong> <i>${supergraphLoadedPlace}</i></p>`); } } return `<section class="supergraph-information">${htmlParts.join('')}</section>`; }; } const readinessCheckPlugin = useReadinessCheck({ endpoint: readinessCheckEndpoint, // @ts-expect-error PromiseLike is not compatible with Promise check: readinessChecker, }); const defaultGatewayPlugin = { onFetch({ setFetchFn }) { setFetchFn(fetchAPI.fetch); }, onPluginInit({ plugins }) { onFetchHooks.splice(0, onFetchHooks.length); onSubgraphExecuteHooks.splice(0, onSubgraphExecuteHooks.length); onDelegateHooks.splice(0, onDelegateHooks.length); for (const plugin of plugins) { if (plugin.onFetch) { onFetchHooks.push(plugin.onFetch); } if (plugin.onSubgraphExecute) { onSubgraphExecuteHooks.push(plugin.onSubgraphExecute); } // @ts-expect-error For backward compatibility if (plugin.onDelegate) { // @ts-expect-error For backward compatibility onDelegateHooks.push(plugin.onDelegate); } if (isDisposable(plugin)) { disposableStack.use(plugin); } } }, }; const productName = config.productName || 'GraphQL Mesh'; const productDescription = config.productDescription || 'Federated architecture for any API service'; const productPackageName = config.productPackageName || '@graphql-mesh/serve-cli'; const productLogo = config.productLogo || defaultProductLogo; const productLink = config.productLink || 'https://the-guild.dev/graphql/mesh'; let graphiqlOptionsOrFactory; if (config.graphiql == null || config.graphiql === true) { graphiqlOptionsOrFactory = { title: productName, defaultQuery: defaultQueryText, }; } else if (config.graphiql === false) { graphiqlOptionsOrFactory = false; } else if (typeof config.graphiql === 'object') { graphiqlOptionsOrFactory = { title: productName, defaultQuery: defaultQueryText, ...config.graphiql, }; } else if (typeof config.graphiql === 'function') { const userGraphiqlFactory = config.graphiql; // @ts-expect-error PromiseLike is not compatible with Promise graphiqlOptionsOrFactory = function graphiqlOptionsFactoryForMesh(...args) { const options = userGraphiqlFactory(...args); return mapMaybePromise(options, resolvedOpts => { if (resolvedOpts === false) { return false; } if (resolvedOpts === true) { return { title: productName, defaultQuery: defaultQueryText, }; } return { title: productName, defaultQuery: defaultQueryText, ...resolvedOpts, }; }); }; } let landingPageRenderer; if (config.landingPage == null || config.landingPage === true) { landingPageRenderer = async function gatewayLandingPageRenderer(opts) { const subgraphHtml = await subgraphInformationHTMLRenderer(); return new opts.fetchAPI.Response(landingPageHtml .replace(/__GRAPHIQL_LINK__/g, opts.graphqlEndpoint) .replace(/__REQUEST_PATH__/g, opts.url.pathname) .replace(/__SUBGRAPH_HTML__/g, subgraphHtml) .replaceAll(/__PRODUCT_NAME__/g, productName) .replaceAll(/__PRODUCT_DESCRIPTION__/g, productDescription) .replaceAll(/__PRODUCT_PACKAGE_NAME__/g, productPackageName) .replace(/__PRODUCT_LINK__/, productLink) .replace(/__PRODUCT_LOGO__/g, productLogo), { status: 200, statusText: 'OK', headers: { 'Content-Type': 'text/html', }, }); }; } else if (typeof config.landingPage === 'function') { landingPageRenderer = config.landingPage; } else if (config.landingPage === false) { landingPageRenderer = false; } const basePlugins = [ defaultGatewayPlugin, unifiedGraphPlugin, readinessCheckPlugin, registryPlugin, persistedDocumentsPlugin, useChangingSchema(getSchema, _setSchema => { setSchema = _setSchema; }), useCompleteSubscriptionsOnDispose(disposableStack), useCompleteSubscriptionsOnSchemaChange(), useRequestId(), useSubgraphExecuteDebug(configContext), useFetchDebug(configContext), ]; const extraPlugins = []; if (config.webhooks) { extraPlugins.push(useWebhooks(configContext)); } if (config.responseCaching) { extraPlugins.push( // @ts-expect-error TODO: what's up with type narrowing useMeshResponseCache({ ...configContext, ...config.responseCaching, })); } if (config.contentEncoding) { extraPlugins.push(useContentEncoding(typeof config.contentEncoding === 'object' ? config.contentEncoding : {})); } if (config.deferStream) { extraPlugins.push(useDeferStream()); } if (config.executionCancellation) { extraPlugins.push(useExecutionCancellation()); } if (config.upstreamCancellation) { extraPlugins.push(useUpstreamCancel()); } if (config.disableIntrospection) { extraPlugins.push(useDisableIntrospection(typeof config.disableIntrospection === 'object' ? config.disableIntrospection : {})); } if (config.csrfPrevention) { extraPlugins.push(useCSRFPrevention(typeof config.csrfPrevention === 'object' ? config.csrfPrevention : {})); } if (config.customAgent) { extraPlugins.push(useCustomAgent(config.customAgent)); } if (config.genericAuth) { extraPlugins.push(useGenericAuth(config.genericAuth)); } if (config.hmacSignature) { extraPlugins.push(useHmacUpstreamSignature(config.hmacSignature)); } if (config.propagateHeaders) { extraPlugins.push(usePropagateHeaders(config.propagateHeaders)); } const yoga = createYoga({ fetchAPI: config.fetchAPI, logging: logger, plugins: [...basePlugins, ...(config.plugins?.(configContext) || []), ...extraPlugins], // @ts-expect-error PromiseLike is not compatible with Promise context({ request, params, ...rest }) { // TODO: I dont like this cast, but it's necessary const { req, connectionParams } = rest; let headers = // Maybe Node-like environment req?.headers ? getHeadersObj(req.headers) : // Fetch environment request?.headers ? getHeadersObj(request.headers) : // Unknown environment {}; if (connectionParams) { headers = { ...headers, ...connectionParams }; } const baseContext = { ...configContext, request, params, headers, connectionParams: headers, }; if (contextBuilder) { return contextBuilder(baseContext); } return baseContext; }, cors: config.cors, graphiql: graphiqlOptionsOrFactory, batching: config.batching, graphqlEndpoint: config.graphqlEndpoint, maskedErrors: config.maskedErrors, healthCheckEndpoint: config.healthCheckEndpoint || '/healthcheck', landingPage: landingPageRenderer, }); fetchAPI ||= yoga.fetchAPI; Object.defineProperties(yoga, { invalidateUnifiedGraph: { value: schemaInvalidator, configurable: true, }, getSchema: { value: getSchema, configurable: true, }, }); return makeAsyncDisposable(yoga, () => disposableStack.disposeAsync()); } function isDynamicUnifiedGraphSchema(schema) { if (isSchema(schema)) { // schema object return false; } if (isDocumentNode(schema)) { // document node that could be a schema return false; } if (typeof schema === 'string') { if (isValidPath(schema) && !isUrl(String(schema))) { // local path return false; } try { // sdl parse(schema); return false; } catch { } } // likely a dynamic schema that can be polled return true; }