summaryrefslogtreecommitdiff
path: root/jstests/aggregation/sources/replaceRoot/address.js
blob: 224691477d5073fab6bf8b807e8a818c8b095984 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
/**
 * $replaceRoot can be used to extract parts of a document; here we test a simple address case.
 */

(function() {
"use strict";

// For arrayEq.
load("jstests/aggregation/extras/utils.js");

Random.setRandomSeed();

/**
 * Helper to get a random entry out of an array.
 */
function randomChoice(array) {
    return array[Random.randInt(array.length)];
}

/**
 * Helper to generate a randomized document with the following schema:
 * {
 *   name: <string>,
 *   address: {number: <3-digit int>, street: <string>, city: <string>, zip: <5-digit int>}
 * }
 */
function generateRandomDocument() {
    let names = ["Asya", "Charlie", "Dan", "Geert", "Kyle"];
    const minNumber = 1;
    const maxNumber = 999;
    let streets = ["3rd", "4th", "5th", "6th", "7th", "8th", "9th"];
    let cities = ["New York", "Palo Alto", "Sydney", "Dublin"];
    const minZip = 10000;
    const maxZip = 99999;

    return {
        names: randomChoice(names),
        address: {
            number: Random.randInt(maxNumber - minNumber + 1) + minNumber,
            street: randomChoice(streets),
            city: randomChoice(cities),
            zip: Random.randInt(maxZip - minZip + 1) + minZip,
        },
    };
}

const dbName = "test";
const collName = jsTest.name();
const coll = db.getCollection(collName);
coll.drop();

// Insert a bunch of documents of the form above.
const nDocs = 10;
let bulk = coll.initializeUnorderedBulkOp();
for (let i = 0; i < nDocs; i++) {
    bulk.insert(generateRandomDocument());
}
assert.commandWorked(bulk.execute());

// Extract the contents of the address field, and make sure that doing the same
// with replaceRoot yields the correct answer.
// First compute each separately, since we know all of the fields in the address,
// to make sure we have the correct results.
let addressPipe = [{
    $project: {
        "_id": 0,
        "number": "$address.number",
        "street": "$address.street",
        "city": "$address.city",
        "zip": "$address.zip"
    }
}];
let correctAddresses = coll.aggregate(addressPipe).toArray();

// Then compute the same results using $replaceRoot.
let replaceWithResult = coll.aggregate([
                                {$replaceRoot: {newRoot: "$address"}},
                                {$sort: {city: 1, zip: 1, street: 1, number: 1}}
                            ])
                            .toArray();

// Then assert they are the same.
assert(arrayEq(replaceWithResult, correctAddresses),
       "$replaceRoot does not work the same as $project-ing the relevant fields to the top level");
}());