Overview
In this guide, you can learn how to set up and configure monitoring in the MongoDB Node.js driver.
Monitoring involves collecting information about the activities of a running program, which you can use with an application performance management library.
Monitoring the Node.js driver lets you understand the driver's resource usage and performance, and can help you make informed decisions when designing and debugging your application.
In this guide you will learn how to perform these tasks:
This guide shows how to use information about the activity of the driver in code. To learn how to record events in the driver, see the Node.js driver's Logging guide.
Monitor Events
You can monitor events using the Node.js driver by subscribing to them in your application.
An event is any action that occurs within the driver during its operation. The Node.js driver includes functionality for listening to a subset of these events.
The Node.js driver organizes the events it defines into the following categories:
- Command Events 
- Server Discovery and Monitoring (SDAM) Events 
- Connection Pool Events 
The following sections show how to monitor each event category.
Command Events
A command event is an event related to a MongoDB database command. You can access one or more command monitoring events using the driver by subscribing to them in your application.
To learn more about MongoDB database commands, see the Database Commands guide in the Server Manual.
Note
Command monitoring is disabled by default. To enable command
monitoring, pass the monitorCommands option as true to
your MongoClient constructor.
Example
The following example demonstrates connecting to a replica set and subscribing to one of the command monitoring events created by the MongoDB deployment:
/* Subscribe to an event */ const { MongoClient } = require("mongodb"); // Replace the following with your MongoDB deployment's connection string const uri = "mongodb+srv://<clusterUrl>/?replicaSet=rs&writeConcern=majority"; const client = new MongoClient(uri, { monitorCommands:true }); // Replace <event name> with the name of the event you are subscribing to const eventName = "<event name>"; // Subscribes to a specified event and print a message when the event is received client.on(eventName, event => console.log(event)); async function run() {   try {     // Establishes and verifies connection to the "admin" database     await client.db("admin").command({ ping: 1 });     console.log("Connected successfully");   } finally {     // Closes the database connection on completion or error     await client.close();   } } run().catch(console.dir); 
Event Descriptions
You can subscribe to any of the following command monitoring events:
| Event Name | Description | 
|---|---|
| 
 | Created when a command is started. | 
| 
 | Created when a command succeeded. | 
| 
 | Created when a command failed. | 
The following sections show sample output for each of the preceding commands. Your output might differ depending on the command you run and the options you set.
commandStarted
CommandStartedEvent {   name: 'commandStarted',   address: 'localhost:27017',   connectionId: 812613,   serviceId: undefined,   requestId: 1534,   commandName: 'find',   command: {     find: { firstName: "Jane", lastName: "Doe" }   },   serverConnectionId: 27177n } 
commandSucceeded
CommandSucceededEvent {   name: 'commandSucceeded',   address: 'localhost:27017',   connectionId: 812613,   serviceId: undefined,   requestId: 1534,   commandName: 'find',   duration: 15,   reply: {     cursor: {       firstBatch: [         {           _id: ObjectId("5e8e2ca217b5324fa9847435"),           firstName: "Jane",           lastName: "Doe"         }       ],       _id: 0,       ns: "app.users"     },     ok: 1,     operationTime: 1586380205   },   serverConnectionId: 27096n, } 
commandFailed
CommandFailedEvent {   name: 'commandFailed',   address: 'localhost:27017',   connectionId: 812613,   serviceId: undefined,   requestId: 1534,   commandName: 'find',   duration: 11,   failure: Error("something failed"),   serverConnectionId: 27208n, } 
Server Discovery and Monitoring Events
The Node.js driver creates topology events, also known as SDAM events, when there is a change in the state of the instance or cluster that you connected to. For example, the driver creates an event when you establish a new connection or if the cluster elects a new primary node.
To learn more about topology events, see the Replication guide in the Server Manual.
The following sections demonstrate how to record topology changes in your application and explore the information provided in these events.
Event Subscription Example
You can access one or more SDAM events by subscribing to them in your application. The following example demonstrates connecting to a replica set and subscribing to one of the SDAM events created by the MongoDB deployment:
/* Subscribe to SDAM event */ const { MongoClient } = require("mongodb"); // Replace the following with your MongoDB deployment's connection string const uri = "mongodb+srv://<clusterUrl>/?replicaSet=rs&writeConcern=majority"; const client = new MongoClient(uri); // Replace <event name> with the name of the event you are subscribing to const eventName = "<event name>"; // Subscribes to a specified event and prints a message when the event is received client.on(eventName, event => {   console.log(`received ${eventName}: ${JSON.stringify(event, null, 2)}`); }); async function run() {   try {     // Establishes and verifies connection to the database     await client.db("admin").command({ ping: 1 });     console.log("Connected successfully");   } finally {     // Closes the database connection on completion or error     await client.close();   } } run().catch(console.dir); 
Event Descriptions
You can subscribe to any of the following SDAM events:
| Event Name | Description | 
|---|---|
| 
 | Created when a connection to an instance opens. | 
| 
 | Created when a connection to an instance closes. | 
| 
 | Created when an instance state changes (such as from secondary to primary). | 
