2017-07-03 10:54:10 +09:00
|
|
|
/*
|
|
|
|
|
|
|
|
Default list, single, and total resolvers
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2018-03-03 11:39:56 +09:00
|
|
|
import { Utils, debug, debugGroup, debugGroupEnd, Connectors } from 'meteor/vulcan:lib';
|
2018-01-04 09:41:48 +09:00
|
|
|
import { createError } from 'apollo-errors';
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
const defaultOptions = {
|
2018-01-29 10:09:31 +09:00
|
|
|
cacheMaxAge: 300,
|
|
|
|
};
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
export const getDefaultResolvers = (collectionName, resolverOptions = defaultOptions) => {
|
|
|
|
return {
|
|
|
|
// resolver for returning a list of documents based on a set of query terms
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
list: {
|
|
|
|
name: `${collectionName}List`,
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2018-01-02 13:04:33 +09:00
|
|
|
description: `A list of ${collectionName} documents matching a set of query terms`,
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2018-01-06 13:40:40 +01:00
|
|
|
async resolver(root, { terms = {}, enableCache = false }, context, { cacheControl }) {
|
2018-01-29 10:09:31 +09:00
|
|
|
debug('');
|
2018-01-06 13:40:40 +01:00
|
|
|
debugGroup(`--------------- start \x1b[35m${collectionName} list\x1b[0m resolver ---------------`);
|
2018-01-06 13:46:31 +01:00
|
|
|
debug(`Options: ${JSON.stringify(resolverOptions)}`);
|
2018-04-23 09:47:04 +09:00
|
|
|
console.log(`Terms: ${JSON.stringify(terms)}`);
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-12-13 18:39:19 +09:00
|
|
|
if (cacheControl && enableCache) {
|
2017-12-08 20:54:23 +09:00
|
|
|
const maxAge = resolverOptions.cacheMaxAge || defaultOptions.cacheMaxAge;
|
|
|
|
cacheControl.setCacheHint({ maxAge });
|
2017-11-30 23:08:27 +09:00
|
|
|
}
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// get currentUser and Users collection from context
|
|
|
|
const { currentUser, Users } = context;
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// get collection based on collectionName argument
|
|
|
|
const collection = context[collectionName];
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// get selector and options from terms and perform Mongo query
|
2018-01-06 13:40:40 +01:00
|
|
|
let { selector, options } = await collection.getParameters(terms, {}, context);
|
2017-11-30 23:08:27 +09:00
|
|
|
options.skip = terms.offset;
|
|
|
|
|
|
|
|
debug({ selector, options });
|
|
|
|
|
2018-03-03 11:39:56 +09:00
|
|
|
const docs = await Connectors.find(collection, selector, options);
|
2017-11-30 23:08:27 +09:00
|
|
|
|
|
|
|
// if collection has a checkAccess function defined, remove any documents that doesn't pass the check
|
2018-01-29 10:09:31 +09:00
|
|
|
const viewableDocs = collection.checkAccess
|
|
|
|
? _.filter(docs, doc => collection.checkAccess(currentUser, doc))
|
|
|
|
: docs;
|
2018-01-06 13:40:40 +01:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// take the remaining documents and remove any fields that shouldn't be accessible
|
|
|
|
const restrictedDocs = Users.restrictViewableFields(currentUser, collection, viewableDocs);
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// prime the cache
|
|
|
|
restrictedDocs.forEach(doc => collection.loader.prime(doc._id, doc));
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2018-01-06 13:40:40 +01:00
|
|
|
debug(`\x1b[33m=> ${restrictedDocs.length} documents returned\x1b[0m`);
|
|
|
|
debugGroupEnd();
|
|
|
|
debug(`--------------- end \x1b[35m${collectionName} list\x1b[0m resolver ---------------`);
|
2018-01-29 10:09:31 +09:00
|
|
|
debug('');
|
2017-11-30 23:08:27 +09:00
|
|
|
|
|
|
|
// return results
|
|
|
|
return restrictedDocs;
|
|
|
|
},
|
2017-07-03 10:54:10 +09:00
|
|
|
},
|
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// resolver for returning a single document queried based on id or slug
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
single: {
|
|
|
|
name: `${collectionName}Single`,
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2018-01-02 13:04:33 +09:00
|
|
|
description: `A single ${collectionName} document fetched by ID or slug`,
|
|
|
|
|
2018-01-06 13:40:40 +01:00
|
|
|
async resolver(root, { documentId, slug, enableCache = false }, context, { cacheControl }) {
|
2018-01-29 10:09:31 +09:00
|
|
|
debug('');
|
2018-01-06 13:40:40 +01:00
|
|
|
debugGroup(`--------------- start \x1b[35m${collectionName} single\x1b[0m resolver ---------------`);
|
2018-01-06 13:46:31 +01:00
|
|
|
debug(`Options: ${JSON.stringify(resolverOptions)}`);
|
2018-04-06 17:59:24 +09:00
|
|
|
debug(`DocumentId: ${documentId}, Slug: ${slug}`);
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-12-13 18:39:19 +09:00
|
|
|
if (cacheControl && enableCache) {
|
2017-12-08 20:54:23 +09:00
|
|
|
const maxAge = resolverOptions.cacheMaxAge || defaultOptions.cacheMaxAge;
|
|
|
|
cacheControl.setCacheHint({ maxAge });
|
2017-11-30 23:08:27 +09:00
|
|
|
}
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
const { currentUser, Users } = context;
|
|
|
|
const collection = context[collectionName];
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// don't use Dataloader if doc is selected by slug
|
2018-01-29 10:09:31 +09:00
|
|
|
const doc = documentId
|
|
|
|
? await collection.loader.load(documentId)
|
2018-03-03 11:39:56 +09:00
|
|
|
: slug ? await Connectors.get(collection, { slug }) : await Connectors.get();
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2018-01-04 09:41:48 +09:00
|
|
|
if (!doc) {
|
2018-01-06 13:40:40 +01:00
|
|
|
const MissingDocumentError = createError('app.missing_document', { message: 'app.missing_document' });
|
|
|
|
throw new MissingDocumentError({ data: { documentId, slug } });
|
2018-01-04 09:41:48 +09:00
|
|
|
}
|
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// if collection has a checkAccess function defined, use it to perform a check on the current document
|
|
|
|
// (will throw an error if check doesn't pass)
|
|
|
|
if (collection.checkAccess) {
|
|
|
|
Utils.performCheck(collection.checkAccess, currentUser, doc, collection, documentId);
|
|
|
|
}
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
const restrictedDoc = Users.restrictViewableFields(currentUser, collection, doc);
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2018-01-06 13:40:40 +01:00
|
|
|
debugGroupEnd();
|
|
|
|
debug(`--------------- end \x1b[35m${collectionName} single\x1b[0m resolver ---------------`);
|
2018-01-29 10:09:31 +09:00
|
|
|
debug('');
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// filter out disallowed properties and return resulting document
|
|
|
|
return restrictedDoc;
|
|
|
|
},
|
2017-07-03 10:54:10 +09:00
|
|
|
},
|
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
// resolver for returning the total number of documents matching a set of query terms
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
total: {
|
|
|
|
name: `${collectionName}Total`,
|
2018-01-02 13:04:33 +09:00
|
|
|
|
|
|
|
description: `The total count of ${collectionName} documents matching a set of query terms`,
|
2018-01-06 13:40:40 +01:00
|
|
|
|
|
|
|
async resolver(root, { terms, enableCache }, context, { cacheControl }) {
|
2017-12-13 18:39:19 +09:00
|
|
|
if (cacheControl && enableCache) {
|
2017-12-08 20:54:23 +09:00
|
|
|
const maxAge = resolverOptions.cacheMaxAge || defaultOptions.cacheMaxAge;
|
|
|
|
cacheControl.setCacheHint({ maxAge });
|
2017-11-30 23:08:27 +09:00
|
|
|
}
|
2017-07-03 10:54:10 +09:00
|
|
|
|
2017-11-30 23:08:27 +09:00
|
|
|
const collection = context[collectionName];
|
2017-09-15 11:14:09 +02:00
|
|
|
|
2018-01-06 13:40:40 +01:00
|
|
|
const { selector } = await collection.getParameters(terms, {}, context);
|
2017-11-30 23:08:27 +09:00
|
|
|
|
2018-03-03 11:39:56 +09:00
|
|
|
const total = await Connectors.count(collection, selector);
|
2018-01-06 13:40:40 +01:00
|
|
|
|
2018-02-11 12:57:51 +09:00
|
|
|
return total;
|
2018-01-28 09:58:57 -06:00
|
|
|
},
|
2018-01-29 10:09:31 +09:00
|
|
|
},
|
|
|
|
};
|
2017-11-30 23:08:27 +09:00
|
|
|
};
|