graphql-compose-mongoose
Version:
Plugin for `graphql-compose` which derive a graphql types from a mongoose model.
131 lines (115 loc) • 3.84 kB
Flow
/* @flow */
import type { TypeComposer, Resolver } from 'graphql-compose';
import { graphql } from 'graphql-compose';
import type { MongooseModel } from 'mongoose';
import { recordHelperArgs } from './helpers';
import type { ExtendedResolveParams, GenResolverOpts } from './index';
async function createSingle(
model: MongooseModel,
tc: TypeComposer,
recordData: any,
resolveParams: ExtendedResolveParams
) {
// eslint-disable-next-line new-cap
let doc = new model(recordData);
if (resolveParams.beforeRecordMutate) {
doc = await resolveParams.beforeRecordMutate(doc, resolveParams);
if (!doc) return null;
}
return doc.save();
}
export default function createMany(
model: MongooseModel,
tc: TypeComposer,
opts?: GenResolverOpts
): Resolver {
if (!model || !model.modelName || !model.schema) {
throw new Error('First arg for Resolver createMany() should be instance of Mongoose Model.');
}
if (!tc || tc.constructor.name !== 'TypeComposer') {
throw new Error('Second arg for Resolver createMany() should be instance of TypeComposer.');
}
const tree = model.schema.obj;
const requiredFields = [];
for (const field in tree) {
if (tree.hasOwnProperty(field)) {
const fieldOptions = tree[field];
if (fieldOptions.required) {
requiredFields.push(field);
}
}
}
const outputTypeName = `CreateMany${tc.getTypeName()}Payload`;
const outputType = tc.constructor.schemaComposer.getOrCreateTC(outputTypeName, t => {
t.addFields({
recordIds: {
type: '[MongoID]!',
description: 'Created document ID',
},
records: {
type: new graphql.GraphQLNonNull(tc.getTypePlural()),
description: 'Created documents',
},
createCount: {
type: 'Int!',
description: 'Count of all documents created',
},
});
});
const resolver = new tc.constructor.schemaComposer.Resolver({
name: 'createMany',
kind: 'mutation',
description: 'Creates Many documents with mongoose defaults, setters, hooks and validation',
type: outputType,
args: {
records: {
type: new graphql.GraphQLNonNull(
new graphql.GraphQLList(
(recordHelperArgs(tc, {
recordTypeName: `CreateMany${tc.getTypeName()}Input`,
removeFields: ['id', '_id'],
isRequired: true,
requiredFields,
...(opts && opts.records),
}).record: any).type
)
),
},
},
resolve: async (resolveParams: ExtendedResolveParams) => {
const recordData = (resolveParams.args && resolveParams.args.records) || [];
if (!Array.isArray(recordData) || recordData.length === 0) {
throw new Error(
`${tc.getTypeName()}.createMany resolver requires args.records to be an Array and must contain at least one record`
);
}
for (const record of recordData) {
if (!(typeof record === 'object') || Object.keys(record).length === 0) {
throw new Error(
`${tc.getTypeName()}.createMany resolver requires args.records to contain non-empty records, with at least one value`
);
}
}
const recordPromises = [];
// concurrently create docs
for (const record of recordData) {
recordPromises.push(createSingle(model, tc, record, resolveParams));
}
const results = await Promise.all(recordPromises);
const returnObj = {
records: [],
recordIds: [],
createCount: 0,
};
for (const doc of results) {
if (doc) {
returnObj.createCount += 1;
returnObj.records.push(doc);
returnObj.recordIds.push(doc._id);
}
}
return returnObj;
},
});
return resolver;
}