"use strict"; var assert = require("assert"), Pipeline = require("../../../lib/pipeline/Pipeline"), FieldPath = require("../../../lib/pipeline/FieldPath"), DocumentSource = require('../../../lib/pipeline/documentSources/DocumentSource'); module.exports = { "Pipeline": { before: function () { Pipeline.stageDesc.$test = (function () { var klass = function TestDocumentSource(options, ctx) { base.call(this, ctx); this.shouldCoalesce = options.coalesce; this.coalesceWasCalled = false; this.optimizeWasCalled = false; this.works = options.works === false ? false : true; // don't judge this.current = 5; }, TestDocumentSource = klass, base = DocumentSource, proto = klass.prototype = Object.create(base.prototype, {constructor: {value: klass}}); proto.coalesce = function () { this.coalesceWasCalled = true; var c = this.shouldCoalesce;//only coalesce with the first thing we find this.shouldCoalesce = false; return c; }; proto.optimize = function () { this.optimizeWasCalled = true; }; proto.getNext = function(callback){ var answer = this.current > 0 ? {val:this.current--} : DocumentSource.EOF, err = null; if (!this.works) err = new Error("doesn't work"), answer = undefined; if(callback) { return callback(err, answer); } else { return answer || err; } }; klass.createFromJson = function (options, ctx) { return new TestDocumentSource(options, ctx); }; return klass; })().createFromJson; }, "parseCommand": { "should throw Error if given non-objects in the array": function () { assert.throws(function () { Pipeline.parseCommand({pipeline: [5]}); }); }, "should throw Error if given objects with more / less than one field": function () { assert.throws(function () { Pipeline.parseCommand({pipeline: [ {} ]}); Pipeline.parseCommand({pipeline: [ {a: 1, b: 2} ]}); }); }, "should throw Error on unknown document sources": function () { assert.throws(function () { Pipeline.parseCommand({pipeline: [ {$foo: "$sdfdf"} ]}); }); }, "should swap $match and $sort if the $match immediately follows the $sort": function () { var p = Pipeline.parseCommand({pipeline: [ {$sort: {"xyz": 1}}, {$match: {}} ]}); assert.equal(p.sourceVector[0].constructor.matchName, "$match"); assert.equal(p.sourceVector[1].constructor.sortName, "$sort"); }, "should attempt to coalesce all sources": function () { var p = Pipeline.parseCommand({pipeline: [ {$test: {coalesce: false}}, {$test: {coalesce: true}}, {$test: {coalesce: false}}, {$test: {coalesce: false}} ]}); assert.equal(p.sourceVector.length, 3); p.sourceVector.slice(0, -1).forEach(function (source) { assert.equal(source.coalesceWasCalled, true); }); assert.equal(p.sources[p.sources.length -1].coalesceWasCalled, false); }, "should optimize all sources": function () { var p = Pipeline.parseCommand({pipeline: [ {$test: {coalesce: false}}, {$test: {coalesce: false}} ]}); p.sourceVector.forEach(function (source) { assert.equal(source.optimizeWasCalled, true); }); } }, "#stitch": { "should set the parent source for all sources in the pipeline except the first one": function () { var p = Pipeline.parseCommand({pipeline:[{$test:{coalesce:false}}, {$test:{coalesce:false}}, {$test:{coalesce:false}}]}); p.stitch(); assert.equal(p.sources[1].source, p.sources[0]); } }, "#_runSync": { "should iterate through sources and return resultant array": function () { var p = Pipeline.parseCommand({pipeline:[{$test:{coalesce:false}}, {$test:{coalesce:false}}, {$test:{coalesce:false}}]}), results = p.run(function(err, results) { assert.deepEqual(results.result, [ { val: 5 }, { val: 4 }, { val: 3 }, { val: 2 }, { val: 1 } ]); }); }, "should catch parse errors": function () { // The $foo part is invalid and causes a throw. assert.throws(function () { Pipeline.parseCommand({pipeline: [ {$match: {$foo: {bar: "baz"}}} ]}); }); }, "should call callback with errors from pipeline components": function (next) { var p = Pipeline.parseCommand({pipeline: [ {$match: {foo: {bar: "baz"}}} ]}); p.run(new DocumentSource({}), function (err, results) { assert(err instanceof Error); return next(); }); } }, "#_runAsync": { "should iterate through sources and return resultant array asynchronously": function () { var p = Pipeline.parseCommand({pipeline:[{$test:{coalesce:false}}, {$test:{coalesce:false}}, {$test:{coalesce:false}}]}), results = p.run(function(err, results) { assert.deepEqual(results.result, [ { val: 5 }, { val: 4 }, { val: 3 }, { val: 2 }, { val: 1 } ]); }); } }, "#addInitialSource": { "should put the given source at the beginning of the pipeline": function () { var p = Pipeline.parseCommand({pipeline:[{$test:{coalesce:false}}, {$test:{coalesce:false}}, {$test:{coalesce:false}}]}), initialSource = Pipeline.stageDesc.$test({coalesce:false}); p.addInitialSource(initialSource); assert.equal(initialSource, p.sources[0]); }, "should be able to addInitialSource then stitch": function () { var p = Pipeline.parseCommand({pipeline:[{$test:{coalesce:false}}, {$test:{coalesce:false}}, {$test:{coalesce:false}}]}), initialSource = Pipeline.stageDesc.$test({coalesce:false}); p.addInitialSource(initialSource); p.stitch(); assert.equal(p.sources[1].source, p.sources[0]); } } } }; if (!module.parent)(new(require("mocha"))()).ui("exports").reporter("spec").addFile(__filename).grep(process.env.MOCHA_GREP || '').run(process.exit);