ObjectExpression.js 24 KB

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