summaryrefslogtreecommitdiff
path: root/jstests/aggregation/sources/replaceRoot/address.js
blob: 32ac3df26265763982ffe96be695fa7b1b7dc186 (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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
/**
 * $replaceRoot can be used to extract parts of a document; here we test a simple address case.
 * @tags: [
 *   requires_sharding,
 *   requires_spawning_own_processes,
 * ]
 */

(function() {
    "use strict";

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

    const dbName = "test";
    const collName = jsTest.name();

    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,
            },
        };
    }

    function doExecutionTest(conn) {
        const coll = conn.getDB(dbName).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.writeOK(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");
    }

    // Test against the standalone started by resmoke.py.
    let conn = db.getMongo();
    doExecutionTest(conn);
    print("Success! Standalone execution test for $replaceRoot passed.");

    // Test against a sharded cluster.
    let st = new ShardingTest({shards: 2});
    doExecutionTest(st.s0);
    st.stop();
    print("Success! Sharding test for $replaceRoot passed.");

}());