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

Commit 478d1e7

Browse files
committed
feat(with-transaction): provide helper for convenient txn api
This introduces a helper (`withTransaction`) that provides a covenient api for automatic retryability around committing transactions. NODE-1741
1 parent 32a5e74 commit 478d1e7

File tree

1 file changed

+103
-1
lines changed

1 file changed

+103
-1
lines changed

lib/sessions.js

+103-1
Original file line numberDiff line numberDiff line change
@@ -243,6 +243,106 @@ class ClientSession extends EventEmitter {
243243
toBSON() {
244244
throw new Error('ClientSession cannot be serialized to BSON.');
245245
}
246+
247+
/**
248+
* A user provided function to be run within a transaction
249+
*
250+
* @callback WithTransactionCallback
251+
* @param {ClientSession} session The parent session of the transaction running the operation. This should be passed into each operation within the lambda.
252+
* @returns {Promise} The resulting Promise of operations run within this transaction
253+
*/
254+
255+
/**
256+
* Runs a provided lambda within a transaction, retrying either the commit operation
257+
* or entire transaction as needed (and when the error permits) to better ensure that
258+
* the transaction can complete successfully.
259+
*
260+
* IMPORTANT: This method requires the user to return a Promise, all lambdas that do not
261+
* return a Promise will result in undefined behavior.
262+
*
263+
* @param {WithTransactionCallback} fn
264+
* @param {TransactionOptions} [options] Optional settings for the transaction
265+
*/
266+
withTransaction(fn, options) {
267+
const self = this;
268+
const startTime = Date.now();
269+
270+
function retryCommit() {
271+
return self.commitTransaction().catch(err => {
272+
if (
273+
!isWriteConcernTimeoutError(err) &&
274+
(err instanceof MongoError && err.hasErrorLabel('UnknownTransactionCommitResult')) &&
275+
Date.now() - startTime < 120000
276+
) {
277+
return retryCommit();
278+
}
279+
280+
if (
281+
err instanceof MongoError &&
282+
err.hasErrorLabel('TransientTransactionError') &&
283+
Date.now() - startTime < 120000
284+
) {
285+
return retryTransaction();
286+
}
287+
288+
throw err;
289+
});
290+
}
291+
292+
function retryTransaction() {
293+
self.startTransaction(options);
294+
295+
// TODO: should we support callbacks?
296+
return fn(self)
297+
.then(() => {
298+
if (
299+
[
300+
TxnState.NO_TRANSACTION,
301+
TxnState.TRANSACTION_COMMITTED,
302+
TxnState.TRANSACTION_ABORTED
303+
].indexOf(self.transaction.state) !== -1
304+
) {
305+
// Assume user provided function intentionally ended the transaction
306+
return;
307+
}
308+
309+
return retryCommit();
310+
})
311+
.catch(err => {
312+
function maybeRetryOrThrow(err) {
313+
if (
314+
err instanceof MongoError &&
315+
err.hasErrorLabel('TransientTransactionError') &&
316+
Date.now() - startTime < 120000
317+
) {
318+
return retryTransaction();
319+
}
320+
321+
throw err;
322+
}
323+
324+
if (self.transaction.isActive) {
325+
return self.abortTransaction().then(() => maybeRetryOrThrow(err));
326+
}
327+
328+
return maybeRetryOrThrow(err);
329+
});
330+
}
331+
332+
return retryTransaction();
333+
}
334+
}
335+
336+
function isWriteConcernTimeoutError(err) {
337+
return err.code === 64 && !!(err.errInfo && err.errInfo.wtimeout === true);
338+
}
339+
340+
function isUnknownTransactionCommitResult(err) {
341+
return (
342+
['CannotSatisfyWriteConcern', 'UnknownReplWriteConcern', 'UnsatisfiableWriteConcern'].indexOf(
343+
err.codeName
344+
) === -1
345+
);
246346
}
247347

248348
function endTransaction(session, commandName, callback) {
@@ -334,7 +434,9 @@ function endTransaction(session, commandName, callback) {
334434
e.errorLabels = [];
335435
}
336436

337-
e.errorLabels.push('UnknownTransactionCommitResult');
437+
if (isUnknownTransactionCommitResult(e)) {
438+
e.errorLabels.push('UnknownTransactionCommitResult');
439+
}
338440
}
339441
} else {
340442
session.transaction.transition(TxnState.TRANSACTION_ABORTED);

0 commit comments

Comments
 (0)