@omnigraph/neo4j
Version:
124 lines (123 loc) • 4.51 kB
JavaScript
import { parse, visit } from 'graphql';
import neo4j from 'neo4j-driver';
import { getDriverFromOpts, getExecutableSchemaFromTypeDefsAndDriver, } from '@graphql-mesh/transport-neo4j';
import { mergeSchemas } from '@graphql-tools/schema';
import { toGraphQLTypeDefs } from '@neo4j/introspector';
import { polyfillStrReplaceAll, revertStrReplaceAllPolyfill } from './strReplaceAllPolyfill.js';
function createAddIntrospectionDirective(subgraph) {
return function addIntrospectionDirective(node) {
if (!node.directives?.some(directive => directive.name.value === 'introspection')) {
return {
...node,
directives: [
...(node.directives || []),
{
kind: 'Directive',
name: {
kind: 'Name',
value: 'introspection',
arguments: [
{
kind: 'Argument',
name: {
kind: 'Name',
value: 'subgraph',
},
value: {
kind: 'StringValue',
value: subgraph,
},
},
],
},
},
],
};
}
};
}
export async function loadGraphQLSchemaFromNeo4J(subgraphName, { endpoint, auth, logger, pubsub, database = 'neo4j', driver }) {
logger?.info('Inferring the schema from the database: ', `"${database}"`);
let closeDriverAfter = false;
if (!driver) {
closeDriverAfter = !pubsub;
driver = getDriverFromOpts({
endpoint,
auth,
logger,
});
}
polyfillStrReplaceAll();
const typeDefsStr = await toGraphQLTypeDefs(() => driver.session({
database,
defaultAccessMode: neo4j.session.READ,
}));
let typeDefs = parse(typeDefsStr, { noLocation: true });
const addIntrospectionDirective = createAddIntrospectionDirective(subgraphName);
typeDefs = visit(typeDefs, {
EnumTypeDefinition: addIntrospectionDirective,
ObjectTypeDefinition: addIntrospectionDirective,
InterfaceTypeDefinition: addIntrospectionDirective,
UnionTypeDefinition: addIntrospectionDirective,
InputObjectTypeDefinition: addIntrospectionDirective,
FieldDefinition: addIntrospectionDirective,
// DirectiveDefinition: addIntrospectionDirective,
ScalarTypeDefinition: addIntrospectionDirective,
EnumValueDefinition: addIntrospectionDirective,
InputValueDefinition: addIntrospectionDirective,
});
typeDefs.definitions.push(...parse(
/* GraphQL */ `
directive @relationshipProperties on OBJECT
directive @relationship(
type: String
direction: _RelationDirections
properties: String
) on FIELD_DEFINITION
enum _RelationDirections {
IN
OUT
}
directive @introspection(
subgraph: String
) on ENUM | OBJECT | INTERFACE | UNION | INPUT_OBJECT | FIELD_DEFINITION | SCALAR | ENUM_VALUE | INPUT_FIELD_DEFINITION
directive @transport(
kind: String
subgraph: String
location: String
options: TransportOptions
) on SCHEMA
scalar TransportOptions
directive @node on OBJECT
`, {
noLocation: true,
}).definitions);
revertStrReplaceAllPolyfill();
const schema = await getExecutableSchemaFromTypeDefsAndDriver({
driver,
pubsub,
typeDefs,
});
if (closeDriverAfter) {
await driver.close();
}
const schemaExtensions = (schema.extensions ||= {});
schemaExtensions.directives = schemaExtensions.directives || {};
schemaExtensions.directives.transport = [
{
kind: 'neo4j',
subgraph: subgraphName,
location: endpoint,
options: {
database,
auth,
},
},
];
return mergeSchemas({
schemas: [schema],
typeDefs,
assumeValid: true,
assumeValidSDL: true,
});
}