0
0
mirror of https://github.com/mongodb/mongo.git synced 2024-12-01 09:32:32 +01:00
mongodb/jstests/sharding/views.js
2019-07-27 11:02:23 -04:00

168 lines
6.3 KiB
JavaScript

/**
* Tests for expected behavior when querying a view that is based on a sharded collection.
* @tags: [requires_find_command]
*/
(function() {
"use strict";
// For profilerHasSingleMatchingEntryOrThrow.
load("jstests/libs/profiler.js");
// Given sharded explain output in 'shardedExplain', verifies that the explain mode 'verbosity'
// affected the output verbosity appropriately, and that the response has the expected format.
// Set 'optimizedAwayPipeline' to true if the pipeline is expected to be optimized away.
function verifyExplainResult(
{shardedExplain = null, verbosity = "", optimizedAwayPipeline = false} = {}) {
assert.commandWorked(shardedExplain);
assert(shardedExplain.hasOwnProperty("shards"), tojson(shardedExplain));
for (let elem in shardedExplain.shards) {
let shard = shardedExplain.shards[elem];
let root;
if (optimizedAwayPipeline) {
assert(shard.hasOwnProperty("queryPlanner"), tojson(shardedExplain));
root = shard;
} else {
assert(shard.stages[0].hasOwnProperty("$cursor"), tojson(shardedExplain));
assert(shard.stages[0].$cursor.hasOwnProperty("queryPlanner"), tojson(shardedExplain));
root = shard.stages[0].$cursor;
}
if (verbosity === "queryPlanner") {
assert(!root.hasOwnProperty("executionStats"), tojson(shardedExplain));
} else if (verbosity === "executionStats") {
assert(root.hasOwnProperty("executionStats"), tojson(shardedExplain));
assert(!root.executionStats.hasOwnProperty("allPlansExecution"),
tojson("shardedExplain"));
} else {
assert.eq(verbosity, "allPlansExecution", tojson(shardedExplain));
assert(root.hasOwnProperty("executionStats"), tojson(shardedExplain));
assert(root.executionStats.hasOwnProperty("allPlansExecution"), tojson(shardedExplain));
}
}
}
let st = new ShardingTest({name: "views_sharded", shards: 2, other: {enableBalancer: false}});
let mongos = st.s;
let config = mongos.getDB("config");
let db = mongos.getDB(jsTestName());
db.dropDatabase();
let coll = db.getCollection("coll");
assert.commandWorked(config.adminCommand({enableSharding: db.getName()}));
st.ensurePrimaryShard(db.getName(), st.shard0.shardName);
assert.commandWorked(config.adminCommand({shardCollection: coll.getFullName(), key: {a: 1}}));
assert.commandWorked(mongos.adminCommand({split: coll.getFullName(), middle: {a: 6}}));
assert.commandWorked(
db.adminCommand({moveChunk: coll.getFullName(), find: {a: 25}, to: st.shard1.shardName}));
for (let i = 0; i < 10; ++i) {
assert.writeOK(coll.insert({a: i}));
}
assert.commandWorked(db.createView("view", coll.getName(), [{$match: {a: {$gte: 4}}}]));
let view = db.getCollection("view");
const explainVerbosities = ["queryPlanner", "executionStats", "allPlansExecution"];
//
// find
//
assert.eq(5, view.find({a: {$lte: 8}}).itcount());
let result = db.runCommand({explain: {find: "view", filter: {a: {$lte: 7}}}});
verifyExplainResult(
{shardedExplain: result, verbosity: "allPlansExecution", optimizedAwayPipeline: true});
for (let verbosity of explainVerbosities) {
result = db.runCommand({explain: {find: "view", filter: {a: {$lte: 7}}}, verbosity: verbosity});
verifyExplainResult(
{shardedExplain: result, verbosity: verbosity, optimizedAwayPipeline: true});
}
//
// aggregate
//
assert.eq(5, view.aggregate([{$match: {a: {$lte: 8}}}]).itcount());
// Test that the explain:true flag for the aggregate command results in queryPlanner verbosity.
result = db.runCommand({aggregate: "view", pipeline: [{$match: {a: {$lte: 8}}}], explain: true});
verifyExplainResult(
{shardedExplain: result, verbosity: "queryPlanner", optimizedAwayPipeline: true});
result =
db.runCommand({explain: {aggregate: "view", pipeline: [{$match: {a: {$lte: 8}}}], cursor: {}}});
verifyExplainResult(
{shardedExplain: result, verbosity: "allPlansExecution", optimizedAwayPipeline: true});
for (let verbosity of explainVerbosities) {
result = db.runCommand({
explain: {aggregate: "view", pipeline: [{$match: {a: {$lte: 8}}}], cursor: {}},
verbosity: verbosity
});
verifyExplainResult(
{shardedExplain: result, verbosity: verbosity, optimizedAwayPipeline: true});
}
//
// count
//
assert.eq(5, view.count({a: {$lte: 8}}));
result = db.runCommand({explain: {count: "view", query: {a: {$lte: 8}}}});
verifyExplainResult({shardedExplain: result, verbosity: "allPlansExecution"});
for (let verbosity of explainVerbosities) {
result = db.runCommand({explain: {count: "view", query: {a: {$lte: 8}}}, verbosity: verbosity});
verifyExplainResult({shardedExplain: result, verbosity: verbosity});
}
//
// distinct
//
result = db.runCommand({distinct: "view", key: "a", query: {a: {$lte: 8}}});
assert.commandWorked(result);
assert.eq([4, 5, 6, 7, 8], result.values.sort());
result = db.runCommand({explain: {distinct: "view", key: "a", query: {a: {$lte: 8}}}});
verifyExplainResult({shardedExplain: result, verbosity: "allPlansExecution"});
for (let verbosity of explainVerbosities) {
result = db.runCommand(
{explain: {distinct: "view", key: "a", query: {a: {$lte: 8}}}, verbosity: verbosity});
verifyExplainResult({shardedExplain: result, verbosity: verbosity});
}
//
// Confirm cleanupOrphaned command fails.
//
result = st.getPrimaryShard(db.getName()).getDB("admin").runCommand({
cleanupOrphaned: view.getFullName()
});
assert.commandFailedWithCode(result, ErrorCodes.CommandNotSupportedOnView);
//
// Confirm getShardVersion command fails.
//
assert.commandFailedWithCode(db.adminCommand({getShardVersion: view.getFullName()}),
ErrorCodes.NamespaceNotSharded);
//
// Confirm that the comment parameter on a find command is retained when rewritten as an
// expanded aggregation on the view.
//
let sdb = st.shard0.getDB(jsTestName());
assert.commandWorked(sdb.setProfilingLevel(2));
assert.eq(5, view.find({a: {$lte: 8}}).comment("agg_comment").itcount());
profilerHasSingleMatchingEntryOrThrow({
profileDB: sdb,
filter: {
"command.aggregate": coll.getName(),
"command.comment": "agg_comment",
"command.needsMerge": true,
"command.pipeline.$mergeCursors": {$exists: false}
}
});
st.stop();
})();