| 
 | Created before attempting a connection to an instance. | 
| 
 | Created after all instance connections in the topology close. | 
| 
 | Created when the topology changes, such as an election of a new primary or a mongos proxy disconnecting. | 
| 
 | Created before issuing a  | 
| 
 | Created when the  | 
| 
 | Created when a  | 
Example Event Documents
The following sections show sample output for each type of SDAM event.
serverDescriptionChanged
ServerDescriptionChangedEvent {   topologyId: 0,   address: 'localhost:27017',   previousDescription: ServerDescription {     address: 'localhost:27017',     error: null,     roundTripTime: 0,     lastUpdateTime: 1571251089030,     lastWriteDate: null,     opTime: null,     type: 'Unknown',     minWireVersion: 0,     maxWireVersion: 0,     hosts: [],     passives: [],     arbiters: [],     tags: []   },   newDescription: ServerDescription {     address: 'localhost:27017',     error: null,     roundTripTime: 0,     lastUpdateTime: 1571251089051,     lastWriteDate: 2019-10-16T18:38:07.000Z,     opTime: { ts: Timestamp, t: 18 },     type: 'RSPrimary',     minWireVersion: 0,     maxWireVersion: 7,     maxBsonObjectSize: 16777216,     maxMessageSizeBytes: 48000000,     maxWriteBatchSize: 100000,     me: 'localhost:27017',     hosts: [ 'localhost:27017' ],     passives: [],     arbiters: [],     tags: [],     setName: 'rs',     setVersion: 1,     electionId: ObjectID,     primary: 'localhost:27017',     logicalSessionTimeoutMinutes: 30,     '$clusterTime': ClusterTime   } } 
The type field of the ServerDescription object in this event contains
one of the following possible values:
| Type | Description | 
|---|---|
| 
 | Unknown instance | 
| 
 | Standalone instance | 
| 
 | Mongos proxy instance | 
| 
 | At least one server recognizes this as the primary, but is not yet verified by all instances. | 
| 
 | Primary instance | 
| 
 | Secondary instance | 
| 
 | Arbiter instance | 
| 
 | See the RSGhost and RSOther specification for more details | 
| 
 | See the RSGhost and RSOther specification for more details | 
serverHeartbeatStarted
ServerHeartbeatStartedEvent {   connectionId: 'localhost:27017' } 
serverHeartbeatSucceeded
ServerHeartbeatSucceededEvent {   duration: 1.939997,   reply:{     hosts: [ 'localhost:27017' ],     setName: 'rs',     setVersion: 1,     isWritablePrimary: true,     secondary: false,     primary: 'localhost:27017',     me: 'localhost:27017',     electionId: ObjectID,     lastWrite: {       opTime: { ts: [Timestamp], t: 18 },       lastWriteDate: 2019-10-16T18:38:17.000Z,       majorityOpTime: { ts: [Timestamp], t: 18 },       majorityWriteDate: 2019-10-16T18:38:17.000Z     },     maxBsonObjectSize: 16777216,     maxMessageSizeBytes: 48000000,     maxWriteBatchSize: 100000,     localTime: 2019-10-16T18:38:19.589Z,     logicalSessionTimeoutMinutes: 30,     minWireVersion: 0,     maxWireVersion: 7,     readOnly: false,     ok: 1,     operationTime: Timestamp,     '$clusterTime': ClusterTime   },   connectionId: 'localhost:27017' } 
serverHeartbeatFailed
ServerHeartbeatFailed {   duration: 20,   failure: MongoError('some error'),   connectionId: 'localhost:27017' } 
serverOpening
ServerOpeningEvent {   topologyId: 0,   address: 'localhost:27017' } 
serverClosed
ServerClosedEvent {   topologyId: 0,   address: 'localhost:27017' } 
topologyOpening
TopologyOpeningEvent {   topologyId: 0 } 
topologyClosed
TopologyClosedEvent {   topologyId: 0 } 
topologyDescriptionChanged
TopologyDescriptionChangedEvent {   topologyId: 0,   previousDescription: TopologyDescription {     type: 'ReplicaSetNoPrimary',     setName: null,     maxSetVersion: null,     maxElectionId: null,     servers: Map {       'localhost:27017' => ServerDescription     },     stale: false,     compatible: true,     compatibilityError: null,     logicalSessionTimeoutMinutes: null,     heartbeatFrequencyMS: 10000,     localThresholdMS: 15,     options: Object,     error: undefined,     commonWireVersion: null   },   newDescription: TopologyDescription {     type: 'ReplicaSetWithPrimary',     setName: 'rs',     maxSetVersion: 1,     maxElectionId: null,     servers: Map {       'localhost:27017' => ServerDescription     },     stale: false,     compatible: true,     compatibilityError: null,     logicalSessionTimeoutMinutes: 30,     heartbeatFrequencyMS: 10000,     localThresholdMS: 15,     options: Object,     error: undefined,     commonWireVersion: 7   } } 
The type field of the TopologyDescription object in this event contains
one of the following possible values:
| Type | Description | 
|---|---|
| 
 | Standalone instance | 
