Skip to content
This repository was archived by the owner on Feb 4, 2022. It is now read-only.

Commit dd4eb9a

Browse files
committed
fix(read-preference): correct server sort for nearest selection
The server sorting during `nearest` server selection was incorrectly using a greater-than sign, instead of following the rules of array sort comparitors. This fixes the issue for `pickNearest` as well as `pickNearestMaxStalenessSeconds`, and contributes tests for both paths. NODE-1577
1 parent 4f2b263 commit dd4eb9a

File tree

2 files changed

+48
-9
lines changed

2 files changed

+48
-9
lines changed

lib/topologies/replset_state.js

+2-8
Original file line numberDiff line numberDiff line change
@@ -878,19 +878,14 @@ function pickNearestMaxStalenessSeconds(self, readPreference) {
878878
// Filter by tags
879879
servers = filterByTags(readPreference, servers);
880880

881-
//
882-
// Locate lowest time (picked servers are lowest time + acceptable Latency margin)
883-
// var lowest = servers.length > 0 ? servers[0].lastIsMasterMS : 0;
884-
885881
// Filter by latency
886882
servers = servers.filter(function(s) {
887883
return s.staleness <= maxStalenessMS;
888884
});
889885

890886
// Sort by time
891887
servers.sort(function(a, b) {
892-
// return a.time > b.time;
893-
return a.lastIsMasterMS > b.lastIsMasterMS;
888+
return a.lastIsMasterMS - b.lastIsMasterMS;
894889
});
895890

896891
// No servers, default to primary
@@ -937,8 +932,7 @@ function pickNearest(self, readPreference) {
937932

938933
// Sort by time
939934
servers.sort(function(a, b) {
940-
// return a.time > b.time;
941-
return a.lastIsMasterMS > b.lastIsMasterMS;
935+
return a.lastIsMasterMS - b.lastIsMasterMS;
942936
});
943937

944938
// Locate lowest time (picked servers are lowest time + acceptable Latency margin)

test/tests/unit/replset/read_preference_tests.js

+46-1
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,10 @@ const ReplSet = require('../../../../lib/topologies/replset');
55
const ReadPreference = require('../../../../lib/topologies/read_preference');
66
const mock = require('mongodb-mock-server');
77
const ReplSetFixture = require('../common').ReplSetFixture;
8+
const ReplSetState = require('../../../../lib/topologies/replset_state');
9+
const MongoError = require('../../../..').MongoError;
810

9-
describe('Secondaries (ReplSet)', function() {
11+
describe('ReadPreference (ReplSet)', function() {
1012
let test;
1113
before(() => (test = new ReplSetFixture()));
1214
afterEach(() => mock.cleanup());
@@ -48,4 +50,47 @@ describe('Secondaries (ReplSet)', function() {
4850
replSet.connect({ readPreference: new ReadPreference('secondary') });
4951
}
5052
});
53+
54+
it('should correctly sort servers by `lastIsMasterMS` during nearest selection', function() {
55+
const state = new ReplSetState();
56+
const sampleData = [
57+
{ type: 'RSPrimary', lastIsMasterMS: 5 },
58+
{ type: 'RSSecondary', lastIsMasterMS: 4 },
59+
{ type: 'RSSecondary', lastIsMasterMS: 4 },
60+
{ type: 'RSSecondary', lastIsMasterMS: 109 },
61+
{ type: 'RSSecondary', lastIsMasterMS: 110 },
62+
{ type: 'RSSecondary', lastIsMasterMS: 110 },
63+
{ type: 'RSSecondary', lastIsMasterMS: 245 },
64+
{ type: 'RSSecondary', lastIsMasterMS: 221 },
65+
{ type: 'RSSecondary', lastIsMasterMS: 199 },
66+
{ type: 'RSSecondary', lastIsMasterMS: 129 },
67+
{ type: 'RSSecondary', lastIsMasterMS: 131 },
68+
{ type: 'RSSecondary', lastIsMasterMS: 284 },
69+
{ type: 'RSSecondary', lastIsMasterMS: 298 },
70+
{ type: 'RSSecondary', lastIsMasterMS: 289 },
71+
{ type: 'RSSecondary', lastIsMasterMS: 312 }
72+
];
73+
74+
// load mock data into replset state
75+
sampleData.forEach(desc => {
76+
desc.ismaster = { maxWireVersion: 6 }; // for maxStalenessSeconds test
77+
desc.staleness = Math.floor(Math.random() * 100);
78+
79+
if (desc.type === 'RSPrimary') {
80+
state.primary = desc;
81+
} else {
82+
state.secondaries.push(desc);
83+
}
84+
});
85+
86+
// select the nearest server without max staleness seconds
87+
let server = state.pickServer(ReadPreference.nearest);
88+
expect(server).to.not.be.an.instanceOf(MongoError);
89+
expect(server.lastIsMasterMS).to.equal(4);
90+
91+
// select the nearest server with max staleness seconds
92+
server = state.pickServer(new ReadPreference('nearest', { maxStalenessSeconds: 100 }));
93+
expect(server).to.not.be.an.instanceOf(MongoError);
94+
expect(server.lastIsMasterMS).to.equal(4);
95+
});
5196
});

0 commit comments

Comments
 (0)