graphql-compose-mongoose
Version:
Plugin for `graphql-compose` which derive a graphql types from a mongoose model.
129 lines (109 loc) • 3.9 kB
Flow
/* @flow */
/* eslint-disable no-use-before-define */
import {
ObjectTypeComposer,
type ObjectTypeComposerArgumentConfigMapDefinition,
} from 'graphql-compose';
import type { MongooseModel } from 'mongoose';
import GraphQLMongoID from '../../types/mongoid';
import { isObject, toMongoFilterDottedObject, getIndexedFieldNamesForGraphQL } from '../../utils';
import type { ExtendedResolveParams } from '../index';
import {
type FilterOperatorsOpts,
addFilterOperators,
processFilterOperators,
} from './filterOperators';
export type FilterHelperArgsOpts = {
filterTypeName?: string,
isRequired?: boolean,
onlyIndexed?: boolean,
requiredFields?: string | string[],
operators?: FilterOperatorsOpts | false,
removeFields?: string | string[],
};
// for merging, discriminators merge-able only
export const getFilterHelperArgOptsMap = () => ({
// filterTypeName? : 'string'
isRequired: 'boolean',
onlyIndexed: 'boolean',
requiredFields: ['string', 'string[]'],
operators: ['FilterOperatorsOptsMap', 'boolean'],
removeFields: ['string', 'string[]'],
});
export const filterHelperArgs = (
typeComposer: ObjectTypeComposer<any, any>,
model: MongooseModel,
opts?: FilterHelperArgsOpts
): ObjectTypeComposerArgumentConfigMapDefinition<> => {
if (!(typeComposer instanceof ObjectTypeComposer)) {
throw new Error('First arg for filterHelperArgs() should be instance of ObjectTypeComposer.');
}
if (!model || !model.modelName || !model.schema) {
throw new Error('Second arg for filterHelperArgs() should be instance of MongooseModel.');
}
if (!opts || !opts.filterTypeName) {
throw new Error('You should provide non-empty `filterTypeName` in options.');
}
const removeFields = [];
if (opts.removeFields) {
if (Array.isArray(opts.removeFields)) {
removeFields.push(...opts.removeFields);
} else {
removeFields.push(opts.removeFields);
}
}
if (opts.onlyIndexed) {
const indexedFieldNames = getIndexedFieldNamesForGraphQL(model);
Object.keys(typeComposer.getFields()).forEach(fieldName => {
if (indexedFieldNames.indexOf(fieldName) === -1) {
removeFields.push(fieldName);
}
});
}
const filterTypeName: string = opts.filterTypeName;
const itc = typeComposer.getInputTypeComposer().clone(filterTypeName);
itc.makeFieldNullable('_id');
itc.addFields({
_ids: [GraphQLMongoID],
});
itc.removeField(removeFields);
if (opts.requiredFields) {
itc.makeFieldNonNull(opts.requiredFields);
}
if (itc.getFieldNames().length === 0) {
return {};
}
addFilterOperators(itc, model, opts);
return {
filter: {
type: opts.isRequired ? itc.getTypeNonNull() : itc,
description: opts.onlyIndexed ? 'Filter only by indexed fields' : 'Filter by fields',
},
};
};
export function filterHelper(resolveParams: ExtendedResolveParams): void {
const filter = resolveParams.args && resolveParams.args.filter;
if (filter && typeof filter === 'object' && Object.keys(filter).length > 0) {
const modelFields = resolveParams.query.schema.paths;
const { _ids, ...filterFields } = filter;
if (_ids && Array.isArray(_ids)) {
// eslint-disable-next-line
resolveParams.query = resolveParams.query.where({ _id: { $in: _ids } });
}
processFilterOperators(filterFields);
const clearedFilter = {};
Object.keys(filterFields).forEach(key => {
if (modelFields[key] || key.indexOf('$') === 0) {
clearedFilter[key] = filterFields[key];
}
});
if (Object.keys(clearedFilter).length > 0) {
// eslint-disable-next-line
resolveParams.query = resolveParams.query.where(toMongoFilterDottedObject(clearedFilter));
}
}
if (isObject(resolveParams.rawQuery)) {
// eslint-disable-next-line
resolveParams.query = resolveParams.query.where(resolveParams.rawQuery);
}
}