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

Commit eb67b1a

Browse files
authored
fix(transactions): do not send txnNumber for non-write commands (#308)
* fix(transactions): do not send txnNumber for non-write commands We started sending txnNumber for non-transaction non-retryable commands, which causes errors against transaction-enabled servers Fixes NODE-1466
1 parent 854cbdb commit eb67b1a

File tree

4 files changed

+151
-4
lines changed

4 files changed

+151
-4
lines changed

lib/topologies/mongos.js

+1
Original file line numberDiff line numberDiff line change
@@ -895,6 +895,7 @@ var executeWriteOperation = function(self, op, ns, ops, options, callback) {
895895
}
896896

897897
// increment and assign txnNumber
898+
options.willRetryWrite = true;
898899
options.session.incrementTransactionNumber();
899900

900901
server[op](ns, ops, options, (err, result) => {

lib/topologies/replset.js

+1
Original file line numberDiff line numberDiff line change
@@ -1231,6 +1231,7 @@ function executeWriteOperation(args, options, callback) {
12311231
// increment and assign txnNumber
12321232
if (willRetryWrite) {
12331233
options.session.incrementTransactionNumber();
1234+
options.willRetryWrite = willRetryWrite;
12341235
}
12351236

12361237
// optionally autostart transaction if requested

lib/wireprotocol/3_2_support.js

+7-4
Original file line numberDiff line numberDiff line change
@@ -18,20 +18,23 @@ var WireProtocol = function(legacyWireProtocol) {
1818
*
1919
* @param {Object} command the command to decorate
2020
* @param {ClientSession} session the session tracking transaction state
21+
* @param {boolean} [isRetryableWrite=false] if true, will be decorated for retryable writes
2122
*/
22-
function decorateWithTransactionsData(command, session) {
23+
function decorateWithTransactionsData(command, session, isRetryableWrite) {
2324
if (!session) {
2425
return;
2526
}
2627

2728
// first apply non-transaction-specific sessions data
2829
const serverSession = session.serverSession;
29-
if (serverSession.txnNumber) {
30+
const inTransaction = session.inTransaction();
31+
32+
if (serverSession.txnNumber && (isRetryableWrite || inTransaction)) {
3033
command.txnNumber = BSON.Long.fromNumber(serverSession.txnNumber);
3134
}
3235

3336
// now try to apply tansaction-specific data
34-
if (!session.inTransaction()) {
37+
if (!inTransaction) {
3538
return;
3639
}
3740

@@ -103,7 +106,7 @@ var executeWrite = function(pool, bson, type, opsField, ns, ops, options, callba
103106
}
104107

105108
// optionally decorate command with transactions data
106-
decorateWithTransactionsData(writeCommand, options.session);
109+
decorateWithTransactionsData(writeCommand, options.session, options.willRetryWrite);
107110

108111
// Options object
109112
var opts = { command: true };
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
'use strict';
2+
3+
const expect = require('chai').expect;
4+
const ReplSet = require('../../../../lib/topologies/replset');
5+
const mock = require('mongodb-mock-server');
6+
const ReplSetFixture = require('../common').ReplSetFixture;
7+
const ClientSession = require('../../../../lib/sessions').ClientSession;
8+
const ServerSessionPool = require('../../../../lib/sessions').ServerSessionPool;
9+
10+
describe('Transaction Feature Decoration', function() {
11+
let test;
12+
const ns = 'db.foo';
13+
const noop = () => {};
14+
const ismaster = Object.assign({}, mock.DEFAULT_ISMASTER_36, { maxWireVersion: 7 });
15+
16+
before(() => (test = new ReplSetFixture()));
17+
afterEach(() => mock.cleanup());
18+
beforeEach(() => test.setup({ ismaster }));
19+
20+
class TestConfig {
21+
constructor(config, flags) {
22+
this.fnName = config.fnName;
23+
this.cmd = config.cmd;
24+
this.arg = config.arg();
25+
this.flags = flags;
26+
this.retryWrites = flags.retryWrites;
27+
this.session = flags.session;
28+
this.transaction = flags.transaction;
29+
}
30+
31+
get shouldPass() {
32+
if (this.session && this.transaction) {
33+
return true;
34+
}
35+
36+
if (this.fnName === 'command') {
37+
return false;
38+
}
39+
40+
return this.session && this.retryWrites;
41+
}
42+
43+
get description() {
44+
const not = this.shouldPass ? '' : 'not ';
45+
const flags = JSON.stringify(this.flags);
46+
47+
return `should ${not}have a txnNumber when command ${this.cmd} is used with ${flags}`;
48+
}
49+
}
50+
51+
[
52+
{ fnName: 'insert', cmd: 'insert', arg: () => [{ foo: 1 }] },
53+
{ fnName: 'update', cmd: 'update', arg: () => [{ foo: 1 }] },
54+
{ fnName: 'remove', cmd: 'delete', arg: () => [{ foo: 1 }] },
55+
{ fnName: 'command', cmd: 'fizzBuzz', arg: () => ({ fizzBuzz: 1 }) }
56+
]
57+
.reduce((testConfigs, op) => {
58+
for (let i = 0; i < 4; i += 1) {
59+
const options = {
60+
retryWrites: i % 2 === 1,
61+
session: i >= 2
62+
};
63+
64+
testConfigs.push(new TestConfig(op, options));
65+
66+
if (options.session) {
67+
testConfigs.push(new TestConfig(op, Object.assign({ transaction: true }, options)));
68+
}
69+
}
70+
return testConfigs;
71+
}, [])
72+
.forEach(config => {
73+
it(config.description, {
74+
metadata: { requires: { topology: 'single', mongodb: '>=3.7.3' } },
75+
test: function(done) {
76+
const replSet = new ReplSet(
77+
[test.primaryServer.address(), test.firstSecondaryServer.address()],
78+
{
79+
setName: 'rs',
80+
connectionTimeout: 3000,
81+
socketTimeout: 0,
82+
haInterval: 100,
83+
size: 1
84+
}
85+
);
86+
87+
function shutdown(err) {
88+
replSet.destroy();
89+
done(err);
90+
}
91+
92+
test.primaryServer.setMessageHandler(request => {
93+
try {
94+
const doc = request.document;
95+
96+
if (doc.ismaster) {
97+
return request.reply(test.primaryStates[0]);
98+
}
99+
100+
if (doc[config.cmd]) {
101+
if (config.shouldPass) {
102+
expect(doc).to.have.property('txnNumber');
103+
} else {
104+
expect(doc).to.not.have.property('txnNumber');
105+
}
106+
107+
request.reply({ ok: 1 });
108+
109+
setTimeout(() => shutdown());
110+
}
111+
} catch (e) {
112+
return shutdown(e);
113+
}
114+
});
115+
116+
const sessionPool = new ServerSessionPool(replSet);
117+
118+
replSet.on('connect', () => {
119+
const options = {};
120+
121+
if (config.retryWrites) {
122+
options.retryWrites = true;
123+
}
124+
125+
if (config.session) {
126+
options.session = new ClientSession(replSet, sessionPool, {}, {});
127+
128+
if (config.transaction) {
129+
options.session.startTransaction();
130+
}
131+
}
132+
133+
replSet[config.fnName](ns, config.arg, options, noop);
134+
});
135+
136+
replSet.on('error', shutdown);
137+
138+
replSet.connect();
139+
}
140+
});
141+
});
142+
});

0 commit comments

Comments
 (0)