@graphql-mesh/fusion-federation
Version:
Conversion tool from Federation to Fusion
260 lines (259 loc) • 15 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.convertSupergraphToFusiongraph = void 0;
const graphql_1 = require("graphql");
const utils_1 = require("@graphql-tools/utils");
const schemaBuildOpts = { noLocation: true, assumeValid: true, assumeValidSDL: true };
function convertSupergraphToFusiongraph(federationSupergraph) {
let supergraphSchema;
if ((0, graphql_1.isSchema)(federationSupergraph)) {
supergraphSchema = federationSupergraph;
}
else if ((0, utils_1.isDocumentNode)(federationSupergraph)) {
supergraphSchema = (0, graphql_1.buildASTSchema)(federationSupergraph, schemaBuildOpts);
}
else {
supergraphSchema = (0, graphql_1.buildSchema)(federationSupergraph, schemaBuildOpts);
}
const subgraphLocationMap = new Map();
const joinGraphEnum = supergraphSchema.getType('join__Graph');
if (!(0, graphql_1.isEnumType)(joinGraphEnum)) {
throw new Error('Expected join__Graph to be an enum');
}
for (const joinGraphEnumValue of joinGraphEnum.getValues()) {
const joinGraphDirective = (0, utils_1.getDirective)(supergraphSchema, joinGraphEnumValue, 'join__graph');
if (!joinGraphDirective?.length) {
throw new Error('Expected join__Graph to have a join__graph directive');
}
const { url } = joinGraphDirective[0];
subgraphLocationMap.set(joinGraphEnumValue.value, url);
}
const typeMap = new Map();
const rootTypeMap = (0, utils_1.getRootTypeMap)(supergraphSchema);
const operationByRootType = new Map();
for (const [operationType, rootType] of rootTypeMap) {
operationByRootType.set(rootType.name, operationType);
}
const fusiongraphSchema = (0, utils_1.mapSchema)(supergraphSchema, {
[utils_1.MapperKind.TYPE](type) {
if (type.name.startsWith('link__') || type.name.startsWith('join__')) {
return null;
}
const typeExtensions = (type.extensions ||= {});
const typeDirectiveExtensions = (typeExtensions.directives ||= {});
const joinTypeDirectives = (0, utils_1.getDirective)(supergraphSchema, type, 'join__type');
const sourceDirectivesForFusion = (typeDirectiveExtensions.source ||= []);
const variableDirectivesForFusion = (typeDirectiveExtensions.variable ||= []);
const fieldMap = type.getFields?.();
const combinedVariableValues = new Set();
const resolverDirectives = (typeDirectiveExtensions.resolver ||= []);
for (const joinTypeDirective of joinTypeDirectives || []) {
if (!joinTypeDirective.external) {
sourceDirectivesForFusion.push({
name: type.name,
subgraph: joinTypeDirective.graph,
});
}
if (joinTypeDirective.key) {
const fakeOpForRequiresSelectionSet = (0, graphql_1.parse)(`{ ${joinTypeDirective.key} }`);
const operationAst = (0, graphql_1.getOperationAST)(fakeOpForRequiresSelectionSet, undefined);
if (!operationAst) {
throw new Error(`Expected requires to be a valid selection set for ${type.name}`);
}
for (const selection of operationAst.selectionSet.selections) {
if (selection.kind === graphql_1.Kind.FIELD) {
const selectionName = selection.name.value;
const varName = `${type.name}_key_${selectionName}`;
const selectionFieldObject = fieldMap[selectionName];
if (!selectionFieldObject) {
throw new Error(`Expected field ${selectionName} to exist on type ${type.name}`);
}
let availableSubgraphsForField = [];
const joinFieldDirectives = (0, utils_1.getDirective)(supergraphSchema, selectionFieldObject, 'join__field');
if (joinFieldDirectives?.length) {
availableSubgraphsForField = joinFieldDirectives
.filter(joinFieldDirective => !joinFieldDirective.external)
.map(joinFieldDirective => joinFieldDirective.graph);
}
else {
availableSubgraphsForField = joinTypeDirectives.map(joinTypeDirective => joinTypeDirective.graph);
}
for (const availableSubgraph of availableSubgraphsForField) {
variableDirectivesForFusion.push({
name: varName,
select: (0, graphql_1.print)(selection),
subgraph: availableSubgraph,
type: (0, graphql_1.getNamedType)(fieldMap[selectionName].type).toString(),
});
}
combinedVariableValues.add(`${selectionName}: $${varName}`);
}
}
const mainVariable = {
name: `representations`,
subgraph: joinTypeDirective.graph,
value: `{ __typename: "${type.name}", ${[...combinedVariableValues].join(', ')} }`,
};
variableDirectivesForFusion.push(mainVariable);
resolverDirectives.push({
operation: `query get${type.name}($representations: [_Any!]!) { _entities(representations: $representations) { ... on ${type.name} { ...__export } } }`,
subgraph: joinTypeDirective.graph,
});
}
}
typeMap.set(type.name, type);
return type;
},
[utils_1.MapperKind.FIELD](fieldConfig, fieldName, typeName) {
const type = typeMap.get(typeName);
if (!type) {
throw new Error(`Expected type ${typeName} to exist`);
}
if (!('getFields' in type)) {
throw new Error(`Expected type ${typeName} to be an object type`);
}
const typeFieldMap = type.getFields();
const typeDirectiveExtensions = ((type.extensions ||= {}).directives ||= {});
const typeVariableDirectives = (typeDirectiveExtensions.variable ||= []);
const representationsVariable = typeVariableDirectives.find((directive) => directive.name === 'representations');
const fieldExtensions = (fieldConfig.extensions ||= {});
const fieldDirectiveExtensions = (fieldExtensions.directives ||= {});
const joinFieldDirectives = (0, utils_1.getDirective)(supergraphSchema, fieldConfig, 'join__field');
const sourceDirectivesForFusion = (fieldDirectiveExtensions.source ||= []);
const variableDirectivesForFusion = (fieldDirectiveExtensions.variable ||= []);
// TODO: Pass this to type's representations
const combinedVariableValues = [];
for (const joinFieldDirective of joinFieldDirectives || []) {
if (!joinFieldDirective.external) {
sourceDirectivesForFusion.push({
subgraph: joinFieldDirective.graph,
name: fieldName,
});
}
if (joinFieldDirective.requires) {
const fakeOpForRequiresSelectionSet = (0, graphql_1.parse)(`{ ${joinFieldDirective.requires} }`);
const operationAst = (0, graphql_1.getOperationAST)(fakeOpForRequiresSelectionSet, undefined);
if (!operationAst) {
throw new Error(`Expected requires to be a valid selection set for ${fieldName}`);
}
for (const selection of operationAst.selectionSet.selections) {
if (selection.kind === 'Field') {
const selectionName = selection.name.value;
const varName = `${typeName}_${fieldName}_requires_${selectionName}`;
const selectionFieldObject = typeFieldMap[selectionName];
if (!selectionFieldObject) {
throw new Error(`Expected field ${selectionName} to exist on type ${typeName}`);
}
const joinFieldDirectives = (0, utils_1.getDirective)(supergraphSchema, selectionFieldObject, 'join__field');
for (const joinFieldDirective of joinFieldDirectives || []) {
if (!joinFieldDirective.external) {
variableDirectivesForFusion.push({
name: varName,
select: (0, graphql_1.print)(selection),
subgraph: joinFieldDirective.graph,
type: (0, graphql_1.getNamedType)(typeFieldMap[selectionName].type).toString(),
});
}
}
combinedVariableValues.push(`${selectionName}: $${varName}`);
}
else {
throw new Error(`Kind ${selection.kind} not supported for requires selection set`);
}
}
}
const operationType = operationByRootType.get(type.name);
if (operationType) {
const operationName = operationType === 'query' ? fieldName : `${operationType}${fieldName}`;
const variableDefinitions = [];
const rootFieldArgs = [];
if ('args' in fieldConfig && fieldConfig.args) {
for (const argName in fieldConfig.args) {
const arg = fieldConfig.args[argName];
let variableDefinitionStr = `$${argName}: ${arg.type}`;
if (arg.defaultValue) {
variableDefinitionStr += ` = ${typeof arg.defaultValue === 'string'
? JSON.stringify(arg.defaultValue)
: arg.defaultValue}`;
}
variableDefinitions.push(variableDefinitionStr);
rootFieldArgs.push(`${argName}: $${argName}`);
}
}
const variableDefinitionsString = variableDefinitions.length
? `(${variableDefinitions.join(', ')})`
: '';
const rootFieldArgsString = rootFieldArgs.length ? `(${rootFieldArgs.join(', ')})` : '';
const operationString = `${operationType} ${operationName}${variableDefinitionsString} { ${fieldName}${rootFieldArgsString} }`;
const resolverDirectives = (fieldDirectiveExtensions.resolver ||= []);
// TODO: Later use global resolvers to batch queries for different types
resolverDirectives.push({
subgraph: joinFieldDirective.graph,
operation: operationString,
});
}
}
if (!joinFieldDirectives?.length) {
const typeSourceDirectives = (typeDirectiveExtensions.source ||= []);
for (const typeSourceDirective of typeSourceDirectives) {
sourceDirectivesForFusion.push({
subgraph: typeSourceDirective.subgraph,
name: fieldName,
});
}
}
if (combinedVariableValues.length) {
const existingVariableValuesStr = representationsVariable.value
.substring(1, representationsVariable.value.length - 2)
.trim();
const variableValues = new Set(existingVariableValuesStr.split(',').map((str) => str.trim()));
for (const combinedVariableValue of combinedVariableValues) {
variableValues.add(combinedVariableValue.trim());
}
representationsVariable.value = `{ ${[...variableValues].join(', ')} }`;
}
const seenTypeVariableKeys = new Set();
typeDirectiveExtensions.variable = typeVariableDirectives.filter((variableDirective) => {
const variableKey = variableDirective.name + '_' + variableDirective.subgraph;
if (seenTypeVariableKeys.has(variableKey)) {
return false;
}
seenTypeVariableKeys.add(variableKey);
return true;
});
return fieldConfig;
},
[utils_1.MapperKind.ENUM_VALUE](enumValueConfig, enumValueName) {
const enumValueExtensions = (enumValueConfig.extensions ||= {});
const enumValueDirectiveExtensions = (enumValueExtensions.directives ||= {});
const joinEnumValueDirectives = (0, utils_1.getDirective)(supergraphSchema, enumValueConfig, 'join__enumValue');
const sourceDirectivesForFusion = (enumValueDirectiveExtensions.source ||= []);
for (const joinEnumValueDirective of joinEnumValueDirectives || []) {
sourceDirectivesForFusion.push({
subgraph: joinEnumValueDirective.graph,
name: enumValueName,
});
}
return enumValueConfig;
},
[utils_1.MapperKind.DIRECTIVE](directiveConfig) {
if (directiveConfig.name.startsWith('join__') ||
directiveConfig.name === 'link__' ||
directiveConfig.name === 'link') {
return null;
}
},
});
const fusiongraphSchemaExtensions = (fusiongraphSchema.extensions ||= {});
const fusiongraphDirectiveExtensions = (fusiongraphSchemaExtensions.directives ||= {});
const fusionTransportDefs = (fusiongraphDirectiveExtensions.transport ||= []);
for (const [subgraph, location] of subgraphLocationMap.entries()) {
fusionTransportDefs.push({
subgraph,
kind: 'http',
location,
});
}
return fusiongraphSchema;
}
exports.convertSupergraphToFusiongraph = convertSupergraphToFusiongraph;
;