| 
 | Replica set with a primary | 
| 
 | Replica set with no primary | 
| 
 | Sharded cluster | 
| 
 | Unknown topology | 
Connection Pool Events
A connection pool is a set of open TCP connections your driver maintains with a MongoDB instance. Connection pools help reduce the number of network handshakes your application needs to perform and can help your application run faster.
The following sections demonstrate how to record connection pool events in your application and explore the information provided in these events.
Event Subscription Examples
You can access one or more connection pool events using the driver by subscribing to them in your application. The following example demonstrates connecting to a replica set and subscribing to one of the connection pool monitoring events created by the MongoDB deployment:
const { MongoClient } = require("mongodb"); // Replace the following with your MongoDB deployment's connection string const uri =   "mongodb+srv://<clusterUrl>/?replicaSet=rs&writeConcern=majority"; const client = new MongoClient(uri); // Replace <event name> with the name of the event you are subscribing to const eventName = "<event name>"; // Subscribes to the event client.on(eventName, (event) =>   console.log("\nreceived event:\n", event) ); async function run() {   try {     // Establishes and verifies connection     await client.db("admin").command({ ping: 1 });     console.log("\nConnected successfully!\n");   } finally {     // Ensures that the client will close when you finish/error     await client.close();   } } run().catch(console.dir); 
Connection pool monitoring events can aid you in debugging and understanding the behavior of your application's connection pool. The following example uses connection pool monitoring events to return a count of checked-out connections in the pool:
function connectionPoolStatus(client) {   let checkedOut = 0;   function onCheckout() {     checkedOut++;   }   function onCheckin() {     checkedOut--;   }   function onClose() {     client.removeListener('connectionCheckedOut', onCheckout);     client.removeListener('connectionCheckedIn', onCheckin);     checkedOut = NaN;   }   // Decreases count of connections checked out of the pool when connectionCheckedIn event is triggered   client.on('connectionCheckedIn', onCheckin);   // Increases count of connections checked out of the pool when connectionCheckedOut event is triggered   client.on('connectionCheckedOut', onCheckout);   // Cleans up event listeners when client is closed   client.on('close', onClose);   return {     count: () => checkedOut,     cleanUp: onClose   }; } 
Event Descriptions
You can subscribe to any of the following connection pool monitoring events:
| Event Name | Description | 
|---|---|
| 
 | Created when a connection pool is created. | 
| 
 | Created when a connection pool is ready. | 
| 
 | Created when a connection pool is closed before server instance destruction. | 
| 
 | Created when a connection is created, but not necessarily when it is used for an operation. | 
| 
 | Created after a connection has successfully completed a handshake and is ready to be used for operations. | 
| 
 | Created when a connection is closed. | 
| 
 | Created when an operation attempts to acquire a connection for execution. | 
| 
 | Created when an operation fails to acquire a connection for execution. | 
| 
 | Created when an operation successfully acquires a connection for execution. | 
| 
 | Created when a connection is checked back into the pool after an operation is executed. | 
| 
 | Created when all connections are closed and the connection pool is cleared. | 
Example Event Documents
The following sections show sample output for each type of connection pool monitoring event.
connectionPoolCreated
ConnectionPoolCreatedEvent {   time: 2023-02-13T15:54:06.944Z,   address: '...',   options: {...} } 
connectionPoolReady
ConnectionPoolReadyEvent {   time: 2023-02-13T15:56:38.440Z,   address: '...' } 
connectionPoolClosed
ConnectionPoolClosedEvent {   time: 2023-02-13T15:56:38.440Z,   address: '...' } 
connectionCreated
ConnectionCreatedEvent {   time: 2023-02-13T15:56:38.291Z,   address: '...',   connectionId: 1 } 
connectionReady
ConnectionReadyEvent {   time: 2023-02-13T15:56:38.291Z,   address: '...',   connectionId: 1,   durationMS: 60 } 
connectionClosed
ConnectionClosedEvent {   time: 2023-02-13T15:56:38.439Z,   address: '...',   connectionId: 1,   reason: 'poolClosed',   serviceId: undefined } 
connectionCheckOutStarted
ConnectionCheckOutStartedEvent {   time: 2023-02-13T15:56:38.291Z,   address: '...', } 
connectionCheckOutFailed
ConnectionCheckOutFailedEvent {   time: 2023-02-13T15:56:38.291Z,   address: '...',   reason: ...,   durationMS: 60 } 
connectionCheckedOut
ConnectionCheckedOutEvent {   time: 2023-02-13T15:54:07.188Z,   address: '...',   connectionId: 1,   durationMS: 60 } 
connectionCheckedIn
ConnectionCheckedInEvent {   time: 2023-02-13T15:54:07.189Z,   address: '...',   connectionId: 1 } 
connectionPoolCleared
ConnectionPoolClearedEvent {   time: 2023-02-13T15:56:38.439Z,   address: '...',   serviceId: undefined,   interruptInUseConnections: true, } 
API Documentation
To learn more about any of the options or types discussed in this guide, see the following API documentation: