From ae820f61d6a911aae495ee757d8e4c8ce420781b Mon Sep 17 00:00:00 2001 From: Dan Aprahamian Date: Thu, 19 Apr 2018 18:26:55 -0400 Subject: [PATCH] fix(pool): ensure that lsid is sent in get requests to mongos There is a strange case, when topology is mongos and readPreference is not primary, where we nest our find queries in a "$query" attribute. This was having the effect of not sending the lsid on find requests, which caused further errors when getMores attempted to send an lsid. Fixes NODE-1420 --- lib/connection/pool.js | 6 ++- test/tests/unit/mongos/sessions_tests.js | 66 ++++++++++++++++++++++++ 2 files changed, 71 insertions(+), 1 deletion(-) diff --git a/lib/connection/pool.js b/lib/connection/pool.js index 4330d0bb2..9923a9aa8 100644 --- a/lib/connection/pool.js +++ b/lib/connection/pool.js @@ -1222,7 +1222,11 @@ Pool.prototype.write = function(commands, options, cb) { // decorate the commands with session-specific details commands.forEach(command => { if (command instanceof Query) { - Object.assign(command.query, sessionOptions); + if (command.query.$query) { + Object.assign(command.query.$query, sessionOptions); + } else { + Object.assign(command.query, sessionOptions); + } } else { Object.assign(command, sessionOptions); } diff --git a/test/tests/unit/mongos/sessions_tests.js b/test/tests/unit/mongos/sessions_tests.js index e14c921bf..7e72b1dd7 100644 --- a/test/tests/unit/mongos/sessions_tests.js +++ b/test/tests/unit/mongos/sessions_tests.js @@ -4,6 +4,11 @@ var Mongos = require('../../../../lib/topologies/mongos'), mock = require('mongodb-mock-server'), genClusterTime = require('../common').genClusterTime; +const sessions = require('../../../../lib/sessions'); +const ServerSessionPool = sessions.ServerSessionPool; +const ClientSession = sessions.ClientSession; +const ReadPreference = require('../../../../lib/topologies/read_preference'); + const test = {}; describe('Sessions (Mongos)', function() { afterEach(() => mock.cleanup()); @@ -161,4 +166,65 @@ describe('Sessions (Mongos)', function() { mongos.connect(); } }); + + it( + 'should ensure that lsid is received within the query object of a find request when read preference is not primary', + { + metadata: { requires: { topology: 'single' } }, + test: function(done) { + const clusterTime = genClusterTime(Date.now()); + test.server.setMessageHandler(request => { + const doc = request.document; + if (doc.ismaster) { + request.reply( + Object.assign({}, mock.DEFAULT_ISMASTER_36, { + msg: 'isdbgrid', + $clusterTime: clusterTime + }) + ); + } else if (doc.$query) { + try { + expect(doc.$readPreference).to.deep.equal({ mode: 'primaryPreferred' }); + expect(doc) + .to.haveOwnProperty('$query') + .to.haveOwnProperty('lsid') + .that.is.an('object'); + done(); + } catch (e) { + done(e); + } + } else { + done('YOU HAVE FAILED. WE WILL FIND ANOTHER WAY. RELEASING CONTROL'); + } + }); + + const mongos = new Mongos([test.server.address()], { + connectionTimeout: 30000, + socketTimeout: 30000, + haInterval: 500, + size: 1 + }); + + mongos.on('error', done); + mongos.once('connect', () => { + const namespace = 'testdb.testcollection'; + const findCommand = { + find: namespace + }; + const pool = new ServerSessionPool(mongos); + const session = new ClientSession(mongos, pool); + const readPreference = new ReadPreference('primaryPreferred'); + + const cursor = mongos.cursor('testdb.testcollection', findCommand, { + session, + readPreference + }); + + cursor.next(() => {}); + }); + + mongos.connect(); + } + } + ); });