-
Notifications
You must be signed in to change notification settings - Fork 5.6k
/
Copy pathquery_settings_server_status.js
220 lines (194 loc) · 8.81 KB
/
query_settings_server_status.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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
// Tests the availability and correctness of query settings server status metrics. It does so by
// creating a sharded cluster environment and verifying that the metrics:
// * increment / decrement as expected on inserts / updates / deletions
// * are propagated throughout all mongod / mongos nodes
// * are restored to their previous values on upon restarts
//
// @tags: [
// requires_fcv_80,
// # This test needs persistence to ensure that query settings metrics survive cluster restarts.
// requires_persistence,
// ]
import "jstests/multiVersion/libs/multi_cluster.js";
import {assertDropAndRecreateCollection} from "jstests/libs/collection_drop_recreate.js";
import {FixtureHelpers} from "jstests/libs/fixture_helpers.js";
import {QuerySettingsUtils} from "jstests/libs/query/query_settings_utils.js";
import {ShardingTest} from "jstests/libs/shardingtest.js";
const st = new ShardingTest({name: jsTestName(), shards: 2, mongos: 2, rs: {nodes: 3}});
const db = st.s.getDB("test");
// Creating the collection, because some sharding passthrough suites are failing when explain
// command is issued on the nonexistent database and collection.
const collName = jsTestName();
assertDropAndRecreateCollection(db, collName);
const primaryQSU = new QuerySettingsUtils(db, collName);
const query = primaryQSU.makeFindQueryInstance({filter: {a: 1}});
const smallerQuerySettings = {
indexHints: {ns: {db: db.getName(), coll: collName}, allowedIndexes: ["a_1"]}
};
const biggerQuerySettings = {
indexHints: {ns: {db: db.getName(), coll: collName}, allowedIndexes: ["a_1", {$natural: 1}]}
};
function restartCluster() {
st.restartMongoses();
// Refresh the db reference so it's possible to later call 'clusterParamRefreshSecs.restore()'.
primaryQSU.db = st.s.getDB(db.getName());
st.stopAllConfigServers({startClean: false}, /* forRestart*/ true);
st.restartAllConfigServers();
st.stopAllShards({startClean: false}, /* forRestart */ true);
st.restartAllShards();
}
// Extends the `jstests/libs/fixture_helpers.js::mapOnEachShardNode()` function to also process
// mongos instances from the current test instance. It runs the `func()` function over all the
// mongod and mongos nodes in the cluster, returning an array according to the defined mapping.
function mapOnEachNode(func) {
const fromMongos = st._mongos.map((connection) => connection.getDB(db.getName())).map(func);
const fromShards = FixtureHelpers.mapOnEachShardNode(
{func, db: st.s.getDB(db.getName()), primaryNodeOnly: false});
return fromMongos.concat(fromShards);
}
function runTest({expectedQueryShapeConfiguration, assertionFunc}) {
// Call `assertQueryShapeConfiguration()` on each mongos instance to ensure that the changes
// introduced by the `setQuerySettings` command have been correctly propagated throughout the
// whole cluster.
st.forEachMongos((connection) =>
new QuerySettingsUtils(connection.getDB(db.getName()), collName)
.assertQueryShapeConfiguration(expectedQueryShapeConfiguration));
// Ensure that each node emits the exact same query settings metrics. Some nodes might be slower
// to startup and load the cluster parameter, therefore the need for wrapping this in an
// 'assert.soon()' construct.
let dataPoints;
assert.soon(() => {
dataPoints = mapOnEachNode(
(nodeDB) => new QuerySettingsUtils(nodeDB, collName).getQuerySettingsServerStatus());
if (!dataPoints || dataPoints.length === 0) {
return false;
}
return dataPoints.every(el => bsonWoCompare(dataPoints[0], el) === 0);
}, `expected all data points to be equal, but found ${tojson(dataPoints)}`);
// Since the test is running in a sharded cluster environment, the number of data points should
// equal 2 (noMongos) + 2 (noShards) * 3 (noNodesPerReplSet) = 8.
assert.eq(dataPoints.length, 8, `expected 8 data points but found ${tojson(dataPoints)}`);
// Validate the first data point via the provided `assertionFunc()` assertion. All the emitted
// data points should be equal thanks to the previous 'assert.soon()', so it's sufficient to
// check only one entry.
assertionFunc(dataPoints[0]);
}
// Expect both the count and the size to be zero, as this is a newly created replica set.
runTest({
expectedQueryShapeConfiguration: [],
assertionFunc: ({count, size, rejectCount}) => {
assert.eq(
count,
0,
"`querySettings.count` server status should be 0 if no query settings are present");
assert.eq(
size,
0,
"`querySettings.size` server status should be 0 if no query settings are present");
assert.eq(
rejectCount,
0,
"`querySettings.rejectCount` should be zero before any reject settings are applied.");
}
});
// Keep track of the previous size from now on to verify that it increases/decreases as expected.
let lastSize;
// Insert the bigger entry and verify that the count increases to 1 and that the size is now
// greater than 0.
st.adminCommand({setQuerySettings: query, settings: biggerQuerySettings});
runTest({
expectedQueryShapeConfiguration:
[primaryQSU.makeQueryShapeConfiguration(biggerQuerySettings, query)],
assertionFunc: ({count, size}) => {
assert.eq(count, 1, "`querySettings.count` server status failed to increase on addition.");
assert.gt(size, 0, "`querySettings.size` server status failed to increase on addition.");
lastSize = size;
}
});
// Replace the query setting with a smaller one and ensure that the size decreases while the
// count remains unchanged.
st.adminCommand({setQuerySettings: query, settings: smallerQuerySettings});
runTest({
expectedQueryShapeConfiguration:
[primaryQSU.makeQueryShapeConfiguration(smallerQuerySettings, query)],
assertionFunc: ({count, size}) => {
assert.eq(count,
1,
"`querySettings.count` server status should remain unchanged on replacements.");
assert.lt(
size,
lastSize,
"`querySettings.size` server status failed to decrease on a smaller replacement.");
lastSize = size;
}
});
// Restart the cluster and ensure that the metrics remain unchanged.
restartCluster();
runTest({
expectedQueryShapeConfiguration:
[primaryQSU.makeQueryShapeConfiguration(smallerQuerySettings, query)],
assertionFunc: ({count, size}) => {
assert.eq(count,
1,
"`querySettings.count` server status should remain unchanged between restarts.");
assert.eq(size,
lastSize,
"`querySettings.size` server status should remain unchanged between restarts.");
lastSize = size;
}
});
// Delete the entry and expect the count to be zero and the size to have decreased.
// Since the cluster parameter holds additional data, other than the provided query settings,
// the size is not expected to go back to zero. (ie: `clusterParameterTime`)
st.adminCommand({removeQuerySettings: query});
runTest({
expectedQueryShapeConfiguration: [],
assertionFunc: ({count, size}) => {
assert.eq(count, 0, "`querySettings.count` server status failed to decrease on deletion.");
assert.lt(
size, lastSize, "`querySettings.size` server status failed to decrease on deletion.");
}
});
const rejectQuerySettings = {
"reject": true
};
const rejectQueryConfig = primaryQSU.makeQueryShapeConfiguration(rejectQuerySettings, query);
// Set reject=true for the test query.
st.adminCommand({setQuerySettings: query, settings: rejectQuerySettings});
// Confirm rejectCount was updated.
runTest({
expectedQueryShapeConfiguration: [rejectQueryConfig],
assertionFunc: ({rejectCount}) => {
assert.eq(
rejectCount,
1,
"`querySettings.rejectCount` should be one after reject has been set for test query.");
}
});
restartCluster();
// Confirm rejectCount persists across restart.
runTest({
expectedQueryShapeConfiguration: [rejectQueryConfig],
assertionFunc: ({rejectCount}) => {
assert.eq(rejectCount, 1, "`querySettings.rejectCount` should still be one after restart.");
}
});
// Remove settings.
st.adminCommand({removeQuerySettings: query});
runTest({
expectedQueryShapeConfiguration: [],
assertionFunc: ({rejectCount}) => {
assert.eq(rejectCount,
0,
"`querySettings.rejectCount` should be zero after settings are removed.");
}
});
restartCluster();
// Confirm reject is not accidentally restored after restart.
runTest({
expectedQueryShapeConfiguration: [],
assertionFunc: ({rejectCount}) => {
assert.eq(rejectCount, 0, "`querySettings.rejectCount` should be zero after restart.");
}
});
st.stop();