/* @flow */ /* eslint-disable no-param-reassign */ import type { Resolver, ObjectTypeComposer } from 'graphql-compose'; import type { MongooseDocument } from 'mongoose'; import type { ExtendedResolveParams, GenResolverOpts } from './index'; import { skipHelperArgs, recordHelperArgs, filterHelperArgs, sortHelperArgs } from './helpers'; import findOne from './findOne'; export default function updateOne( model: Class, // === MongooseModel tc: ObjectTypeComposer, opts?: GenResolverOpts ): Resolver { if (!model || !model.modelName || !model.schema) { throw new Error('First arg for Resolver updateOne() should be instance of Mongoose Model.'); } if (!tc || tc.constructor.name !== 'ObjectTypeComposer') { throw new Error( 'Second arg for Resolver updateOne() should be instance of ObjectTypeComposer.' ); } const findOneResolver = findOne(model, tc, opts); const outputTypeName = `UpdateOne${tc.getTypeName()}Payload`; const outputType = tc.schemaComposer.getOrCreateOTC(outputTypeName, t => { t.addFields({ recordId: { type: 'MongoID', description: 'Updated document ID', }, record: { type: tc, description: 'Updated document', }, }); }); const resolver = tc.schemaComposer.createResolver({ name: 'updateOne', kind: 'mutation', description: 'Update one document: ' + '1) Retrieve one document via findOne. ' + '2) Apply updates to mongoose document. ' + '3) Mongoose applies defaults, setters, hooks and validation. ' + '4) And save it.', type: outputType, args: { ...recordHelperArgs(tc, { recordTypeName: `UpdateOne${tc.getTypeName()}Input`, removeFields: ['id', '_id'], isRequired: true, ...(opts && opts.record), }), ...filterHelperArgs(tc, model, { filterTypeName: `FilterUpdateOne${tc.getTypeName()}Input`, model, ...(opts && opts.filter), }), ...sortHelperArgs(tc, model, { sortTypeName: `SortUpdateOne${tc.getTypeName()}Input`, ...(opts && opts.sort), }), ...skipHelperArgs(), }, resolve: async (resolveParams: ExtendedResolveParams) => { const recordData = (resolveParams.args && resolveParams.args.record) || null; const filterData = (resolveParams.args && resolveParams.args.filter) || {}; if (!(typeof filterData === 'object') || Object.keys(filterData).length === 0) { return Promise.reject( new Error( `${tc.getTypeName()}.updateOne resolver requires at least one value in args.filter` ) ); } // We should get all data for document, cause Mongoose model may have hooks/middlewares // which required some fields which not in graphql projection // So empty projection returns all fields. resolveParams.projection = {}; let doc = await findOneResolver.resolve(resolveParams); if (resolveParams.beforeRecordMutate) { doc = await resolveParams.beforeRecordMutate(doc, resolveParams); } if (doc && recordData) { doc.set(recordData); await doc.save(); } if (doc) { return { record: doc, recordId: tc.getRecordIdFn()(doc), }; } return null; }, }); return resolver; }