summaryrefslogtreecommitdiff
path: root/jstests/libs/override_methods/enable_sessions.js
blob: 846143da9990b26bf784eaaf4e0e3de16577a863 (plain)
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
/**
 * Enables sessions on the db object
 */
(function() {
"use strict";

load("jstests/libs/override_methods/override_helpers.js");

const getDBOriginal = Mongo.prototype.getDB;

const sessionMap = new WeakMap();
const sessionOptions = TestData.sessionOptions;

// Override the runCommand to check for any command obj that does not contain a logical session
// and throw an error.
function runCommandWithLsidCheck(conn, dbName, cmdName, cmdObj, func, makeFuncArgs) {
    if (jsTest.options().disableEnableSessions) {
        return func.apply(conn, makeFuncArgs(cmdObj));
    }

    // If the command is in a wrapped form, then we look for the actual command object
    // inside the query/$query object.
    let cmdObjUnwrapped = cmdObj;
    if (cmdName === "query" || cmdName === "$query") {
        cmdObj[cmdName] = Object.assign({}, cmdObj[cmdName]);
        cmdObjUnwrapped = cmdObj[cmdName];
    }

    if (!cmdObjUnwrapped.hasOwnProperty("lsid")) {
        // TODO: SERVER-30848 fixes getMore requests to use a session in the mongo shell.
        // Until that happens, we bypass throwing an error for getMore and only throw an error
        // for other requests not using sessions.
        if (cmdName !== "getMore") {
            throw new Error("command object does not have session id: " + tojson(cmdObj));
        }
    }
    return func.apply(conn, makeFuncArgs(cmdObj));
}

// Override the getDB to return a db object with the correct driverSession. We use a WeakMap
// to cache the session for each connection instance so we can retrieve the same session on
// subsequent calls to getDB.
Mongo.prototype.getDB = function(dbName) {
    if (jsTest.options().disableEnableSessions) {
        return getDBOriginal.apply(this, arguments);
    }

    if (!sessionMap.has(this)) {
        const session = this.startSession(sessionOptions);
        // Override the endSession function to be a no-op so jstestfuzz doesn't accidentally
        // end the session.
        session.endSession = Function.prototype;
        sessionMap.set(this, session);
    }

    const db = getDBOriginal.apply(this, arguments);
    db._session = sessionMap.get(this);
    return db;
};

// Override the global `db` object to be part of a session.
db = db.getMongo().getDB(db.getName());

OverrideHelpers.prependOverrideInParallelShell("jstests/libs/override_methods/enable_sessions.js");
OverrideHelpers.overrideRunCommand(runCommandWithLsidCheck);
})();