ObjectExpression_test.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662
  1. "use strict";
  2. if (!module.parent) return require.cache[__filename] = 0, (new(require("mocha"))()).addFile(__filename).ui("exports").run(process.exit);
  3. var assert = require("assert"),
  4. ObjectExpression = require("../../../../lib/pipeline/expressions/ObjectExpression"),
  5. Expression = require("../../../../lib/pipeline/expressions/Expression"),
  6. ConstantExpression = require("../../../../lib/pipeline/expressions/ConstantExpression"),
  7. FieldPathExpression = require("../../../../lib/pipeline/expressions/FieldPathExpression"),
  8. AndExpression = require("../../../../lib/pipeline/expressions/AndExpression"),
  9. Variables = require("../../../../lib/pipeline/expressions/Variables"),
  10. DepsTracker = require("../../../../lib/pipeline/DepsTracker");
  11. //SKIPPED: assertBinaryEqual
  12. //SKIPPED: toJson
  13. function expressionToJson(expr) {
  14. return expr.serialize(false);
  15. }
  16. //SKIPPED: fromJson
  17. //SKIPPED: valueFromBson
  18. function assertDependencies(expectedDependencies, expression, includePath) {
  19. if (includePath === undefined) includePath = true;
  20. var path = [],
  21. dependencies = new DepsTracker();
  22. expression.addDependencies(dependencies, includePath ? path : undefined);
  23. var bab = Object.keys(dependencies.fields);
  24. assert.deepEqual(bab.sort(), expectedDependencies.sort());
  25. assert.strictEqual(dependencies.needWholeDocument, false);
  26. assert.strictEqual(dependencies.needTextScore, false);
  27. }
  28. /// An assertion for `ObjectExpression` instances based on Mongo's `ExpectedResultBase` class
  29. function assertExpectedResult(args) {
  30. {// check for required args
  31. if (args === undefined) throw new TypeError("missing arg: `args` is required");
  32. if (!("expected" in args)) throw new Error("missing arg: `args.expected` is required");
  33. if (!("expectedDependencies" in args)) throw new Error("missing arg: `args.expectedDependencies` is required");
  34. if (!("expectedJsonRepresentation" in args)) throw new Error("missing arg: `args.expectedJsonRepresentation` is required");
  35. }
  36. {// base args if none provided
  37. if (args.source === undefined) args.source = {_id:0, a:1, b:2};
  38. if (args.expectedIsSimple === undefined) args.expectedIsSimple = true;
  39. if (args.expression === undefined) args.expression = ObjectExpression.createRoot(); //NOTE: replace prepareExpression + _expression=
  40. }
  41. // run implementation
  42. var doc = args.source,
  43. result = {},
  44. vars = new Variables(0, doc);
  45. args.expression.addToDocument(result, doc, vars);
  46. assert.deepEqual(result, args.expected);
  47. assertDependencies(args.expectedDependencies, args.expression);
  48. assert.deepEqual(expressionToJson(args.expression), args.expectedJsonRepresentation);
  49. assert.deepEqual(args.expression.isSimple(), args.expectedIsSimple);
  50. }
  51. exports.ObjectExpression = {
  52. "constructor()": {
  53. "should return instance if given arg": function() {
  54. assert(new ObjectExpression(false) instanceof Expression);
  55. assert(new ObjectExpression(true) instanceof Expression);
  56. },
  57. "should throw Error when constructing without args": function() {
  58. assert.throws(function() {
  59. new ObjectExpression();
  60. });
  61. },
  62. },
  63. "#addDependencies": {
  64. "should be able to get dependencies for non-inclusion expressions": function nonInclusionDependencies() {
  65. /** Dependencies for non inclusion expressions. */
  66. var expr = ObjectExpression.createRoot();
  67. expr.addField("a", ConstantExpression.create(5));
  68. assertDependencies(["_id"], expr, true);
  69. assertDependencies([], expr, false);
  70. expr.addField("b", FieldPathExpression.create("c.d"));
  71. assertDependencies(["_id", "c.d"], expr, true);
  72. assertDependencies(["c.d"], expr, false);
  73. },
  74. "should be able to get dependencies for inclusion expressions": function inclusionDependencies() {
  75. /** Dependencies for inclusion expressions. */
  76. var expr = ObjectExpression.createRoot();
  77. expr.includePath("a");
  78. assertDependencies(["_id", "a"], expr, true);
  79. var unused = new DepsTracker();
  80. assert.throws(function() {
  81. expr.addDependencies(unused);
  82. }, Error);
  83. },
  84. },
  85. "#serialize": {
  86. "should be able to convert to JSON representation and have constants represented by expressions": function json() {
  87. /** Serialize to a BSONObj, with constants represented by expressions. */
  88. var expr = ObjectExpression.createRoot();
  89. expr.addField("foo.a", ConstantExpression.create(5));
  90. assert.deepEqual({foo:{a:{$const:5}}}, expr.serialize());
  91. },
  92. },
  93. "#optimize": {
  94. "should be able to optimize expression and sub-expressions": function optimize() {
  95. /** Optimizing an object expression optimizes its sub expressions. */
  96. var expr = ObjectExpression.createRoot();
  97. // Add inclusion.
  98. expr.includePath("a");
  99. // Add non inclusion.
  100. var andExpr = new AndExpression();
  101. expr.addField("b", andExpr);
  102. expr.optimize();
  103. // Optimizing 'expression' optimizes its non inclusion sub expressions, while
  104. // inclusion sub expressions are passed through.
  105. assert.deepEqual({a:true, b:{$const:true}}, expressionToJson(expr));
  106. },
  107. },
  108. "#evaluate()": {
  109. "should be able to provide an empty object": function empty() {
  110. /** Empty object spec. */
  111. var expr = ObjectExpression.createRoot();
  112. assertExpectedResult({
  113. expression: expr,
  114. expected: {_id:0},
  115. expectedDependencies: ["_id"],
  116. expectedJsonRepresentation: {}
  117. });
  118. },
  119. "should be able to include 'a' field only": function include() {
  120. /** Include 'a' field only. */
  121. var expr = ObjectExpression.createRoot();
  122. expr.includePath("a");
  123. assertExpectedResult({
  124. expression: expr,
  125. expected: {_id:0, a:1},
  126. expectedDependencies: ["_id", "a"],
  127. expectedJsonRepresentation: {a:true}
  128. });
  129. },
  130. "should NOT be able to include missing 'a' field": function missingInclude() {
  131. /** Cannot include missing 'a' field. */
  132. var expr = ObjectExpression.createRoot();
  133. expr.includePath("a");
  134. assertExpectedResult({
  135. source: {_id:0, b:2},
  136. expression: expr,
  137. expected: {_id:0},
  138. expectedDependencies: ["_id", "a"],
  139. expectedJsonRepresentation: {a:true}
  140. });
  141. },
  142. "should be able to include '_id' field only": function includeId() {
  143. /** Include '_id' field only. */
  144. var expr = ObjectExpression.createRoot();
  145. expr.includePath("_id");
  146. assertExpectedResult({
  147. expression: expr,
  148. expected: {_id:0},
  149. expectedDependencies: ["_id"],
  150. expectedJsonRepresentation: {_id:true}
  151. });
  152. },
  153. "should be able to exclude '_id' field": function excludeId() {
  154. /** Exclude '_id' field. */
  155. var expr = ObjectExpression.createRoot();
  156. expr.includePath("b");
  157. expr.excludeId = true;
  158. assertExpectedResult({
  159. expression: expr,
  160. expected: {b:2},
  161. expectedDependencies: ["b"],
  162. expectedJsonRepresentation: {_id:false, b:true}
  163. });
  164. },
  165. "should be able to include fields in source document order regardless of inclusion order": function sourceOrder() {
  166. /** Result order based on source document field order, not inclusion spec field order. */
  167. var expr = ObjectExpression.createRoot();
  168. expr.includePath("b");
  169. expr.includePath("a");
  170. assertExpectedResult({
  171. expression: expr,
  172. get expected() { return this.source; },
  173. expectedDependencies: ["_id", "a", "b"],
  174. expectedJsonRepresentation: {b:true, a:true}
  175. });
  176. },
  177. "should be able to include a nested field": function includeNested() {
  178. /** Include a nested field. */
  179. var expr = ObjectExpression.createRoot();
  180. expr.includePath("a.b");
  181. assertExpectedResult({
  182. expression: expr,
  183. expected: {_id:0, a:{b:5}},
  184. source: {_id:0, a:{b:5, c:6}, z:2},
  185. expectedDependencies: ["_id", "a.b"],
  186. expectedJsonRepresentation: {a:{b:true}}
  187. });
  188. },
  189. "should be able to include two nested fields": function includeTwoNested() {
  190. /** Include two nested fields. */
  191. var expr = ObjectExpression.createRoot();
  192. expr.includePath("a.b");
  193. expr.includePath("a.c");
  194. assertExpectedResult({
  195. expression: expr,
  196. expected: {_id:0, a:{b:5, c:6}},
  197. source: {_id:0, a:{b:5,c:6}, z:2},
  198. expectedDependencies: ["_id", "a.b", "a.c"],
  199. expectedJsonRepresentation: {a:{b:true, c:true}}
  200. });
  201. },
  202. "should be able to include two fields nested within different parents": function includeTwoParentNested() {
  203. /** Include two fields nested within different parents. */
  204. var expr = ObjectExpression.createRoot();
  205. expr.includePath("a.b");
  206. expr.includePath("c.d");
  207. assertExpectedResult({
  208. expression: expr,
  209. expected: {_id:0, a:{b:5}, c:{d:6}},
  210. source: {_id:0, a:{b:5}, c:{d:6}, z:2},
  211. expectedDependencies: ["_id", "a.b", "c.d"],
  212. expectedJsonRepresentation: {a:{b:true}, c:{d:true}}
  213. });
  214. },
  215. "should be able to attempt to include a missing nested field": function includeMissingNested() {
  216. /** Attempt to include a missing nested field. */
  217. var expr = ObjectExpression.createRoot();
  218. expr.includePath("a.b");
  219. assertExpectedResult({
  220. expression: expr,
  221. expected: {_id:0, a:{}},
  222. source: {_id:0, a:{c:6}, z:2},
  223. expectedDependencies: ["_id", "a.b"],
  224. expectedJsonRepresentation: {a:{b:true}}
  225. });
  226. },
  227. "should be able to attempt to include a nested field within a non object": function includeNestedWithinNonObject() {
  228. /** Attempt to include a nested field within a non object. */
  229. var expr = ObjectExpression.createRoot();
  230. expr.includePath("a.b");
  231. assertExpectedResult({
  232. expression: expr,
  233. expected: {_id:0},
  234. source: {_id:0, a:2, z:2},
  235. expectedDependencies: ["_id", "a.b"],
  236. expectedJsonRepresentation: {a:{b:true}}
  237. });
  238. },
  239. "should be able to include a nested field within an array": function includeArrayNested() {
  240. /** Include a nested field within an array. */
  241. var expr = ObjectExpression.createRoot();
  242. expr.includePath("a.b");
  243. assertExpectedResult({
  244. expression: expr,
  245. expected: {_id:0,a:[{b:5},{b:2},{}]},
  246. source: {_id:0,a:[{b:5,c:6},{b:2,c:9},{c:7},[],2],z:1},
  247. expectedDependencies: ["_id", "a.b"],
  248. expectedJsonRepresentation: {a:{b:true}}
  249. });
  250. },
  251. "should NOT include non-root '_id' field implicitly": function excludeNonRootId() {
  252. /** Don't include not root '_id' field implicitly. */
  253. var expr = ObjectExpression.createRoot();
  254. expr.includePath("a.b");
  255. assertExpectedResult({
  256. expression: expr,
  257. source: {_id:0, a:{_id:1, b:1}},
  258. expected: {_id:0, a:{b:1}},
  259. expectedDependencies: ["_id", "a.b"],
  260. expectedJsonRepresentation: {a:{b:true}}
  261. });
  262. },
  263. "should project a computed expression": function computed() {
  264. /** Project a computed expression. */
  265. var expr = ObjectExpression.createRoot();
  266. expr.addField("a", ConstantExpression.create(5));
  267. assertExpectedResult({
  268. expression: expr,
  269. source: {_id:0},
  270. expected: {_id:0, a:5},
  271. expectedDependencies: ["_id"],
  272. expectedJsonRepresentation: {a:{$const:5}},
  273. expectedIsSimple: false
  274. });
  275. },
  276. "should project a computed expression replacing an existing field": function computedReplacement() {
  277. /** Project a computed expression replacing an existing field. */
  278. var expr = ObjectExpression.createRoot();
  279. expr.addField("a", ConstantExpression.create(5));
  280. assertExpectedResult({
  281. expression: expr,
  282. source: {_id:0, a:99},
  283. expected: {_id:0, a:5},
  284. expectedDependencies: ["_id"],
  285. expectedJsonRepresentation: {a:{$const:5}},
  286. expectedIsSimple: false
  287. });
  288. },
  289. "should NOT be able to project an undefined value": function computedUndefined() {
  290. /** An undefined value is passed through */
  291. var expr = ObjectExpression.createRoot();
  292. expr.addField("a", ConstantExpression.create(undefined));
  293. assertExpectedResult({
  294. expression: expr,
  295. source: {_id:0},
  296. expected: {_id:0, a:undefined},
  297. expectedDependencies: ["_id"],
  298. expectedJsonRepresentation: {a:{$const:undefined}},
  299. expectedIsSimple: false
  300. });
  301. },
  302. "should project a computed expression replacing an existing field with Undefined": function computedUndefinedReplacement() {
  303. /** Project a computed expression replacing an existing field with Undefined. */
  304. var expr = ObjectExpression.createRoot();
  305. expr.addField("a", ConstantExpression.create(undefined));
  306. assertExpectedResult({
  307. expression: expr,
  308. source: {_id:0, a:99},
  309. expected: {_id:0, a:undefined},
  310. expectedDependencies: ["_id"],
  311. expectedJsonRepresentation: {a:{$const:undefined}},
  312. expectedIsSimple: false
  313. });
  314. },
  315. "should project a null value": function computedNull() {
  316. /** A null value is projected. */
  317. var expr = ObjectExpression.createRoot();
  318. expr.addField("a", ConstantExpression.create(null));
  319. assertExpectedResult({
  320. expression: expr,
  321. source: {_id:0},
  322. expected: {_id:0, a:null},
  323. expectedDependencies: ["_id"],
  324. expectedJsonRepresentation: {a:{$const:null}},
  325. expectedIsSimple: false
  326. });
  327. },
  328. "should project a nested value": function computedNested() {
  329. /** A nested value is projected. */
  330. var expr = ObjectExpression.createRoot();
  331. expr.addField("a.b", ConstantExpression.create(5));
  332. assertExpectedResult({
  333. expression: expr,
  334. source: {_id:0},
  335. expected: {_id:0, a:{b:5}},
  336. expectedDependencies: ["_id"],
  337. expectedJsonRepresentation: {a:{b:{$const:5}}},
  338. expectedIsSimple: false
  339. });
  340. },
  341. "should project a field path": function computedFieldPath() {
  342. /** A field path is projected. */
  343. var expr = ObjectExpression.createRoot();
  344. expr.addField("a", FieldPathExpression.create("x"));
  345. assertExpectedResult({
  346. expression: expr,
  347. source: {_id:0, x:4},
  348. expected: {_id:0, a:4},
  349. expectedDependencies: ["_id", "x"],
  350. expectedJsonRepresentation: {a:"$x"},
  351. expectedIsSimple: false
  352. });
  353. },
  354. "should project a nested field path": function computedNestedFieldPath() {
  355. /** A nested field path is projected. */
  356. var expr = ObjectExpression.createRoot();
  357. expr.addField("a.b", FieldPathExpression.create("x.y"));
  358. assertExpectedResult({
  359. expression: expr,
  360. source: {_id:0, x:{y:4}},
  361. expected: {_id:0, a:{b:4}},
  362. expectedDependencies: ["_id", "x.y"],
  363. expectedJsonRepresentation: {a:{b:"$x.y"}},
  364. expectedIsSimple: false
  365. });
  366. },
  367. "should NOT project an empty subobject expression for a missing field": function emptyNewSubobject() {
  368. /** An empty subobject expression for a missing field is not projected. */
  369. var expr = ObjectExpression.createRoot();
  370. // Create a sub expression returning an empty object.
  371. var subExpr = ObjectExpression.create();
  372. subExpr.addField("b", FieldPathExpression.create("a.b"));
  373. expr.addField("a", subExpr);
  374. assertExpectedResult({
  375. expression: expr,
  376. source: {_id:0},
  377. expected: {_id:0},
  378. expectedDependencies: ["_id", "a.b"],
  379. expectedJsonRepresentation: {a:{b:"$a.b"}},
  380. expectedIsSimple: false
  381. });
  382. },
  383. "should project a non-empty new subobject": function nonEmptyNewSubobject() {
  384. /** A non empty subobject expression for a missing field is projected. */
  385. var expr = ObjectExpression.createRoot();
  386. // Create a sub expression returning an empty object.
  387. var subExpr = ObjectExpression.create();
  388. subExpr.addField("b", ConstantExpression.create(6));
  389. expr.addField("a", subExpr);
  390. assertExpectedResult({
  391. expression: expr,
  392. source: {_id:0},
  393. expected: {_id:0, a:{b:6}},
  394. expectedDependencies: ["_id"],
  395. expectedJsonRepresentation: {a:{b:{$const:6}}},
  396. expectedIsSimple: false
  397. });
  398. },
  399. "should project two computed fields in a common parent": function adjacentDottedComputedFields() {
  400. /** Two computed fields within a common parent. */
  401. var expr = ObjectExpression.createRoot();
  402. expr.addField("a.b", ConstantExpression.create(6));
  403. expr.addField("a.c", ConstantExpression.create(7));
  404. assertExpectedResult({
  405. expression: expr,
  406. source: {_id:0},
  407. expected: {_id:0, a:{b:6, c:7}},
  408. expectedDependencies: ["_id"],
  409. expectedJsonRepresentation: {a:{b:{$const:6},c:{$const:7}}},
  410. expectedIsSimple: false
  411. });
  412. },
  413. "should project two computed fields in a common parent (w/ one case dotted)": function adjacentDottedAndNestedComputedFields() {
  414. /** Two computed fields within a common parent, in one case dotted. */
  415. var expr = ObjectExpression.createRoot();
  416. expr.addField("a.b", ConstantExpression.create(6));
  417. var subExpr = ObjectExpression.create();
  418. subExpr.addField("c", ConstantExpression.create(7));
  419. expr.addField("a", subExpr);
  420. assertExpectedResult({
  421. expression: expr,
  422. source: {_id:0},
  423. expected: {_id:0, a:{b:6, c:7}},
  424. expectedDependencies: ["_id"],
  425. expectedJsonRepresentation: {a:{b:{$const:6},c:{$const:7}}},
  426. expectedIsSimple: false
  427. });
  428. },
  429. "should project two computed fields in a common parent (in another case dotted)": function adjacentNestedAndDottedComputedFields() {
  430. /** Two computed fields within a common parent, in another case dotted. */
  431. var expr = ObjectExpression.createRoot();
  432. var subExpr = ObjectExpression.create();
  433. subExpr.addField("b", ConstantExpression.create(6));
  434. expr.addField("a", subExpr);
  435. expr.addField("a.c", ConstantExpression.create(7));
  436. assertExpectedResult({
  437. expression: expr,
  438. source: {_id:0},
  439. expected: {_id:0, a:{b:6, c:7}},
  440. expectedDependencies: ["_id"],
  441. expectedJsonRepresentation: {a:{b:{$const:6},c:{$const:7}}},
  442. expectedIsSimple: false
  443. });
  444. },
  445. "should project two computed fields in a common parent (nested rather than dotted)": function adjacentNestedComputedFields() {
  446. /** Two computed fields within a common parent, nested rather than dotted. */
  447. var expr = ObjectExpression.createRoot();
  448. var subExpr1 = ObjectExpression.create();
  449. subExpr1.addField("b", ConstantExpression.create(6));
  450. expr.addField("a", subExpr1);
  451. var subExpr2 = ObjectExpression.create();
  452. subExpr2.addField("c", ConstantExpression.create(7));
  453. expr.addField("a", subExpr2);
  454. assertExpectedResult({
  455. expression: expr,
  456. source: {_id:0},
  457. expected: {_id:0, a:{b:6, c:7}},
  458. expectedDependencies: ["_id"],
  459. expectedJsonRepresentation: {a:{b:{$const:6},c:{$const:7}}},
  460. expectedIsSimple: false
  461. });
  462. },
  463. "should project multiple nested fields out of order without affecting output order": function adjacentNestedOrdering() {
  464. /** Field ordering is preserved when nested fields are merged. */
  465. var expr = ObjectExpression.createRoot();
  466. expr.addField("a.b", ConstantExpression.create(6));
  467. var subExpr = ObjectExpression.create();
  468. // Add field 'd' then 'c'. Expect the same field ordering in the result doc.
  469. subExpr.addField("d", ConstantExpression.create(7));
  470. subExpr.addField("c", ConstantExpression.create(8));
  471. expr.addField("a", subExpr);
  472. assertExpectedResult({
  473. expression: expr,
  474. source: {_id:0},
  475. expected: {_id:0, a:{b:6, d:7, c:8}},
  476. expectedDependencies: ["_id"],
  477. expectedJsonRepresentation: {a:{b:{$const:6},d:{$const:7},c:{$const:8}}},
  478. expectedIsSimple: false
  479. });
  480. },
  481. "should project adjacent fields two levels deep": function multipleNestedFields() {
  482. /** Adjacent fields two levels deep. */
  483. var expr = ObjectExpression.createRoot();
  484. expr.addField("a.b.c", ConstantExpression.create(6));
  485. var bSubExpression = ObjectExpression.create();
  486. bSubExpression.addField("d", ConstantExpression.create(7));
  487. var aSubExpression = ObjectExpression.create();
  488. aSubExpression.addField("b", bSubExpression);
  489. expr.addField("a", aSubExpression);
  490. assertExpectedResult({
  491. expression: expr,
  492. source: {_id:0},
  493. expected: {_id:0, a:{b:{c:6, d:7}}},
  494. expectedDependencies: ["_id"],
  495. expectedJsonRepresentation: {a:{b:{c:{$const:6},d:{$const:7}}}},
  496. expectedIsSimple: false
  497. });
  498. },
  499. "should error if two expressions generate the same field": function conflictingExpressionFields() {
  500. /** Two expressions cannot generate the same field. */
  501. var expr = ObjectExpression.createRoot();
  502. expr.addField("a", ConstantExpression.create(5));
  503. assert.throws(function() {
  504. expr.addField("a", ConstantExpression.create(6)); // Duplicate field.
  505. }, Error);
  506. },
  507. "should error if an expression field conflicts with an inclusion field": function conflictingInclusionExpressionFields() {
  508. /** An expression field conflicts with an inclusion field. */
  509. var expr = ObjectExpression.createRoot();
  510. expr.includePath("a");
  511. assert.throws(function() {
  512. expr.addField("a", ConstantExpression.create(6));
  513. }, Error);
  514. },
  515. "should error if an inclusion field conflicts with an expression field": function conflictingExpressionInclusionFields() {
  516. /** An inclusion field conflicts with an expression field. */
  517. var expr = ObjectExpression.createRoot();
  518. expr.addField("a", ConstantExpression.create(5));
  519. assert.throws(function() {
  520. expr.includePath("a");
  521. }, Error);
  522. },
  523. "should error if an object expression conflicts with a constant expression": function conflictingObjectConstantExpressionFields() {
  524. /** An object expression conflicts with a constant expression. */
  525. var expr = ObjectExpression.createRoot();
  526. var subExpr = ObjectExpression.create();
  527. subExpr.includePath("b");
  528. expr.addField("a", subExpr);
  529. assert.throws(function() {
  530. expr.addField("a.b", ConstantExpression.create(6));
  531. }, Error);
  532. },
  533. "should error if a constant expression conflicts with an object expression": function conflictingConstantObjectExpressionFields() {
  534. /** A constant expression conflicts with an object expression. */
  535. var expr = ObjectExpression.createRoot();
  536. expr.addField("a.b", ConstantExpression.create(6));
  537. var subExpr = ObjectExpression.create();
  538. subExpr.includePath("b");
  539. assert.throws(function() {
  540. expr.addField("a", subExpr);
  541. }, Error);
  542. },
  543. "should error if two nested expressions cannot generate the same field": function conflictingNestedFields() {
  544. /** Two nested expressions cannot generate the same field. */
  545. var expr = ObjectExpression.createRoot();
  546. expr.addField("a.b", ConstantExpression.create(5));
  547. assert.throws(function() {
  548. expr.addField("a.b", ConstantExpression.create(6)); // Duplicate field.
  549. }, Error);
  550. },
  551. "should error if an expression is created for a subfield of another expression": function conflictingFieldAndSubfield() {
  552. /** An expression cannot be created for a subfield of another expression. */
  553. var expr = ObjectExpression.createRoot();
  554. expr.addField("a", ConstantExpression.create(5));
  555. assert.throws(function() {
  556. expr.addField("a.b", ConstantExpression.create(5));
  557. }, Error);
  558. },
  559. "should error if an expression is created for a nested field of another expression.": function conflictingFieldAndNestedField() {
  560. /** An expression cannot be created for a nested field of another expression. */
  561. var expr = ObjectExpression.createRoot();
  562. expr.addField("a", ConstantExpression.create(5));
  563. var subExpr = ObjectExpression.create();
  564. subExpr.addField("b", ConstantExpression.create(5));
  565. assert.throws(function() {
  566. expr.addField("a", subExpr);
  567. }, Error);
  568. },
  569. "should error if an expression is created for a parent field of another expression": function conflictingSubfieldAndField() {
  570. /** An expression cannot be created for a parent field of another expression. */
  571. var expr = ObjectExpression.createRoot();
  572. expr.addField("a.b", ConstantExpression.create(5));
  573. assert.throws(function() {
  574. expr.addField("a", ConstantExpression.create(5));
  575. }, Error);
  576. },
  577. "should error if an expression is created for a parent of a nested field": function conflictingNestedFieldAndField() {
  578. /** An expression cannot be created for a parent of a nested field. */
  579. var expr = ObjectExpression.createRoot();
  580. var subExpr = ObjectExpression.create();
  581. subExpr.addField("b", ConstantExpression.create(5));
  582. expr.addField("a", subExpr);
  583. assert.throws(function() {
  584. expr.addField("a", ConstantExpression.create(5));
  585. }, Error);
  586. },
  587. "should be able to evaluate expressions in general": function evaluate() {
  588. /**
  589. * evaluate() does not supply an inclusion document.
  590. * Inclusion spec'd fields are not included.
  591. * (Inclusion specs are not generally expected/allowed in cases where evaluate is called instead of addToDocument.)
  592. */
  593. var expr = ObjectExpression.createRoot();
  594. expr.includePath("a");
  595. expr.addField("b", ConstantExpression.create(5));
  596. expr.addField("c", FieldPathExpression.create("a"));
  597. var res = expr.evaluateInternal(new Variables(1, {_id:0, a:1}));
  598. assert.deepEqual({"b":5, "c":1}, res);
  599. },
  600. },
  601. };