UNPKG

@graphql-mesh/transport-neo4j

Version:
113 lines (112 loc) 3.96 kB
import { extendSchema, parse, visit } from 'graphql'; import { GraphQLBigInt } from 'graphql-scalars'; import { makeAsyncDisposable } from '@graphql-mesh/utils'; import { createDefaultExecutor } from '@graphql-tools/delegate'; import { asArray, getDirectiveExtensions, getDocumentNodeFromSchema, } from '@graphql-tools/utils'; import { Neo4jGraphQL } from '@neo4j/graphql'; import { getDriverFromOpts } from './driver.js'; import { getEventEmitterFromPubSub } from './eventEmitterForPubSub.js'; function filterIntrospectionDefinitions(node) { if (!node.directives?.some(directive => directive.name.value === 'introspection')) { return null; } return node; } export async function getNeo4JExecutor(opts) { const schemaDirectives = getDirectiveExtensions(opts.schema); if (!opts.schema.getDirective('relationship')) { opts.schema = extendSchema(opts.schema, parse( /* GraphQL */ ` directive @relationship( type: String direction: _RelationDirections properties: String ) on FIELD_DEFINITION `, { noLocation: true })); } const transportDirectives = schemaDirectives?.transport; if (!transportDirectives?.length) { throw new Error('No transport directive found on the schema!'); } const { location: endpoint, options: { database, auth }, } = transportDirectives[0]; let driver = opts.driver; if (!driver) { driver = getDriverFromOpts({ endpoint, auth, logger: opts.logger, }); } let typeDefs = getDocumentNodeFromSchema(opts.schema); const astVisitor = { enter: filterIntrospectionDefinitions, }; typeDefs = visit(typeDefs, { EnumTypeDefinition: astVisitor, ObjectTypeDefinition: astVisitor, InterfaceTypeDefinition: astVisitor, UnionTypeDefinition: astVisitor, InputObjectTypeDefinition: astVisitor, FieldDefinition: astVisitor, // DirectiveDefinition: astVisitor, ScalarTypeDefinition: astVisitor, EnumValueDefinition: astVisitor, InputValueDefinition: astVisitor, }); typeDefs.definitions.push(...parse(/* GraphQL */ ` directive @source( subgraph: String name: String type: String ) on ENUM | OBJECT | INTERFACE | UNION | INPUT_OBJECT | FIELD_DEFINITION | SCALAR | ENUM_VALUE | INPUT_FIELD_DEFINITION `).definitions); const executableSchema = await getExecutableSchemaFromTypeDefsAndDriver({ driver, pubsub: opts.pubsub, typeDefs, }); const defaultExecutor = createDefaultExecutor(executableSchema); const sessionConfig = { database, }; return makeAsyncDisposable(function neo4JExecutor(args) { return defaultExecutor({ ...args, context: { ...args.context, sessionConfig, }, }); }, () => driver.close()); } export function getExecutableSchemaFromTypeDefsAndDriver({ driver, pubsub, typeDefs, }) { let features; if (pubsub) { features = { subscriptions: { events: getEventEmitterFromPubSub(pubsub), publish: eventMeta => pubsub.publish(eventMeta.event, eventMeta), close: () => { }, }, }; } const extendedTypeDefs = [ ...asArray(typeDefs), /* GraphQL */ ` directive @introspection( subgraph: String ) on ENUM | OBJECT | INTERFACE | UNION | INPUT_OBJECT | FIELD_DEFINITION | SCALAR | ENUM_VALUE | INPUT_FIELD_DEFINITION `, ]; const neo4jGraphQL = new Neo4jGraphQL({ typeDefs: extendedTypeDefs, driver, validate: false, debug: !!process.env.DEBUG, resolvers: { BigInt: GraphQLBigInt, }, features, }); return neo4jGraphQL.getSchema(); }