@graphql-codegen/fragment-matcher
Version:
graphql-code-generate plugin for generating fragments matcher introspection file
116 lines (115 loc) • 3.88 kB
JavaScript
import { extname } from 'path';
import { removeFederation } from '@graphql-codegen/plugin-helpers';
import { execute, parse } from 'graphql';
const extensions = {
ts: ['.ts', '.tsx'],
js: ['.js', '.jsx'],
json: ['.json'],
};
export const plugin = async (schema, _documents, pluginConfig, info) => {
const config = {
module: 'es2015',
federation: false,
apolloClientVersion: 3,
useExplicitTyping: false,
...pluginConfig,
};
const apolloClientVersion = parseInt(config.apolloClientVersion);
const cleanSchema = config.federation ? removeFederation(schema) : schema;
const { useExplicitTyping } = config;
const introspection = (await execute({
schema: cleanSchema,
document: parse(`
{
__schema {
types {
kind
name
possibleTypes {
name
}
}
}
}
`),
}));
const ext = extname(info.outputFile).toLowerCase();
if (!introspection.data) {
throw new Error(`Plugin "fragment-matcher" couldn't introspect the schema`);
}
const filterUnionAndInterfaceTypes = type => type.kind === 'UNION' || type.kind === 'INTERFACE';
const createPossibleTypesCollection = (acc, type) => {
return { ...acc, [type.name]: type.possibleTypes.map(possibleType => possibleType.name) };
};
const filteredData = apolloClientVersion === 2
? {
__schema: {
...introspection.data.__schema,
types: introspection.data.__schema.types.filter(type => type.kind === 'UNION' || type.kind === 'INTERFACE'),
},
}
: {
possibleTypes: introspection.data.__schema.types
.filter(filterUnionAndInterfaceTypes)
.reduce(createPossibleTypesCollection, {}),
};
const content = JSON.stringify(filteredData, null, 2);
if (extensions.json.includes(ext)) {
return content;
}
if (extensions.js.includes(ext)) {
const defaultExportStatement = config.module === 'es2015' ? `export default` : 'module.exports =';
return `
${defaultExportStatement} ${content}
`;
}
if (extensions.ts.includes(ext)) {
let typename;
if (apolloClientVersion === 2) {
typename = `IntrospectionResultData`;
}
else if (apolloClientVersion === 3) {
typename = `PossibleTypesResultData`;
}
let type;
if (useExplicitTyping) {
type = `export type ${typename} = ${content};`;
}
else if (apolloClientVersion === 2) {
type = `export interface ${typename} {
__schema: {
types: {
kind: string;
name: string;
possibleTypes: {
name: string;
}[];
}[];
};
}`;
}
else if (apolloClientVersion === 3) {
type = `export interface ${typename} {
possibleTypes: {
[key: string]: string[]
}
}`;
}
return `
${type}
const result: ${typename} = ${content};
export default result;
`;
}
throw new Error(`Extension ${ext} is not supported`);
};
export const validate = async (_schema, _documents, config, outputFile) => {
const ext = extname(outputFile).toLowerCase();
const all = Object.values(extensions).reduce((acc, exts) => [...acc, ...exts], []);
if (!all.includes(ext)) {
throw new Error(`Plugin "fragment-matcher" requires extension to be one of ${all.map(val => val.replace('.', '')).join(', ')}!`);
}
if (config.module === 'commonjs' && extensions.ts.includes(ext)) {
throw new Error(`Plugin "fragment-matcher" doesn't support commonjs modules combined with TypeScript!`);
}
};