@syngrisi/syngrisi
Version:
Syngrisi - Visual Testing Tool
81 lines (74 loc) • 3.21 kB
JavaScript
/* eslint-disable no-param-reassign, valid-jsdoc */
const { EJSON } = require('bson');
const paginateDistinct = (schema) => {
/**
* @typedef {Object} QueryResult
* @property {Document[]} results - Results found
* @property {number} page - Current page
* @property {number} limit - Maximum number of results per page
* @property {number} totalPages - Total number of pages
* @property {number} totalResults - Total number of documents
*/
/**
* Query for documents with pagination
* @param {Object} [filter] - Mongo filter
* @param {Object} [options] - Query options
* @param {string} [options.sortBy] - Sorting criteria using the format: sortField:(desc|asc).
* Multiple sorting criteria should be separated by commas (,)
* @param {string} [options.populate] - Populate data fields. Hierarchy of fields should be separated by (.).
* Multiple populating criteria should be separated by commas (,)
* @param {number} [options.limit] - Maximum number of results per page (default = 10)
* @param {number} [options.page] - Current page (default = 1)
* @returns {Promise<QueryResult>}
*/
// eslint-disable-next-line func-names
schema.statics.paginateDistinct = async function (filter, options) {
let sort = {};
if (options.sortBy) {
options.sortBy.split(',')
.forEach((sortOption) => {
const [key, order] = sortOption.split(':');
sort[key] = (order === 'desc') ? -1 : 1;
});
} else {
sort = { _id: -1 };
}
let limit = options.limit && parseInt(options.limit, 10) >= 0 ? parseInt(options.limit, 10) : 10;
limit = limit === 0 ? 9007199254740991 : limit;
const page = options.page && parseInt(options.page, 10) > 0 ? parseInt(options.page, 10) : 1;
const skip = (page - 1) * limit;
const groupAggregateObj = { $group: { _id: `$${options.field}` } };
const documentsCount = (await this.aggregate([groupAggregateObj])
.exec()).length;
const aggregateArr = [
{ $match: EJSON.parse(filter.filter || '{}') },
groupAggregateObj,
{ $sort: sort },
{ $skip: skip },
{ $limit: limit },
];
const aggregatedDocs = (await this.aggregate(aggregateArr))
.filter((x) => x._id)
.map((x) => {
if (x[options.field]) {
return x[options.field][0];
}
return { name: x._id };
});
return Promise.all([documentsCount, aggregatedDocs])
.then((values) => {
const [totalResults, results] = values;
const totalPages = Math.ceil(totalResults / limit);
const result = {
results,
page,
limit,
totalPages,
totalResults,
timestamp: new Date() / 1,
};
return Promise.resolve(result);
});
};
};
module.exports = paginateDistinct;