/jstests/noPassthrough/server_transaction_metrics.js

https://github.com/paralect/mongo · JavaScript · 202 lines · 136 code · 24 blank · 42 comment · 0 complexity · 4c5ea8e512af24e087e892626958bad8 MD5 · raw file

  1. // Tests multi-document transactions metrics in the serverStatus output.
  2. // @tags: [uses_transactions]
  3. (function() {
  4. "use strict";
  5. // Verifies that the server status response has the fields that we expect.
  6. function verifyServerStatusFields(serverStatusResponse) {
  7. assert(serverStatusResponse.hasOwnProperty("transactions"),
  8. "Expected the serverStatus response to have a 'transactions' field\n" +
  9. tojson(serverStatusResponse));
  10. assert(serverStatusResponse.transactions.hasOwnProperty("currentActive"),
  11. "The 'transactions' field in serverStatus did not have the 'currentActive' field\n" +
  12. tojson(serverStatusResponse.transactions));
  13. assert(serverStatusResponse.transactions.hasOwnProperty("currentInactive"),
  14. "The 'transactions' field in serverStatus did not have the 'currentInactive' field\n" +
  15. tojson(serverStatusResponse.transactions));
  16. assert(serverStatusResponse.transactions.hasOwnProperty("currentOpen"),
  17. "The 'transactions' field in serverStatus did not have the 'currentOpen' field\n" +
  18. tojson(serverStatusResponse.transactions));
  19. assert(serverStatusResponse.transactions.hasOwnProperty("totalAborted"),
  20. "The 'transactions' field in serverStatus did not have the 'totalAborted' field\n" +
  21. tojson(serverStatusResponse.transactions));
  22. assert(serverStatusResponse.transactions.hasOwnProperty("totalCommitted"),
  23. "The 'transactions' field in serverStatus did not have the 'totalCommitted' field\n" +
  24. tojson(serverStatusResponse.transactions));
  25. assert(serverStatusResponse.transactions.hasOwnProperty("totalStarted"),
  26. "The 'transactions' field in serverStatus did not have the 'totalStarted' field\n" +
  27. tojson(serverStatusResponse.transactions));
  28. }
  29. // Verifies that the given value of the server status response is incremented in the way
  30. // we expect.
  31. function verifyServerStatusChange(initialStats, newStats, valueName, expectedIncrement) {
  32. assert.eq(initialStats[valueName] + expectedIncrement,
  33. newStats[valueName],
  34. "expected " + valueName + " to increase by " + expectedIncrement);
  35. }
  36. // Set up the replica set.
  37. const rst = new ReplSetTest({nodes: 1});
  38. rst.startSet();
  39. rst.initiate();
  40. const primary = rst.getPrimary();
  41. // Set up the test database.
  42. const dbName = "test";
  43. const collName = "server_transactions_metrics";
  44. const testDB = primary.getDB(dbName);
  45. const adminDB = rst.getPrimary().getDB('admin');
  46. testDB.runCommand({drop: collName, writeConcern: {w: "majority"}});
  47. assert.commandWorked(testDB.runCommand({create: collName, writeConcern: {w: "majority"}}));
  48. // Start the session.
  49. const sessionOptions = {
  50. causalConsistency: false
  51. };
  52. const session = testDB.getMongo().startSession(sessionOptions);
  53. const sessionDb = session.getDatabase(dbName);
  54. const sessionColl = sessionDb[collName];
  55. // Get state of server status before the transaction.
  56. let initialStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  57. verifyServerStatusFields(initialStatus);
  58. // This transaction will commit.
  59. jsTest.log("Start a transaction and then commit it.");
  60. // Compare server status after starting a transaction with the server status before.
  61. session.startTransaction();
  62. assert.commandWorked(sessionColl.insert({_id: "insert-1"}));
  63. let newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  64. verifyServerStatusFields(newStatus);
  65. // Verify that the open transaction counter is incremented while inside the transaction.
  66. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 1);
  67. // Verify that when not running an operation, the transaction is inactive.
  68. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  69. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 1);
  70. // Compare server status after the transaction commit with the server status before.
  71. assert.commandWorked(session.commitTransaction_forTesting());
  72. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  73. verifyServerStatusFields(newStatus);
  74. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalStarted", 1);
  75. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalCommitted", 1);
  76. // Verify that current open counter is decremented on commit.
  77. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 0);
  78. // Verify that both active and inactive are 0 on commit.
  79. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  80. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 0);
  81. // This transaction will abort.
  82. jsTest.log("Start a transaction and then abort it.");
  83. // Compare server status after starting a transaction with the server status before.
  84. session.startTransaction();
  85. assert.commandWorked(sessionColl.insert({_id: "insert-2"}));
  86. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  87. verifyServerStatusFields(newStatus);
  88. // Verify that the open transaction counter is incremented while inside the transaction.
  89. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 1);
  90. // Verify that when not running an operation, the transaction is inactive.
  91. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  92. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 1);
  93. // Compare server status after the transaction abort with the server status before.
  94. assert.commandWorked(session.abortTransaction_forTesting());
  95. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  96. verifyServerStatusFields(newStatus);
  97. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalStarted", 2);
  98. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalCommitted", 1);
  99. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalAborted", 1);
  100. // Verify that current open counter is decremented on abort.
  101. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 0);
  102. // Verify that both active and inactive are 0 on abort.
  103. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  104. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 0);
  105. // This transaction will abort due to a duplicate key insert.
  106. jsTest.log("Start a transaction that will abort on a duplicated key error.");
  107. // Compare server status after starting a transaction with the server status before.
  108. session.startTransaction();
  109. // Inserting a new document will work fine, and the transaction starts.
  110. assert.commandWorked(sessionColl.insert({_id: "insert-3"}));
  111. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  112. verifyServerStatusFields(newStatus);
  113. // Verify that the open transaction counter is incremented while inside the transaction.
  114. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 1);
  115. // Verify that when not running an operation, the transaction is inactive.
  116. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  117. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 1);
  118. // Compare server status after the transaction abort with the server status before.
  119. // The duplicated insert will fail, causing the transaction to abort.
  120. assert.commandFailedWithCode(sessionColl.insert({_id: "insert-3"}), ErrorCodes.DuplicateKey);
  121. // Ensure that the transaction was aborted on failure.
  122. assert.commandFailedWithCode(session.commitTransaction_forTesting(), ErrorCodes.NoSuchTransaction);
  123. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  124. verifyServerStatusFields(newStatus);
  125. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalStarted", 3);
  126. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalCommitted", 1);
  127. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "totalAborted", 2);
  128. // Verify that current open counter is decremented on abort caused by an error.
  129. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 0);
  130. // Verify that both active and inactive are 0 on abort.
  131. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  132. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 0);
  133. // Hang the transaction on a failpoint in the middle of an operation to check active and
  134. // inactive counters while operation is running inside a transaction.
  135. jsTest.log("Start a transaction that will hang in the middle of an operation due to a fail point.");
  136. assert.commandWorked(
  137. testDB.adminCommand({configureFailPoint: 'hangDuringBatchUpdate', mode: 'alwaysOn'}));
  138. const transactionFn = function() {
  139. const collName = 'server_transactions_metrics';
  140. const session = db.getMongo().startSession({causalConsistency: false});
  141. const sessionDb = session.getDatabase('test');
  142. const sessionColl = sessionDb[collName];
  143. session.startTransaction({readConcern: {level: 'snapshot'}});
  144. assert.commandWorked(sessionColl.update({}, {"update-1": 2}));
  145. assert.commandWorked(session.commitTransaction_forTesting());
  146. };
  147. const transactionProcess = startParallelShell(transactionFn, primary.port);
  148. // Keep running currentOp() until we see the transaction subdocument.
  149. assert.soon(function() {
  150. const transactionFilter = {
  151. active: true,
  152. 'lsid': {$exists: true},
  153. 'transaction.parameters.txnNumber': {$eq: 0}
  154. };
  155. return 1 === adminDB.aggregate([{$currentOp: {}}, {$match: transactionFilter}]).itcount();
  156. });
  157. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  158. verifyServerStatusFields(newStatus);
  159. // Verify that the open transaction counter is incremented while inside the transaction.
  160. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 1);
  161. // Verify that the metrics show that the transaction is active while inside the operation.
  162. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 1);
  163. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 0);
  164. // Now the transaction can proceed.
  165. assert.commandWorked(
  166. testDB.adminCommand({configureFailPoint: 'hangDuringBatchUpdate', mode: 'off'}));
  167. transactionProcess();
  168. newStatus = assert.commandWorked(testDB.adminCommand({serverStatus: 1}));
  169. verifyServerStatusFields(newStatus);
  170. // Verify that current open counter is decremented on commit.
  171. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentOpen", 0);
  172. // Verify that both active and inactive are 0 after the transaction finishes.
  173. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentActive", 0);
  174. verifyServerStatusChange(initialStatus.transactions, newStatus.transactions, "currentInactive", 0);
  175. // End the session and stop the replica set.
  176. session.endSession();
  177. rst.stopSet();
  178. }());