From eba674a7c50843bc5742a720d7af15b2e10b0cdc Mon Sep 17 00:00:00 2001 From: eastolfi Date: Tue, 14 Jun 2016 14:49:50 +0000 Subject: [PATCH] feat(aggregation): Add aggregation stages: match, sort --- lib/Aggregation.js | 44 +- lib/Collection.js | 7 +- lib/Cursor.js | 113 +++- lib/Selector.js | 10 +- src/Aggregation.js | 42 +- src/Collection.js | 5 +- src/Cursor.js | 106 +++- src/Selector.js | 8 +- test/3_Cursor.js | 6 +- test/5_Aggregation.js | 1194 +++++------------------------------------ 10 files changed, 407 insertions(+), 1128 deletions(-) diff --git a/lib/Aggregation.js b/lib/Aggregation.js index 8adcce9..927839b 100644 --- a/lib/Aggregation.js +++ b/lib/Aggregation.js @@ -16,20 +16,22 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope */ var _ = require("lodash"), - Logger = require("jsw-logger"); + Logger = require("jsw-logger"), + Cursor = require("./Cursor"), + Selector = require("./Selector"); var logger = null; var stages = { '$project': false, - '$match': false, + '$match': true, '$redact': false, '$limit': false, '$skip': false, '$unwind': false, '$group': true, '$sample': false, - '$sort': false, + '$sort': true, '$geoNear': false, '$lookup': false, '$out': false, @@ -163,11 +165,29 @@ var do_single_group = function do_single_group(group_id, group_stage, documents) var do_complex_group = function do_complex_group() {}; +var do_sort = function do_sort(documents, sort_stage) { + return documents.sort(new Selector(sort_stage, Selector.SORT_SELECTOR)); +}; + +var do_match = function do_match(documents, match_stage) { + var cursor = new Cursor(documents, match_stage); + + return cursor.fetch(); +}; + var do_group = function do_group(documents, group_stage) { if (!_.hasIn(group_stage, '_id')) logger.throw('The field "_id" is required in the "$group" stage'); var new_id = group_stage['_id']; + if (!_.isNull(new_id)) { + if (new_id.substr(0, 1) !== '$') { + logger.throw("Field names references in a right side assignement must be preceded by '$'"); + } else { + new_id = new_id.substr(1, new_id.length); + } + } + if (_.isPlainObject(new_id)) { // complex_id // do_complex_group(); @@ -189,16 +209,30 @@ var Aggregation = function () { _createClass(Aggregation, [{ key: "aggregate", value: function aggregate(collection) { + var docs = collection.docs; + for (var i = 0; i < this.pipeline.length; i++) { var stage = this.pipeline[i]; for (var key in stage) { switch (key) { + case '$match': + docs = do_match(docs, stage[key]); + + break; case '$group': - return do_group(collection.docs, stage[key]); + docs = do_group(docs, stage[key]); + + break; + case '$sort': + docs = do_sort(docs, stage[key]); + + break; } } } + + return docs; // move to cursor } }, { key: "validStage", @@ -215,4 +249,4 @@ var Aggregation = function () { }(); module.exports = Aggregation; -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Aggregation.js"],"names":[],"mappings":"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","file":"Aggregation.js","sourcesContent":["/**\n * @file Cursor.js - based on Monglo#Cursor ({@link https://github.com/Monglo}) by Christian Sullivan <cs@euforic.co> | Copyright (c) 2012\n * @version 1.0.0\n * \n * @author Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @copyright 2016 Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @license MIT Licensed\n */\n\nvar _ = require(\"lodash\"),\n    Logger = require(\"jsw-logger\");\n    \nvar logger = null;\n\nvar stages = {\n    '$project': false,\n    '$match': false,\n    '$redact': false,\n    '$limit': false,\n    '$skip': false,\n    '$unwind': false,\n    '$group': true,\n    '$sample': false,\n    '$sort': false,\n    '$geoNear': false,\n    '$lookup': false,\n    '$out': false,\n    '$indexStats': false\n};\n\nvar group_operators = {\n    $sum: function(documents, new_id, new_field, value, isCount) {\n        var new_docs = {};\n        \n        for (let i = 0; i < documents.length; i++) {\n            let doc = documents[i];\n            let val = value;\n            \n            if (!isCount) {\n                val = doc[value.substr(1, value.length)] || 0;\n            }\n            \n            if (_.hasIn(doc, new_id)) {\n                let _id = doc[new_id];\n                \n                if (!_.hasIn(new_docs, _id)) {\n                    new_docs[_id] = {\n                        _id: _id,\n                        [new_field]: _.toNumber(val)\n                    };\n                } else {\n                    new_docs[_id][new_field] += _.toNumber(val);\n                }\n            }\n        }\n        \n        return new_docs;\n    },\n    \n    $avg: function(documents, new_id, new_field, value, isCount) {\n        var new_docs = {};\n        \n        for (let i = 0; i < documents.length; i++) {\n            let doc = documents[i];\n            let val = value;\n            \n            if (!isCount) {\n                val = doc[value.substr(1, value.length)] || 0;\n            }\n            \n            if (_.hasIn(doc, new_id) || _.isNull(new_id)) {\n                let _id = doc[new_id] || null;\n                \n                if (!_.hasIn(new_docs, _id)) {\n                    new_docs[_id] = {\n                        _id: _id,\n                        [new_field]: _.toNumber(val),\n                        __COUNT__: 1\n                    };\n                } else {\n                    new_docs[_id][new_field] += _.toNumber(val);\n                    new_docs[_id].__COUNT__++;\n                }\n            }\n        }\n        \n        for (let key in new_docs) {\n            new_docs[key][new_field] = new_docs[key][new_field] / new_docs[key].__COUNT__;\n            delete new_docs[key].__COUNT__;\n        }\n        \n        return new_docs;\n    } \n};\n\nvar do_single_group = function(group_id, group_stage, documents) {\n    // var operators = {};\n    \n    let docs = {};\n    \n    for (let field in group_stage) {\n        if (field !== '_id') {\n            // handle group field\n            // let group_key = key;\n            let group_field = group_stage[field];\n            \n            for (let key in group_field) {\n                if (!_.hasIn(group_operators, key)) logger.throw(`Unknown accumulator operator \"${key}\" for group stage`);\n                \n                // loop through all documents\n                // var new_docs = {};\n                // for (let i = 0; i < documents.length; i++) {\n                //     let doc = documents[i];\n                    \n                //     if (_.hasIn(doc, group_id)) {\n                //         let _id = doc[group_id];\n                        \n                //         if (!_.hasIn(new_docs, _id)) {\n                //             new_docs[_id] = {\n                //                 _id: _id,\n                //                 [new_field]: value\n                //             };\n                //         } else {\n                //             new_docs[_id][new_field] += value;\n                //         }\n                //     }\n                // }\n                \n                // if (!_.hasIn(operators, key)) operators[key] = [];\n                \n                // operators[key].push({\n                //     new_field: field,\n                //     value: group_field[key]\n                // });\n                \n                let count = true;\n                if (_.isString(group_field[key])) {\n                    if (group_field[key].substr(0, 1) !== '$') logger.throw(\"Field names references in a right side assignement must be preceded by '$'\");\n                    \n                    if (!_.isFinite(_.toNumber(group_field[key]))) {\n                        count = false;\n                    }\n                }\n                \n                let operator = group_operators[key];\n                \n                _.merge(docs, operator(documents, group_id, field, group_field[key], count));\n                \n                break;\n            }\n        }\n    }\n    \n    return _.values(docs);\n};\n\nvar do_complex_group = function() {\n    \n};\n\nvar do_group = function(documents, group_stage) {\n    if (!_.hasIn(group_stage, '_id')) logger.throw('The field \"_id\" is required in the \"$group\" stage');\n    \n    let new_id = group_stage['_id'];\n                \n    if (_.isPlainObject(new_id)) {\n        // complex_id\n        // do_complex_group();\n    } else {\n        // single_id\n        return do_single_group(new_id, group_stage, documents);\n    }\n};\n\nclass Aggregation {\n    constructor(pipeline) {\n        logger = Logger.instance;\n        \n        this.pipeline = pipeline;\n    }\n    \n    aggregate(collection) {\n        for (let i = 0; i < this.pipeline.length; i++) {\n            let stage = this.pipeline[i];\n            \n            for (let key in stage) {\n                switch (key) {\n                    case '$group':\n                        return do_group(collection.docs, stage[key]);\n                }\n            }\n        }\n    }\n    \n    validStage(stage) {\n        if (!_.hasIn(stages, stage)) return logger.throw(`Unknown stage \"${stage}\"`);\n        \n        if (stages[stage] === false) return logger.throw(`Unsupported stage \"${stage}\"`);\n        \n        return true;\n    }\n}\n\n\nmodule.exports = Aggregation;"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Aggregation.js"],"names":[],"mappings":"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","file":"Aggregation.js","sourcesContent":["/**\n * @file Cursor.js - based on Monglo#Cursor ({@link https://github.com/Monglo}) by Christian Sullivan <cs@euforic.co> | Copyright (c) 2012\n * @version 1.0.0\n * \n * @author Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @copyright 2016 Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @license MIT Licensed\n */\n\nvar _ = require(\"lodash\"),\n    Logger = require(\"jsw-logger\"),\n    Cursor = require(\"./Cursor\"),\n    Selector = require(\"./Selector\");\n    \nvar logger = null;\n\nvar stages = {\n    '$project': false,\n    '$match': true,\n    '$redact': false,\n    '$limit': false,\n    '$skip': false,\n    '$unwind': false,\n    '$group': true,\n    '$sample': false,\n    '$sort': true,\n    '$geoNear': false,\n    '$lookup': false,\n    '$out': false,\n    '$indexStats': false\n};\n\nvar group_operators = {\n    $sum: function(documents, new_id, new_field, value, isCount) {\n        var new_docs = {};\n        \n        for (let i = 0; i < documents.length; i++) {\n            let doc = documents[i];\n            let val = value;\n            \n            if (!isCount) {\n                val = doc[value.substr(1, value.length)] || 0;\n            }\n            \n            if (_.hasIn(doc, new_id)) {\n                let _id = doc[new_id];\n                \n                if (!_.hasIn(new_docs, _id)) {\n                    new_docs[_id] = {\n                        _id: _id,\n                        [new_field]: _.toNumber(val)\n                    };\n                } else {\n                    new_docs[_id][new_field] += _.toNumber(val);\n                }\n            }\n        }\n        \n        return new_docs;\n    },\n    \n    $avg: function(documents, new_id, new_field, value, isCount) {\n        var new_docs = {};\n        \n        for (let i = 0; i < documents.length; i++) {\n            let doc = documents[i];\n            let val = value;\n            \n            if (!isCount) {\n                val = doc[value.substr(1, value.length)] || 0;\n            }\n            \n            if (_.hasIn(doc, new_id) || _.isNull(new_id)) {\n                let _id = doc[new_id] || null;\n                \n                if (!_.hasIn(new_docs, _id)) {\n                    new_docs[_id] = {\n                        _id: _id,\n                        [new_field]: _.toNumber(val),\n                        __COUNT__: 1\n                    };\n                } else {\n                    new_docs[_id][new_field] += _.toNumber(val);\n                    new_docs[_id].__COUNT__++;\n                }\n            }\n        }\n        \n        for (let key in new_docs) {\n            new_docs[key][new_field] = new_docs[key][new_field] / new_docs[key].__COUNT__;\n            delete new_docs[key].__COUNT__;\n        }\n        \n        return new_docs;\n    } \n};\n\nvar do_single_group = function(group_id, group_stage, documents) {\n    // var operators = {};\n    \n    let docs = {};\n    \n    for (let field in group_stage) {\n        if (field !== '_id') {\n            // handle group field\n            // let group_key = key;\n            let group_field = group_stage[field];\n            \n            for (let key in group_field) {\n                if (!_.hasIn(group_operators, key)) logger.throw(`Unknown accumulator operator \"${key}\" for group stage`);\n                \n                // loop through all documents\n                // var new_docs = {};\n                // for (let i = 0; i < documents.length; i++) {\n                //     let doc = documents[i];\n                    \n                //     if (_.hasIn(doc, group_id)) {\n                //         let _id = doc[group_id];\n                        \n                //         if (!_.hasIn(new_docs, _id)) {\n                //             new_docs[_id] = {\n                //                 _id: _id,\n                //                 [new_field]: value\n                //             };\n                //         } else {\n                //             new_docs[_id][new_field] += value;\n                //         }\n                //     }\n                // }\n                \n                // if (!_.hasIn(operators, key)) operators[key] = [];\n                \n                // operators[key].push({\n                //     new_field: field,\n                //     value: group_field[key]\n                // });\n                \n                let count = true;\n                if (_.isString(group_field[key])) {\n                    if (group_field[key].substr(0, 1) !== '$') logger.throw(\"Field names references in a right side assignement must be preceded by '$'\");\n                    \n                    if (!_.isFinite(_.toNumber(group_field[key]))) {\n                        count = false;\n                    }\n                }\n                \n                let operator = group_operators[key];\n                \n                _.merge(docs, operator(documents, group_id, field, group_field[key], count));\n                \n                break;\n            }\n        }\n    }\n    \n    return _.values(docs);\n};\n\nvar do_complex_group = function() {\n    \n};\n\nvar do_sort = function(documents, sort_stage) {\n    return documents.sort(new Selector(sort_stage, Selector.SORT_SELECTOR));\n};\n\nvar do_match = function(documents, match_stage) {\n    var cursor = new Cursor(documents, match_stage);\n    \n    return cursor.fetch();\n};\n\nvar do_group = function(documents, group_stage) {\n    if (!_.hasIn(group_stage, '_id')) logger.throw('The field \"_id\" is required in the \"$group\" stage');\n    \n    let new_id = group_stage['_id'];\n    \n    if (!_.isNull(new_id)) {\n        if (new_id.substr(0, 1) !== '$') {\n            logger.throw(\"Field names references in a right side assignement must be preceded by '$'\");\n        } else {\n            new_id = new_id.substr(1, new_id.length);\n        }\n    }\n                \n    if (_.isPlainObject(new_id)) {\n        // complex_id\n        // do_complex_group();\n    } else {\n        // single_id\n        return do_single_group(new_id, group_stage, documents);\n    }\n};\n\nclass Aggregation {\n    constructor(pipeline) {\n        logger = Logger.instance;\n        \n        this.pipeline = pipeline;\n    }\n    \n    aggregate(collection) {\n        var docs = collection.docs;\n        \n        for (let i = 0; i < this.pipeline.length; i++) {\n            let stage = this.pipeline[i];\n            \n            for (let key in stage) {\n                switch (key) {\n                    case '$match':\n                        docs = do_match(docs, stage[key]);\n                        \n                        break;\n                    case '$group':\n                        docs = do_group(docs, stage[key]);\n                        \n                        break;\n                    case '$sort':\n                        docs = do_sort(docs, stage[key]);\n                        \n                        break;\n                }\n            }\n        }\n        \n        return docs;    // move to cursor\n    }\n    \n    validStage(stage) {\n        if (!_.hasIn(stages, stage)) return logger.throw(`Unknown stage \"${stage}\"`);\n        \n        if (stages[stage] === false) return logger.throw(`Unsupported stage \"${stage}\"`);\n        \n        return true;\n    }\n}\n\n\nmodule.exports = Aggregation;"]} diff --git a/lib/Collection.js b/lib/Collection.js index 1350921..1d940f0 100644 --- a/lib/Collection.js +++ b/lib/Collection.js @@ -203,8 +203,7 @@ Collection.prototype.find = function (selection, fields, options, callback) { options = params.options; callback = params.callback; - // callback for backward compatibility - var cursor = new Cursor(this.db, this, selection, fields, options); + var cursor = new Cursor(this.docs, selection, fields, options); /** * "find" event. @@ -262,7 +261,7 @@ Collection.prototype.findOne = function (selection, fields, options, callback) { options = params.options; callback = params.callback; - var cursor = new Cursor(this.db, this, selection, fields, options); + var cursor = new Cursor(this.docs, selection, fields, options); /** * "findOne" event. @@ -1254,4 +1253,4 @@ var _ensureFindParams = function _ensureFindParams(params) { return params; }; -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Collection.js"],"names":[],"mappings":"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","file":"Collection.js","sourcesContent":["/**\n * @file Collection.js - based on Monglo#Collection ({@link https://github.com/Monglo}) by Christian Sullivan <cs@euforic.co> | Copyright (c) 2012\n * @version 1.0.0\n * \n * @author Eduardo Astolfi <eastolfi91@gmail.com>\n * @copyright 2016 Eduardo Astolfi <eastolfi91@gmail.com>\n * @license MIT Licensed\n */\n\nvar Logger = require(\"jsw-logger\"),\n    EventEmitter = require(\"./utils/EventEmitter\"),\n    _ = require(\"lodash\"),\n    Aggregation = require(\"./Aggregation\"),\n    Cursor = require(\"./Cursor\"),\n    ObjectId = require('./ObjectId'),\n    Selector = require(\"./Selector\"),\n    SelectorMatcher = require(\"./SelectorMatcher\");\n    \nvar logger = null;\n    \n/**\n * Collection\n * \n * @module Collection\n * @constructor\n * @since 0.0.1\n * \n * @classdesc Collection class that maps a MongoDB-like collection\n * \n * @param {MongoPortable} db - Additional options\n * @param {String} collectionName - The name of the collection\n * @param {Object} [options] - Database object\n * \n * @param {Object} [options.pkFactory=null] - Object overriding the basic \"ObjectId\" primary key generation.\n * \n */\nvar database = null;\nclass Collection extends EventEmitter {\n// var Collection = function(db, collectionName, options) {\n    constructor(db, collectionName, options) {\n        super();\n        \n        if (!(this instanceof Collection)) return new Collection(db, collectionName, options);\n        \n        logger = Logger.instance;\n    \n        if (_.isNil(db)) logger.throw(\"db parameter required\");\n        \n        if (_.isNil(collectionName)) logger.throw(\"collectionName parameter required\");\n        \n        if (_.isNil(options) || !_.isPlainObject(options)) options = {};\n        \n        Collection.checkCollectionName(collectionName);\n    \n        // this.db = db;\n        database = db;\n        this.name = collectionName;\n        this.databaseName = db.databaseName;\n        this.fullName = this.databaseName + '.' + this.name;\n        this.docs = [];\n        this.doc_indexes = {};\n        this.snapshots = [];\n        this.opts = {}; // Default options\n        \n        _.merge(this.opts, options);\n        \n        // this.emit = db.emit;\n    }\n    \n    emit(name, args, cb) {\n        super.emit(name, args, cb, database._stores);\n    }\n}\n\n// TODO enforce rule that field names can't start with '$' or contain '.'\n// (real mongodb does in fact enforce this)\n// TODO possibly enforce that 'undefined' does not appear (we assume\n// this in our handling of null and $exists)\n/**\n * Inserts a document into the collection\n * \n * @method Collection#insert\n * \n * @param {Object} doc - Document to be inserted\n * @param {Object} [options] - Additional options\n * \n * @param {Boolean} [options.chain=false] - If set to \"true\" returns this instance, so it can be chained with other methods\n * \n * @param {Function} [callback=null] Callback function to be called at the end with the results\n * \n * @returns {Object|Collection} If \"options.chain\" set to \"true\" returns this instance, otherwise returns the inserted document\n */\nCollection.prototype.insert = function (doc, options, callback) {\n    if (_.isNil(doc)) logger.throw(\"doc parameter required\");\n    \n    if (!_.isPlainObject(doc)) logger.throw(\"doc must be an object\");\n    \n    if (_.isNil(options)) options = {};\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    // Creating a safe copy of the document\n    var _doc = _.cloneDeep(doc);\n\n    // If the document comes with a number ID, parse it to String\n    if (_.isNumber(_doc._id)) {\n        _doc._id = _.toString(_doc._id);\n    }\n\n    if (_.isNil(_doc._id) || (!_doc._id instanceof ObjectId && (!_.isString(_doc._id) || !_doc._id.length))) {\n        _doc._id = new ObjectId();\n    }\n\n    // Add options to more dates\n    _doc.timestamp = new ObjectId().generationTime;\n    \n    // Reverse\n    this.doc_indexes[_.toString(_doc._id)] = this.docs.length;\n    this.docs.push(_doc);\n    \n    /**\n     * \"insert\" event.\n     *\n     * @event MongoPortable~insert\n     * \n     * @param {Object} collection - Information about the collection\n     * @param {Object} doc - Information about the document inserted\n     */\n    this.emit(\n        'insert',\n        {\n            collection: this,\n            doc: _doc\n        }\n    );\n\n    if (callback) callback(null, _doc);\n\n    if (options.chain) return this;\n    \n    return _doc;\n};\n\n/**\n * Finds all matching documents\n * \n * @method Collection#find\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object|Array|String} [fields={}] - The fields of the document to show\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.skip] - Number of documents to be skipped\n * @param {Number} [options.limit] - Max number of documents to display\n * @param {Object|Array|String} [options.fields] - Same as \"fields\" parameter (if both passed, \"options.fields\" will be ignored)\n * @param {Boolean} [options.forceFetch=false] - If set to'\"true\" returns the array of documents already fetched\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Array|Cursor} If \"options.forceFetch\" set to true returns the array of documents, otherwise returns a cursor\n */\nCollection.prototype.find = function (selection, fields, options, callback) {\n    let params = _ensureFindParams({\n        selection: selection, \n        fields: fields,\n        options: options, \n        callback: callback\n    });\n    \n    selection = params.selection;\n    fields = params.fields;\n    options = params.options;\n    callback = params.callback;\n    \n    // callback for backward compatibility\n    var cursor = new Cursor(this.db, this, selection, fields, options);\n\n    /**\n     * \"find\" event.\n     *\n     * @event MongoPortable~find\n     * \n     * @property {Object} collection - Information about the collection\n     * @property {Object} selector - The selection of the query\n     * @property {Object} fields - The fields showed in the query\n     */\n    this.emit(\n        'find',\n        {\n            collection: this,\n            selector: selection,\n            fields: fields\n        }\n    );\n    \n    // Pass the cursor fetched to the callback\n    // Add [options.noFetchCallback = true]\n    if (callback) callback(null, cursor.fetch());\n\n    if (options.forceFetch) {\n        return cursor.fetch();\n    } else {\n        return cursor;\n    }\n};\n\n/**\n * Finds the first matching document\n * \n * @method Collection#findOne\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object|Array|String} [fields={}] - The fields of the document to show\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.skip] - Number of documents to be skipped\n * @param {Number} [options.limit] - Max number of documents to display\n * @param {Object|Array|String} [options.fields] - Same as \"fields\" parameter (if both passed, \"options.fields\" will be ignored)\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} Returns the first matching document of the collection\n */\nCollection.prototype.findOne = function (selection, fields, options, callback) {\n    let params = _ensureFindParams({\n        selection: selection, \n        fields: fields,\n        options: options, \n        callback: callback\n    });\n    \n    selection = params.selection;\n    fields = params.fields;\n    options = params.options;\n    callback = params.callback;\n    \n    var cursor = new Cursor(this.db, this, selection, fields, options);\n\n    /**\n     * \"findOne\" event.\n     *\n     * @event MongoPortable~findOne\n     * \n     * @property {Object} collection - Information about the collection\n     * @property {Object} selector - The selection of the query\n     * @property {Object} fields - The fields showed in the query\n     */\n    this.emit(\n        'findOne',\n        {\n            collection: this,\n            selector: selection,\n            fields: fields\n        }\n    );\n    \n    var res = null;\n    \n    if (cursor.hasNext()) {\n        res = cursor.next();\n    }\n    \n    // Pass the cursor fetched to the callback\n    // Add [options.noFetchCallback = true]\n    if (callback) callback(null, res);\n    \n    return res;\n};\n\n\n/**\n * Updates one or many documents\n * \n * @method Collection#update\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object} [update={}] - The update operation\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.updateAsMongo=true] - By default: \n *      If the [update] object contains update operator modifiers, such as those using the \"$set\" modifier, then:\n *          <ul>\n *              <li>The [update] object must contain only update operator expressions</li>\n *              <li>The Collection#update method updates only the corresponding fields in the document</li>\n *          <ul>\n *      If the [update] object contains only \"field: value\" expressions, then:\n *          <ul>\n *              <li>The Collection#update method replaces the matching document with the [update] object. The Collection#update method does not replace the \"_id\" value</li>\n *              <li>Collection#update cannot update multiple documents</li>\n *          <ul>\n * \n * @param {Number} [options.override=false] - Replaces the whole document (only apllies when [updateAsMongo=false])\n * @param {Number} [options.upsert=false] - Creates a new document when no document matches the query criteria\n * @param {Number} [options.multi=false] - Updates multiple documents that meet the criteria\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} Object with the update/insert (if upsert=true) information\n */\nCollection.prototype.update = function (selection, update, options, callback) {\n    if (_.isNil(selection)) selection = {};\n    \n    if (_.isNil(update)) logger.throw(\"You must specify the update operation\");\n    \n    if (_.isNil(options)) {\n        options = {\n            skip: 0,\n            limit: 15   // for no limit pass [options.limit = -1]\n        };\n    }\n    \n    if (_.isFunction(selection)) logger.throw(\"You must specify the update operation\");\n    \n    if (_.isFunction(update)) logger.throw(\"You must specify the update operation\");\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    // Check special case where we are using an objectId\n    if(selection instanceof ObjectId) {\n        selection = {\n            _id: selection\n        };\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n\n    var res = null;\n\n    var docs = null;\n    if (options.multi) {\n        docs = this.find(selection, null, { forceFetch: true });\n    } else {\n        docs = this.findOne(selection);\n    }\n    \n    if (_.isNil(docs)) {\n        docs = [];\n    }\n    \n    if (!_.isArray(docs)) {\n        docs = [docs];\n    }\n    \n    if (docs.length === 0) {\n        if (options.upsert) {\n            var inserted = this.insert(update);\n\n            res = {\n                updated: {\n                    documents: null,\n                    count: 0\n                },\n                inserted: {\n                    documents: [inserted],\n                    count: 1\n                }\n            };\n        } else {\n            // No documents found\n            res = {\n                updated: {\n                    documents: null,\n                    count: 0\n                },\n                inserted: {\n                    documents: null,\n                    count: 0\n                }\n            };\n        }\n    } else {\n        var updatedDocs = [];\n        \n        for (var i = 0; i < docs.length; i++) {\n            var doc = docs[i];\n            \n            var override = null;\n            \n            var hasModifier = false;\n            \n            for (let key in update) {\n                // IE7 doesn't support indexing into strings (eg, key[0] or key.indexOf('$') ), so use substr.\n                // Testing over the first letter:\n                //      Bests result with 1e8 loops => key[0](~3s) > substr(~5s) > regexp(~6s) > indexOf(~16s)\n                \n                var modifier = (key.substr(0, 1) === '$');\n                if (modifier) {\n                    hasModifier = true;\n                }\n                \n                if (options.updateAsMongo) {\n                    if (hasModifier && !modifier) logger.throw(\"All update fields must be an update operator\");\n                    \n                    if (!hasModifier && options.multi) logger.throw(\"You can not update several documents when no update operators are included\");\n                    \n                    if (hasModifier) override = false;\n                    \n                    if (!hasModifier) override = true;\n                } else {\n                    override = !!options.override;\n                }\n            }\n            \n            var _docUpdate = null;\n            \n            if (override) {\n                // Overrides the document except for the \"_id\"\n                _docUpdate = {\n                    _id: doc._id\n                };\n                \n                // Must ignore fields starting with '$', '.'...\n                for (let key in update) {\n                    if (key.substr(0, 1) === '$' || /\\./g.test(key)) {\n                        logger.warn(`The field ${key} can not begin with '$' or contain '.'`);\n                    } else {\n                        _docUpdate[key] = update[key];\n                    }\n                }\n            } else {\n                _docUpdate = _.cloneDeep(doc);\n                \n                for (let key in update) {\n                    let val = update[key];\n                    \n                    if (key.substr(0, 1) === '$') {\n                        _docUpdate = _applyModifier(_docUpdate, key, val);\n                    } else {\n                        if (!_.isNil(_docUpdate[key])) {\n                            if (key !== '_id') {\n                                _docUpdate[key] = val;\n                            } else {\n                                logger.warn(\"The field '_id' can not be updated\");\n                            }\n                        } else {\n                            logger.warn(`The document does not contains the field ${key}`);\n                        }\n                    }\n                }\n            }\n            \n            updatedDocs.push(_docUpdate);\n            \n            let idx = this.doc_indexes[_docUpdate._id];\n            this.docs[idx] = _docUpdate;\n        }\n        \n        /**\n         * \"update\" event.\n         *\n         * @event MongoPortable~update\n         * \n         * @property {Object} collection - Information about the collection\n         * @property {Object} selector - The selection of the query\n         * @property {Object} modifier - The modifier used in the query\n         * @property {Object} docs - The updated/inserted documents information\n         */\n        this.emit(\n            'update',\n            {\n                collection: this,\n                selector: selection,\n                modifier: update,\n                docs: updatedDocs\n            }\n        );\n        \n        res = {\n            updated: {\n                documents: updatedDocs,\n                count: updatedDocs.length\n            },\n            inserted: {\n                documents: null,\n                count: 0\n            }\n        };\n    }\n    \n    \n    if (callback) callback(null, res);\n    \n    return res;\n};\n\nvar _applyModifier = function(_docUpdate, key, val) {\n    var doc = _.cloneDeep(_docUpdate);\n    // var mod = _modifiers[key];\n                        \n    if (!_modifiers[key]) {\n        logger.throw(`Invalid modifier specified: ${key}`);\n    }\n    \n    for (var keypath in val) {\n        var value = val[keypath];\n        var keyparts = keypath.split('.');\n        \n        _modify(doc, keyparts, value, key);\n        \n        // var no_create = !!Collection._noCreateModifiers[key];\n        // var forbid_array = (key === \"$rename\");\n        // var target = Collection._findModTarget(_docUpdate, keyparts, no_create, forbid_array);\n        // var field = keyparts.pop();\n\n        // mod(target, field, value, keypath, _docUpdate);\n    }\n    \n    return doc;\n};\n\nvar _modify = function(document, keyparts, value, key, level = 0) {\n    for (let i = level; i < keyparts.length; i++) {\n        let path = keyparts[i];\n        let isNumeric = /^[0-9]+$/.test(path);\n        let target = document[path];\n        \n        var create = _.hasIn(Collection._noCreateModifiers, key) ? false : true;\n        if (!create && (!_.isObject(document) || _.isNil(target))) {\n            logger.throw(`The element \"${path}\" must exists in \"${JSON.stringify(document)}\"`);\n        }\n        \n        if (_.isArray(document)) {\n            // Do not allow $rename on arrays\n            if (key === \"$rename\") return null;\n            \n            // Only let the use of \"arrayfield.<numeric_index>.subfield\"\n            if (isNumeric) {\n                path = _.toNumber(path);\n            } else {\n                logger.throw(`The field \"${path}\" can not be appended to an array`);\n            }\n            \n            // Fill the array to the desired length\n            while (document.length < path) {\n                document.push(null);\n            }\n        }\n        \n        if (i < keyparts.length - 1) {\n            if (_.isNil(target)) {\n                // If we are accessing with \"arrayField.<numeric_index>\"\n                if (_.isFinite(_.toNumber(keyparts[i + 1]))) {  //  || keyparts[i + 1] === '$'  // TODO \"arrayField.$\"\n                    target = [];\n                } else {\n                    target = {};\n                }\n            }\n            \n            document[path] = _modify(target, keyparts, value, key, level + 1);\n\n            return document;\n        } else {\n            _modifiers[key](document, path, value);\n            \n            return document;\n        }\n    }\n};\n\n/**\n * Removes one or many documents\n * \n * @method Collection#remove\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.justOne=false] - Deletes the first occurrence of the selection\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} Object with the deleted documents\n */\nCollection.prototype.remove = function (selection, options, callback) {\n    if (_.isNil(selection)) selection = {};\n    \n    if (_.isFunction(selection)) {\n        callback = selection;\n        selection = {};\n    }\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    if (_.isNil(options)) options = { justOne: false };\n    \n    // If we are not passing a selection and we are not removing just one, is the same as a drop\n    if (Object.size(selection) === 0 && !options.justOne) return this.drop(options, callback);\n    \n    // Check special case where we are using an objectId\n    if(selection instanceof ObjectId) {\n        selection = {\n            _id: selection\n        };\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    var cursor = this.find(selection);\n    \n    var docs = [];\n    cursor.forEach(doc => {\n        var idx = this.doc_indexes[doc._id];\n        \n        delete this.doc_indexes[doc._id];\n        this.docs.splice(idx, 1);\n        \n        docs.push(doc);\n    });\n    \n    /**\n     * \"remove\" event.\n     *\n     * @event MongoPortable~remove\n     * \n     * @property {Object} collection - Information about the collection\n     * @property {Object} selector - The selection of the query\n     * @property {Object} docs - The deleted documents information\n     */\n    this.emit(\n        'remove',\n        {\n            collection: this,\n            selector: selection,\n            docs: docs\n        }\n    );\n    \n    if (callback) callback(null, docs);\n    \n    return docs;\n};\n\n/**\n * Alias for {@link Collection#remove}\n * \n * @method Collection#delete\n */\nCollection.prototype.delete = function (selection, options, callback) {\n    return this.remove(selection, options, callback);\n};\n \n /**\n * Alias for {@link Collection#remove}\n * \n * @method Collection#destroy\n */\nCollection.prototype.destroy = function (selection, options, callback) {\n    return this.remove(selection, options, callback);\n};\n\n/**\n * Drops a collection\n * \n * @method Collection#drop\n * \n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.dropIndexes=false] - True if we want to drop the indexes too\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} True when the collection is dropped\n */\nCollection.prototype.drop = function(options, callback) {\n    if (_.isNil(options)) options = {};\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    this.doc_indexes = {};\n    this.docs = [];\n    \n    if (options.dropIndexes) {} // TODO\n    \n    this.emit(\n        'dropCollection',\n        {\n            collection: this,\n            indexes: !!options.dropIndexes\n        }\n    );\n    \n    if (callback) callback(null, true);\n    \n    return true;\n};\n\n/**\n * Insert or update a document. If the document has an \"_id\" is an update (with upsert), if not is an insert.\n * \n * @method Collection#save\n * \n * @param {Object} doc - Document to be inserted/updated\n * \n * @param {Number} [options.dropIndexes=false] - True if we want to drop the indexes too\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} True when the collection is dropped\n */\nCollection.prototype.save = function(doc, options, callback) {\n    if (_.isNil(doc) || _.isFunction(doc)) logger.throw(\"You must pass a document\");\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n\n    if (_.hasIn(doc, '_id')) {\n        options.upsert = true;\n        \n        return this.update(\n            { _id: doc._id },\n            doc,\n            options,\n            callback\n        );\n    } else {\n        return this.insert(doc, options, callback);\n    }\n};\n\n/**\n* @ignore\n*/\nCollection.prototype.ensureIndex = function() {\n    //TODO Implement EnsureIndex\n    logger.throw('Collection#ensureIndex unimplemented by driver');\n};\n\n// TODO document (at some point)\n// TODO test\n// TODO obviously this particular implementation will not be very efficient\n/**\n* @ignore\n*/\nCollection.prototype.backup = function (backupID, callback) {\n    if (_.isFunction(backupID)) {\n        callback = backupID;\n        backupID = new ObjectId().toString();\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n\n    this.snapshots[backupID] = _.cloneDeep(this.docs);\n    this.emit(\n        'snapshot',\n        {\n            collection: this,\n            backupID: backupID,\n            documents: this.snapshots[backupID] \n        }\n    );\n\n    var result = {\n        backupID: backupID,\n        documents: this.snapshots[backupID]\n    };\n    \n    if (callback) callback(null, result);\n\n    return result;\n};\n\n// Lists available Backups\n/**\n* @ignore\n*/\nCollection.prototype.backups = function (callback) {\n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    var backups = [];\n\n    for (let id in this.snapshots) {\n        backups.push({id: id, documents: this.snapshots[id]});\n    }\n\n    if (callback) callback(null, backups);\n\n    return backups;\n};\n\n// Lists available Backups\n/**\n* @ignore\n*/\nCollection.prototype.removeBackup = function (backupID, callback) {\n    if (_.isFunction(backupID)) {\n        callback = backupID;\n        backupID = null;\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    let result = false;\n    \n    if (backupID) {\n        delete this.snapshots[_.toString(backupID)];\n        \n        result = backupID;\n    } else {\n        this.snapshots = {};\n        \n        result = true;\n    }\n    \n    if (callback) callback(null, result);\n\n    return result;\n};\n\n\n// Restore the snapshot. If no snapshot exists, raise an exception;\n/**\n* @ignore\n*/\nCollection.prototype.restore = function (backupID, callback) {\n    if (_.isFunction(backupID)) {\n        callback = backupID;\n        backupID = null;\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    var snapshotCount = Object.size(this.snapshots);\n    var backupData = null;\n\n    if (snapshotCount === 0) {\n        logger.throw(\"There is no snapshots\");\n    } else {\n        if (!backupID) {\n            if (snapshotCount === 1) {\n                logger.info(\"No backupID passed. Restoring the only snapshot\");\n                \n                // Retrieve the only snapshot\n                for (let key in this.snapshots) backupID = key;\n            } else {\n                logger.throw(\"The are several snapshots. Please specify one backupID\");\n            }\n        }\n    }\n    \n    backupData = this.snapshots[backupID];\n            \n    if (!backupData) {\n        logger.throw(`Unknown Backup ID: ${backupID}`);\n    }\n\n    this.docs = backupData;\n    this.emit(\n        'restore',\n        {\n            collection: this,\n            backupID: backupID\n        }\n    );\n\n    if (callback) callback(null);\n\n    return this;\n};\n\n/**\n * Calculates aggregate values for the data in a collection\n * \n * @method Collection#aggregate\n * \n * @param {Array} pipeline - A sequence of data aggregation operations or stages\n * @param {Object} [options] - Additional options\n * \n * @param {Boolean} [options.forceFetch=false] - If set to'\"true\" returns the array of documents already fetched\n * \n * @returns {Array|Cursor} If \"options.forceFetch\" set to true returns the array of documents, otherwise returns a cursor\n */\nCollection.prototype.aggregate = function(pipeline, options = { forceFetch: false }) {\n    if (_.isNil(pipeline) || !_.isArray(pipeline)) logger.throw('The \"pipeline\" param must be an array');\n    \n    var aggregation = new Aggregation(pipeline);\n    \n    for (let i = 0; i < pipeline.length; i++) {\n        let stage = pipeline[i];\n        \n        for (let key in stage) {\n            if (key.substr(0, 1) !== '$') logger.throw(\"The pipeline stages must begin with '$'\");\n            \n            if (!aggregation.validStage(key)) logger.throw(`Invalid stage \"${key}\"`);\n            \n            break;\n        }\n    }\n    \n    var result = aggregation.aggregate(this);\n    \n    return result;  // change to cursor\n};\n\n/**\n* @ignore\n*/\nCollection._noCreateModifiers = {\n    $unset: true,\n    $pop: true,\n    $rename: true,\n    $pull: true,\n    $pullAll: true\n};\n\n/**\n* @ignore\n*/\nvar _modifiers = {\n    $inc: function (target, field, arg) {\n        if (!_.isNumber(arg)) {\n            logger.throw(\"Modifier $inc allowed for numbers only\");\n        }\n\n        if (field in target) {\n            if (!_.isNumber(target[field])) {\n                logger.throw(\"Cannot apply $inc modifier to non-number\");\n            }\n\n            target[field] += arg;\n        } else {\n            target[field] = arg;\n        }\n    },\n\n    $set: function (target, field, arg) {\n        target[field] = _.cloneDeep(arg);\n    },\n\n    $unset: function (target, field, arg) {\n        if (!_.isNil(target)) {\n            if (_.isArray(target)) {\n                if (field in target) {\n                    target[field] = null;\n                }\n            } else {\n                delete target[field];\n            }\n        }\n    },\n\n    $push: function (target, field, arg) {\n        var x = target[field];\n\n        if (_.isNil(x)) {\n            target[field] = [arg];\n        } else if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $push modifier to non-array\");\n        } else {\n            x.push(_.cloneDeep(arg));\n        }\n    },\n\n    $pushAll: function (target, field, arg) {\n        var x = target[field];\n\n        if (_.isNil(x)) {\n            target[field] = arg;\n        } else if (!_.isArray(x)) {\n            logger.throw(\"Modifier $pushAll/pullAll allowed for arrays only\");\n        } else {\n            for (var i = 0; i < arg.length; i++) {\n                x.push(arg[i]);\n            }\n        }\n    },\n\n    $addToSet: function (target, field, arg) {\n        var x = target[field];\n\n        if (_.isNil(x)) {\n            target[field] = [arg];\n        } else if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $addToSet modifier to non-array\");\n        } else {\n            let isEach = false;\n            if (_.isPlainObject(arg)) {\n                for (let k in arg) {\n                    if (k === \"$each\") {\n                        isEach = true;\n                    }\n                    \n                    break;\n                }\n            }\n\n            let values = isEach ? arg[\"$each\"] : [arg];\n            _.forEach(values, function (value) {\n                for (let i = 0; i < x.length; i++) {\n                    if (SelectorMatcher.equal(value, x[i])) return;\n                }\n\n                x.push(value);\n            });\n        }\n    },\n\n    $pop: function (target, field, arg) {\n        if (_.isNil(target) || _.isNil(target[field])) return;\n\n        var x = target[field];\n\n        if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $pop modifier to non-array\");\n        } else {\n            if (_.isNumber(arg) && arg < 0) {\n                x.splice(0, 1);\n            } else {\n                x.pop();\n            }\n        }\n    },\n\n    $pull: function (target, field, arg) {\n        if (_.isNil(target) || _.isNil(target[field])) return;\n\n        var x = target[field];\n\n        if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $pull/pullAll modifier to non-array\");\n        } else {\n            var out = [];\n            \n            if (typeof arg === \"object\" && !(arg instanceof Array)) {\n                // XXX would be much nicer to compile this once, rather than\n                // for each document we modify.. but usually we're not\n                // modifying that many documents, so we'll let it slide for\n                // now\n\n                // XXX _compileSelector isn't up for the job, because we need\n                // to permit stuff like {$pull: {a: {$gt: 4}}}.. something\n                // like {$gt: 4} is not normally a complete selector.\n                var match = new Selector({\n                    \"__matching__\": arg\n                });\n                for (var i = 0; i < x.length; i++) {\n                    var _doc_ = {\n                        __matching__: x[i]\n                    };\n                    if (!match.test(_doc_)) {\n                        out.push(x[i]);\n                    }\n                }\n            } else {\n                for (var i = 0; i < x.length; i++) {\n                    if (!SelectorMatcher.equal(x[i], arg)) {\n                        out.push(x[i]);\n                    }\n                }\n            }\n\n            target[field] = out;\n        }\n    },\n\n    $pullAll: function (target, field, arg) {\n        if (_.isNil(target) || _.isNil(target[field])) return;\n\n        var x = target[field];\n\n        if (!_.isNil(x) && !_.isArray(x)) {\n            logger.throw(\"Modifier $pushAll/pullAll allowed for arrays only\");\n        } else if (!_.isNil(x)) {\n            var out = [];\n\n            for (var i = 0; i < x.length; i++) {\n                var exclude = false;\n\n                for (var j = 0; j < arg.length; j++) {\n                    if (SelectorMatcher.equal(x[i], arg[j])) {\n                        exclude = true;\n                        \n                        break;\n                    }\n                }\n\n                if (!exclude) {\n                    out.push(x[i]);\n                }\n            }\n\n            target[field] = out;\n        }\n    },\n\n    $rename: function (target, field, value) {\n        if (field === value) {\n            // no idea why mongo has this restriction..\n            logger.throw(\"The new field name must be different\");\n        }\n\n        if (!_.isString(value) || value.trim() === '') {\n            logger.throw(\"The new name must be a non-empty string\");\n        }\n\n        target[value] = target[field];\n        delete target[field];\n    },\n\n    $bit: function (target, field, arg) {\n        // XXX mongo only supports $bit on integers, and we only support\n        // native javascript numbers (doubles) so far, so we can't support $bit\n        logger.throw(\"$bit is not supported\");\n    }\n};\n\n/**\n* @ignore\n*/\nCollection.checkCollectionName = function(collectionName) {\n    if (!_.isString(collectionName)) {\n        logger.throw(\"collection name must be a String\");\n    }\n\n    if (!collectionName || collectionName.indexOf('..') !== -1) {\n        logger.throw(\"collection names cannot be empty\");\n    }\n\n    if (collectionName.indexOf('$') !== -1 && collectionName.match(/((^\\$cmd)|(oplog\\.\\$main))/) === null) {\n        logger.throw(\"collection names must not contain '$'\");\n    }\n\n    if (collectionName.match(/^system\\./) !== null) {\n        logger.throw(\"collection names must not start with 'system.' (reserved for internal use)\");\n    }\n    \n    if (collectionName.match(/^\\.|\\.$/) !== null) {\n        logger.throw(\"collection names must not start or end with '.'\");\n    }\n};\n\n/**\n* @ignore\n*/\nCollection.prototype.rename = function(newName) {\n    if (_.isString(newName)) {\n        if (this.name !== newName) {\n            Collection.checkCollectionName(newName);\n            \n            var dbName = this.name.split('.').length > 1 ? this.name.split('.')[0] : '';\n            \n            this.name = newName;\n            this.fullName = dbName + '.' + this.name;\n            \n            return this;\n        }\n    } else {\n        // Error\n    }\n};\n\nmodule.exports = Collection;\n\n/**\n * Gets the size of an object.\n * \n * @method Object#size\n * \n * @param {Object} obj - The object\n * \n * @returns {Number} The size of the object\n */\nObject.size = function(obj) {\n    var size = 0, \n        key;\n    \n    for (key in obj) {\n        if (obj.hasOwnProperty(key)) {\n            size++;\n        }\n    }\n    \n    return size;\n};\n\nvar _ensureFindParams = function(params) {\n    // selection, fields, options, callback\n    if (_.isNil(params.selection)) params.selection = {};\n\n    if (_.isNil(params.selection)) params.selection = {};\n\n    if (_.isNil(params.fields)) params.fields = [];\n\n    if (_.isNil(params.options)) {\n        params.options = {\n            skip: 0,\n            limit: 15 // for no limit pass [options.limit = -1]\n        };\n    }\n\n    // callback as first parameter\n    if (_.isFunction(params.selection)) {\n        params.callback = params.selection;\n        params.selection = {};\n    }\n\n    // callback as second parameter\n    if (_.isFunction(params.fields)) {\n        params.callback = params.fields;\n        params.fields = [];\n    }\n\n    // callback as third parameter\n    if (_.isFunction(params.options)) {\n        params.callback = params.options;\n        params.options = {};\n    }\n\n    // Check special case where we are using an objectId\n    if (params.selection instanceof ObjectId) {\n        params.selection = {\n            _id: params.selection\n        };\n    }\n\n    if (!_.isNil(params.callback) && !_.isFunction(params.callback)) {\n        logger.throw(\"callback must be a function\");\n    }\n\n    if (params.options.fields) {\n        if (_.isNil(params.fields) || params.fields.length === 0) {\n            params.fields = params.options.fields;\n        } else {\n            logger.warn(\"Fields already present. Ignoring 'options.fields'.\");\n        }\n    }\n    \n    return params;\n};"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Collection.js"],"names":[],"mappings":"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","file":"Collection.js","sourcesContent":["/**\n * @file Collection.js - based on Monglo#Collection ({@link https://github.com/Monglo}) by Christian Sullivan <cs@euforic.co> | Copyright (c) 2012\n * @version 1.0.0\n * \n * @author Eduardo Astolfi <eastolfi91@gmail.com>\n * @copyright 2016 Eduardo Astolfi <eastolfi91@gmail.com>\n * @license MIT Licensed\n */\n\nvar Logger = require(\"jsw-logger\"),\n    EventEmitter = require(\"./utils/EventEmitter\"),\n    _ = require(\"lodash\"),\n    Aggregation = require(\"./Aggregation\"),\n    Cursor = require(\"./Cursor\"),\n    ObjectId = require('./ObjectId'),\n    Selector = require(\"./Selector\"),\n    SelectorMatcher = require(\"./SelectorMatcher\");\n    \nvar logger = null;\n    \n/**\n * Collection\n * \n * @module Collection\n * @constructor\n * @since 0.0.1\n * \n * @classdesc Collection class that maps a MongoDB-like collection\n * \n * @param {MongoPortable} db - Additional options\n * @param {String} collectionName - The name of the collection\n * @param {Object} [options] - Database object\n * \n * @param {Object} [options.pkFactory=null] - Object overriding the basic \"ObjectId\" primary key generation.\n * \n */\nvar database = null;\nclass Collection extends EventEmitter {\n// var Collection = function(db, collectionName, options) {\n    constructor(db, collectionName, options) {\n        super();\n        \n        if (!(this instanceof Collection)) return new Collection(db, collectionName, options);\n        \n        logger = Logger.instance;\n    \n        if (_.isNil(db)) logger.throw(\"db parameter required\");\n        \n        if (_.isNil(collectionName)) logger.throw(\"collectionName parameter required\");\n        \n        if (_.isNil(options) || !_.isPlainObject(options)) options = {};\n        \n        Collection.checkCollectionName(collectionName);\n    \n        // this.db = db;\n        database = db;\n        this.name = collectionName;\n        this.databaseName = db.databaseName;\n        this.fullName = this.databaseName + '.' + this.name;\n        this.docs = [];\n        this.doc_indexes = {};\n        this.snapshots = [];\n        this.opts = {}; // Default options\n        \n        _.merge(this.opts, options);\n        \n        // this.emit = db.emit;\n    }\n    \n    emit(name, args, cb) {\n        super.emit(name, args, cb, database._stores);\n    }\n}\n\n// TODO enforce rule that field names can't start with '$' or contain '.'\n// (real mongodb does in fact enforce this)\n// TODO possibly enforce that 'undefined' does not appear (we assume\n// this in our handling of null and $exists)\n/**\n * Inserts a document into the collection\n * \n * @method Collection#insert\n * \n * @param {Object} doc - Document to be inserted\n * @param {Object} [options] - Additional options\n * \n * @param {Boolean} [options.chain=false] - If set to \"true\" returns this instance, so it can be chained with other methods\n * \n * @param {Function} [callback=null] Callback function to be called at the end with the results\n * \n * @returns {Object|Collection} If \"options.chain\" set to \"true\" returns this instance, otherwise returns the inserted document\n */\nCollection.prototype.insert = function (doc, options, callback) {\n    if (_.isNil(doc)) logger.throw(\"doc parameter required\");\n    \n    if (!_.isPlainObject(doc)) logger.throw(\"doc must be an object\");\n    \n    if (_.isNil(options)) options = {};\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    // Creating a safe copy of the document\n    var _doc = _.cloneDeep(doc);\n\n    // If the document comes with a number ID, parse it to String\n    if (_.isNumber(_doc._id)) {\n        _doc._id = _.toString(_doc._id);\n    }\n\n    if (_.isNil(_doc._id) || (!_doc._id instanceof ObjectId && (!_.isString(_doc._id) || !_doc._id.length))) {\n        _doc._id = new ObjectId();\n    }\n\n    // Add options to more dates\n    _doc.timestamp = new ObjectId().generationTime;\n    \n    // Reverse\n    this.doc_indexes[_.toString(_doc._id)] = this.docs.length;\n    this.docs.push(_doc);\n    \n    /**\n     * \"insert\" event.\n     *\n     * @event MongoPortable~insert\n     * \n     * @param {Object} collection - Information about the collection\n     * @param {Object} doc - Information about the document inserted\n     */\n    this.emit(\n        'insert',\n        {\n            collection: this,\n            doc: _doc\n        }\n    );\n\n    if (callback) callback(null, _doc);\n\n    if (options.chain) return this;\n    \n    return _doc;\n};\n\n/**\n * Finds all matching documents\n * \n * @method Collection#find\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object|Array|String} [fields={}] - The fields of the document to show\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.skip] - Number of documents to be skipped\n * @param {Number} [options.limit] - Max number of documents to display\n * @param {Object|Array|String} [options.fields] - Same as \"fields\" parameter (if both passed, \"options.fields\" will be ignored)\n * @param {Boolean} [options.forceFetch=false] - If set to'\"true\" returns the array of documents already fetched\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Array|Cursor} If \"options.forceFetch\" set to true returns the array of documents, otherwise returns a cursor\n */\nCollection.prototype.find = function (selection, fields, options, callback) {\n    let params = _ensureFindParams({\n        selection: selection, \n        fields: fields,\n        options: options, \n        callback: callback\n    });\n    \n    selection = params.selection;\n    fields = params.fields;\n    options = params.options;\n    callback = params.callback;\n    \n    var cursor = new Cursor(this.docs, selection, fields, options);\n\n    /**\n     * \"find\" event.\n     *\n     * @event MongoPortable~find\n     * \n     * @property {Object} collection - Information about the collection\n     * @property {Object} selector - The selection of the query\n     * @property {Object} fields - The fields showed in the query\n     */\n    this.emit(\n        'find',\n        {\n            collection: this,\n            selector: selection,\n            fields: fields\n        }\n    );\n    \n    // Pass the cursor fetched to the callback\n    // Add [options.noFetchCallback = true]\n    if (callback) callback(null, cursor.fetch());\n\n    if (options.forceFetch) {\n        return cursor.fetch();\n    } else {\n        return cursor;\n    }\n};\n\n/**\n * Finds the first matching document\n * \n * @method Collection#findOne\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object|Array|String} [fields={}] - The fields of the document to show\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.skip] - Number of documents to be skipped\n * @param {Number} [options.limit] - Max number of documents to display\n * @param {Object|Array|String} [options.fields] - Same as \"fields\" parameter (if both passed, \"options.fields\" will be ignored)\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} Returns the first matching document of the collection\n */\nCollection.prototype.findOne = function (selection, fields, options, callback) {\n    let params = _ensureFindParams({\n        selection: selection, \n        fields: fields,\n        options: options, \n        callback: callback\n    });\n    \n    selection = params.selection;\n    fields = params.fields;\n    options = params.options;\n    callback = params.callback;\n    \n    var cursor = new Cursor(this.docs, selection, fields, options);\n\n    /**\n     * \"findOne\" event.\n     *\n     * @event MongoPortable~findOne\n     * \n     * @property {Object} collection - Information about the collection\n     * @property {Object} selector - The selection of the query\n     * @property {Object} fields - The fields showed in the query\n     */\n    this.emit(\n        'findOne',\n        {\n            collection: this,\n            selector: selection,\n            fields: fields\n        }\n    );\n    \n    var res = null;\n    \n    if (cursor.hasNext()) {\n        res = cursor.next();\n    }\n    \n    // Pass the cursor fetched to the callback\n    // Add [options.noFetchCallback = true]\n    if (callback) callback(null, res);\n    \n    return res;\n};\n\n\n/**\n * Updates one or many documents\n * \n * @method Collection#update\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object} [update={}] - The update operation\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.updateAsMongo=true] - By default: \n *      If the [update] object contains update operator modifiers, such as those using the \"$set\" modifier, then:\n *          <ul>\n *              <li>The [update] object must contain only update operator expressions</li>\n *              <li>The Collection#update method updates only the corresponding fields in the document</li>\n *          <ul>\n *      If the [update] object contains only \"field: value\" expressions, then:\n *          <ul>\n *              <li>The Collection#update method replaces the matching document with the [update] object. The Collection#update method does not replace the \"_id\" value</li>\n *              <li>Collection#update cannot update multiple documents</li>\n *          <ul>\n * \n * @param {Number} [options.override=false] - Replaces the whole document (only apllies when [updateAsMongo=false])\n * @param {Number} [options.upsert=false] - Creates a new document when no document matches the query criteria\n * @param {Number} [options.multi=false] - Updates multiple documents that meet the criteria\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} Object with the update/insert (if upsert=true) information\n */\nCollection.prototype.update = function (selection, update, options, callback) {\n    if (_.isNil(selection)) selection = {};\n    \n    if (_.isNil(update)) logger.throw(\"You must specify the update operation\");\n    \n    if (_.isNil(options)) {\n        options = {\n            skip: 0,\n            limit: 15   // for no limit pass [options.limit = -1]\n        };\n    }\n    \n    if (_.isFunction(selection)) logger.throw(\"You must specify the update operation\");\n    \n    if (_.isFunction(update)) logger.throw(\"You must specify the update operation\");\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    // Check special case where we are using an objectId\n    if(selection instanceof ObjectId) {\n        selection = {\n            _id: selection\n        };\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n\n    var res = null;\n\n    var docs = null;\n    if (options.multi) {\n        docs = this.find(selection, null, { forceFetch: true });\n    } else {\n        docs = this.findOne(selection);\n    }\n    \n    if (_.isNil(docs)) {\n        docs = [];\n    }\n    \n    if (!_.isArray(docs)) {\n        docs = [docs];\n    }\n    \n    if (docs.length === 0) {\n        if (options.upsert) {\n            var inserted = this.insert(update);\n\n            res = {\n                updated: {\n                    documents: null,\n                    count: 0\n                },\n                inserted: {\n                    documents: [inserted],\n                    count: 1\n                }\n            };\n        } else {\n            // No documents found\n            res = {\n                updated: {\n                    documents: null,\n                    count: 0\n                },\n                inserted: {\n                    documents: null,\n                    count: 0\n                }\n            };\n        }\n    } else {\n        var updatedDocs = [];\n        \n        for (var i = 0; i < docs.length; i++) {\n            var doc = docs[i];\n            \n            var override = null;\n            \n            var hasModifier = false;\n            \n            for (let key in update) {\n                // IE7 doesn't support indexing into strings (eg, key[0] or key.indexOf('$') ), so use substr.\n                // Testing over the first letter:\n                //      Bests result with 1e8 loops => key[0](~3s) > substr(~5s) > regexp(~6s) > indexOf(~16s)\n                \n                var modifier = (key.substr(0, 1) === '$');\n                if (modifier) {\n                    hasModifier = true;\n                }\n                \n                if (options.updateAsMongo) {\n                    if (hasModifier && !modifier) logger.throw(\"All update fields must be an update operator\");\n                    \n                    if (!hasModifier && options.multi) logger.throw(\"You can not update several documents when no update operators are included\");\n                    \n                    if (hasModifier) override = false;\n                    \n                    if (!hasModifier) override = true;\n                } else {\n                    override = !!options.override;\n                }\n            }\n            \n            var _docUpdate = null;\n            \n            if (override) {\n                // Overrides the document except for the \"_id\"\n                _docUpdate = {\n                    _id: doc._id\n                };\n                \n                // Must ignore fields starting with '$', '.'...\n                for (let key in update) {\n                    if (key.substr(0, 1) === '$' || /\\./g.test(key)) {\n                        logger.warn(`The field ${key} can not begin with '$' or contain '.'`);\n                    } else {\n                        _docUpdate[key] = update[key];\n                    }\n                }\n            } else {\n                _docUpdate = _.cloneDeep(doc);\n                \n                for (let key in update) {\n                    let val = update[key];\n                    \n                    if (key.substr(0, 1) === '$') {\n                        _docUpdate = _applyModifier(_docUpdate, key, val);\n                    } else {\n                        if (!_.isNil(_docUpdate[key])) {\n                            if (key !== '_id') {\n                                _docUpdate[key] = val;\n                            } else {\n                                logger.warn(\"The field '_id' can not be updated\");\n                            }\n                        } else {\n                            logger.warn(`The document does not contains the field ${key}`);\n                        }\n                    }\n                }\n            }\n            \n            updatedDocs.push(_docUpdate);\n            \n            let idx = this.doc_indexes[_docUpdate._id];\n            this.docs[idx] = _docUpdate;\n        }\n        \n        /**\n         * \"update\" event.\n         *\n         * @event MongoPortable~update\n         * \n         * @property {Object} collection - Information about the collection\n         * @property {Object} selector - The selection of the query\n         * @property {Object} modifier - The modifier used in the query\n         * @property {Object} docs - The updated/inserted documents information\n         */\n        this.emit(\n            'update',\n            {\n                collection: this,\n                selector: selection,\n                modifier: update,\n                docs: updatedDocs\n            }\n        );\n        \n        res = {\n            updated: {\n                documents: updatedDocs,\n                count: updatedDocs.length\n            },\n            inserted: {\n                documents: null,\n                count: 0\n            }\n        };\n    }\n    \n    \n    if (callback) callback(null, res);\n    \n    return res;\n};\n\nvar _applyModifier = function(_docUpdate, key, val) {\n    var doc = _.cloneDeep(_docUpdate);\n    // var mod = _modifiers[key];\n                        \n    if (!_modifiers[key]) {\n        logger.throw(`Invalid modifier specified: ${key}`);\n    }\n    \n    for (var keypath in val) {\n        var value = val[keypath];\n        var keyparts = keypath.split('.');\n        \n        _modify(doc, keyparts, value, key);\n        \n        // var no_create = !!Collection._noCreateModifiers[key];\n        // var forbid_array = (key === \"$rename\");\n        // var target = Collection._findModTarget(_docUpdate, keyparts, no_create, forbid_array);\n        // var field = keyparts.pop();\n\n        // mod(target, field, value, keypath, _docUpdate);\n    }\n    \n    return doc;\n};\n\nvar _modify = function(document, keyparts, value, key, level = 0) {\n    for (let i = level; i < keyparts.length; i++) {\n        let path = keyparts[i];\n        let isNumeric = /^[0-9]+$/.test(path);\n        let target = document[path];\n        \n        var create = _.hasIn(Collection._noCreateModifiers, key) ? false : true;\n        if (!create && (!_.isObject(document) || _.isNil(target))) {\n            logger.throw(`The element \"${path}\" must exists in \"${JSON.stringify(document)}\"`);\n        }\n        \n        if (_.isArray(document)) {\n            // Do not allow $rename on arrays\n            if (key === \"$rename\") return null;\n            \n            // Only let the use of \"arrayfield.<numeric_index>.subfield\"\n            if (isNumeric) {\n                path = _.toNumber(path);\n            } else {\n                logger.throw(`The field \"${path}\" can not be appended to an array`);\n            }\n            \n            // Fill the array to the desired length\n            while (document.length < path) {\n                document.push(null);\n            }\n        }\n        \n        if (i < keyparts.length - 1) {\n            if (_.isNil(target)) {\n                // If we are accessing with \"arrayField.<numeric_index>\"\n                if (_.isFinite(_.toNumber(keyparts[i + 1]))) {  //  || keyparts[i + 1] === '$'  // TODO \"arrayField.$\"\n                    target = [];\n                } else {\n                    target = {};\n                }\n            }\n            \n            document[path] = _modify(target, keyparts, value, key, level + 1);\n\n            return document;\n        } else {\n            _modifiers[key](document, path, value);\n            \n            return document;\n        }\n    }\n};\n\n/**\n * Removes one or many documents\n * \n * @method Collection#remove\n * \n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.justOne=false] - Deletes the first occurrence of the selection\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} Object with the deleted documents\n */\nCollection.prototype.remove = function (selection, options, callback) {\n    if (_.isNil(selection)) selection = {};\n    \n    if (_.isFunction(selection)) {\n        callback = selection;\n        selection = {};\n    }\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    if (_.isNil(options)) options = { justOne: false };\n    \n    // If we are not passing a selection and we are not removing just one, is the same as a drop\n    if (Object.size(selection) === 0 && !options.justOne) return this.drop(options, callback);\n    \n    // Check special case where we are using an objectId\n    if(selection instanceof ObjectId) {\n        selection = {\n            _id: selection\n        };\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    var cursor = this.find(selection);\n    \n    var docs = [];\n    cursor.forEach(doc => {\n        var idx = this.doc_indexes[doc._id];\n        \n        delete this.doc_indexes[doc._id];\n        this.docs.splice(idx, 1);\n        \n        docs.push(doc);\n    });\n    \n    /**\n     * \"remove\" event.\n     *\n     * @event MongoPortable~remove\n     * \n     * @property {Object} collection - Information about the collection\n     * @property {Object} selector - The selection of the query\n     * @property {Object} docs - The deleted documents information\n     */\n    this.emit(\n        'remove',\n        {\n            collection: this,\n            selector: selection,\n            docs: docs\n        }\n    );\n    \n    if (callback) callback(null, docs);\n    \n    return docs;\n};\n\n/**\n * Alias for {@link Collection#remove}\n * \n * @method Collection#delete\n */\nCollection.prototype.delete = function (selection, options, callback) {\n    return this.remove(selection, options, callback);\n};\n \n /**\n * Alias for {@link Collection#remove}\n * \n * @method Collection#destroy\n */\nCollection.prototype.destroy = function (selection, options, callback) {\n    return this.remove(selection, options, callback);\n};\n\n/**\n * Drops a collection\n * \n * @method Collection#drop\n * \n * @param {Object} [options] - Additional options\n * \n * @param {Number} [options.dropIndexes=false] - True if we want to drop the indexes too\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} True when the collection is dropped\n */\nCollection.prototype.drop = function(options, callback) {\n    if (_.isNil(options)) options = {};\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    this.doc_indexes = {};\n    this.docs = [];\n    \n    if (options.dropIndexes) {} // TODO\n    \n    this.emit(\n        'dropCollection',\n        {\n            collection: this,\n            indexes: !!options.dropIndexes\n        }\n    );\n    \n    if (callback) callback(null, true);\n    \n    return true;\n};\n\n/**\n * Insert or update a document. If the document has an \"_id\" is an update (with upsert), if not is an insert.\n * \n * @method Collection#save\n * \n * @param {Object} doc - Document to be inserted/updated\n * \n * @param {Number} [options.dropIndexes=false] - True if we want to drop the indexes too\n * @param {Object} [options.writeConcern=null] - An object expressing the write concern\n * \n * @param {Function} [callback=null] - Callback function to be called at the end with the results\n * \n * @returns {Object} True when the collection is dropped\n */\nCollection.prototype.save = function(doc, options, callback) {\n    if (_.isNil(doc) || _.isFunction(doc)) logger.throw(\"You must pass a document\");\n    \n    if (_.isFunction(options)) {\n        callback = options;\n        options = {};\n    }\n\n    if (_.hasIn(doc, '_id')) {\n        options.upsert = true;\n        \n        return this.update(\n            { _id: doc._id },\n            doc,\n            options,\n            callback\n        );\n    } else {\n        return this.insert(doc, options, callback);\n    }\n};\n\n/**\n* @ignore\n*/\nCollection.prototype.ensureIndex = function() {\n    //TODO Implement EnsureIndex\n    logger.throw('Collection#ensureIndex unimplemented by driver');\n};\n\n// TODO document (at some point)\n// TODO test\n// TODO obviously this particular implementation will not be very efficient\n/**\n* @ignore\n*/\nCollection.prototype.backup = function (backupID, callback) {\n    if (_.isFunction(backupID)) {\n        callback = backupID;\n        backupID = new ObjectId().toString();\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n\n    this.snapshots[backupID] = _.cloneDeep(this.docs);\n    this.emit(\n        'snapshot',\n        {\n            collection: this,\n            backupID: backupID,\n            documents: this.snapshots[backupID] \n        }\n    );\n\n    var result = {\n        backupID: backupID,\n        documents: this.snapshots[backupID]\n    };\n    \n    if (callback) callback(null, result);\n\n    return result;\n};\n\n// Lists available Backups\n/**\n* @ignore\n*/\nCollection.prototype.backups = function (callback) {\n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    var backups = [];\n\n    for (let id in this.snapshots) {\n        backups.push({id: id, documents: this.snapshots[id]});\n    }\n\n    if (callback) callback(null, backups);\n\n    return backups;\n};\n\n// Lists available Backups\n/**\n* @ignore\n*/\nCollection.prototype.removeBackup = function (backupID, callback) {\n    if (_.isFunction(backupID)) {\n        callback = backupID;\n        backupID = null;\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    let result = false;\n    \n    if (backupID) {\n        delete this.snapshots[_.toString(backupID)];\n        \n        result = backupID;\n    } else {\n        this.snapshots = {};\n        \n        result = true;\n    }\n    \n    if (callback) callback(null, result);\n\n    return result;\n};\n\n\n// Restore the snapshot. If no snapshot exists, raise an exception;\n/**\n* @ignore\n*/\nCollection.prototype.restore = function (backupID, callback) {\n    if (_.isFunction(backupID)) {\n        callback = backupID;\n        backupID = null;\n    }\n    \n    if (!_.isNil(callback) && !_.isFunction(callback)) logger.throw(\"callback must be a function\");\n    \n    var snapshotCount = Object.size(this.snapshots);\n    var backupData = null;\n\n    if (snapshotCount === 0) {\n        logger.throw(\"There is no snapshots\");\n    } else {\n        if (!backupID) {\n            if (snapshotCount === 1) {\n                logger.info(\"No backupID passed. Restoring the only snapshot\");\n                \n                // Retrieve the only snapshot\n                for (let key in this.snapshots) backupID = key;\n            } else {\n                logger.throw(\"The are several snapshots. Please specify one backupID\");\n            }\n        }\n    }\n    \n    backupData = this.snapshots[backupID];\n            \n    if (!backupData) {\n        logger.throw(`Unknown Backup ID: ${backupID}`);\n    }\n\n    this.docs = backupData;\n    this.emit(\n        'restore',\n        {\n            collection: this,\n            backupID: backupID\n        }\n    );\n\n    if (callback) callback(null);\n\n    return this;\n};\n\n/**\n * Calculates aggregate values for the data in a collection\n * \n * @method Collection#aggregate\n * \n * @param {Array} pipeline - A sequence of data aggregation operations or stages\n * @param {Object} [options] - Additional options\n * \n * @param {Boolean} [options.forceFetch=false] - If set to'\"true\" returns the array of documents already fetched\n * \n * @returns {Array|Cursor} If \"options.forceFetch\" set to true returns the array of documents, otherwise returns a cursor\n */\nCollection.prototype.aggregate = function(pipeline, options = { forceFetch: false }) {\n    if (_.isNil(pipeline) || !_.isArray(pipeline)) logger.throw('The \"pipeline\" param must be an array');\n    \n    var aggregation = new Aggregation(pipeline);\n    \n    for (let i = 0; i < pipeline.length; i++) {\n        let stage = pipeline[i];\n        \n        for (let key in stage) {\n            if (key.substr(0, 1) !== '$') logger.throw(\"The pipeline stages must begin with '$'\");\n            \n            if (!aggregation.validStage(key)) logger.throw(`Invalid stage \"${key}\"`);\n            \n            break;\n        }\n    }\n    \n    var result = aggregation.aggregate(this);\n    \n    return result;  // change to cursor\n};\n\n/**\n* @ignore\n*/\nCollection._noCreateModifiers = {\n    $unset: true,\n    $pop: true,\n    $rename: true,\n    $pull: true,\n    $pullAll: true\n};\n\n/**\n* @ignore\n*/\nvar _modifiers = {\n    $inc: function (target, field, arg) {\n        if (!_.isNumber(arg)) {\n            logger.throw(\"Modifier $inc allowed for numbers only\");\n        }\n\n        if (field in target) {\n            if (!_.isNumber(target[field])) {\n                logger.throw(\"Cannot apply $inc modifier to non-number\");\n            }\n\n            target[field] += arg;\n        } else {\n            target[field] = arg;\n        }\n    },\n\n    $set: function (target, field, arg) {\n        target[field] = _.cloneDeep(arg);\n    },\n\n    $unset: function (target, field, arg) {\n        if (!_.isNil(target)) {\n            if (_.isArray(target)) {\n                if (field in target) {\n                    target[field] = null;\n                }\n            } else {\n                delete target[field];\n            }\n        }\n    },\n\n    $push: function (target, field, arg) {\n        var x = target[field];\n\n        if (_.isNil(x)) {\n            target[field] = [arg];\n        } else if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $push modifier to non-array\");\n        } else {\n            x.push(_.cloneDeep(arg));\n        }\n    },\n\n    $pushAll: function (target, field, arg) {\n        var x = target[field];\n\n        if (_.isNil(x)) {\n            target[field] = arg;\n        } else if (!_.isArray(x)) {\n            logger.throw(\"Modifier $pushAll/pullAll allowed for arrays only\");\n        } else {\n            for (var i = 0; i < arg.length; i++) {\n                x.push(arg[i]);\n            }\n        }\n    },\n\n    $addToSet: function (target, field, arg) {\n        var x = target[field];\n\n        if (_.isNil(x)) {\n            target[field] = [arg];\n        } else if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $addToSet modifier to non-array\");\n        } else {\n            let isEach = false;\n            if (_.isPlainObject(arg)) {\n                for (let k in arg) {\n                    if (k === \"$each\") {\n                        isEach = true;\n                    }\n                    \n                    break;\n                }\n            }\n\n            let values = isEach ? arg[\"$each\"] : [arg];\n            _.forEach(values, function (value) {\n                for (let i = 0; i < x.length; i++) {\n                    if (SelectorMatcher.equal(value, x[i])) return;\n                }\n\n                x.push(value);\n            });\n        }\n    },\n\n    $pop: function (target, field, arg) {\n        if (_.isNil(target) || _.isNil(target[field])) return;\n\n        var x = target[field];\n\n        if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $pop modifier to non-array\");\n        } else {\n            if (_.isNumber(arg) && arg < 0) {\n                x.splice(0, 1);\n            } else {\n                x.pop();\n            }\n        }\n    },\n\n    $pull: function (target, field, arg) {\n        if (_.isNil(target) || _.isNil(target[field])) return;\n\n        var x = target[field];\n\n        if (!_.isArray(x)) {\n            logger.throw(\"Cannot apply $pull/pullAll modifier to non-array\");\n        } else {\n            var out = [];\n            \n            if (typeof arg === \"object\" && !(arg instanceof Array)) {\n                // XXX would be much nicer to compile this once, rather than\n                // for each document we modify.. but usually we're not\n                // modifying that many documents, so we'll let it slide for\n                // now\n\n                // XXX _compileSelector isn't up for the job, because we need\n                // to permit stuff like {$pull: {a: {$gt: 4}}}.. something\n                // like {$gt: 4} is not normally a complete selector.\n                var match = new Selector({\n                    \"__matching__\": arg\n                });\n                for (var i = 0; i < x.length; i++) {\n                    var _doc_ = {\n                        __matching__: x[i]\n                    };\n                    if (!match.test(_doc_)) {\n                        out.push(x[i]);\n                    }\n                }\n            } else {\n                for (var i = 0; i < x.length; i++) {\n                    if (!SelectorMatcher.equal(x[i], arg)) {\n                        out.push(x[i]);\n                    }\n                }\n            }\n\n            target[field] = out;\n        }\n    },\n\n    $pullAll: function (target, field, arg) {\n        if (_.isNil(target) || _.isNil(target[field])) return;\n\n        var x = target[field];\n\n        if (!_.isNil(x) && !_.isArray(x)) {\n            logger.throw(\"Modifier $pushAll/pullAll allowed for arrays only\");\n        } else if (!_.isNil(x)) {\n            var out = [];\n\n            for (var i = 0; i < x.length; i++) {\n                var exclude = false;\n\n                for (var j = 0; j < arg.length; j++) {\n                    if (SelectorMatcher.equal(x[i], arg[j])) {\n                        exclude = true;\n                        \n                        break;\n                    }\n                }\n\n                if (!exclude) {\n                    out.push(x[i]);\n                }\n            }\n\n            target[field] = out;\n        }\n    },\n\n    $rename: function (target, field, value) {\n        if (field === value) {\n            // no idea why mongo has this restriction..\n            logger.throw(\"The new field name must be different\");\n        }\n\n        if (!_.isString(value) || value.trim() === '') {\n            logger.throw(\"The new name must be a non-empty string\");\n        }\n\n        target[value] = target[field];\n        delete target[field];\n    },\n\n    $bit: function (target, field, arg) {\n        // XXX mongo only supports $bit on integers, and we only support\n        // native javascript numbers (doubles) so far, so we can't support $bit\n        logger.throw(\"$bit is not supported\");\n    }\n};\n\n/**\n* @ignore\n*/\nCollection.checkCollectionName = function(collectionName) {\n    if (!_.isString(collectionName)) {\n        logger.throw(\"collection name must be a String\");\n    }\n\n    if (!collectionName || collectionName.indexOf('..') !== -1) {\n        logger.throw(\"collection names cannot be empty\");\n    }\n\n    if (collectionName.indexOf('$') !== -1 && collectionName.match(/((^\\$cmd)|(oplog\\.\\$main))/) === null) {\n        logger.throw(\"collection names must not contain '$'\");\n    }\n\n    if (collectionName.match(/^system\\./) !== null) {\n        logger.throw(\"collection names must not start with 'system.' (reserved for internal use)\");\n    }\n    \n    if (collectionName.match(/^\\.|\\.$/) !== null) {\n        logger.throw(\"collection names must not start or end with '.'\");\n    }\n};\n\n/**\n* @ignore\n*/\nCollection.prototype.rename = function(newName) {\n    if (_.isString(newName)) {\n        if (this.name !== newName) {\n            Collection.checkCollectionName(newName);\n            \n            var dbName = this.name.split('.').length > 1 ? this.name.split('.')[0] : '';\n            \n            this.name = newName;\n            this.fullName = dbName + '.' + this.name;\n            \n            return this;\n        }\n    } else {\n        // Error\n    }\n};\n\nmodule.exports = Collection;\n\n/**\n * Gets the size of an object.\n * \n * @method Object#size\n * \n * @param {Object} obj - The object\n * \n * @returns {Number} The size of the object\n */\nObject.size = function(obj) {\n    var size = 0, \n        key;\n    \n    for (key in obj) {\n        if (obj.hasOwnProperty(key)) {\n            size++;\n        }\n    }\n    \n    return size;\n};\n\nvar _ensureFindParams = function(params) {\n    // selection, fields, options, callback\n    if (_.isNil(params.selection)) params.selection = {};\n\n    if (_.isNil(params.selection)) params.selection = {};\n\n    if (_.isNil(params.fields)) params.fields = [];\n\n    if (_.isNil(params.options)) {\n        params.options = {\n            skip: 0,\n            limit: 15 // for no limit pass [options.limit = -1]\n        };\n    }\n\n    // callback as first parameter\n    if (_.isFunction(params.selection)) {\n        params.callback = params.selection;\n        params.selection = {};\n    }\n\n    // callback as second parameter\n    if (_.isFunction(params.fields)) {\n        params.callback = params.fields;\n        params.fields = [];\n    }\n\n    // callback as third parameter\n    if (_.isFunction(params.options)) {\n        params.callback = params.options;\n        params.options = {};\n    }\n\n    // Check special case where we are using an objectId\n    if (params.selection instanceof ObjectId) {\n        params.selection = {\n            _id: params.selection\n        };\n    }\n\n    if (!_.isNil(params.callback) && !_.isFunction(params.callback)) {\n        logger.throw(\"callback must be a function\");\n    }\n\n    if (params.options.fields) {\n        if (_.isNil(params.fields) || params.fields.length === 0) {\n            params.fields = params.options.fields;\n        } else {\n            logger.warn(\"Fields already present. Ignoring 'options.fields'.\");\n        }\n    }\n    \n    return params;\n};"]} diff --git a/lib/Cursor.js b/lib/Cursor.js index c858eac..2c790bd 100644 --- a/lib/Cursor.js +++ b/lib/Cursor.js @@ -27,7 +27,7 @@ var logger = null; * @classdesc Cursor class that maps a MongoDB-like cursor * * @param {MongoPortable} db - Additional options - * @param {Collection} collection - The collection instance + * @param {Array} documents - The list of documents * @param {Object|Array|String} [selection={}] - The selection for matching documents * @param {Object|Array|String} [fields={}] - The fields of the document to show * @param {Object} [options] - Database object @@ -36,13 +36,12 @@ var logger = null; * */ -var Cursor = function Cursor(db, collection, selection, fields) { - var options = arguments.length <= 4 || arguments[4] === undefined ? {} : arguments[4]; +var Cursor = function Cursor(documents, selection, fields) { + var options = arguments.length <= 3 || arguments[3] === undefined ? {} : arguments[3]; _classCallCheck(this, Cursor); - this.db = db; - this.collection = collection; + this.documents = documents; this.selector = selection; this.skipValue = options.skip || 0; this.limitValue = options.limit || 15; @@ -51,6 +50,7 @@ var Cursor = function Cursor(db, collection, selection, fields) { logger = Logger.instance; + /** ADD IDX **/ if (Selector.isSelectorCompiled(this.selector)) { this.selector_compiled = this.selector; } else { @@ -73,6 +73,27 @@ var Cursor = function Cursor(db, collection, selection, fields) { } } + /** ADD IDX **/ + + this.fetch_mode = Cursor.COLSCAN || Cursor.IDXSCAN; + this.indexex = null; //findUsableIndexes(); + + // if (cursor.fetch_mode === Cursor.COLSCAN) { + // // COLSCAN, wi will iterate over all documents + // docs = _.cloneDeep(cursor.collection.docs); + // } else if (cursor.fetch_mode === Cursor.IDXSCAN) { + // // IDXSCAN, wi will iterate over all needed documents + // for (let i = 0; i < cursor.indexes.length; i++) { + // let index = cursor.indexes[i]; + + // for (let i = index.start; i < index.end; i++) { + // let idx_id = cursor.collection.getIndex(index.name)[i]; + + // docs.push(cursor.collection.docs[idx_id]); + // } + // } + // } + this.fields = new Selector(fields, Selector.FIELD_SELECTOR); this.sort_compiled = new Selector(this.sortValue, Selector.SORT_SELECTOR); @@ -81,13 +102,14 @@ var Cursor = function Cursor(db, collection, selection, fields) { this.cursor_pos = 0; }; +Cursor.COLSCAN = 'colscan'; +Cursor.IDXSCAN = 'idxscan'; + /** * Moves a cursor to the begining * * @method Cursor#rewind */ - - Cursor.prototype.rewind = function () { this.db_objects = null; this.cursor_pos = 0; @@ -135,7 +157,7 @@ Cursor.prototype.map = function (callback) { * @returns {Boolean} True if we can fetch one more document */ Cursor.prototype.hasNext = function () { - return this.cursor_pos < this.collection.docs.length; + return this.cursor_pos < this.documents.length; }; /** @@ -243,25 +265,44 @@ var _mapFields = function _mapFields(doc, fields) { var _getDocuments = function _getDocuments(cursor) { var justOne = arguments.length <= 1 || arguments[1] === undefined ? false : arguments[1]; - if (cursor.selector_id) { - if (_.hasIn(cursor.collection.doc_indexes, _.toString(cursor.selector_id))) { - var idx = cursor.collection.doc_indexes[_.toString(cursor.selector_id)]; + var docs = []; - return _mapFields(cursor.collection.docs[idx], cursor.fields); - } else { - if (justOne) { - return null; - } else { - return []; + if (cursor.fetch_mode === Cursor.COLSCAN) { + // COLSCAN, wi will iterate over all documents + docs = _.cloneDeep(cursor.documents); + } else if (cursor.fetch_mode === Cursor.IDXSCAN) { + // IDXSCAN, wi will iterate over all needed documents + for (var i = 0; i < cursor.indexes.length; i++) { + var index = cursor.indexes[i]; + + for (var _i2 = index.start; _i2 < index.end; _i2++) { + // let idx_id = cursor.collection.getIndex(index.name)[i]; + var idx_id = index.index[_i2]; + + docs.push(cursor.documents[idx_id]); } } } + // if (cursor.selector_id) { + // if (_.hasIn(cursor.collection.doc_indexes, _.toString(cursor.selector_id))) { + // let idx = cursor.collection.doc_indexes[_.toString(cursor.selector_id)]; + + // return _mapFields(cursor.collection.docs[idx], cursor.fields); + // } else { + // if (justOne) { + // return null; + // } else { + // return []; + // } + // } + // } + // TODO add warning when sort/skip/limit and fetching one // TODO add warning when skip/limit without order // TODO index - while (cursor.cursor_pos < cursor.collection.docs.length) { - var _doc = cursor.collection.docs[cursor.cursor_pos]; + while (cursor.cursor_pos < docs.length) { + var _doc = docs[cursor.cursor_pos]; cursor.cursor_pos++; if (cursor.selector_compiled.test(_doc)) { @@ -299,6 +340,26 @@ Cursor.prototype.count = function () { return this.fetchAll().length; }; +/** + * Set the sorting of the cursor + * + * @method Cursor#sort + * + * @param {Object|Array|String} spec - The sorting specification + * + * @returns {Cursor} This instance so it can be chained with other methods + */ +Cursor.prototype.setSorting = function (spec) { + if (_.isNil(spec)) logger.throw("You need to specify a sorting"); + + if (spec) { + this.sortValue = spec; + this.sort_compiled = new Selector(spec, Selector.SORT_SELECTOR); + } + + return this; +}; + /** * Applies a sorting on the cursor * @@ -316,15 +377,11 @@ Cursor.prototype.sort = function (spec) { } if (_sort) { - if (spec) { - this.sortValue = spec; - this.sort_compiled = _sort; + if (!_.isNil(this.db_objects) && _.isArray(this.db_objects)) { + this.db_objects = this.db_objects.sort(_sort); + this.sorted = true; } else { - // If no spec, do sort - if (!_.isNil(this.db_objects) && _.isArray(this.db_objects)) { - this.db_objects = this.db_objects.sort(_sort); - this.sorted = true; - } + this.setSorting(spec); } } @@ -551,4 +608,4 @@ Cursor.prototype.toArray = function () { }; module.exports = Cursor; -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Cursor.js"],"names":[],"mappings":"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","file":"Cursor.js","sourcesContent":["/**\n * @file Cursor.js - based on Monglo#Cursor ({@link https://github.com/Monglo}) by Christian Sullivan <cs@euforic.co> | Copyright (c) 2012\n * @version 1.0.0\n * \n * @author Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @copyright 2016 Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @license MIT Licensed\n */\n\nvar Logger = require(\"jsw-logger\"),\n    _ = require(\"lodash\"),\n    Selector = require('./Selector');\n    \nvar logger = null;\n\n/**\n * Cursor\n * \n * @module Cursor\n * @constructor\n * @since 0.0.1\n * \n * @classdesc Cursor class that maps a MongoDB-like cursor\n * \n * @param {MongoPortable} db - Additional options\n * @param {Collection} collection - The collection instance\n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object|Array|String} [fields={}] - The fields of the document to show\n * @param {Object} [options] - Database object\n * \n * @param {Object} [options.pkFactory=null] - Object overriding the basic \"ObjectId\" primary key generation.\n * \n */\nclass Cursor {\n    constructor(db, collection, selection, fields, options = {}) {\n        this.db = db;\n        this.collection = collection;\n        this.selector = selection;\n        this.skipValue = options.skip || 0;\n        this.limitValue = options.limit || 15;\n        this.sortValue = options.sort || null;\n        this.sorted = false;\n        \n        logger = Logger.instance;\n    \n        if (Selector.isSelectorCompiled(this.selector)) {\n            this.selector_compiled = this.selector;\n        } else {\n            this.selector_compiled = new Selector(this.selector, Selector.MATCH_SELECTOR);\n        }\n        \n        for (let i = 0; i < this.selector_compiled.clauses.length; i++) {\n            if (this.selector_compiled.clauses[i].key === '_id') {\n                this.selector_id = this.selector_compiled.clauses[i].value;\n            }\n        }\n        \n        for (let i = 0; i < this.selector_compiled.clauses.length; i++) {\n            if (this.selector_compiled.clauses[i].key === '_id') {\n                var _val = this.selector_compiled.clauses[i].value;\n                \n                if (_.isString(_val) || _.isNumber(_val)) {\n                    this.selector_id = _val;\n                }\n            }\n        }\n\n        \n        this.fields = new Selector(fields, Selector.FIELD_SELECTOR);\n        \n        this.sort_compiled = new Selector(this.sortValue, Selector.SORT_SELECTOR);\n    \n        this.db_objects = null;\n        this.cursor_pos = 0;\n    }\n}\n\n/**\n * Moves a cursor to the begining\n * \n * @method Cursor#rewind\n */\nCursor.prototype.rewind = function() {\n    this.db_objects = null;\n    this.cursor_pos = 0;\n};\n\n/**\n * Iterates over the cursor, calling a callback function\n * \n * @method Cursor#forEach\n * \n * @param {Function} [callback=null] - Callback function to be called for each document\n */\nCursor.prototype.forEach = function(callback) {\n    let docs = this.fetchAll();\n    \n    for (let i = 0; i < docs.length; i++) {\n        callback(docs[i]);\n    }\n};\n\n/**\n * Iterates over the cursor, returning a new array with the documents affected by the callback function\n * \n * @method Cursor#map\n * \n * @param {Function} [callback=null] - Callback function to be called for each document\n * \n * @returns {Array} The documents after being affected with the callback function\n */\nCursor.prototype.map = function(callback) {\n    var res = [];\n\n    this.forEach(function (doc) {\n        res.push(callback(doc));\n    });\n\n    return res;\n};\n\n/**\n * Checks if the cursor has one document to be fetched\n * \n * @method Cursor#hasNext\n * \n * @returns {Boolean} True if we can fetch one more document\n */\nCursor.prototype.hasNext = function() {\n    return (this.cursor_pos < this.collection.docs.length);\n};\n\n/**\n * Alias for {@link Cursor#fetchOne}\n * \n * @method Cursor#next\n */\nCursor.prototype.next = function() {\n    return this.fetchOne();\n};\n\n/**\n * Alias for {@link Cursor#fetchAll}\n * \n * @method Cursor#fetch\n */\nCursor.prototype.fetch = function() {\n    return this.fetchAll();\n};\n\n/**\n * Fetch all documents in the cursor\n * \n * @method Cursor#fetchAll\n * \n * @returns {Array} All the documents contained in the cursor\n */\nCursor.prototype.fetchAll = function() {\n    return _getDocuments(this, false) || [];\n};\n\n/**\n * Retrieves the next document in the cursor\n * \n * @method Cursor#fetchOne\n * \n * @returns {Object} The next document in the cursor\n */\nCursor.prototype.fetchOne = function() {\n    return _getDocuments(this, true);\n};\n\nvar _mapFields = function(doc, fields) {\n    var _doc = _.cloneDeep(doc);\n\n    if (!_.isNil(fields) && _.isPlainObject(fields) && !_.isEqual(fields, {})) {\n        var showId = true,\n            showing = null;\n\n        // Whether if we showing the _id field\n        if (_.hasIn(fields, '_id') && fields._id === -1) {\n            showId = false;\n        }\n\n        var tmp = null;\n\n        for (var field in fields) {\n            // Whether if we are showing or hidding fields\n            if (field !== '_id' && showing === null) {\n                showing = fields[field] === 1 ? true : false;\n            }\n\n            if (showing != null) {\n                if (tmp === null) {\n                    if (showing) {\n                        tmp = {};\n                    } else {\n                        tmp = _.cloneDeep(doc);\n                    }\n                }\n        \n                // Add or remove the field\n                if (showing) {\n                    tmp[field] = doc[field];\n                } else {\n                    delete tmp[field];\n                }\n            }\n        }\n\n        // Add or remove the _id field\n        if (showId) {\n            tmp._id = doc._id;\n        } else {\n            delete tmp._id;\n        }\n\n        _doc = tmp;\n    }\n\n    return _doc;\n};\n\n/**\n * Retrieves one or all the documents in the cursor\n * \n * @method _getDocuments\n * @private\n * \n * @param {Cursor} cursor - The cursor with the documents\n * @param {Boolean} [justOne=false] - Whether it retrieves one or all the documents\n * \n * @returns {Array|Object} If [justOne=true] returns the next document, otherwise returns all the documents\n */\nvar _getDocuments = function(cursor, justOne = false) {\n    if (cursor.selector_id) {\n        if (_.hasIn(cursor.collection.doc_indexes, _.toString(cursor.selector_id))) {\n            let idx = cursor.collection.doc_indexes[_.toString(cursor.selector_id)];\n            \n            return _mapFields(cursor.collection.docs[idx], cursor.fields);\n        } else {\n            if (justOne) {\n                return null;\n            } else {\n                return [];\n            }\n        }\n    }\n    \n    // TODO add warning when sort/skip/limit and fetching one\n    // TODO add warning when skip/limit without order\n    // TODO index\n    while (cursor.cursor_pos < cursor.collection.docs.length) {\n        var _doc = cursor.collection.docs[cursor.cursor_pos];\n        cursor.cursor_pos++;\n        \n        if (cursor.selector_compiled.test(_doc)) {\n            if (_.isNil(cursor.db_objects)) cursor.db_objects = [];\n            \n            _doc = _mapFields(_doc, cursor.fields);\n            \n            cursor.db_objects.push(_doc);\n            \n            if (justOne) {\n                // Add force sort\n                return _doc;\n            }\n        }\n    }\n    \n    if (_.isNil(cursor.db_objects)) return null;\n    \n    if (!cursor.sorted && hasSorting(cursor)) cursor.sort();\n    \n    var idxFrom = cursor.skipValue;\n    var idxTo = cursor.limitValue !== -1 ? (cursor.limitValue + idxFrom) : cursor.db_objects.length;\n    \n    return cursor.db_objects.slice(idxFrom, idxTo);\n    \n};\n\n/**\n * Obtains the total of documents of the cursor\n * \n * @method Cursor#count\n * \n * @returns {Number} The total of documents in the cursor\n */\nCursor.prototype.count = function() {\n    return this.fetchAll().length;\n};\n\n/**\n * Applies a sorting on the cursor\n * \n * @method Cursor#sort\n * \n * @param {Object|Array|String} spec - The sorting specification\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.sort = function(spec) {\n    var _sort = this.sort_compiled || null;\n    \n    if (spec) {\n        _sort = new Selector(spec, Selector.SORT_SELECTOR);\n    }\n    \n    if (_sort) {\n        if (spec) {\n            this.sortValue = spec;\n            this.sort_compiled = _sort;\n        } else {\n            // If no spec, do sort\n            if (!_.isNil(this.db_objects) && _.isArray(this.db_objects)) {\n                this.db_objects = this.db_objects.sort(_sort);\n                this.sorted = true;\n            }\n        }\n    }\n    \n    return this;\n};\n\n/**\n * Set the number of document to skip when fetching the cursor\n * \n * @method Cursor#skip\n * \n * @param {Number} skip - The number of documents to skip\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.skip = function(skip) {\n    if (_.isNil(skip) || _.isNaN(skip)) throw new Error(\"Must pass a number\");\n    \n    this.skipValue = skip;\n    \n    return this;\n};\n\n/**\n * Set the max number of document to fetch\n * \n * @method Cursor#limit\n * \n * @param {Number} limit - The max number of documents\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.limit = function(limit) {\n    if (_.isNil(limit) || _.isNaN(limit)) throw new Error(\"Must pass a number\");\n    \n    this.limitValue = limit;\n    \n    return this;\n};\n\n/**\n * Checks if a cursor has a sorting defined\n * \n * @method hasSorting\n * @private\n * \n * @param {Cursor} cursor - The cursor\n * \n * @returns {Boolean} Whether the cursor has sorting or not\n */\nvar hasSorting = function(cursor) {\n    if (_.isNil(cursor.sortValue)) return false;\n    \n    return true;\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.batchSize = function() {\n    // Controls the number of documents MongoDB will return to the client in a single network message.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.close = function() {\n    // Close a cursor and free associated server resources.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.comment = function() {\n    // Attaches a comment to the query to allow for traceability in the logs and the system.profile collection.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.explain = function() {\n    // Reports on the query execution plan for a cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.hint = function() {\n    // Forces MongoDB to use a specific index for a query.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.itcount = function() {\n    // Computes the total number of documents in the cursor client-side by fetching and iterating the result set.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.maxScan = function() {\n    // Specifies the maximum number of items to scan; documents for collection scans, keys for index scans.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.maxTimeMS = function() {\n    // Specifies a cumulative time limit in milliseconds for processing operations on a cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.max = function() {\n    // Specifies an exclusive upper index bound for a cursor. For use with cursor.hint()\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.min = function() {\n    // Specifies an inclusive lower index bound for a cursor. For use with cursor.hint()\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.noCursorTimeout = function() {\n    // Instructs the server to avoid closing a cursor automatically after a period of inactivity.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.objsLeftInBatch = function() {\n    // Returns the number of documents left in the current cursor batch.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.pretty = function() {\n    // Configures the cursor to display results in an easy-to-read format.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.readConcern = function() {\n    // Specifies a read concern for a find() operation.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.readPref = function() {\n    // Specifies a read preference to a cursor to control how the client directs queries to a replica set.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.returnKey = function() {\n    // Modifies the cursor to return index keys rather than the documents.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.showRecordId = function() {\n    // Adds an internal storage engine ID field to each document returned by the cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.size = function() {\n    // Returns a count of the documents in the cursor after applying skip() and limit() methods.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.snapshot = function() {\n    // Forces the cursor to use the index on the _id field. Ensures that the cursor returns each document, \n    // with regards to the value of the _id field, only once.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.tailable = function() {\n    // Marks the cursor as tailable. Only valid for cursors over capped collections.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.toArray = function() {\n    // Returns an array that contains all documents returned by the cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\nmodule.exports = Cursor;"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Cursor.js"],"names":[],"mappings":"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","file":"Cursor.js","sourcesContent":["/**\n * @file Cursor.js - based on Monglo#Cursor ({@link https://github.com/Monglo}) by Christian Sullivan <cs@euforic.co> | Copyright (c) 2012\n * @version 1.0.0\n * \n * @author Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @copyright 2016 Eduardo Astolfi <eduardo.astolfi91@gmail.com>\n * @license MIT Licensed\n */\n\nvar Logger = require(\"jsw-logger\"),\n    _ = require(\"lodash\"),\n    Selector = require('./Selector');\n    \nvar logger = null;\n\n/**\n * Cursor\n * \n * @module Cursor\n * @constructor\n * @since 0.0.1\n * \n * @classdesc Cursor class that maps a MongoDB-like cursor\n * \n * @param {MongoPortable} db - Additional options\n * @param {Array} documents - The list of documents\n * @param {Object|Array|String} [selection={}] - The selection for matching documents\n * @param {Object|Array|String} [fields={}] - The fields of the document to show\n * @param {Object} [options] - Database object\n * \n * @param {Object} [options.pkFactory=null] - Object overriding the basic \"ObjectId\" primary key generation.\n * \n */\nclass Cursor {\n    constructor(documents, selection, fields, options = {}) {\n        this.documents = documents;\n        this.selector = selection;\n        this.skipValue = options.skip || 0;\n        this.limitValue = options.limit || 15;\n        this.sortValue = options.sort || null;\n        this.sorted = false;\n        \n        logger = Logger.instance;\n    \n        /** ADD IDX **/\n        if (Selector.isSelectorCompiled(this.selector)) {\n            this.selector_compiled = this.selector;\n        } else {\n            this.selector_compiled = new Selector(this.selector, Selector.MATCH_SELECTOR);\n        }\n        \n        for (let i = 0; i < this.selector_compiled.clauses.length; i++) {\n            if (this.selector_compiled.clauses[i].key === '_id') {\n                this.selector_id = this.selector_compiled.clauses[i].value;\n            }\n        }\n        \n        for (let i = 0; i < this.selector_compiled.clauses.length; i++) {\n            if (this.selector_compiled.clauses[i].key === '_id') {\n                var _val = this.selector_compiled.clauses[i].value;\n                \n                if (_.isString(_val) || _.isNumber(_val)) {\n                    this.selector_id = _val;\n                }\n            }\n        }\n\n        /** ADD IDX **/\n        \n        this.fetch_mode = Cursor.COLSCAN || Cursor.IDXSCAN;\n        this.indexex = null;//findUsableIndexes();\n        \n        // if (cursor.fetch_mode === Cursor.COLSCAN) {\n        //     // COLSCAN, wi will iterate over all documents\n        //     docs = _.cloneDeep(cursor.collection.docs);\n        // } else if (cursor.fetch_mode === Cursor.IDXSCAN) {\n        //     // IDXSCAN, wi will iterate over all needed documents\n        //     for (let i = 0; i < cursor.indexes.length; i++) {\n        //         let index = cursor.indexes[i];\n                \n        //         for (let i = index.start; i < index.end; i++) {\n        //             let idx_id = cursor.collection.getIndex(index.name)[i];\n                    \n        //             docs.push(cursor.collection.docs[idx_id]);\n        //         }\n        //     }\n        // }\n        \n        this.fields = new Selector(fields, Selector.FIELD_SELECTOR);\n        \n        this.sort_compiled = new Selector(this.sortValue, Selector.SORT_SELECTOR);\n    \n        this.db_objects = null;\n        this.cursor_pos = 0;\n    }\n}\n\nCursor.COLSCAN = 'colscan';\nCursor.IDXSCAN = 'idxscan';\n\n/**\n * Moves a cursor to the begining\n * \n * @method Cursor#rewind\n */\nCursor.prototype.rewind = function() {\n    this.db_objects = null;\n    this.cursor_pos = 0;\n};\n\n/**\n * Iterates over the cursor, calling a callback function\n * \n * @method Cursor#forEach\n * \n * @param {Function} [callback=null] - Callback function to be called for each document\n */\nCursor.prototype.forEach = function(callback) {\n    let docs = this.fetchAll();\n    \n    for (let i = 0; i < docs.length; i++) {\n        callback(docs[i]);\n    }\n};\n\n/**\n * Iterates over the cursor, returning a new array with the documents affected by the callback function\n * \n * @method Cursor#map\n * \n * @param {Function} [callback=null] - Callback function to be called for each document\n * \n * @returns {Array} The documents after being affected with the callback function\n */\nCursor.prototype.map = function(callback) {\n    var res = [];\n\n    this.forEach(function (doc) {\n        res.push(callback(doc));\n    });\n\n    return res;\n};\n\n/**\n * Checks if the cursor has one document to be fetched\n * \n * @method Cursor#hasNext\n * \n * @returns {Boolean} True if we can fetch one more document\n */\nCursor.prototype.hasNext = function() {\n    return (this.cursor_pos < this.documents.length);\n};\n\n/**\n * Alias for {@link Cursor#fetchOne}\n * \n * @method Cursor#next\n */\nCursor.prototype.next = function() {\n    return this.fetchOne();\n};\n\n/**\n * Alias for {@link Cursor#fetchAll}\n * \n * @method Cursor#fetch\n */\nCursor.prototype.fetch = function() {\n    return this.fetchAll();\n};\n\n/**\n * Fetch all documents in the cursor\n * \n * @method Cursor#fetchAll\n * \n * @returns {Array} All the documents contained in the cursor\n */\nCursor.prototype.fetchAll = function() {\n    return _getDocuments(this, false) || [];\n};\n\n/**\n * Retrieves the next document in the cursor\n * \n * @method Cursor#fetchOne\n * \n * @returns {Object} The next document in the cursor\n */\nCursor.prototype.fetchOne = function() {\n    return _getDocuments(this, true);\n};\n\nvar _mapFields = function(doc, fields) {\n    var _doc = _.cloneDeep(doc);\n\n    if (!_.isNil(fields) && _.isPlainObject(fields) && !_.isEqual(fields, {})) {\n        var showId = true,\n            showing = null;\n\n        // Whether if we showing the _id field\n        if (_.hasIn(fields, '_id') && fields._id === -1) {\n            showId = false;\n        }\n\n        var tmp = null;\n\n        for (var field in fields) {\n            // Whether if we are showing or hidding fields\n            if (field !== '_id' && showing === null) {\n                showing = fields[field] === 1 ? true : false;\n            }\n\n            if (showing != null) {\n                if (tmp === null) {\n                    if (showing) {\n                        tmp = {};\n                    } else {\n                        tmp = _.cloneDeep(doc);\n                    }\n                }\n        \n                // Add or remove the field\n                if (showing) {\n                    tmp[field] = doc[field];\n                } else {\n                    delete tmp[field];\n                }\n            }\n        }\n\n        // Add or remove the _id field\n        if (showId) {\n            tmp._id = doc._id;\n        } else {\n            delete tmp._id;\n        }\n\n        _doc = tmp;\n    }\n\n    return _doc;\n};\n\n/**\n * Retrieves one or all the documents in the cursor\n * \n * @method _getDocuments\n * @private\n * \n * @param {Cursor} cursor - The cursor with the documents\n * @param {Boolean} [justOne=false] - Whether it retrieves one or all the documents\n * \n * @returns {Array|Object} If [justOne=true] returns the next document, otherwise returns all the documents\n */\nvar _getDocuments = function(cursor, justOne = false) {\n    var docs = [];\n    \n    if (cursor.fetch_mode === Cursor.COLSCAN) {\n        // COLSCAN, wi will iterate over all documents\n        docs = _.cloneDeep(cursor.documents);\n    } else if (cursor.fetch_mode === Cursor.IDXSCAN) {\n        // IDXSCAN, wi will iterate over all needed documents\n        for (let i = 0; i < cursor.indexes.length; i++) {\n            let index = cursor.indexes[i];\n            \n            for (let i = index.start; i < index.end; i++) {\n                // let idx_id = cursor.collection.getIndex(index.name)[i];\n                let idx_id = index.index[i];\n                \n                docs.push(cursor.documents[idx_id]);\n            }\n        }\n    }\n    \n    // if (cursor.selector_id) {\n    //     if (_.hasIn(cursor.collection.doc_indexes, _.toString(cursor.selector_id))) {\n    //         let idx = cursor.collection.doc_indexes[_.toString(cursor.selector_id)];\n            \n    //         return _mapFields(cursor.collection.docs[idx], cursor.fields);\n    //     } else {\n    //         if (justOne) {\n    //             return null;\n    //         } else {\n    //             return [];\n    //         }\n    //     }\n    // }\n    \n    // TODO add warning when sort/skip/limit and fetching one\n    // TODO add warning when skip/limit without order\n    // TODO index\n    while (cursor.cursor_pos < docs.length) {\n        var _doc = docs[cursor.cursor_pos];\n        cursor.cursor_pos++;\n        \n        if (cursor.selector_compiled.test(_doc)) {\n            if (_.isNil(cursor.db_objects)) cursor.db_objects = [];\n            \n            _doc = _mapFields(_doc, cursor.fields);\n            \n            cursor.db_objects.push(_doc);\n            \n            if (justOne) {\n                // Add force sort\n                return _doc;\n            }\n        }\n    }\n    \n    if (_.isNil(cursor.db_objects)) return null;\n    \n    if (!cursor.sorted && hasSorting(cursor)) cursor.sort();\n    \n    var idxFrom = cursor.skipValue;\n    var idxTo = cursor.limitValue !== -1 ? (cursor.limitValue + idxFrom) : cursor.db_objects.length;\n    \n    return cursor.db_objects.slice(idxFrom, idxTo);\n    \n};\n\n/**\n * Obtains the total of documents of the cursor\n * \n * @method Cursor#count\n * \n * @returns {Number} The total of documents in the cursor\n */\nCursor.prototype.count = function() {\n    return this.fetchAll().length;\n};\n\n/**\n * Set the sorting of the cursor\n * \n * @method Cursor#sort\n * \n * @param {Object|Array|String} spec - The sorting specification\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.setSorting = function(spec) {\n    if (_.isNil(spec)) logger.throw(\"You need to specify a sorting\");\n    \n    if (spec) {\n        this.sortValue = spec;\n        this.sort_compiled = (new Selector(spec, Selector.SORT_SELECTOR));\n    }\n    \n    return this;\n};\n\n/**\n * Applies a sorting on the cursor\n * \n * @method Cursor#sort\n * \n * @param {Object|Array|String} spec - The sorting specification\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.sort = function(spec) {\n    var _sort = this.sort_compiled || null;\n    \n    if (spec) {\n        _sort = new Selector(spec, Selector.SORT_SELECTOR);\n    }\n    \n    if (_sort) {\n        if (!_.isNil(this.db_objects) && _.isArray(this.db_objects)) {\n            this.db_objects = this.db_objects.sort(_sort);\n            this.sorted = true;\n        } else {\n            this.setSorting(spec);\n        }\n    }\n    \n    return this;\n};\n\n/**\n * Set the number of document to skip when fetching the cursor\n * \n * @method Cursor#skip\n * \n * @param {Number} skip - The number of documents to skip\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.skip = function(skip) {\n    if (_.isNil(skip) || _.isNaN(skip)) throw new Error(\"Must pass a number\");\n    \n    this.skipValue = skip;\n    \n    return this;\n};\n\n/**\n * Set the max number of document to fetch\n * \n * @method Cursor#limit\n * \n * @param {Number} limit - The max number of documents\n * \n * @returns {Cursor} This instance so it can be chained with other methods\n */\nCursor.prototype.limit = function(limit) {\n    if (_.isNil(limit) || _.isNaN(limit)) throw new Error(\"Must pass a number\");\n    \n    this.limitValue = limit;\n    \n    return this;\n};\n\n/**\n * Checks if a cursor has a sorting defined\n * \n * @method hasSorting\n * @private\n * \n * @param {Cursor} cursor - The cursor\n * \n * @returns {Boolean} Whether the cursor has sorting or not\n */\nvar hasSorting = function(cursor) {\n    if (_.isNil(cursor.sortValue)) return false;\n    \n    return true;\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.batchSize = function() {\n    // Controls the number of documents MongoDB will return to the client in a single network message.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.close = function() {\n    // Close a cursor and free associated server resources.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.comment = function() {\n    // Attaches a comment to the query to allow for traceability in the logs and the system.profile collection.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.explain = function() {\n    // Reports on the query execution plan for a cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.hint = function() {\n    // Forces MongoDB to use a specific index for a query.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.itcount = function() {\n    // Computes the total number of documents in the cursor client-side by fetching and iterating the result set.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.maxScan = function() {\n    // Specifies the maximum number of items to scan; documents for collection scans, keys for index scans.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.maxTimeMS = function() {\n    // Specifies a cumulative time limit in milliseconds for processing operations on a cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.max = function() {\n    // Specifies an exclusive upper index bound for a cursor. For use with cursor.hint()\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.min = function() {\n    // Specifies an inclusive lower index bound for a cursor. For use with cursor.hint()\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.noCursorTimeout = function() {\n    // Instructs the server to avoid closing a cursor automatically after a period of inactivity.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.objsLeftInBatch = function() {\n    // Returns the number of documents left in the current cursor batch.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.pretty = function() {\n    // Configures the cursor to display results in an easy-to-read format.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.readConcern = function() {\n    // Specifies a read concern for a find() operation.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.readPref = function() {\n    // Specifies a read preference to a cursor to control how the client directs queries to a replica set.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.returnKey = function() {\n    // Modifies the cursor to return index keys rather than the documents.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.showRecordId = function() {\n    // Adds an internal storage engine ID field to each document returned by the cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.size = function() {\n    // Returns a count of the documents in the cursor after applying skip() and limit() methods.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.snapshot = function() {\n    // Forces the cursor to use the index on the _id field. Ensures that the cursor returns each document, \n    // with regards to the value of the _id field, only once.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.tailable = function() {\n    // Marks the cursor as tailable. Only valid for cursors over capped collections.\n    throw new Error(\"Not yet implemented\");\n};\n\n/**\n * @todo Implement\n */\nCursor.prototype.toArray = function() {\n    // Returns an array that contains all documents returned by the cursor.\n    throw new Error(\"Not yet implemented\");\n};\n\nmodule.exports = Cursor;"]} diff --git a/lib/Selector.js b/lib/Selector.js index 367cbbc..4ff6588 100644 --- a/lib/Selector.js +++ b/lib/Selector.js @@ -6,7 +6,8 @@ function _classCallCheck(instance, Constructor) { if (!(instance instanceof Cons var Logger = require("jsw-logger"), _ = require("lodash"), - SelectorMatcher = require("./SelectorMatcher"); + SelectorMatcher = require("./SelectorMatcher"), + ObjectId = require("./ObjectId"); var logger = null; @@ -430,6 +431,11 @@ var _buildKeypathSelector = function _buildKeypathSelector(keypath, value) { clause.type = 'operator_object'; } + } else if (value instanceof ObjectId) { + logger.debug('clause of type ObjectId -> String'); + + clause.type = 'string'; + clause.value = value.toString(); } else { clause.type = '__invalid__'; } @@ -457,4 +463,4 @@ Selector.SORT_SELECTOR = 'sort'; Selector.FIELD_SELECTOR = 'field'; module.exports = Selector; -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Selector.js"],"names":[],"mappings":"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","file":"Selector.js","sourcesContent":["var Logger = require(\"jsw-logger\"),\n    _ = require(\"lodash\"),\n    SelectorMatcher = require(\"./SelectorMatcher\");\n    \nvar logger = null;\n\nclass Selector {\n    constructor(selector, type = Selector.MATCH_SELECTOR) {\n        logger = Logger.instance;\n        \n        this.selector_compiled = null;\n\t\t\n\t\tif (type === Selector.MATCH_SELECTOR) {\n\t\t\tthis.selector_compiled = this.compile(selector);\n\t\t} else if (type === Selector.SORT_SELECTOR) {\n\t\t\treturn this.compileSort(selector);\n\t\t} else if (type === Selector.FIELD_SELECTOR) {\n\t\t\treturn this.compileFields(selector);\n\t\t} else {\n\t\t\tlogger.throw(\"You need to specify the selector type\");\n\t\t}\n    }\n    \n    test(doc) {\n        return this.selector_compiled.test(doc);\n    }\n    \n    compile(selector) {\n\t\tif (_.isNil(selector)) {\n\t\t\tlogger.debug('selector -> null');\n\t\t\t\n\t\t\tselector = {};\n\t\t} else {\n\t\t\tlogger.debug('selector -> not null');\n\t\t\t\n\t\t\tif (!selector || (_.hasIn(selector, '_id') && !selector._id)) {\n\t\t\t\tlogger.debug('selector -> false value || { _id: false value }');\n\t\t\t\t\n\t\t\t\tselector = {\n\t\t\t\t\t_id: false\n\t\t\t\t};\n\t\t\t}\n\t\t}\n\t\t\n\t\tif (_.isFunction(selector)) {\n\t\t\tlogger.debug('selector -> function(doc) { ... }');\n\t\t\t\n\t\t\t//_initFunction.call(matcher, selector);\n\t\t\tthis.clauses = [{\n\t\t\t\tkind: 'function',\n\t\t\t\tvalue: selector\n\t\t\t}];\n\t\t\t\n\t\t\tlogger.debug('clauses created: ' + JSON.stringify(this.clauses));\n\t\t} else if (_.isString(selector) || _.isNumber(selector)) {\n\t\t\tlogger.debug('selector -> \"123456789\" || 123456798');\n\t\t\t\n\t\t\tselector = {\n\t\t\t\t_id: selector\n\t\t\t};\n\t\t\t\n\t\t\t//_initObject.call(matcher, selector);\n\t\t\tthis.clauses = _buildSelector(selector);\n\t\t\t\n\t\t\tlogger.debug('clauses created: ' + JSON.stringify(this.clauses));\n\t\t} else {\n\t\t\tlogger.debug('selector -> { field: value }');\n\t\t\t\n\t\t\t//_initObject.call(matcher, selector);\n\t\t\tthis.clauses = _buildSelector(selector);\n\t\t\t\n\t\t\tlogger.debug('clauses created: ' + JSON.stringify(this.clauses));\n\t\t}\n\t\t\n\t\tvar matcher = new SelectorMatcher(this);\n\t\t\n\t\treturn matcher;\n    }\n    \n    compileSort(spec) {\n        if (_.isNil(spec))  {\n            return function () {\n                return 0;\n            };\n        }\n        \n        var keys = [];\n        var asc = [];\n        \n        if (_.isString(spec)) {\n            spec = spec.replace(/( )+/ig, ' ').trim();\n            \n            if (spec.indexOf(',') !== -1) {\n                // Replace commas by spaces, and treat it as a spaced-separated string\n                return this.compileSort(spec.replace(/,/ig, ' '));\n            } else if (spec.indexOf(' ') !== -1) {\n                var fields = spec.split(' ');\n                \n                for (var i = 0; i < fields.length; i++) {\n                    var field = fields[i].trim();\n                    \n                    if ((field === 'desc'  || field === 'asc') ||\n                        (field === '-1'    || field === '1') ||\n                        (field === 'false' || field === 'true')) {\n                            \n                        throw Error(\"Bad sort specification: \", JSON.stringify(spec));\n                    } else {\n                        var next = _.toString(fields[i+1]);\n                        \n                        if (next === 'desc' || next === 'asc') {\n                            keys.push(field);\n                            asc.push((next === 'asc') ? true : false);\n                            \n                            i++;\n                        } else if (next === '-1' || next === '1') {\n                            keys.push(field);\n                            asc.push((next === '1') ? true : false);\n                            \n                            i++;\n                        } else if (next === 'false' || next === 'true') {\n                            keys.push(field);\n                            asc.push((next === 'true') ? true : false);\n                            \n                            i++;\n                        } else {\n                            keys.push(field);\n                            asc.push(true); // Default sort\n                        }\n                    }\n                }\n            } else {\n                //.sort(\"field1\")\n                \n                keys.push(spec);\n                asc.push(true);\n            }\n        } else if (_.isArray(spec)) {\n            // Join the array with spaces, and treat it as a spaced-separated string\n            return this.compileSort(spec.join(' '));\n            // for (var i = 0; i < spec.length; i++) {\n            //     if (_.isString(spec[i])) {\n            //         keys.push(spec[i]);\n            //         asc.push(true);\n            //     } else {\n            //         keys.push(spec[i][0]);\n            //         asc.push(spec[i][1] !== \"desc\");\n            //     }\n            // }\n        } else if (_.isPlainObject(spec)) {\n            // TODO Nested path -> .sort({ \"field1.field12\": \"asc\" })\n            var _spec = [];\n            for (var key in spec) {\n                if (_.hasIn(spec, key)) {\n                    _spec.push(key);\n                    _spec.push(spec[key]);\n                }\n            }\n            \n            return this.compileSort(_spec);\n        } else {\n            throw Error(\"Bad sort specification: \", JSON.stringify(spec));\n        }\n    \n        // return {keys: keys, asc: asc};\n        return function(a, b) {\n            var x = 0;\n            \n            for (var i = 0; i < keys.length; i++) {\n                if (i !== 0 && x !== 0) return x;   // Non reachable?\n                \n                \n                // x = Selector._f._cmp(a[JSON.stringify(keys[i])], b[JSON.stringify(keys[i])]);\n                x = SelectorMatcher.cmp(a[keys[i]], b[keys[i]]);\n                \n                if (!asc[i]) {\n                    x *= -1;\n                }\n            }\n            \n            return x;\n        };\n        \n        // eval() does not return a value in IE8, nor does the spec say it\n        // should. Assign to a local to get the value, instead.\n        \n        // var _func;\n        // var code = \"_func = (function(c){return function(a,b){var x;\";\n        // for (var i = 0; i < keys.length; i++) {\n        //     if (i !== 0) {\n        //         code += \"if(x!==0)return x;\";\n        //     }\n    \n        //     code += \"x=\" + (asc[i] ? \"\" : \"-\") + \"c(a[\" + JSON.stringify(keys[i]) + \"],b[\" + JSON.stringify(keys[i]) + \"]);\";\n        // }\n    \n        // code += \"return x;};})\";\n    \n        // eval(code);\n    \n        // return _func(Selector._f._cmp);\n    }\n    \n    compileFields(spec) {\n        var projection = {};\n        \n        if (_.isNil(spec)) return projection;\n        \n        if (_.isString(spec)) {\n            spec = spec.replace(/( )+/ig, ' ').trim();\n            \n            if (spec.indexOf(',') !== -1) {\n                // Replace commas by spaces, and treat it as a spaced-separated string\n                return this.compileFields(spec.replace(/,/ig, ' '));\n            } else if (spec.indexOf(' ') !== -1) {\n                var fields = spec.split(' ');\n                \n                for (var i = 0; i < fields.length; i++) {\n                    var field = fields[i].trim();\n                    \n                    if ((field === '-1'    || field === '1') ||\n                        (field === 'false' || field === 'true')) {\n                            \n                        throw Error(\"Bad fields specification: \", JSON.stringify(spec));\n                    } else {\n                        var next = _.toString(fields[i+1]);\n                        \n                        if (next === '-1' || next === '1') {\n                            if (next === '-1') {\n                                for (let _key in projection) {\n                                    if (field !== '_id' && projection[_key] === 1) {\n                                        throw new Error(\"A projection cannot contain both include and exclude specifications\");\n                                    }\n                                }\n                                \n                                projection[field] = -1;\n                            } else {\n                                projection[field] = 1;\n                            }\n                            \n                            i++;\n                        } else if (next === 'false' || next === 'true') {\n                            if (next === 'false') {\n                                if (field === '_id') {\n                                    projection[field] = -1;\n                                } else {\n                                    throw new Error(\"A projection cannot contain both include and exclude specifications\");\n                                }\n                            } else {\n                                projection[field] = 1;\n                            }\n                            \n                            i++;\n                        } else {\n                            projection[field] = 1;\n                        }\n                    }\n                }\n            } else if (spec.length > 0) {\n                //.find({}, \"field1\")\n                \n                projection[spec] = 1;\n            }\n        } else if (_.isArray(spec)) {\n            // Join the array with spaces, and treat it as a spaced-separated string\n            return this.compileFields(spec.join(' '));\n        } else if (_.isPlainObject(spec)) {\n            // TODO Nested path -> .find({}, { \"field1.field12\": \"asc\" })\n            var _spec = [];\n            for (var key in spec) {\n                if (_.hasIn(spec, key)) {\n                    _spec.push(key);\n                    _spec.push(spec[key]);\n                }\n            }\n            \n            return this.compileFields(_spec);\n        } else {\n            throw Error(\"Bad fields specification: \", JSON.stringify(spec));\n        }\n        \n        return projection;\n    }\n\t\n\t/* STATIC METHODS */\n\tstatic isSelectorCompiled(selector) {\n\t\tif (!_.isNil(selector) && (\n\t\t    selector instanceof SelectorMatcher || (selector instanceof Selector && \n\t\t                                            selector.selector_compiled instanceof SelectorMatcher)\n\t    )) {\n\t\t\treturn true;\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n\t\n\tstatic matches(selector, doc) {\n        return (new Selector(selector)).test(doc);\n    }\n}\n\nvar _buildSelector = function(selector) {\n\tlogger.debug('Called: _buildSelector');\n    \n    var clauses = [];\n    \n    for (var key in selector) {\n        var value = selector[key];\n        \n        if (key.charAt(0) === '$') {\n            logger.debug('selector -> operator => { $and: [{...}, {...}] }');\n            \n            clauses.push(_buildDocumentSelector(key, value));\n        } else {\n            logger.debug('selector -> plain => { field1: <value> }');\n            \n            clauses.push(_buildKeypathSelector(key, value));\n        }\n    }\n    \n    return clauses;\n};\n\nvar _buildDocumentSelector = function(key, value) {\n    var clause = {};\n    \n    switch (key) {\n        case '$or':\n        case '$and':\n        case '$nor':\n            clause.key = key.replace(/\\$/, '');\n            // The rest will be handled by '_operator_'\n        case '_operator_':\n            // Generic handler for operators ($or, $and, $nor)\n            \n            clause.kind = 'operator';\n            clause.type = 'array';\n            \n            clause.value = [];\n            for (let i = 0; i < value.length; i++) {\n                clause.value = _.union(clause.value, _buildSelector(value[i]));\n            }\n            \n            break;\n        default:\n            throw Error(\"Unrecogized key in selector: \", key);\n    }\n    \n    // TODO cases: $where, $elemMatch\n    \n    logger.debug('clause created: ' + JSON.stringify(clause));\n    \n    return clause;\n};\n\nvar _buildKeypathSelector = function (keypath, value) {\n    logger.debug('Called: _buildKeypathSelector');\n    \n    var clause = {};\n    \n    clause.value = value;\n    \n    if (_.isNil(value)) {\n        logger.debug('clause of type null');\n        \n        clause.type = 'null';\n    } else if (_.isRegExp(value)) {\n        logger.debug('clause of type RegExp');\n\n        clause.type = 'regexp';\n        \n        var source = value.toString().split('/');\n\n        clause.value = {\n            $regex: source[1]   // The first item splitted is an empty string\n        };\n        \n        if (source[2] != \"\") {\n            clause.value[\"$options\"] = source[2];\n        }\n    } else if (_.isArray(value)) {\n        logger.debug('clause of type Array');\n        \n        clause.type = 'array';\n    } else if (_.isString(value)) {\n        logger.debug('clause of type String');\n        \n        clause.type = 'string';\n    } else if (_.isNumber(value)) {\n        logger.debug('clause of type Number');\n        \n        clause.type = 'number';\n    } else if (_.isBoolean(value)) {\n        logger.debug('clause of type Boolean');\n        \n        clause.type = 'boolean';\n    } else if (_.isFunction(value)) {\n        logger.debug('clause of type Function');\n        \n        clause.type = 'function';\n    } else if (_.isPlainObject(value)) {\n        var literalObject = true;\n        for (var key in value) {\n            if (key.charAt(0) === '$') {\n                literalObject = false;\n                break;\n            }\n        }\n        \n        if (literalObject) {\n            logger.debug('clause of type Object => { field: { field_1: <value>, field_2: <value> } }');\n            \n            clause.type = 'literal_object';\n        } else {\n            logger.debug('clause of type Operator => { field: { $gt: 2, $lt 5 } }');\n            \n            clause.type = 'operator_object';\n        }\n    } else {\n        clause.type = '__invalid__';\n    }\n    \n    var parts = keypath.split('.');\n    if (parts.length > 1) {\n        logger.debug('clause over Object field => { \"field1.field1_2\": <value> }');\n        \n        clause.kind = 'object';\n        clause.key = parts;\n    } else {\n        logger.debug('clause over Plain field => { \"field\": <value> }');\n        \n        clause.kind = 'plain';\n        clause.key = parts[0];\n    }\n    \n    logger.debug('clause created: ' + JSON.stringify(clause));\n    \n    return clause;\n};\n\nSelector.MATCH_SELECTOR = 'match';\nSelector.SORT_SELECTOR = 'sort';\nSelector.FIELD_SELECTOR = 'field';\n\nmodule.exports = Selector;"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/Selector.js"],"names":[],"mappings":"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","file":"Selector.js","sourcesContent":["var Logger = require(\"jsw-logger\"),\n    _ = require(\"lodash\"),\n    SelectorMatcher = require(\"./SelectorMatcher\"),\n    ObjectId = require(\"./ObjectId\");\n    \nvar logger = null;\n\nclass Selector {\n    constructor(selector, type = Selector.MATCH_SELECTOR) {\n        logger = Logger.instance;\n        \n        this.selector_compiled = null;\n\t\t\n\t\tif (type === Selector.MATCH_SELECTOR) {\n\t\t\tthis.selector_compiled = this.compile(selector);\n\t\t} else if (type === Selector.SORT_SELECTOR) {\n\t\t\treturn this.compileSort(selector);\n\t\t} else if (type === Selector.FIELD_SELECTOR) {\n\t\t\treturn this.compileFields(selector);\n\t\t} else {\n\t\t\tlogger.throw(\"You need to specify the selector type\");\n\t\t}\n    }\n    \n    test(doc) {\n        return this.selector_compiled.test(doc);\n    }\n    \n    compile(selector) {\n\t\tif (_.isNil(selector)) {\n\t\t\tlogger.debug('selector -> null');\n\t\t\t\n\t\t\tselector = {};\n\t\t} else {\n\t\t\tlogger.debug('selector -> not null');\n\t\t\t\n\t\t\tif (!selector || (_.hasIn(selector, '_id') && !selector._id)) {\n\t\t\t\tlogger.debug('selector -> false value || { _id: false value }');\n\t\t\t\t\n\t\t\t\tselector = {\n\t\t\t\t\t_id: false\n\t\t\t\t};\n\t\t\t}\n\t\t}\n\t\t\n\t\tif (_.isFunction(selector)) {\n\t\t\tlogger.debug('selector -> function(doc) { ... }');\n\t\t\t\n\t\t\t//_initFunction.call(matcher, selector);\n\t\t\tthis.clauses = [{\n\t\t\t\tkind: 'function',\n\t\t\t\tvalue: selector\n\t\t\t}];\n\t\t\t\n\t\t\tlogger.debug('clauses created: ' + JSON.stringify(this.clauses));\n\t\t} else if (_.isString(selector) || _.isNumber(selector)) {\n\t\t\tlogger.debug('selector -> \"123456789\" || 123456798');\n\t\t\t\n\t\t\tselector = {\n\t\t\t\t_id: selector\n\t\t\t};\n\t\t\t\n\t\t\t//_initObject.call(matcher, selector);\n\t\t\tthis.clauses = _buildSelector(selector);\n\t\t\t\n\t\t\tlogger.debug('clauses created: ' + JSON.stringify(this.clauses));\n\t\t} else {\n\t\t\tlogger.debug('selector -> { field: value }');\n\t\t\t\n\t\t\t//_initObject.call(matcher, selector);\n\t\t\tthis.clauses = _buildSelector(selector);\n\t\t\t\n\t\t\tlogger.debug('clauses created: ' + JSON.stringify(this.clauses));\n\t\t}\n\t\t\n\t\tvar matcher = new SelectorMatcher(this);\n\t\t\n\t\treturn matcher;\n    }\n    \n    compileSort(spec) {\n        if (_.isNil(spec))  {\n            return function () {\n                return 0;\n            };\n        }\n        \n        var keys = [];\n        var asc = [];\n        \n        if (_.isString(spec)) {\n            spec = spec.replace(/( )+/ig, ' ').trim();\n            \n            if (spec.indexOf(',') !== -1) {\n                // Replace commas by spaces, and treat it as a spaced-separated string\n                return this.compileSort(spec.replace(/,/ig, ' '));\n            } else if (spec.indexOf(' ') !== -1) {\n                var fields = spec.split(' ');\n                \n                for (var i = 0; i < fields.length; i++) {\n                    var field = fields[i].trim();\n                    \n                    if ((field === 'desc'  || field === 'asc') ||\n                        (field === '-1'    || field === '1') ||\n                        (field === 'false' || field === 'true')) {\n                            \n                        throw Error(\"Bad sort specification: \", JSON.stringify(spec));\n                    } else {\n                        var next = _.toString(fields[i+1]);\n                        \n                        if (next === 'desc' || next === 'asc') {\n                            keys.push(field);\n                            asc.push((next === 'asc') ? true : false);\n                            \n                            i++;\n                        } else if (next === '-1' || next === '1') {\n                            keys.push(field);\n                            asc.push((next === '1') ? true : false);\n                            \n                            i++;\n                        } else if (next === 'false' || next === 'true') {\n                            keys.push(field);\n                            asc.push((next === 'true') ? true : false);\n                            \n                            i++;\n                        } else {\n                            keys.push(field);\n                            asc.push(true); // Default sort\n                        }\n                    }\n                }\n            } else {\n                //.sort(\"field1\")\n                \n                keys.push(spec);\n                asc.push(true);\n            }\n        } else if (_.isArray(spec)) {\n            // Join the array with spaces, and treat it as a spaced-separated string\n            return this.compileSort(spec.join(' '));\n            // for (var i = 0; i < spec.length; i++) {\n            //     if (_.isString(spec[i])) {\n            //         keys.push(spec[i]);\n            //         asc.push(true);\n            //     } else {\n            //         keys.push(spec[i][0]);\n            //         asc.push(spec[i][1] !== \"desc\");\n            //     }\n            // }\n        } else if (_.isPlainObject(spec)) {\n            // TODO Nested path -> .sort({ \"field1.field12\": \"asc\" })\n            var _spec = [];\n            for (var key in spec) {\n                if (_.hasIn(spec, key)) {\n                    _spec.push(key);\n                    _spec.push(spec[key]);\n                }\n            }\n            \n            return this.compileSort(_spec);\n        } else {\n            throw Error(\"Bad sort specification: \", JSON.stringify(spec));\n        }\n    \n        // return {keys: keys, asc: asc};\n        return function(a, b) {\n            var x = 0;\n            \n            for (var i = 0; i < keys.length; i++) {\n                if (i !== 0 && x !== 0) return x;   // Non reachable?\n                \n                \n                // x = Selector._f._cmp(a[JSON.stringify(keys[i])], b[JSON.stringify(keys[i])]);\n                x = SelectorMatcher.cmp(a[keys[i]], b[keys[i]]);\n                \n                if (!asc[i]) {\n                    x *= -1;\n                }\n            }\n            \n            return x;\n        };\n        \n        // eval() does not return a value in IE8, nor does the spec say it\n        // should. Assign to a local to get the value, instead.\n        \n        // var _func;\n        // var code = \"_func = (function(c){return function(a,b){var x;\";\n        // for (var i = 0; i < keys.length; i++) {\n        //     if (i !== 0) {\n        //         code += \"if(x!==0)return x;\";\n        //     }\n    \n        //     code += \"x=\" + (asc[i] ? \"\" : \"-\") + \"c(a[\" + JSON.stringify(keys[i]) + \"],b[\" + JSON.stringify(keys[i]) + \"]);\";\n        // }\n    \n        // code += \"return x;};})\";\n    \n        // eval(code);\n    \n        // return _func(Selector._f._cmp);\n    }\n    \n    compileFields(spec) {\n        var projection = {};\n        \n        if (_.isNil(spec)) return projection;\n        \n        if (_.isString(spec)) {\n            spec = spec.replace(/( )+/ig, ' ').trim();\n            \n            if (spec.indexOf(',') !== -1) {\n                // Replace commas by spaces, and treat it as a spaced-separated string\n                return this.compileFields(spec.replace(/,/ig, ' '));\n            } else if (spec.indexOf(' ') !== -1) {\n                var fields = spec.split(' ');\n                \n                for (var i = 0; i < fields.length; i++) {\n                    var field = fields[i].trim();\n                    \n                    if ((field === '-1'    || field === '1') ||\n                        (field === 'false' || field === 'true')) {\n                            \n                        throw Error(\"Bad fields specification: \", JSON.stringify(spec));\n                    } else {\n                        var next = _.toString(fields[i+1]);\n                        \n                        if (next === '-1' || next === '1') {\n                            if (next === '-1') {\n                                for (let _key in projection) {\n                                    if (field !== '_id' && projection[_key] === 1) {\n                                        throw new Error(\"A projection cannot contain both include and exclude specifications\");\n                                    }\n                                }\n                                \n                                projection[field] = -1;\n                            } else {\n                                projection[field] = 1;\n                            }\n                            \n                            i++;\n                        } else if (next === 'false' || next === 'true') {\n                            if (next === 'false') {\n                                if (field === '_id') {\n                                    projection[field] = -1;\n                                } else {\n                                    throw new Error(\"A projection cannot contain both include and exclude specifications\");\n                                }\n                            } else {\n                                projection[field] = 1;\n                            }\n                            \n                            i++;\n                        } else {\n                            projection[field] = 1;\n                        }\n                    }\n                }\n            } else if (spec.length > 0) {\n                //.find({}, \"field1\")\n                \n                projection[spec] = 1;\n            }\n        } else if (_.isArray(spec)) {\n            // Join the array with spaces, and treat it as a spaced-separated string\n            return this.compileFields(spec.join(' '));\n        } else if (_.isPlainObject(spec)) {\n            // TODO Nested path -> .find({}, { \"field1.field12\": \"asc\" })\n            var _spec = [];\n            for (var key in spec) {\n                if (_.hasIn(spec, key)) {\n                    _spec.push(key);\n                    _spec.push(spec[key]);\n                }\n            }\n            \n            return this.compileFields(_spec);\n        } else {\n            throw Error(\"Bad fields specification: \", JSON.stringify(spec));\n        }\n        \n        return projection;\n    }\n\t\n\t/* STATIC METHODS */\n\tstatic isSelectorCompiled(selector) {\n\t\tif (!_.isNil(selector) && (\n\t\t    selector instanceof SelectorMatcher || (selector instanceof Selector && \n\t\t                                            selector.selector_compiled instanceof SelectorMatcher)\n\t    )) {\n\t\t\treturn true;\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n\t\n\tstatic matches(selector, doc) {\n        return (new Selector(selector)).test(doc);\n    }\n}\n\nvar _buildSelector = function(selector) {\n\tlogger.debug('Called: _buildSelector');\n    \n    var clauses = [];\n    \n    for (var key in selector) {\n        var value = selector[key];\n        \n        if (key.charAt(0) === '$') {\n            logger.debug('selector -> operator => { $and: [{...}, {...}] }');\n            \n            clauses.push(_buildDocumentSelector(key, value));\n        } else {\n            logger.debug('selector -> plain => { field1: <value> }');\n            \n            clauses.push(_buildKeypathSelector(key, value));\n        }\n    }\n    \n    return clauses;\n};\n\nvar _buildDocumentSelector = function(key, value) {\n    var clause = {};\n    \n    switch (key) {\n        case '$or':\n        case '$and':\n        case '$nor':\n            clause.key = key.replace(/\\$/, '');\n            // The rest will be handled by '_operator_'\n        case '_operator_':\n            // Generic handler for operators ($or, $and, $nor)\n            \n            clause.kind = 'operator';\n            clause.type = 'array';\n            \n            clause.value = [];\n            for (let i = 0; i < value.length; i++) {\n                clause.value = _.union(clause.value, _buildSelector(value[i]));\n            }\n            \n            break;\n        default:\n            throw Error(\"Unrecogized key in selector: \", key);\n    }\n    \n    // TODO cases: $where, $elemMatch\n    \n    logger.debug('clause created: ' + JSON.stringify(clause));\n    \n    return clause;\n};\n\nvar _buildKeypathSelector = function (keypath, value) {\n    logger.debug('Called: _buildKeypathSelector');\n    \n    var clause = {};\n    \n    clause.value = value;\n    \n    if (_.isNil(value)) {\n        logger.debug('clause of type null');\n        \n        clause.type = 'null';\n    } else if (_.isRegExp(value)) {\n        logger.debug('clause of type RegExp');\n\n        clause.type = 'regexp';\n        \n        var source = value.toString().split('/');\n\n        clause.value = {\n            $regex: source[1]   // The first item splitted is an empty string\n        };\n        \n        if (source[2] != \"\") {\n            clause.value[\"$options\"] = source[2];\n        }\n    } else if (_.isArray(value)) {\n        logger.debug('clause of type Array');\n        \n        clause.type = 'array';\n    } else if (_.isString(value)) {\n        logger.debug('clause of type String');\n        \n        clause.type = 'string';\n    } else if (_.isNumber(value)) {\n        logger.debug('clause of type Number');\n        \n        clause.type = 'number';\n    } else if (_.isBoolean(value)) {\n        logger.debug('clause of type Boolean');\n        \n        clause.type = 'boolean';\n    } else if (_.isFunction(value)) {\n        logger.debug('clause of type Function');\n        \n        clause.type = 'function';\n    } else if (_.isPlainObject(value)) {\n        var literalObject = true;\n        for (var key in value) {\n            if (key.charAt(0) === '$') {\n                literalObject = false;\n                break;\n            }\n        }\n        \n        if (literalObject) {\n            logger.debug('clause of type Object => { field: { field_1: <value>, field_2: <value> } }');\n            \n            clause.type = 'literal_object';\n        } else {\n            logger.debug('clause of type Operator => { field: { $gt: 2, $lt 5 } }');\n            \n            clause.type = 'operator_object';\n        }\n    } else if (value instanceof ObjectId) {\n        logger.debug('clause of type ObjectId -> String');\n        \n        clause.type = 'string';\n        clause.value = value.toString();\n    } else {\n        clause.type = '__invalid__';\n    }\n    \n    var parts = keypath.split('.');\n    if (parts.length > 1) {\n        logger.debug('clause over Object field => { \"field1.field1_2\": <value> }');\n        \n        clause.kind = 'object';\n        clause.key = parts;\n    } else {\n        logger.debug('clause over Plain field => { \"field\": <value> }');\n        \n        clause.kind = 'plain';\n        clause.key = parts[0];\n    }\n    \n    logger.debug('clause created: ' + JSON.stringify(clause));\n    \n    return clause;\n};\n\nSelector.MATCH_SELECTOR = 'match';\nSelector.SORT_SELECTOR = 'sort';\nSelector.FIELD_SELECTOR = 'field';\n\nmodule.exports = Selector;"]} diff --git a/src/Aggregation.js b/src/Aggregation.js index cf1f305..8693c99 100644 --- a/src/Aggregation.js +++ b/src/Aggregation.js @@ -8,20 +8,22 @@ */ var _ = require("lodash"), - Logger = require("jsw-logger"); + Logger = require("jsw-logger"), + Cursor = require("./Cursor"), + Selector = require("./Selector"); var logger = null; var stages = { '$project': false, - '$match': false, + '$match': true, '$redact': false, '$limit': false, '$skip': false, '$unwind': false, '$group': true, '$sample': false, - '$sort': false, + '$sort': true, '$geoNear': false, '$lookup': false, '$out': false, @@ -158,10 +160,28 @@ var do_complex_group = function() { }; +var do_sort = function(documents, sort_stage) { + return documents.sort(new Selector(sort_stage, Selector.SORT_SELECTOR)); +}; + +var do_match = function(documents, match_stage) { + var cursor = new Cursor(documents, match_stage); + + return cursor.fetch(); +}; + var do_group = function(documents, group_stage) { if (!_.hasIn(group_stage, '_id')) logger.throw('The field "_id" is required in the "$group" stage'); let new_id = group_stage['_id']; + + if (!_.isNull(new_id)) { + if (new_id.substr(0, 1) !== '$') { + logger.throw("Field names references in a right side assignement must be preceded by '$'"); + } else { + new_id = new_id.substr(1, new_id.length); + } + } if (_.isPlainObject(new_id)) { // complex_id @@ -180,16 +200,30 @@ class Aggregation { } aggregate(collection) { + var docs = collection.docs; + for (let i = 0; i < this.pipeline.length; i++) { let stage = this.pipeline[i]; for (let key in stage) { switch (key) { + case '$match': + docs = do_match(docs, stage[key]); + + break; case '$group': - return do_group(collection.docs, stage[key]); + docs = do_group(docs, stage[key]); + + break; + case '$sort': + docs = do_sort(docs, stage[key]); + + break; } } } + + return docs; // move to cursor } validStage(stage) { diff --git a/src/Collection.js b/src/Collection.js index 26afc09..5b92125 100644 --- a/src/Collection.js +++ b/src/Collection.js @@ -177,8 +177,7 @@ Collection.prototype.find = function (selection, fields, options, callback) { options = params.options; callback = params.callback; - // callback for backward compatibility - var cursor = new Cursor(this.db, this, selection, fields, options); + var cursor = new Cursor(this.docs, selection, fields, options); /** * "find" event. @@ -239,7 +238,7 @@ Collection.prototype.findOne = function (selection, fields, options, callback) { options = params.options; callback = params.callback; - var cursor = new Cursor(this.db, this, selection, fields, options); + var cursor = new Cursor(this.docs, selection, fields, options); /** * "findOne" event. diff --git a/src/Cursor.js b/src/Cursor.js index 12d4edc..3a2846d 100644 --- a/src/Cursor.js +++ b/src/Cursor.js @@ -23,7 +23,7 @@ var logger = null; * @classdesc Cursor class that maps a MongoDB-like cursor * * @param {MongoPortable} db - Additional options - * @param {Collection} collection - The collection instance + * @param {Array} documents - The list of documents * @param {Object|Array|String} [selection={}] - The selection for matching documents * @param {Object|Array|String} [fields={}] - The fields of the document to show * @param {Object} [options] - Database object @@ -32,9 +32,8 @@ var logger = null; * */ class Cursor { - constructor(db, collection, selection, fields, options = {}) { - this.db = db; - this.collection = collection; + constructor(documents, selection, fields, options = {}) { + this.documents = documents; this.selector = selection; this.skipValue = options.skip || 0; this.limitValue = options.limit || 15; @@ -43,6 +42,7 @@ class Cursor { logger = Logger.instance; + /** ADD IDX **/ if (Selector.isSelectorCompiled(this.selector)) { this.selector_compiled = this.selector; } else { @@ -65,6 +65,26 @@ class Cursor { } } + /** ADD IDX **/ + + this.fetch_mode = Cursor.COLSCAN || Cursor.IDXSCAN; + this.indexex = null;//findUsableIndexes(); + + // if (cursor.fetch_mode === Cursor.COLSCAN) { + // // COLSCAN, wi will iterate over all documents + // docs = _.cloneDeep(cursor.collection.docs); + // } else if (cursor.fetch_mode === Cursor.IDXSCAN) { + // // IDXSCAN, wi will iterate over all needed documents + // for (let i = 0; i < cursor.indexes.length; i++) { + // let index = cursor.indexes[i]; + + // for (let i = index.start; i < index.end; i++) { + // let idx_id = cursor.collection.getIndex(index.name)[i]; + + // docs.push(cursor.collection.docs[idx_id]); + // } + // } + // } this.fields = new Selector(fields, Selector.FIELD_SELECTOR); @@ -75,6 +95,9 @@ class Cursor { } } +Cursor.COLSCAN = 'colscan'; +Cursor.IDXSCAN = 'idxscan'; + /** * Moves a cursor to the begining * @@ -127,7 +150,7 @@ Cursor.prototype.map = function(callback) { * @returns {Boolean} True if we can fetch one more document */ Cursor.prototype.hasNext = function() { - return (this.cursor_pos < this.collection.docs.length); + return (this.cursor_pos < this.documents.length); }; /** @@ -233,25 +256,44 @@ var _mapFields = function(doc, fields) { * @returns {Array|Object} If [justOne=true] returns the next document, otherwise returns all the documents */ var _getDocuments = function(cursor, justOne = false) { - if (cursor.selector_id) { - if (_.hasIn(cursor.collection.doc_indexes, _.toString(cursor.selector_id))) { - let idx = cursor.collection.doc_indexes[_.toString(cursor.selector_id)]; + var docs = []; + + if (cursor.fetch_mode === Cursor.COLSCAN) { + // COLSCAN, wi will iterate over all documents + docs = _.cloneDeep(cursor.documents); + } else if (cursor.fetch_mode === Cursor.IDXSCAN) { + // IDXSCAN, wi will iterate over all needed documents + for (let i = 0; i < cursor.indexes.length; i++) { + let index = cursor.indexes[i]; - return _mapFields(cursor.collection.docs[idx], cursor.fields); - } else { - if (justOne) { - return null; - } else { - return []; + for (let i = index.start; i < index.end; i++) { + // let idx_id = cursor.collection.getIndex(index.name)[i]; + let idx_id = index.index[i]; + + docs.push(cursor.documents[idx_id]); } } } + // if (cursor.selector_id) { + // if (_.hasIn(cursor.collection.doc_indexes, _.toString(cursor.selector_id))) { + // let idx = cursor.collection.doc_indexes[_.toString(cursor.selector_id)]; + + // return _mapFields(cursor.collection.docs[idx], cursor.fields); + // } else { + // if (justOne) { + // return null; + // } else { + // return []; + // } + // } + // } + // TODO add warning when sort/skip/limit and fetching one // TODO add warning when skip/limit without order // TODO index - while (cursor.cursor_pos < cursor.collection.docs.length) { - var _doc = cursor.collection.docs[cursor.cursor_pos]; + while (cursor.cursor_pos < docs.length) { + var _doc = docs[cursor.cursor_pos]; cursor.cursor_pos++; if (cursor.selector_compiled.test(_doc)) { @@ -290,6 +332,26 @@ Cursor.prototype.count = function() { return this.fetchAll().length; }; +/** + * Set the sorting of the cursor + * + * @method Cursor#sort + * + * @param {Object|Array|String} spec - The sorting specification + * + * @returns {Cursor} This instance so it can be chained with other methods + */ +Cursor.prototype.setSorting = function(spec) { + if (_.isNil(spec)) logger.throw("You need to specify a sorting"); + + if (spec) { + this.sortValue = spec; + this.sort_compiled = (new Selector(spec, Selector.SORT_SELECTOR)); + } + + return this; +}; + /** * Applies a sorting on the cursor * @@ -307,15 +369,11 @@ Cursor.prototype.sort = function(spec) { } if (_sort) { - if (spec) { - this.sortValue = spec; - this.sort_compiled = _sort; + if (!_.isNil(this.db_objects) && _.isArray(this.db_objects)) { + this.db_objects = this.db_objects.sort(_sort); + this.sorted = true; } else { - // If no spec, do sort - if (!_.isNil(this.db_objects) && _.isArray(this.db_objects)) { - this.db_objects = this.db_objects.sort(_sort); - this.sorted = true; - } + this.setSorting(spec); } } diff --git a/src/Selector.js b/src/Selector.js index 84096a3..4b39d99 100644 --- a/src/Selector.js +++ b/src/Selector.js @@ -1,6 +1,7 @@ var Logger = require("jsw-logger"), _ = require("lodash"), - SelectorMatcher = require("./SelectorMatcher"); + SelectorMatcher = require("./SelectorMatcher"), + ObjectId = require("./ObjectId"); var logger = null; @@ -415,6 +416,11 @@ var _buildKeypathSelector = function (keypath, value) { clause.type = 'operator_object'; } + } else if (value instanceof ObjectId) { + logger.debug('clause of type ObjectId -> String'); + + clause.type = 'string'; + clause.value = value.toString(); } else { clause.type = '__invalid__'; } diff --git a/test/3_Cursor.js b/test/3_Cursor.js index 074623d..7df269d 100644 --- a/test/3_Cursor.js +++ b/test/3_Cursor.js @@ -32,11 +32,10 @@ describe("Cursor", function() { expect(selector).to.exist; - var c = new Cursor(null, null, selector, ["field1, field2"], { sort: { field2: -1 } }); + var c = new Cursor([], selector, ["field1, field2"], { sort: { field2: -1 } }); expect(c).to.exist; - expect(c.db).to.not.exist; expect(c.collection).to.not.exist; expect(c.selector).to.exist; expect(c.fields).to.exist; @@ -52,11 +51,10 @@ describe("Cursor", function() { }); it("should be able to create a new instance from a compiled selector", function() { - var c = new Cursor(null, null, { field1: { $gte: 3 } }, ["field1, field2"], { sort: { field2: -1 } }); + var c = new Cursor([], { field1: { $gte: 3 } }, ["field1, field2"], { sort: { field2: -1 } }); expect(c).to.exist; - expect(c.db).to.not.exist; expect(c.collection).to.not.exist; expect(c.selector).to.exist; expect(c.fields).to.exist; diff --git a/test/5_Aggregation.js b/test/5_Aggregation.js index 75acc1d..3aa41b4 100644 --- a/test/5_Aggregation.js +++ b/test/5_Aggregation.js @@ -42,7 +42,7 @@ describe("Aggregation", function() { describe("#Group", function() { it("should be able to group several documents ($sum)", function() { - var coll = db.collection("coll_1"); + var coll = db.collection("coll_group_1"); coll.insert({ _id: 1111, @@ -72,7 +72,7 @@ describe("Aggregation", function() { var docs = coll.aggregate([{ $group: { - _id: "age", + _id: "$age", count: { $sum: 1 }, @@ -93,7 +93,7 @@ describe("Aggregation", function() { }); it("should be able to group several documents ($avg)", function() { - var coll = db.collection("coll_2"); + var coll = db.collection("coll_group_2"); coll.insert({ _id: 1111, @@ -138,1071 +138,159 @@ describe("Aggregation", function() { expect(docs[0].aver_age).to.be.equal(23); }); }); - // describe(" - Create", function() { - // it("should be able to insert a document", function(done) { - // var coll = db.collection(TEST_COLL); - - // buildDoc(); - - // coll.insert(TEST_DOC, function(error, inserted) { - // expect(error).to.not.exist; - // expect(inserted).to.exist; - - // expect(inserted.stringField).to.be.equal(TEST_DOC.stringField); - // expect(inserted.numberField).to.be.equal(TEST_DOC.numberField); - // expect(coll.docs).to.have.length(1); - - // done(); - // }); - // }); - - // it("should be able to insert several documents chained", function() { - // var coll = db.collection("coll_2"); - - // coll.insert({ field: 'a' }, {chain: true}).insert({ _id: 12345, field: 'b' }); - - // expect(coll.docs).to.have.length(2); - // }); - - // it("should be able to save a document", function(done) { - // var coll = db.collection(TEST_COLL); - - // // new document - // var inserted = coll.save({ - // stringField: "save", - // numberField: 3 - // }); - - // expect(inserted).to.exist; - - // expect(inserted._id).to.exist; - // expect(inserted.stringField).to.be.equal("save"); - // expect(inserted.numberField).to.be.equal(3); - - // // update / upsert - - // coll.save({ _id: 12345, stringField: "save" }, function(error, updatedInfo) { - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.not.exist; - // expect(updatedInfo.updated.count).to.be.equal(0); - - // expect(updatedInfo.inserted.documents).to.be.instanceof(Array); - // expect(updatedInfo.inserted.count).to.be.equal(1); - - // var inserted = updatedInfo.inserted.documents[0]; - - // expect(inserted._id).to.be.equal("12345"); - // expect(inserted.stringField).to.be.equal("save"); - - // done(); - // }); - // }); - // }); - - // describe(" - Read", function() { - // it("should be able to read the first document", function(done) { - // var coll = db.collection(TEST_COLL); - - // coll.findOne(function(error, doc) { - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.exist; - // expect(doc.numberField).to.be.equal(TEST_DOC.numberField); - - // done(); - // }); - // }); - - // it("should be able to read a document", function(done) { - // var coll = db.collection(TEST_COLL); - - // var doc = coll.findOne({stringField: "yes"}, {_id: -1, numberField: 1}, { fields: { numberField: 1 }}); - - // expect(doc).to.exist; - - // expect(doc._id).to.not.exist; - // expect(doc.stringField).to.not.exist; - // expect(doc.numberField).to.be.equal(TEST_DOC.numberField); - - // doc = coll.findOne({_id: 123456789}, null, { fields: { numberField: 1 }}); - - // expect(doc).to.not.exist; - - // doc = coll.findOne(TEST_DOC._id); - - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.be.equal(TEST_DOC.stringField); - // expect(doc.numberField).to.be.equal(TEST_DOC.numberField); - - // doc = coll.findOne({_id: new ObjectId()}); - - // expect(doc).to.not.exist; - - // coll.findOne({_id: TEST_DOC._id}, function(error, doc) { - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.exist; - // expect(doc.numberField).to.be.equal(TEST_DOC.numberField); - - // done(); - // }); - // }); - - // it("should be able to read several documents", function(done) { - // var coll = db.collection(TEST_COLL); - - // var cursor = coll.find({stringField: "yes"}); - - // expect(cursor).to.exist; - - // cursor.forEach(function (doc) { - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.exist; - // expect(doc.numberField).to.exist; - // }); - - // var docs = coll.find({_id: new ObjectId()}, null, { forceFetch: true }); - - // expect(docs).to.exist; - // expect(docs).to.have.length(0); - - // // Callback - - // var calls = 3; - - // coll.find(function(error, docs) { - // expect(error).to.not.exist; - // expect(docs).to.exist; - - // expect(docs).to.have.length(3); - - // var doc = docs[0]; - - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.exist; - // expect(doc.numberField).to.exist; - - // if (calls === 1) { - // done(); - // } else { - // calls--; - // } - // }); - - // coll.find({ stringField: "yes" }, function(error, docs) { - // expect(error).to.not.exist; - // expect(docs).to.exist; - - // expect(docs).to.have.length(1); - - // var doc = docs[0]; - - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.exist; - // expect(doc.numberField).to.exist; - - // if (calls === 1) { - // done(); - // } else { - // calls--; - // } - // }); - - // coll.find({ stringField: "yes" }, { stringField: -1 }, function(error, docs) { - // expect(error).to.not.exist; - // expect(docs).to.exist; - - // expect(docs).to.have.length(1); - - // var doc = docs[0]; - - // expect(doc).to.exist; - - // expect(doc._id).to.exist; - // expect(doc.stringField).to.not.exist; - // expect(doc.numberField).to.exist; - - // if (calls === 1) { - // done(); - // } else { - // calls--; - // } - // }); - // }); - // }); - - // describe(" - Update", function() { - // it("should be able to update a document", function() { - // var coll = db.collection(TEST_COLL); - - // /**/ - // // coll.insert({ - // // _id: "TESTING", - // // objectField1: { - // // arrayField1: [1, 2, "3"], - // // arrayField2: [{ - // // subField1: "test1" - // // }, { - // // subField1: "test2" - // // }] - // // }, - // // arrayField1: [1, 2, "3"], - // // arrayField2: [{ - // // subField1: "test1" - // // }, { - // // subField1: "test2" - // // }] - // // }); - - // // coll.update( - // // { - // // _id: "TESTING" - // // }, { - // // $set: { - // // "objectField1.arrayField1.1": "setted", - // // "objectField1.arrayField2.1.subField1": "setted", - // // "arrayField1.1": "setted", - // // "arrayField2.1.subField1": "setted" - // // } - // // }); - // /**/ - - // // By _id - // var updatedInfo = coll.update( - // TEST_DOC._id, - // { - // numberField: 5, - // unexistingField: 1, - // _id: "non updated" - // } - // ); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.be.instanceof(Array); - // expect(updatedInfo.updated.count).to.be.equal(1); - - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - - // var updated = updatedInfo.updated.documents[0]; - - // expect(updated.unexistingField).to.not.exist; - // expect(updated._id).to.be.eql(TEST_DOC._id); - // expect(updated.stringField).to.be.equal("yes"); - // expect(updated.numberField).to.be.equal(5); - - // var doc = coll.findOne({stringField: "yes"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("yes"); - // expect(doc.numberField).to.be.equal(5); - - // // By field - // updatedInfo = coll.update({stringField: "yes"}, {numberField: 1}); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.be.instanceof(Array); - // expect(updatedInfo.updated.count).to.be.equal(1); - - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - - // updated = updatedInfo.updated.documents[0]; - - // expect(updated.stringField).to.be.equal("yes"); - // expect(updated.numberField).to.be.equal(1); - - // doc = coll.findOne({stringField: "yes"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("yes"); - // expect(doc.numberField).to.be.equal(1); - - // // No matches - // updatedInfo = coll.update({stringField: "nope"}, {numberField: 1}); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.not.exist; - // expect(updatedInfo.updated.count).to.be.equal(0); - - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - // }); - - // describe("should be able to use update operators", function() { - - // var expectUpdateInfo = function(error, updatedInfo, nUpdated, nInserted) { - // expect(error).to.not.exist; - // expect(updatedInfo).to.exist; - - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // if (nUpdated > 0) { - // expect(updatedInfo.updated.documents).to.be.instanceof(Array); - // expect(updatedInfo.updated.count).to.be.equal(nUpdated); - // } else { - // expect(updatedInfo.updated.documents).to.not.exist; - // expect(updatedInfo.updated.count).to.be.equal(0); - // } - - // if (nInserted > 0) { - // expect(updatedInfo.inserted.documents).to.be.instanceof(Array); - // expect(updatedInfo.inserted.count).to.be.equal(nInserted); - // } else { - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - // } - // }; - - // before(function() { - // db.collection("FIELD_OP") - // .insert({ stringField: "field_op", numberField: 3 }, { chain: true }) - // .insert({ stringField: "array_op", arrayField: ["first", "inside", "other", "last"] }, { chain: true }) - // .insert({ stringField: "yep6", numberField: 7 }, { chain: true }) - // .insert({ stringField: "yep8", numberField: 9 }); - // }); - - // describe("- Field Update Operators", function() { - // it("should update with the $inc operator", function(done) { - // var coll = db.collection("FIELD_OP"); - - // coll.update( - // { - // stringField: "field_op" - // }, { - // $inc: { - // numberField: 2, - // otherNumberField: 3 - // } - // }, - // function(error, updatedInfo) { - // expectUpdateInfo(error, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "field_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("field_op"); - // expect(doc.numberField).to.be.equal(5); - // expect(doc.otherNumberField).to.be.equal(3); - - // done(); - // } - // ); - // }); - - // it.skip("should update with the $mul operator", function() { - - // }); - - // it("should update with the $rename operator", function(done) { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "field_op" - // }; - // var update = { - // $rename: { - // numberField: 'numericField' - // } - // }; - - // coll.update(selector, update, function(error, updatedInfo) { - - // expectUpdateInfo(error, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "field_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("field_op"); - // expect(doc.numberField).to.not.exist; - // expect(doc.numericField).to.be.equal(5); - - // done(); - // }); - // }); - - // it.skip("should update with the $setOnInsert operator", function() { - - // }); - - // it("should update with the $set operator", function(done) { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "field_op" - // }; - // var update = { - // $set: { - // booleanField: true, - // "objectField.field": ["yes!", "noo"], - // newField: "ok", - // "newArray.1": "second" - // } - // }; - - // coll.update(selector, update, function(error, updatedInfo) { - // expectUpdateInfo(error, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "field_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("field_op"); - // expect(doc.numberField).to.not.exist; - // expect(doc.numericField).to.be.equal(5); - // expect(doc.booleanField).to.be.true; - // expect(doc.objectField).to.be.eql({ field: ["yes!", "noo"] }); - // expect(doc.newField).to.be.equal("ok"); - - // expect(doc.newArray).to.exist; - // expect(doc.newArray).to.be.instanceof(Array); - // expect(doc.newArray).to.have.length(2); - // expect(doc.newArray[1]).to.be.equal("second"); - - // done(); - // }); - // }); - - // it("should update with the $unset operator", function() { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "field_op", - // "objectField.field.1": "noo" - // }; - // var update = { - // $unset: { - // booleanField: "does not matter", - // "newArray.1": "neither does this" // TODO - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "field_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("field_op"); - // expect(doc.numberField).to.not.exist; - // expect(doc.numericField).to.be.equal(5); - // expect(doc.booleanField).to.not.exist; - // expect(doc.objectField).to.be.eql({ field: ["yes!", "noo"] }); - // expect(doc.newArray).to.exist; - // expect(doc.newArray).to.have.length(2); - // expect(doc.newArray[1]).to.be.equal(null); - // }); - - // it.skip("should update with the $min operator", function() { - - // }); - - // it.skip("should update with the $max operator", function() { - - // }); - - // it.skip("should update with the $currentDate operator", function() { - - // }); - // }); - - // describe("- Array Update Operators", function() { - // it("should update with the $addToSet operator", function() { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "array_op" - // }; - // var update = { - // $addToSet: { - // arrayField: "newly", - // unexistingArray: "first" - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField).to.have.length(5); - // expect(doc.arrayField[1]).to.be.equal("inside"); - // expect(doc.arrayField[4]).to.be.equal("newly"); - // expect(doc.unexistingArray).to.exist; - // expect(doc.unexistingArray).to.have.length(1); - // expect(doc.unexistingArray[0]).to.be.equal("first"); - - // // operator $each - // update = { - // $addToSet: { - // unexistingArray: { - // $each: ["first", "second"] - // } - // } - // }; - - // updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField).to.have.length(5); - // expect(doc.arrayField[1]).to.be.equal("inside"); - // expect(doc.arrayField[4]).to.be.equal("newly"); - // expect(doc.unexistingArray).to.exist; - // expect(doc.unexistingArray).to.have.length(2); - // expect(doc.unexistingArray[1]).to.be.equal("second"); - - // }); - - // it("should update with the $pop operator", function() { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "array_op" - // }; - // var update = { - // $pop: { - // arrayField: 1 - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // update['$pop'].arrayField = -1; - // updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField).to.have.length(3); - // expect(doc.arrayField[0]).to.not.be.equal("first"); - // expect(doc.arrayField[2]).to.be.equal("last"); - // }); - - // it("should update with the $push operator", function() { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "array_op" - // }; - // var update = { - // $push: { - // arrayField: "added", - // arrayField2: "first" - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField).to.have.length(4); - // expect(doc.arrayField[3]).to.be.equal("added"); - // expect(doc.arrayField2).to.exist; - // expect(doc.arrayField2).to.have.length(1); - // expect(doc.arrayField2[0]).to.be.equal("first"); - // }); - - // it("should update with the $pushAll operator", function() { // TODO Change, as is deprecated in MongoDB 2.4+ - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "array_op" - // }; - // var update = { - // $pushAll: { - // arrayField2: ["second", "third", "ot99her", "some99thing", "last"], - // addedArray: ["new"] - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField2).to.exist; - // expect(doc.arrayField2).to.have.length(6); - // expect(doc.arrayField2[0]).to.be.equal("first"); - // expect(doc.arrayField2[1]).to.be.equal("second"); - // expect(doc.arrayField2[2]).to.be.equal("third"); - // expect(doc.arrayField2[5]).to.be.equal("last"); - // expect(doc.addedArray).to.exist; - // expect(doc.addedArray).to.have.length(1); - // expect(doc.addedArray[0]).to.be.equal("new"); - // }); - - // it("should update with the $pull operator", function() { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "array_op" - // }; - // var update = { - // $pull: { - // arrayField: "first", - // arrayField2: { - // $regex: /^[\D][a-z]+[\d]+[a-z]+$/, - // $options: 'ig' - // } - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField).to.have.length(4); - // expect(doc.arrayField2).to.exist; - // expect(doc.arrayField2).to.have.length(4); - // expect(doc.arrayField2[0]).to.be.equal("first"); - // expect(doc.arrayField2[1]).to.be.equal("second"); - // expect(doc.arrayField2[2]).to.be.equal("third"); - // expect(doc.arrayField2[3]).to.be.equal("last"); - // }); - - // it("should update with the $pullAll operator", function() { - // var coll = db.collection("FIELD_OP"); - - // var selector = { - // stringField: "array_op" - // }; - // var update = { - // $pullAll: { - // arrayField2: ["first", "last"] - // } - // }; - - // var updatedInfo = coll.update(selector, update); - - // expectUpdateInfo(null, updatedInfo, 1, 0); - - // var doc = coll.findOne({stringField: "array_op"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("array_op"); - // expect(doc.arrayField).to.exist; - // expect(doc.arrayField).to.have.length(4); - // expect(doc.arrayField2).to.exist; - // expect(doc.arrayField2).to.have.length(2); - // expect(doc.arrayField2[0]).to.not.be.equal("first"); - // expect(doc.arrayField2[1]).to.not.be.equal("last"); - // }); - // }); - - // describe.skip("- Bitwise Update Operators", function() { - // it("should update with the $bit operator", function() { - - // }); - // }); - - // describe.skip("- Isolation Update Operators", function() { - // it("should update with the $isolated operator", function() { - - // }); - // }); - // }); + + describe("#Match", function() { + it("should be able to match documents", function() { + var coll = db.collection("coll_match_1"); - // it("should be able to use upsert a document", function() { - // var coll = db.collection(TEST_COLL); - - // var updatedInfo = coll.update( - // { - // stringField: "noope" - // }, { - // stringField: "yees", - // numberField: 2 - // }, { - // upsert: true - // } - // ); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.not.exist; - // expect(updatedInfo.updated.count).to.be.equal(0); - - // expect(updatedInfo.inserted.documents).to.be.instanceof(Array); - // expect(updatedInfo.inserted.count).to.be.equal(1); - - // var created = updatedInfo.inserted.documents[0]; - - // expect(created).to.exist; - - // expect(created.stringField).to.be.equal("yees"); - // expect(created.numberField).to.be.equal(2); - - // var doc = coll.findOne({stringField: "yees"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("yees"); - // expect(doc.numberField).to.be.equal(2); - // }); + coll.insert({ + _id: 1111, + name: 'eeee', + age: 22 + }, {chain: true}) + .insert({ + _id: 1112, + name: 'dddd', + age: 23 + }, {chain: true}) + .insert({ + _id: 1113, + name: 'cccc', + age: 22 + }, {chain: true}) + .insert({ + _id: 1114, + name: 'bbbb', + age: 25 + }, {chain: true}) + .insert({ + _id: 1115, + name: 'aaaa', + age: 23 + }); - // it("should be able to update several documents", function() { - // var coll = db.collection(TEST_COLL); - - // // add _id - // var updatedInfo = coll.update( - // { - // stringField: /^ye+s$/ig - // }, { - // numberField: 9 - // }, { - // multi: true - // } - // ); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.be.instanceof(Array); - // expect(updatedInfo.updated.count).to.be.equal(2); - - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - - // var updated1 = updatedInfo.updated.documents[0]; - // var updated2 = updatedInfo.updated.documents[1]; - - // expect(updated1.stringField).to.be.equal("yes"); - // expect(updated1.numberField).to.be.equal(9); - - // expect(updated2.stringField).to.be.equal("yees"); - // expect(updated2.numberField).to.be.equal(9); - - // var cursor = coll.find({numberField: 9}); - - // expect(cursor.count()).to.be.equal(2); - // }); + var docs = coll.aggregate([{ + $match: { + age: { + $gte: 23 + } + } + }]); - // it("should be able to override a document", function() { - // var coll = db.collection(TEST_COLL); - - // // add _id - // var updatedInfo = coll.update( - // { - // stringField: "yees" - // }, { - // numberField: 10, - // booleanField: true, - // $inc: { - // numberField: 1 - // } - // }, { - // override: true - // } - // ); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.be.instanceof(Array); - // expect(updatedInfo.updated.count).to.be.equal(1); - - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - - // var updated = updatedInfo.updated.documents[0]; - - // expect(updated.stringField).to.not.exist; - // expect(updated.numberField).to.be.equal(10); - // expect(updated.booleanField).to.be.true; - - // var cursor = coll.find({numberField: 10}); - - // expect(cursor.count()).to.be.equal(1); - // }); + console.log(docs); - // it.skip("should be able to update as MongoDB does", function() { - // var coll = db.collection(TEST_COLL); - - // var updatedInfo = coll.update( - // { - // stringField: "yes" - // }, { - // $inc: { - // numberField: 2 - // } - // }, { - // updateAsMongo: true - // } - // ); - - // expect(updatedInfo).to.exist; - // expect(updatedInfo.updated).to.exist; - // expect(updatedInfo.inserted).to.exist; - - // expect(updatedInfo.updated.documents).to.be.instanceof(Array); - // expect(updatedInfo.updated.count).to.be.equal(1); - - // expect(updatedInfo.inserted.documents).to.not.exist; - // expect(updatedInfo.inserted.count).to.be.equal(0); - - // var updated = updatedInfo.updated.documents[0]; - - // expect(updated).to.exist; - - // expect(updated.stringField).to.be.equal("yes"); - // expect(updated.numberField).to.be.equal(3); - - // var doc = coll.findOne({stringField: "yes"}); - - // expect(doc).to.exist; - - // expect(doc.stringField).to.be.equal("yes"); - // expect(doc.numberField).to.be.equal(3); - // }); - // }); - - // describe(" - Delete", function() { - // it("should be able to remove a document", function(done) { - // var coll = db.collection(TEST_COLL); - - // var removed = coll.remove({stringField: "yes"}); - - // expect(removed).to.exist; - // expect(removed).to.be.instanceof(Array); - - // expect(removed).to.be.have.length(1); - // expect(removed[0].stringField).to.be.equal("yes"); - - // var doc = coll.findOne({stringField: "yes"}); - - // expect(doc).to.not.exist; - - // // Remove by _id - // coll.delete(TEST_DOC._id, function(error, removed) { - // expect(error).to.not.exist; - // expect(removed).to.exist; - - // expect(removed).to.be.instanceof(Array); - - // expect(removed).to.be.have.length(0); - - // done(); - // }); - // }); + expect(docs).to.exist; + expect(docs).to.be.instanceof(Array); + expect(docs).to.have.length(3); + }); + }); + + describe("#Sort", function() { + it("should be able to sort documents", function() { + var coll = db.collection("coll_sort_1"); - // it("should be able to remove all documents", function(done) { - // var coll = db.collection(TEST_COLL); - - // coll.destroy(function (error, removed) { - // expect(error).to.not.exist; - // expect(removed).to.exist; - - // expect(removed).to.be.true; - - // var doc = coll.findOne({stringField: "yes"}); - - // expect(doc).to.not.exist; - - // done(); - // }); - // }); + coll.insert({ + _id: 1111, + name: 'eeee', + age: 22 + }, {chain: true}) + .insert({ + _id: 1112, + name: 'dddd', + age: 23 + }, {chain: true}) + .insert({ + _id: 1113, + name: 'cccc', + age: 22 + }, {chain: true}) + .insert({ + _id: 1114, + name: 'bbbb', + age: 25 + }, {chain: true}) + .insert({ + _id: 1115, + name: 'aaaa', + age: 23 + }); - // it("should be able to drop the collection", function(done) { - // var coll = db.collection(TEST_COLL); - - // coll.drop(function (error, removed) { - // expect(error).to.not.exist; - // expect(removed).to.exist; - - // expect(removed).to.be.true; - - // var doc = coll.findOne({stringField: "yes"}); - - // expect(doc).to.not.exist; - - // done(); - // }); - // }); - // }); - // }); - - describe("#Backups", function() { - var ID = null; - - before(function() { - if (db) db.dropDatabase(); + var docs = coll.aggregate([{ + $sort: { + age: -1 + } + }]); - db = new MongoPortable("BACKUPS"); + console.log(docs); - db.collection("pruebas") - .insert({ stringField: "first" }, { chain: true }) - .insert({ stringField: "second" }, { chain: true }) - .insert({ stringField: "third" }, { chain: true }) - .insert({ stringField: "fourth" }); + expect(docs).to.exist; + expect(docs).to.be.instanceof(Array); + expect(docs).to.have.length(5); }); - - describe("#Backups", function() { - it("should create a new backup", function(done) { - var coll = db.collection("pruebas"); - - var snapshot = coll.backup("FIRST"); - - expect(snapshot).to.exist; - - expect(snapshot.backupID).to.exist; - expect(snapshot.documents).to.exist; - - expect(snapshot.backupID).to.be.equal("FIRST"); - expect(snapshot.documents).to.be.instanceof(Array); - expect(snapshot.documents).to.have.length(4); - - coll.insert({ stringField: "new" }); - - coll.backup(function(error, snapshot) { - expect(snapshot.backupID).to.exist; - expect(snapshot.documents).to.exist; - - expect(snapshot.documents).to.be.instanceof(Array); - expect(snapshot.documents).to.have.length(5); - - ID = snapshot.backupID; - - done(); - }); - }); + }); + + describe("#Mixed", function() { + it("should be able to aggregate several stages", function() { + var coll = db.collection("coll_mixed_1"); - it("should retrieve all backups", function() { - var coll = db.collection("pruebas"); - - var snapshots = coll.backups(); - - expect(snapshots).to.exist; - - expect(snapshots).to.be.instanceof(Array); - expect(snapshots).to.have.length(2); - - expect(snapshots[0].id).to.be.equal("FIRST"); - expect(snapshots[0].documents).to.be.instanceof(Array); - expect(snapshots[0].documents).to.have.length(4); - - expect(snapshots[1].documents).to.be.instanceof(Array); - expect(snapshots[1].documents).to.have.length(5); + coll.insert({ + _id: 1111, + name: 'eeee', + age: 22 + }, {chain: true}) + .insert({ + _id: 1112, + name: 'dddd', + age: 23 + }, {chain: true}) + .insert({ + _id: 1113, + name: 'cccc', + age: 22 + }, {chain: true}) + .insert({ + _id: 1114, + name: 'bbbb', + age: 25 + }, {chain: true}) + .insert({ + _id: 1115, + name: 'aaaa', + age: 23 + }, {chain: true}) + .insert({ + _id: 1115, + name: 'abab', + age: 22 }); - it("should restore a backup", function() { - var coll = db.collection("pruebas"); - - expect(coll.find().count()).to.be.equal(5); - - // By backupID - var cursor = coll.restore("FIRST").find(); - - expect(cursor.count()).to.be.equal(4); - }); + var docs = coll.aggregate([{ + $match: { + age: { + $lt: 25 + } + } + }, { + $group: { + _id: "$age", + total: { + $sum: 1 + } + } + }, { + $sort: { + total: -1 + } + }]); - it("should remove a backup", function() { - var coll = db.collection("pruebas"); - - var removed = coll.removeBackup(ID); - - expect(removed).to.exist; - expect(removed).to.be.equal(ID); - - var snapshots = coll.backups(); - - expect(snapshots).to.exist; - - expect(snapshots).to.be.instanceof(Array); - expect(snapshots).to.have.length(1); - }); + console.log(docs); - it("should restore the only backup", function() { - var coll = db.collection("pruebas"); - - expect(coll.find().count()).to.be.equal(4); - - // By backupID - var cursor = coll.restore(function(error) { - expect(error).to.not.exist; - }).find(); - - expect(cursor.count()).to.be.equal(4); - }); + expect(docs).to.exist; + expect(docs).to.be.instanceof(Array); + expect(docs).to.have.length(2); - it("should drop all the backups", function(done) { - var coll = db.collection("pruebas"); - - coll.removeBackup(function(error, removed) { - expect(error).to.not.exist; - expect(removed).to.be.true; - - done(); - }); - }); + expect(docs[0]).to.be.eql({ _id: 22, total: 3 }); + expect(docs[1]).to.be.eql({ _id: 23, total: 2 }); }); }); }); \ No newline at end of file