AndExpression.js 3.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. "use strict";
  2. /**
  3. * Create an expression that finds the conjunction of n operands. The
  4. * conjunction uses short-circuit logic; the expressions are evaluated in the
  5. * order they were added to the conjunction, and the evaluation stops and
  6. * returns false on the first operand that evaluates to false.
  7. *
  8. * @class AndExpression
  9. * @namespace mungedb-aggregate.pipeline.expressions
  10. * @module mungedb-aggregate
  11. * @constructor
  12. **/
  13. var AndExpression = module.exports = function AndExpression() {
  14. if (arguments.length !== 0) throw new Error("zero args expected");
  15. base.call(this);
  16. }, klass = AndExpression, base = require("./VariadicExpressionT")(klass), proto = klass.prototype = Object.create(base.prototype, {constructor: {value: klass}});
  17. // DEPENDENCIES
  18. var Value = require("../Value"),
  19. ConstantExpression = require("./ConstantExpression"),
  20. CoerceToBoolExpression = require("./CoerceToBoolExpression"),
  21. Expression = require("./Expression");
  22. // PROTOTYPE MEMBERS
  23. klass.opName = "$and";
  24. proto.getOpName = function getOpName() {
  25. return klass.opName;
  26. };
  27. /**
  28. * Takes an array one or more values and returns true if all of the values in the array are true. Otherwise $and returns false.
  29. * @method evaluate
  30. **/
  31. proto.evaluateInternal = function evaluateInternal(vars) {
  32. for (var i = 0, n = this.operands.length; i < n; ++i) {
  33. var value = this.operands[i].evaluateInternal(vars);
  34. if (!Value.coerceToBool(value)) return false;
  35. }
  36. return true;
  37. };
  38. proto.isAssociativeAndCommutative = function isAssociativeAndCommutative() { return true; };
  39. proto.optimize = function optimize() {
  40. var expr = base.prototype.optimize.call(this); //optimize the conjunction as much as possible
  41. // if the result isn't a conjunction, we can't do anything
  42. if (!(expr instanceof AndExpression)) return expr;
  43. var andExpr = expr;
  44. // Check the last argument on the result; if it's not constant (as promised by ExpressionNary::optimize(),) then there's nothing we can do.
  45. var n = andExpr.operands.length;
  46. // ExpressionNary::optimize() generates an ExpressionConstant for {$and:[]}.
  47. if (!n) throw new Error("requires operands!");
  48. var lastExpr = andExpr.operands[n - 1];
  49. if (!(lastExpr instanceof ConstantExpression)) return expr;
  50. // Evaluate and coerce the last argument to a boolean. If it's false, then we can replace this entire expression.
  51. var last = Value.coerceToBool(lastExpr.evaluateInternal());
  52. if (!last) return new ConstantExpression(false);
  53. // If we got here, the final operand was true, so we don't need it anymore.
  54. // If there was only one other operand, we don't need the conjunction either.
  55. // Note we still need to keep the promise that the result will be a boolean.
  56. if (n == 2) return new CoerceToBoolExpression(andExpr.operands[0]);
  57. //Remove the final "true" value, and return the new expression.
  58. //CW TODO: Note that because of any implicit conversions, we may need to apply an implicit boolean conversion.
  59. andExpr.operands.length = n - 1; //truncate the array
  60. return expr;
  61. };
  62. /** Register Expression */
  63. Expression.registerExpression(klass.opName, base.parse);
  64. //TODO: proto.toMatcherBson