-
Notifications
You must be signed in to change notification settings - Fork 5.6k
/
Copy pathimplicit_query_settings_fallback.js
90 lines (79 loc) · 3.61 KB
/
implicit_query_settings_fallback.js
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
import {
getCollectionName,
getCommandName,
getExplainCommand,
getInnerCommand,
isInternalDbName,
isSystemCollectionName
} from "jstests/libs/cmd_object_utils.js";
import {OverrideHelpers} from "jstests/libs/override_methods/override_helpers.js";
import {
everyWinningPlan,
getNestedProperties,
isIdhackOrExpress
} from "jstests/libs/query/analyze_plan.js";
import {QuerySettingsIndexHintsTests} from "jstests/libs/query/query_settings_index_hints_tests.js";
import {QuerySettingsUtils} from "jstests/libs/query/query_settings_utils.js";
/**
* Override which applies 'bad' query settings over supported commands in order to test the fallback
* mechanism. Asserts that the query plans generated by the fallback should be identical to those
* generated without any query settings.
*/
function runCommandOverride(conn, dbName, _cmdName, cmdObj, clientFunction, makeFuncArgs) {
const assertFallbackPlanMatchesOriginalPlan = () => {
if (isInternalDbName(dbName)) {
// Query settings cannot be set over internal databases.
return;
}
const db = conn.getDB(dbName);
const innerCmd = getInnerCommand(cmdObj);
if (!QuerySettingsUtils.isSupportedCommand(getCommandName(innerCmd))) {
return;
}
const explain = db.runCommand(getExplainCommand(innerCmd));
if (!explain.ok) {
// Some commands such as $collStats cannot be explained and will lead to failures.
return;
}
if (explain.hasOwnProperty("warning")) {
// The explain output exceeded the 16MB BSON size limit and was truncated.
return;
}
// If the query explain has no 'winningPlan', we can not assert for query settings
// fallback.
if (getNestedProperties(explain, "winningPlan").length === 0) {
return;
}
const isIdHackQuery =
everyWinningPlan(explain, (winningPlan) => isIdhackOrExpress(db, winningPlan));
if (isIdHackQuery) {
// Query settings cannot be applied over IDHACK or Express queries.
return;
}
const collectionName = getCollectionName(db, innerCmd);
if (!collectionName || isSystemCollectionName(collectionName)) {
// Can't test the fallback on queries not involving any collections or queries targeting
// the system collection:
// - Queries not involving any collections will always yield to EOF plans.
// - Query settings validate against queries targeting system collections.
return;
}
const ns = {db: dbName, coll: collectionName};
const qsutils = new QuerySettingsUtils(db, collectionName);
const qstests = new QuerySettingsIndexHintsTests(qsutils);
const representativeQuery = qsutils.makeQueryInstance(innerCmd);
qstests.assertQuerySettingsFallback(representativeQuery, ns);
};
const res = clientFunction.apply(conn, makeFuncArgs(cmdObj));
if (res.ok) {
// Only run the test if the original command works. Some tests assert on commands failing,
// so we should simply bubble these commands through without any additional checks.
OverrideHelpers.withPreOverrideRunCommand(assertFallbackPlanMatchesOriginalPlan);
}
return res;
}
// Override the default runCommand with our custom version.
OverrideHelpers.overrideRunCommand(runCommandOverride);
// Always apply the override if a test spawns a parallel shell.
OverrideHelpers.prependOverrideInParallelShell(
"jstests/libs/override_methods/implicit_query_settings_fallback.js");