Files
mongo/jstests/core/diagnostics/opcounters_write_cmd.js
Steve McClure 1ffbc6c2e9 SERVER-109432: Autofix JS var usage to favor let (#40637)
GitOrigin-RevId: 9674b7db36a0f3f650d39c1e3fb2ad6ff2141cfb
2025-08-28 19:21:01 +00:00

227 lines
6.9 KiB
JavaScript

// Test that opcounters get incremented properly.
// @tags: [
// # unrecognized command "invalid" is not allowed with security tolen.
// not_allowed_with_signed_security_token,
// uses_multiple_connections,
// assumes_standalone_mongod,
// # The config fuzzer may run logical session cache refreshes in the background, which modifies
// # some serverStatus metrics read in this test.
// does_not_support_config_fuzzer,
// inspects_command_opcounters,
// does_not_support_repeated_reads,
// ]
let mongo = new Mongo(db.getMongo().host);
let newdb = mongo.getDB(db.toString());
// Deletes in the system.profile collection can interfere with the opcounters tests below.
newdb.setProfilingLevel(0);
newdb.system.profile.drop();
let t = newdb.opcounters;
let opCounters;
let res;
//
// Count ops attempted in write commands in mongod and mongos
//
//
// 1. Insert.
//
// - unordered insert of N:
// counted as N ops, regardless of errors
// - ordered insert of N:
// counted as K + 1 ops, where K is number of docs successfully inserted,
// adding the failed attempt
//
t.drop();
// Single insert, no error.
opCounters = newdb.serverStatus().opcounters;
res = t.insert({_id: 0});
assert.commandWorked(res);
assert.eq(opCounters.insert + 1, newdb.serverStatus().opcounters.insert);
// Bulk insert, no error.
opCounters = newdb.serverStatus().opcounters;
res = t.insert([{_id: 1}, {_id: 2}]);
assert.commandWorked(res);
assert.eq(opCounters.insert + 2, newdb.serverStatus().opcounters.insert);
// Single insert, with error.
opCounters = newdb.serverStatus().opcounters;
res = t.insert({_id: 0});
assert.writeError(res);
assert.eq(opCounters.insert + 1, newdb.serverStatus().opcounters.insert);
// Bulk insert, with error, ordered.
opCounters = newdb.serverStatus().opcounters;
res = t.insert([{_id: 3}, {_id: 3}, {_id: 4}]);
assert.writeError(res);
assert.eq(opCounters.insert + 2, newdb.serverStatus().opcounters.insert);
// Bulk insert, with error, unordered.
let continueOnErrorFlag = 1;
opCounters = newdb.serverStatus().opcounters;
res = t.insert([{_id: 5}, {_id: 5}, {_id: 6}], continueOnErrorFlag);
assert.writeError(res);
assert.eq(opCounters.insert + 3, newdb.serverStatus().opcounters.insert);
//
// 2. Update.
//
t.drop();
t.insert({_id: 0});
// Update, no error.
opCounters = newdb.serverStatus().opcounters;
res = t.update({_id: 0}, {$set: {a: 1}});
assert.commandWorked(res);
assert.eq(opCounters.update + 1, newdb.serverStatus().opcounters.update);
// Update, with error.
opCounters = newdb.serverStatus().opcounters;
res = t.update({_id: 0}, {$set: {_id: 1}});
assert.writeError(res);
assert.eq(opCounters.update + 1, newdb.serverStatus().opcounters.update);
//
// 3. Delete.
//
t.drop();
t.insert([{_id: 0}, {_id: 1}]);
// Delete, no error.
opCounters = newdb.serverStatus().opcounters;
res = t.remove({_id: 0});
assert.commandWorked(res);
assert.eq(opCounters.delete + 1, newdb.serverStatus().opcounters.delete);
// Delete, with error.
opCounters = newdb.serverStatus().opcounters;
res = t.remove({_id: {$invalidOp: 1}});
assert.writeError(res);
assert.eq(opCounters.delete + 1, newdb.serverStatus().opcounters.delete);
//
// 4. Query.
//
// - counted as 1 op, regardless of errors
//
t.drop();
t.insert({_id: 0});
// Query, no error.
opCounters = newdb.serverStatus().opcounters;
t.findOne();
assert.eq(opCounters.query + 1, newdb.serverStatus().opcounters.query);
// Query, with error.
opCounters = newdb.serverStatus().opcounters;
assert.throws(function () {
t.findOne({_id: {$invalidOp: 1}});
});
assert.eq(opCounters.query + 1, newdb.serverStatus().opcounters.query);
t.drop();
t.insert([{_id: 0}, {_id: 1}, {_id: 2}]);
opCounters = newdb.serverStatus().opcounters;
t.aggregate({$match: {_id: 1}});
assert.eq(opCounters.query + 1, newdb.serverStatus().opcounters.query);
// Query, with error.
opCounters = newdb.serverStatus().opcounters;
assert.throws(function () {
t.aggregate({$match: {$invalidOp: 1}});
});
assert.eq(opCounters.query + 1, newdb.serverStatus().opcounters.query);
//
// 5. Getmore.
//
// - counted as 1 op per getmore issued, regardless of errors
//
t.drop();
t.insert([{_id: 0}, {_id: 1}, {_id: 2}]);
// Getmore, no error.
opCounters = newdb.serverStatus().opcounters;
t.find().batchSize(2).toArray(); // 3 documents, batchSize=2 => 1 query + 1 getmore
assert.eq(opCounters.query + 1, newdb.serverStatus().opcounters.query);
assert.eq(opCounters.getmore + 1, newdb.serverStatus().opcounters.getmore);
// Getmore, with error.
opCounters = newdb.serverStatus().opcounters;
assert.commandFailedWithCode(
t.getDB().runCommand({getMore: NumberLong(123), collection: t.getName()}),
ErrorCodes.CursorNotFound,
);
assert.eq(opCounters.getmore + 1, newdb.serverStatus().opcounters.getmore);
//
// 6. Command.
//
// - unrecognized commands not counted
// - recognized commands counted as 1 op, regardless of errors
// - some (recognized) commands can suppress command counting (i.e. aren't counted as commands)
//
t.drop();
t.insert({_id: 0});
// Send a command that attaches a databaseVersion to ensure the shard refreshes its cached database
// version before starting to record opcounters.
assert.commandWorked(newdb.runCommand({listCollections: 1}));
// Command, recognized, no error.
let serverStatus = newdb.runCommand({serverStatus: 1});
opCounters = serverStatus.opcounters;
let metricsObj = serverStatus.metrics.commands;
assert.eq(opCounters.command + 1, newdb.serverStatus().opcounters.command); // "serverStatus" counted
// Count this and the last run of "serverStatus"
assert.eq(
metricsObj.serverStatus.total + 2,
newdb.serverStatus().metrics.commands.serverStatus.total,
"total ServerStatus command counter did not increment",
); // "serverStatus" counted
assert.eq(
metricsObj.serverStatus.failed,
newdb.serverStatus().metrics.commands.serverStatus.failed,
"failed ServerStatus command counter incremented!",
); // "serverStatus" counted
// Command, recognized, with error.
let countVal = {"total": 0, "failed": 0};
if (metricsObj.count != null) {
countVal = metricsObj.count;
}
res = t.runCommand("count", {query: {$invalidOp: 1}}); // "count command" counted
assert.eq(0, res.ok);
assert.eq(opCounters.command + 5, newdb.serverStatus().opcounters.command); // "serverStatus", "count" counted
assert.eq(
countVal.total + 1,
newdb.serverStatus().metrics.commands.count.total,
"total count command counter did not incremented",
); // "serverStatus", "count" counted
assert.eq(
countVal.failed + 1,
newdb.serverStatus().metrics.commands.count.failed,
"failed count command counter did not increment",
); // "serverStatus", "count" counted
// Command, unrecognized.
res = t.runCommand("invalid");
assert.eq(0, res.ok);
assert.eq(opCounters.command + 8, newdb.serverStatus().opcounters.command); // "serverStatus" counted
assert.eq(null, newdb.serverStatus().metrics.commands.invalid);
assert.eq(metricsObj["<UNKNOWN>"] + 1, newdb.serverStatus().metrics.commands["<UNKNOWN>"]);