-
Notifications
You must be signed in to change notification settings - Fork 49
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
DOCSP-4466 Add CSOT #951
base: master
Are you sure you want to change the base?
DOCSP-4466 Add CSOT #951
Changes from 15 commits
1178d8d
0b8b950
0d5a095
703252c
381da04
9145ed6
20827e2
74770f0
d87f15c
f4134ad
0d5f77a
5f4f261
75909a9
b5d5f6d
1ed5035
06a08d7
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
import { MongoClient } from "mongodb"; | ||
|
||
// start-operation | ||
const uri = "<connection string uri>"; | ||
const client = new MongoClient(uri, { | ||
timeoutMS: 10000 | ||
}); | ||
|
||
async function run() { | ||
try { | ||
const db = client.db("test-db"); | ||
const coll = db.collection("test-collection"); | ||
|
||
const result = await coll.insertOne({ name: "Yngwie" }); | ||
console.log("Insert result:", result); | ||
} finally { | ||
await client.close(); | ||
} | ||
} | ||
|
||
run().catch(console.dir); | ||
// end-operation |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
import { MongoClient } from "mongodb"; | ||
|
||
const { MongoClient } = require('mongodb'); | ||
|
||
//start-csot | ||
// Creates a new MongoClient with a client-level timeoutMS configuration | ||
const uri = "<connection string uri>"; | ||
const client = new MongoClient(uri, { | ||
// Client-level timeout: 15 seconds | ||
timeoutMS: 15000 | ||
}); | ||
|
||
async function run() { | ||
try { | ||
const db = client.db("test-db"); | ||
const coll = db.collection("test-collection"); | ||
|
||
// Performs a query operation with an operation-level timeoutMS configuration | ||
const docs = await coll.find({}, | ||
// Operation-level timeout: 10 seconds | ||
{ timeoutMS: 10000 }) | ||
.toArray(); | ||
|
||
console.log(docs); | ||
} finally { | ||
await client.close(); | ||
} | ||
} | ||
|
||
run().catch(console.dir); | ||
//end-csot |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,227 @@ | ||
.. _node-csot: | ||
|
||
Limit Server Execution Time | ||
=========================== | ||
|
||
.. contents:: On this page | ||
:local: | ||
:backlinks: none | ||
:depth: 2 | ||
:class: singlecol | ||
|
||
.. facet:: | ||
:name: genre | ||
:values: reference | ||
|
||
.. meta:: | ||
:keywords: error, blocking, thread, task | ||
|
||
Overview | ||
-------- | ||
|
||
When you use {+driver-short+} to perform a server operation, you can also limit | ||
stephmarie17 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
the duration allowed for the server to finish the operation. To do so, | ||
specify a **client-side operation timeout (CSOT)**. The timeout applies to all steps | ||
needed to complete the operation, including server selection, connection | ||
checkout, serialization, and server-side execution. When the timeout expires, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. checkout, It is only possible to timeout asynchronous operations serialization is not async, so if it takes time to serialize it won't be interrupted by a time out, rather the next async thing performed will be interrupted by the expired timer. Encryption is a good example of something that is async before we reach server-side execution. I don't think this detail is to critical to users but we don't want to be inaccurate. |
||
{+driver-short+} raises a timeout exception. | ||
stephmarie17 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
.. note:: Experimental Feature | ||
|
||
The CSOT feature is experimental and might change in future driver releases. | ||
|
||
timeoutMS Option | ||
---------------- | ||
|
||
To specify a timeout when connecting to a MongoDB deployment, set the | ||
``timeoutMS`` connection option to the timeout length in milliseconds. You can | ||
specify the ``timeoutMS`` option in your ``MongoClientOptions`` instance or at | ||
the database, collection, session, transaction, or operation levels. | ||
Comment on lines
+38
to
+39
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We should probably mention it's also supported as a URI option here |
||
|
||
The following code example uses the ``timeoutMS`` option to specify a timeout of | ||
10 seconds when instantiating a new ``MongoClient`` instance: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't necessarily want us to change this but I figure I'd bring it up to get your thoughts. Folks should probably be setting a timeoutMS north of 30 seconds for most use cases, the famous serverSelectionTimeoutMS defaults to that value as a reasonable amount of time for an app to recover when a replicaset is reconfiguring. Nginx defaults timing out http requests to 60 seconds. Now of course we're just offering an example, but I worry about the copy paste-able code here encouraging timeouts that are lower than necessary and lead to a degraded experience. Of course users should feel empowered to set any value, but reasonably 10s can be small. |
||
|
||
.. code-block:: javascript | ||
|
||
const uri = "mongodb://<hostname:<port>"; | ||
const client = new MongoClient(uri, { | ||
timeoutMS: 10000 | ||
}); | ||
|
||
.. note:: | ||
|
||
The ``timeoutMS`` connection option takes precedence over the | ||
following options: | ||
|
||
- ``socketTimeoutMS`` | ||
- ``waitQueueTimeoutMS`` | ||
- ``wTimeoutMS`` | ||
- ``maxTimeMS`` | ||
- ``maxCommitTimeMS`` | ||
|
||
When the CSOT feature is no longer experimental, the preceding options will | ||
be deprecated. | ||
|
||
If you specify the ``timeoutMS`` option, the driver automatically applies the | ||
specified timeout to each server operation. The following code example specifies | ||
Comment on lines
+65
to
+66
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. q: i thought the timeout applied to the total time for all operations in the aggregate. does this depend on the level at which you specify it? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's been awhile since I initially wrote this but if I recall correctly this was how CSOT is also described in the PyMongo doc so I carried this intro over but will see if there is a better way to describe this during technical review! There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's per operation. Cursors introduce their own complexity of lifetime or per iteration. But there's no concept of a total timeout for all operations performed on a client. |
||
a timeout of 10 seconds at the client level, and then calls the ``insertOne()`` | ||
method: | ||
|
||
.. literalinclude:: /code-snippets/connection/csot-operation.js | ||
:language: javascript | ||
:start-after: start-operation | ||
:end-before: end-operation | ||
|
||
Timeout Inheritance | ||
~~~~~~~~~~~~~~~~~~~ | ||
|
||
When you specify a ``timeoutMS`` option, the driver applies the timeout | ||
according to the same inheritance behaviors as other {+driver-short+} options. | ||
stephmarie17 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
The following table describes how the timeout value is inherited at each level: | ||
|
||
.. list-table:: | ||
:header-rows: 1 | ||
:widths: 30 70 | ||
|
||
* - Level | ||
- Inheritance Description | ||
|
||
* - Operation | ||
- Takes the highest precedence and will override ``timeoutMS`` | ||
options set at any other level. | ||
|
||
* - Transaction | ||
- Takes precedence over ``timeoutMS`` set at the session, | ||
collection, database, or client level. | ||
|
||
* - Session | ||
- Applies to all transactions and operations within | ||
that session, unless the option is overridden by options set at those levels. | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Missing database level from this list |
||
* - Collection | ||
- Applies to all sessions and operations on that | ||
collection, unless the option is overridden by options set at those levels. | ||
|
||
* - Client | ||
- Applies to all databases, collections, sessions, transactions, and | ||
operations within that client that do not otherwise specify | ||
``timeoutMS``. | ||
|
||
For more information on overrides and specific options, see the :ref:`Overrides | ||
<node-csot-overrides>` section. | ||
|
||
.. _node-csot-overrides: | ||
|
||
Overrides | ||
--------- | ||
|
||
The {+driver-short+} supports various levels of configuration to control the | ||
behavior and performance of database operations. | ||
|
||
You can specify a ``timeoutMS`` option at the operation level to override the | ||
client-level configuration for a specific operation. This allows you to | ||
customize timeouts based on the needs of individual queries. | ||
|
||
The following example demonstrates how an operation-level ``timeoutMS`` | ||
configuration can override a client-level ``timeoutMS`` configuration: | ||
|
||
.. literalinclude:: /code-snippets/connection/csot.js | ||
:language: javascript | ||
:start-after: start-csot | ||
:end-before: end-csot | ||
|
||
Transactions | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. q: can you help me understand the relationship between Overrides / Transactions / Client Encryption? it almost seems like overrides should be appended to the previous section on inheritance. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah I was debating how to organize this and actually ended up adding the inheritance section after writing these for additional clarity around the expected behavior. I added that as a subsection thinking it was more connected to the option, while the Overrides section calls out some behavior specific to the example use case, transactions, and client encryption. Mostly based on what was requested in the ticket. Does this make sense as is or do you think it needs further tweaking on the section headings? |
||
~~~~~~~~~~~~ | ||
|
||
When you create a new ``ClientSession`` instance to implement a transaction, use | ||
the ``defaultTimeoutMS`` option. You can set ``defaultTimeoutMS`` to specify the | ||
``timeoutMS`` values to use for: | ||
|
||
- `commitTransaction() | ||
<{+api+}/classes/ClientSession.html#commitTransaction>`__ | ||
- `abortTransaction() | ||
<{+api+}/classes/ClientSession.html#abortTransaction>`__ | ||
- `withTransaction() <{+api+}/classes/ClientSession.html#withTransaction>`__ | ||
- `endSession() | ||
<{+api+}/classes/ClientSession.html#endSession>`__ | ||
|
||
If you do not specify ``defaultTimeoutMS``, the driver uses the ``timeoutMS`` | ||
value set on the parent ``MongoClient``. | ||
|
||
You cannot override ``defaultTimeoutMS`` by setting the ``timeoutMS`` option on an | ||
operation in a transaction session provided by the ``withTransaction()`` callback. | ||
Doing so throws an error. | ||
|
||
Client Encryption | ||
~~~~~~~~~~~~~~~~~ | ||
|
||
When you use Client-Side Field Level Encryption (CSFLE), the driver uses the | ||
``timeoutMS`` option to limit the time allowed for encryption and decryption | ||
operations. | ||
|
||
If you specify the ``timeoutMS`` option when you construct a | ||
``ClientEncryption`` instance, it controls the lifetime of all operations | ||
performed on that instance. If you do not provide ``timeoutMS``, the instance | ||
inherits the ``timeoutMS`` setting from the ``MongoClient`` used in the | ||
``ClientEncryption`` constructor. | ||
|
||
If you set ``timeoutMS`` on both the client and directly in | ||
``ClientEncryption``, the value provided to ``ClientEncryption`` takes | ||
precedence. | ||
|
||
Cursors | ||
------- | ||
|
||
Cursors require special handling when you use the CSOT feature. You can configure | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. To me, this implies complexity when the defaults are going to be appropriate for most folks. Cursors do not require special handling, they offer configuration if needed, the timeoutMode default (lifetime) is likely the common denominator. |
||
cursors to interact with CSOT by using either the cursor lifetime or cursor | ||
iteration mode. | ||
|
||
Cursor Lifetime Mode | ||
~~~~~~~~~~~~~~~~~~~~ | ||
|
||
The cursor lifetime mode uses ``timeoutMS`` to limit the entire lifetime of a | ||
cursor. This is the default timeout mode for non-tailable cursors, such as those | ||
returned by the ``find()``, ``aggregate()``, and ``listCollections()`` methods. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. find and aggregate both can be invoked with the right options to get a tailable cursor. Maybe it would be clearer to just say "This is the default timeout mode" [for default cursors]. We can expand further later (if you're making a tailable cursor, then timeoutMode... etc) |
||
In this mode, the initialization of the cursor and all subsequent calls to the | ||
``getMore()`` method must finish within the limit specified by the ``timeoutMS`` | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Users do not call |
||
option. If they do not, the system throws a timeout error. | ||
|
||
When you close a cursor by calling the ``toArray()``or ``close()`` method, the | ||
stephmarie17 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
timeout resets. | ||
|
||
The following example shows how to set the ``timeoutMS`` option to ensure that | ||
the cursor is initialized and all documents are retrieved within 10 seconds: | ||
|
||
.. code-block:: javascript | ||
|
||
const docs = await collection.find({}, {timeoutMS: 10000}).toArray(); | ||
|
||
Cursor Iteration Mode | ||
~~~~~~~~~~~~~~~~~~~~~ | ||
|
||
The cursor iteration mode uses the ``timeoutMS`` option to limit each call to | ||
the ``next()``, ``hasNext()``, or ``tryNext()`` method. The timeout refreshes | ||
after each call completes. This is the default mode for all tailable cursors, | ||
such as the tailable cursors returned by the ``find()`` method on capped | ||
collections or change streams. | ||
|
||
As shown in the following example, the cursor continues to fetch new documents | ||
as they are added to a collection, and then times out if it takes longer than 10 | ||
seconds to retrieve documents: | ||
stephmarie17 marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
.. code-block:: javascript | ||
|
||
for await (const doc of cappedCollection.find({}, | ||
{tailable: true, timeoutMS: 10000})) { | ||
// Handle each document | ||
}; | ||
Comment on lines
+214
to
+217
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. While tailable cursors are defaulted to the other timeoutMode I think that their usage is rare enough that it is unnecessary for us to combine their use with the timeoutMode setting for our only example for iteration mode. If I am using a normal find cursor and I want to perform some async action for each document timeoutMode=iteration is what I'm going to want to use so I don't bound my "some action" code by the driver's timeoutMS setting: for await (const movie of mflix.find({}, { timeoutMode: 'iteration' })) {
const imdbResponse = await fetch(movie.imdb_url);
console.log(await imdbResponse.text());
} If you left the timeoutMode in lifetime here then the cursor's time to finish would likely be deeply cut into by the http fetch operation. I think an example like this is a bit more practical to the average user compared to tailable cursors. |
||
|
||
API Documentation | ||
----------------- | ||
|
||
To learn more about using timeouts with the {+driver-short+}, see the following | ||
API documentation: | ||
|
||
- `MongoClient <{+api+}/classes/MongoClient.html>`__ | ||
- `timeoutMS <{+api+}/classes/MongoClient.html#timeoutMS>`__ | ||
- `ClientSession <{+api+}/classes/ClientSession.html>`__ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.