mongo/jstests/libs/cmd_object_utils.js

205 lines
5.8 KiB
JavaScript

import {sysCollNamePrefix} from "jstests/core/timeseries/libs/timeseries_writes_util.js";
export const kGenericArgFieldNames = [
"apiVersion",
"apiStrict",
"apiDeprecationErrors",
"maxTimeMS",
"readConcern",
"writeConcern",
"lsid",
"clientOperationKey",
"txnNumber",
"autocommit",
"startTransaction",
"stmtId",
"comment",
"$readPreference",
"$clusterTime",
"$audit",
"$client",
"$configServerState",
"$oplogQueryData",
"$queryOptions",
"$replData",
"$traceCtx",
"databaseVersion",
"help",
"shardVersion",
"tracking_info",
"coordinator",
"maxTimeMSOpOnly",
"usesDefaultMaxTimeMS",
"$configTime",
"$topologyTime",
"txnRetryCounter",
"versionContext",
"mayBypassWriteBlocking",
"expectPrefix",
"requestGossipRoutingCache",
"startOrContinueTransaction",
"rawData",
"serialization_context",
];
/**
* Resolves the command name for the given 'cmdObj'.
*/
export function getCommandName(cmdObj) {
return Object.keys(cmdObj)[0];
}
/**
* Returns the inner command if 'cmdObj' represents an explain command, or simply 'cmdObj'
* otherwise.
*/
export function getInnerCommand(cmdObj) {
const isExplain = "explain" in cmdObj;
if (!isExplain) {
return cmdObj;
}
if (typeof cmdObj.explain === "object") {
const [{explain}, genericArgs] = extractGenericArgs(cmdObj);
return {...explain, ...genericArgs};
}
const {explain, ...cmdWithoutExplain} = cmdObj;
return cmdWithoutExplain;
}
/**
* Splits 'cmdObj' into a pair of [<command without generic args>, <generic args>].
*/
function extractGenericArgs(cmdObj) {
let cmd = {},
genericArgs = {};
for (const [key, value] of Object.entries(cmdObj)) {
const isGenericArg = kGenericArgFieldNames.includes(key);
if (isGenericArg) {
genericArgs[key] = value;
} else {
cmd[key] = value;
}
}
return [cmd, genericArgs];
}
/**
* Returns the explain command object for the given 'cmdObj'.
*/
export function getExplainCommand(cmdObj, verbosity = "queryPlanner") {
// Extract the generic arguments out of 'cmd' so they can be re-added on root of the final
// explain command.
const [cmd, genericArgs] = extractGenericArgs(cmdObj);
if ("rawData" in genericArgs) {
// Unlike the rest of the generic arguments, 'rawData' must be passed to the inner command.
cmd.rawData = genericArgs.rawData;
delete genericArgs.rawData;
}
// Remove explain incompatible generic arguments.
const explainIncompatibleGenericArgs = [
// Cannot run 'explain' with "writeConcern".
"writeConcern",
// Cannot run 'explain' in a multi-document transaction.
"txnNumber",
"autocommit",
"startTransaction",
"stmtId",
// Don't pass in the "$readPreference" to ensure that the explain command can be executed
// against any node in a cluster.
"$readPreference",
];
for (const arg of explainIncompatibleGenericArgs) {
delete genericArgs[arg];
}
// Explained aggregate commands require a cursor argument.
const isAggregateCmd = getCommandName(cmdObj) === "aggregate";
if (isAggregateCmd) {
cmd.cursor = {};
}
return {explain: cmd, ...genericArgs, verbosity};
}
/**
* Returns the value of the rawData parameter for the given 'cmdObj'.
* Returns false if the parameter is not included.
*/
export function getRawData(cmdObj) {
const [cmd, genericArgs] = extractGenericArgs(cmdObj);
if ("rawData" in genericArgs) {
return genericArgs.rawData;
}
return false;
}
/**
* Resolves the collection name for the given 'cmdObj'. If the command targets a view, then this
* will recursively find the underlying collection's name and return it. Returns 'undefined' if the
* collection does not exist.
*/
export function getCollectionName(db, cmdObj) {
try {
const collectionsInfo = db.getCollectionInfos();
if (!collectionsInfo || collectionsInfo.length === 0) {
return undefined;
}
let viewOn;
let name = cmdObj[getCommandName(cmdObj)];
do {
let collInfo = collectionsInfo.find((c) => c.name === name);
if (!collInfo) {
name = undefined;
}
viewOn = collInfo?.options?.viewOn;
if (viewOn) {
name = viewOn;
}
} while (viewOn);
return name;
} catch (ex) {
switch (ex.code) {
case ErrorCodes.InvalidViewDefinition: {
// The 'DB.prototype.getCollectionInfos()' implementation may throw an exception
// when faced with a malformed view definition. This is analogous to a missing
// collection for the purpose of passthrough suites.
return undefined;
}
default:
throw ex;
}
}
}
export function isSystemCollectionName(collectionName) {
return collectionName.startsWith("system.");
}
export function isInternalDbName(dbName) {
return ["admin", "local", "config"].includes(dbName);
}
/**
* Returns true iff the 'collectionName' exists and it is a timeseries collection.
*/
export function isTimeSeriesCollection(db, collectionName) {
const collectionInfo = db.getCollectionInfos({name: collectionName});
if (!collectionInfo || collectionInfo.length === 0) {
return false;
}
return collectionInfo[0].type === "timeseries" || collectionName.startsWith("system.bucket.");
}
/**
* Return true iff this is a "system.bucket.*" collection.
*/
export function isSystemBucketNss(innerCmd) {
const nss = innerCmd[getCommandName(innerCmd)];
return typeof nss === "string" && nss.startsWith(sysCollNamePrefix);
}