'use strict';
|
|
const ReadPreference = require('../topologies/read_preference');
|
const MongoError = require('../error').MongoError;
|
const ServerType = require('../sdam/server_description').ServerType;
|
const TopologyDescription = require('../sdam/topology_description').TopologyDescription;
|
|
const MESSAGE_HEADER_SIZE = 16;
|
const COMPRESSION_DETAILS_SIZE = 9; // originalOpcode + uncompressedSize, compressorID
|
|
// OPCODE Numbers
|
// Defined at https://docs.mongodb.com/manual/reference/mongodb-wire-protocol/#request-opcodes
|
var opcodes = {
|
OP_REPLY: 1,
|
OP_UPDATE: 2001,
|
OP_INSERT: 2002,
|
OP_QUERY: 2004,
|
OP_GETMORE: 2005,
|
OP_DELETE: 2006,
|
OP_KILL_CURSORS: 2007,
|
OP_COMPRESSED: 2012,
|
OP_MSG: 2013
|
};
|
|
var getReadPreference = function(cmd, options) {
|
// Default to command version of the readPreference
|
var readPreference = cmd.readPreference || new ReadPreference('primary');
|
// If we have an option readPreference override the command one
|
if (options.readPreference) {
|
readPreference = options.readPreference;
|
}
|
|
if (typeof readPreference === 'string') {
|
readPreference = new ReadPreference(readPreference);
|
}
|
|
if (!(readPreference instanceof ReadPreference)) {
|
throw new MongoError('read preference must be a ReadPreference instance');
|
}
|
|
return readPreference;
|
};
|
|
// Parses the header of a wire protocol message
|
var parseHeader = function(message) {
|
return {
|
length: message.readInt32LE(0),
|
requestId: message.readInt32LE(4),
|
responseTo: message.readInt32LE(8),
|
opCode: message.readInt32LE(12)
|
};
|
};
|
|
function applyCommonQueryOptions(queryOptions, options) {
|
Object.assign(queryOptions, {
|
raw: typeof options.raw === 'boolean' ? options.raw : false,
|
promoteLongs: typeof options.promoteLongs === 'boolean' ? options.promoteLongs : true,
|
promoteValues: typeof options.promoteValues === 'boolean' ? options.promoteValues : true,
|
promoteBuffers: typeof options.promoteBuffers === 'boolean' ? options.promoteBuffers : false,
|
monitoring: typeof options.monitoring === 'boolean' ? options.monitoring : false,
|
fullResult: typeof options.fullResult === 'boolean' ? options.fullResult : false
|
});
|
|
if (typeof options.socketTimeout === 'number') {
|
queryOptions.socketTimeout = options.socketTimeout;
|
}
|
|
if (options.session) {
|
queryOptions.session = options.session;
|
}
|
|
if (typeof options.documentsReturnedIn === 'string') {
|
queryOptions.documentsReturnedIn = options.documentsReturnedIn;
|
}
|
|
return queryOptions;
|
}
|
|
function isSharded(topologyOrServer) {
|
if (topologyOrServer.type === 'mongos') return true;
|
if (topologyOrServer.description && topologyOrServer.description.type === ServerType.Mongos) {
|
return true;
|
}
|
|
// NOTE: This is incredibly inefficient, and should be removed once command construction
|
// happens based on `Server` not `Topology`.
|
if (topologyOrServer.description && topologyOrServer.description instanceof TopologyDescription) {
|
const servers = Array.from(topologyOrServer.description.servers.values());
|
return servers.some(server => server.type === ServerType.Mongos);
|
}
|
|
return false;
|
}
|
|
function databaseNamespace(ns) {
|
return ns.split('.')[0];
|
}
|
function collectionNamespace(ns) {
|
return ns
|
.split('.')
|
.slice(1)
|
.join('.');
|
}
|
|
module.exports = {
|
getReadPreference,
|
MESSAGE_HEADER_SIZE,
|
COMPRESSION_DETAILS_SIZE,
|
opcodes,
|
parseHeader,
|
applyCommonQueryOptions,
|
isSharded,
|
databaseNamespace,
|
collectionNamespace
|
};
|