-
Notifications
You must be signed in to change notification settings - Fork 5.6k
/
Copy pathsimulate_atlas_proxy.js
230 lines (203 loc) · 8.01 KB
/
simulate_atlas_proxy.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
221
222
223
224
225
226
227
228
229
230
/**
* Overrides the runCommand method to prefix all databases and namespaces ("config", "admin",
* "local" excluded) with a tenant prefix.
*/
import {OverrideHelpers} from "jstests/libs/override_methods/override_helpers.js";
import {
createCmdObjWithTenantId,
getTenantIdForDatabase,
isCmdObjWithTenantId,
prependTenantIdToDbNameIfApplicable,
removeTenantIdAndMaybeCheckPrefixes,
} from "jstests/serverless/libs/tenant_prefixing.js";
// Assert that some tenantIds are provided
assert(!!TestData.tenantId || (TestData.tenantIds && TestData.tenantIds.length > 0),
"Missing required tenantId or tenantIds");
// Save references to the original methods in the IIFE's scope.
// This scoping allows the original methods to be called by the overrides below.
const originalRunCommand = Mongo.prototype.runCommand;
// Save a reference to the connection created at shell startup. This will be used as a proxy for
// multiple internal routing connections for the lifetime of the test execution. If there is no
// initial connection, then we will not perform connection routing when using this override.
const initialConn = (typeof db !== 'undefined') ? db.getMongo() : undefined;
/**
* Asserts that the provided connection is an internal routing connection, not the top-level proxy
* connection. The proxy connection also has an internal routing connection, so it is excluded from
* this check.
*/
function assertRoutingConnection(conn) {
if (conn !== initialConn) {
assert.eq(null,
conn._internalRoutingConnection,
"Expected connection to have no internal routing connection.");
}
}
/**
* @returns The internal routing connection for a provided connection
*/
function getRoutingConnection(conn) {
if (conn === initialConn && conn._internalRoutingConnection == null) {
conn._internalRoutingConnection = conn;
}
// Since we are patching the prototype below, there must eventually be a "base case" for
// determining which connection to run a method on. If the provided `conn` has no internal
// routing connection, we assume that it _is_ the internal routing connection, and return
// here.
if (conn._internalRoutingConnection == null) {
return conn;
}
// Sanity check ensuring we have not accidentally created an internal routing connection on an
// internal routing connection.
assertRoutingConnection(conn._internalRoutingConnection);
return conn._internalRoutingConnection;
}
function toIndexSet(indexedDocs) {
let set = new Set();
if (indexedDocs) {
for (let doc of indexedDocs) {
set.add(doc.index);
}
}
return set;
}
/**
* Remove the indices for non-upsert writes that succeeded.
*/
function removeSuccessfulOpIndexesExceptForUpserted(resObj, indexMap, ordered) {
// Optimization to only look through the indices in a set rather than in an array.
let indexSetForUpserted = toIndexSet(resObj.upserted);
let indexSetForWriteErrors = toIndexSet(resObj.writeErrors);
for (let index in Object.keys(indexMap)) {
if ((!indexSetForUpserted.has(parseInt(index)) &&
!(ordered && resObj.writeErrors && (index > resObj.writeErrors[0].index)) &&
!indexSetForWriteErrors.has(parseInt(index)))) {
delete indexMap[index];
}
}
return indexMap;
}
/**
* Rewrites a server connection string (ex: rsName/host,host,host) to a URI that the shell can
* connect to.
*/
function convertServerConnectionStringToURI(input) {
const inputParts = input.split('/');
return `mongodb://${inputParts[1]}/?replicaSet=${inputParts[0]}`;
}
/**
* Executes 'cmdObjWithTenantId'
* 'dbNameWithTenantId' is only used for logging.
*/
function runCommandWithTenantId(
conn, securityToken, dbNameWithTenantId, cmdObjWithTenantId, options) {
// 'indexMap' is a mapping from a write's index in the current cmdObj to its index in the
// original cmdObj.
let indexMap = {};
if (cmdObjWithTenantId.documents) {
for (let i = 0; i < cmdObjWithTenantId.documents.length; i++) {
indexMap[i] = i;
}
}
if (cmdObjWithTenantId.updates) {
for (let i = 0; i < cmdObjWithTenantId.updates.length; i++) {
indexMap[i] = i;
}
}
if (cmdObjWithTenantId.deletes) {
for (let i = 0; i < cmdObjWithTenantId.deletes.length; i++) {
indexMap[i] = i;
}
}
const newConn = getRoutingConnection(conn);
if (securityToken) {
newConn._setSecurityToken(securityToken);
}
return originalRunCommand.apply(newConn, [dbNameWithTenantId, cmdObjWithTenantId, options]);
}
Mongo.prototype.runCommand = function(dbName, cmdObj, options) {
const useSecurityToken = !!TestData.useSecurityToken;
const useResponsePrefixChecking = !!TestData.useResponsePrefixChecking;
const tenantId = getTenantIdForDatabase(dbName);
const dbNameWithTenantId = prependTenantIdToDbNameIfApplicable(dbName, tenantId);
const securityToken = useSecurityToken
? _createTenantToken({tenant: ObjectId(tenantId), expectPrefix: true})
: undefined;
// If the command is already prefixed, just run it
if (isCmdObjWithTenantId(cmdObj)) {
return runCommandWithTenantId(this, securityToken, dbNameWithTenantId, cmdObj, options);
}
// Prepend a tenant prefix to all database names and namespaces, where applicable.
const cmdObjWithTenantId = createCmdObjWithTenantId(cmdObj, tenantId);
const resObj = runCommandWithTenantId(
this, securityToken, dbNameWithTenantId, cmdObjWithTenantId, options);
// Remove the tenant prefix from all database names and namespaces in the result since tests
// assume the command was run against the original database.
const cmdName = Object.keys(cmdObj)[0];
let checkPrefixOptions = !useResponsePrefixChecking ? {} : {
checkPrefix: true,
expectPrefix: true,
tenantId,
dbName,
cmdName,
debugLog: "Failed to check tenant prefix in response : " + tojsononeline(resObj) +
". The request command obj is " + tojsononeline(cmdObjWithTenantId)
};
removeTenantIdAndMaybeCheckPrefixes(resObj, checkPrefixOptions);
return resObj;
};
Mongo.prototype.getDbNameWithTenantPrefix = function(dbName) {
const tenantId = getTenantIdForDatabase(dbName);
return prependTenantIdToDbNameIfApplicable(dbName, tenantId);
};
// Override base methods on the Mongo prototype to try to proxy the call to the underlying
// internal routing connection, if one exists.
// NOTE: This list is derived from scripting/mozjs/mongo.cpp:62.
['auth',
'cleanup',
'close',
'compact',
'getAutoEncryptionOptions',
'isAutoEncryptionEnabled',
'cursorHandleFromId',
'find',
'generateDataKey',
'getDataKeyCollection',
'logout',
'encrypt',
'decrypt',
'isReplicaSetConnection',
'_markNodeAsFailed',
'getMinWireVersion',
'getMaxWireVersion',
'isReplicaSetMember',
'isMongos',
'isTLS',
'getApiParameters',
'_startSession',
'_refreshAccessToken',
// Don't override this method, since it is never called directly in jstests. The expectation of is
// that it will be run on the connection `Mongo.prototype.runCommand` chose.
// '_runCommandImpl',
].forEach(methodName => {
const $method = Mongo.prototype[methodName];
Mongo.prototype[methodName] = function() {
return $method.apply(getRoutingConnection(this), arguments);
};
});
// The following methods are overridden so that the method applies to both
// the proxy connection and the underlying internal routing connection, if one exists.
['toggleAutoEncryption',
'unsetAutoEncryption',
'setAutoEncryption',
].forEach(methodName => {
const $method = Mongo.prototype[methodName];
Mongo.prototype[methodName] = function() {
let rc = getRoutingConnection(this);
if (rc !== this) {
$method.apply(rc, arguments);
}
return $method.apply(this, arguments);
};
});
OverrideHelpers.prependOverrideInParallelShell(
"jstests/libs/override_methods/simulate_atlas_proxy.js");