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

Commit 0060ad7

Browse files
authored
fix(secondaries): fixes connection with secondary readPreference (#258)
Ensures that when readPreference is `secondary`, the `connect` event is not triggered until connection has been established with a secondary. Fixes NODE-1089
1 parent ec2b7a6 commit 0060ad7

File tree

2 files changed

+68
-4
lines changed

2 files changed

+68
-4
lines changed

lib/topologies/replset.js

+17-4
Original file line numberDiff line numberDiff line change
@@ -773,6 +773,22 @@ function applyAuthenticationContexts(self, server, callback) {
773773
applyAuth(authContexts, server, callback);
774774
}
775775

776+
function shouldTriggerConnect(self) {
777+
const isConnecting = self.state === CONNECTING;
778+
const hasPrimary = self.s.replicaSetState.hasPrimary();
779+
const hasSecondary = self.s.replicaSetState.hasSecondary();
780+
const secondaryOnlyConnectionAllowed = self.s.options.secondaryOnlyConnectionAllowed;
781+
const readPreferenceSecondary =
782+
self.s.connectOptions.readPreference &&
783+
self.s.connectOptions.readPreference.equals(ReadPreference.secondary);
784+
785+
return (
786+
(isConnecting &&
787+
((readPreferenceSecondary && hasSecondary) || (!readPreferenceSecondary && hasPrimary))) ||
788+
(hasSecondary && secondaryOnlyConnectionAllowed)
789+
);
790+
}
791+
776792
function handleInitialConnectEvent(self, event) {
777793
return function() {
778794
var _this = this;
@@ -835,10 +851,7 @@ function handleInitialConnectEvent(self, event) {
835851
_this.on('parseError', handleEvent(self, 'parseError'));
836852

837853
// Do we have a primary or primaryAndSecondary
838-
if (
839-
(self.state === CONNECTING && self.s.replicaSetState.hasPrimary()) ||
840-
(self.s.replicaSetState.hasSecondary() && self.s.options.secondaryOnlyConnectionAllowed)
841-
) {
854+
if (shouldTriggerConnect(self)) {
842855
// We are connected
843856
self.state = CONNECTED;
844857

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
'use strict';
2+
3+
const expect = require('chai').expect;
4+
const ReplSet = require('../../../../lib/topologies/replset');
5+
const ReadPreference = require('../../../../lib/topologies/read_preference');
6+
const mock = require('../../../mock');
7+
const ReplSetFixture = require('../common').ReplSetFixture;
8+
9+
describe('Secondaries (ReplSet)', function() {
10+
let test;
11+
before(() => (test = new ReplSetFixture()));
12+
afterEach(() => mock.cleanup());
13+
beforeEach(() => test.setup());
14+
15+
it('Should not be "connected" with ReadPreference secondary unless secondary is connected', {
16+
metadata: {
17+
requires: {
18+
topology: 'single'
19+
}
20+
},
21+
22+
test: function(done) {
23+
const replSet = new ReplSet(
24+
[test.primaryServer.address(), test.firstSecondaryServer.address()],
25+
{
26+
setName: 'rs',
27+
connectionTimeout: 3000,
28+
socketTimeout: 0,
29+
haInterval: 100,
30+
size: 1
31+
}
32+
);
33+
34+
replSet.on('error', done);
35+
36+
replSet.on('connect', server => {
37+
let err;
38+
try {
39+
expect(server.s.replicaSetState.hasSecondary()).to.equal(true);
40+
} catch (e) {
41+
err = e;
42+
}
43+
44+
replSet.destroy();
45+
done(err);
46+
});
47+
48+
replSet.connect({ readPreference: new ReadPreference('secondary') });
49+
}
50+
});
51+
});

0 commit comments

Comments
 (0)