Skip to content

Commit 708ce16

Browse files
authored
PYTHON-4724 - Prohibit AsyncMongoClient from being used across multiple event loops (#2256)
1 parent 1c813dc commit 708ce16

File tree

5 files changed

+60
-1
lines changed

5 files changed

+60
-1
lines changed

pymongo/asynchronous/mongo_client.py

+8
Original file line numberDiff line numberDiff line change
@@ -878,6 +878,7 @@ def __init__(
878878

879879
self._opened = False
880880
self._closed = False
881+
self._loop: Optional[asyncio.AbstractEventLoop] = None
881882
if not is_srv:
882883
self._init_background()
883884

@@ -1709,6 +1710,13 @@ async def _get_topology(self) -> Topology:
17091710
If this client was created with "connect=False", calling _get_topology
17101711
launches the connection process in the background.
17111712
"""
1713+
if not _IS_SYNC:
1714+
if self._loop is None:
1715+
self._loop = asyncio.get_running_loop()
1716+
elif self._loop != asyncio.get_running_loop():
1717+
raise RuntimeError(
1718+
"Cannot use AsyncMongoClient in different event loop. AsyncMongoClient uses low-level asyncio APIs that bind it to the event loop it was created on."
1719+
)
17121720
if not self._opened:
17131721
if self._resolve_srv_info["is_srv"]:
17141722
await self._resolve_srv()

pymongo/synchronous/mongo_client.py

+8
Original file line numberDiff line numberDiff line change
@@ -876,6 +876,7 @@ def __init__(
876876

877877
self._opened = False
878878
self._closed = False
879+
self._loop: Optional[asyncio.AbstractEventLoop] = None
879880
if not is_srv:
880881
self._init_background()
881882

@@ -1703,6 +1704,13 @@ def _get_topology(self) -> Topology:
17031704
If this client was created with "connect=False", calling _get_topology
17041705
launches the connection process in the background.
17051706
"""
1707+
if not _IS_SYNC:
1708+
if self._loop is None:
1709+
self._loop = asyncio.get_running_loop()
1710+
elif self._loop != asyncio.get_running_loop():
1711+
raise RuntimeError(
1712+
"Cannot use MongoClient in different event loop. MongoClient uses low-level asyncio APIs that bind it to the event loop it was created on."
1713+
)
17061714
if not self._opened:
17071715
if self._resolve_srv_info["is_srv"]:
17081716
self._resolve_srv()

pyproject.toml

+2
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,8 @@ filterwarnings = [
117117
"module:unclosed <ssl.SSLSocket:ResourceWarning",
118118
"module:unclosed <socket object:ResourceWarning",
119119
"module:unclosed transport:ResourceWarning",
120+
# pytest-asyncio known issue: https://github.com/pytest-dev/pytest-asyncio/issues/724
121+
"module:unclosed event loop:ResourceWarning",
120122
# https://github.com/eventlet/eventlet/issues/818
121123
"module:please use dns.resolver.Resolver.resolve:DeprecationWarning",
122124
# https://github.com/dateutil/dateutil/issues/1314
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
# Copyright 2025-present MongoDB, Inc.
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
15+
"""Test that the asynchronous API detects event loop changes and fails correctly."""
16+
from __future__ import annotations
17+
18+
import asyncio
19+
import unittest
20+
21+
from pymongo import AsyncMongoClient
22+
23+
24+
class TestClientLoopSafety(unittest.TestCase):
25+
def test_client_errors_on_different_loop(self):
26+
client = AsyncMongoClient()
27+
loop1 = asyncio.new_event_loop()
28+
loop1.run_until_complete(client.aconnect())
29+
loop2 = asyncio.new_event_loop()
30+
with self.assertRaisesRegex(
31+
RuntimeError, "Cannot use AsyncMongoClient in different event loop"
32+
):
33+
loop2.run_until_complete(client.aconnect())
34+
loop1.run_until_complete(client.close())
35+
loop1.close()
36+
loop2.close()

tools/synchro.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -180,7 +180,12 @@
180180

181181
def async_only_test(f: str) -> bool:
182182
"""Return True for async tests that should not be converted to sync."""
183-
return f in ["test_locks.py", "test_concurrency.py", "test_async_cancellation.py"]
183+
return f in [
184+
"test_locks.py",
185+
"test_concurrency.py",
186+
"test_async_cancellation.py",
187+
"test_async_loop_safety.py",
188+
]
184189

185190

186191
test_files = [

0 commit comments

Comments
 (0)