From b5e2b0ba14cdbd9ce2f56b5eda6f0ba8d9403b57 Mon Sep 17 00:00:00 2001 From: "J. Nick Koston" Date: Tue, 1 Oct 2024 20:24:49 -0500 Subject: [PATCH] [PR #7368/8a8913b backport][3.10] Fixed failure to try next host after single-host connection timeout (#9390) Co-authored-by: Sam Bull Co-authored-by: pre-commit-ci[bot] Co-authored-by: J. Nick Koston Co-authored-by: Brett Higgins --- CHANGES/7342.breaking.rst | 3 + CONTRIBUTORS.txt | 1 + aiohttp/client.py | 2 +- aiohttp/connector.py | 2 +- docs/client_quickstart.rst | 3 +- docs/client_reference.rst | 8 ++- tests/test_client_session.py | 4 +- tests/test_connector.py | 115 ++++++++++++++++++++++++++++++++++- 8 files changed, 130 insertions(+), 8 deletions(-) create mode 100644 CHANGES/7342.breaking.rst diff --git a/CHANGES/7342.breaking.rst b/CHANGES/7342.breaking.rst new file mode 100644 index 00000000000..1fa511c4c97 --- /dev/null +++ b/CHANGES/7342.breaking.rst @@ -0,0 +1,3 @@ +Fixed failure to try next host after single-host connection timeout -- by :user:`brettdh`. + +The default client :class:`aiohttp.ClientTimeout` params has changed to include a ``sock_connect`` timeout of 30 seconds so that this correct behavior happens by default. diff --git a/CONTRIBUTORS.txt b/CONTRIBUTORS.txt index c318f7cc669..52cb1d59ff3 100644 --- a/CONTRIBUTORS.txt +++ b/CONTRIBUTORS.txt @@ -60,6 +60,7 @@ Bob Haddleton Boris Feld Boyi Chen Brett Cannon +Brett Higgins Brian Bouterse Brian C. Lane Brian Muller diff --git a/aiohttp/client.py b/aiohttp/client.py index 5c83099258a..7f42e4b8d4d 100644 --- a/aiohttp/client.py +++ b/aiohttp/client.py @@ -208,7 +208,7 @@ class ClientTimeout: # 5 Minute default read timeout -DEFAULT_TIMEOUT: Final[ClientTimeout] = ClientTimeout(total=5 * 60) +DEFAULT_TIMEOUT: Final[ClientTimeout] = ClientTimeout(total=5 * 60, sock_connect=30) # https://www.rfc-editor.org/rfc/rfc9110#section-9.2.2 IDEMPOTENT_METHODS = frozenset({"GET", "HEAD", "OPTIONS", "TRACE", "PUT", "DELETE"}) diff --git a/aiohttp/connector.py b/aiohttp/connector.py index 81d49083837..1c1283190d4 100644 --- a/aiohttp/connector.py +++ b/aiohttp/connector.py @@ -1318,7 +1318,7 @@ async def _create_direct_connection( req=req, client_error=client_error, ) - except ClientConnectorError as exc: + except (ClientConnectorError, asyncio.TimeoutError) as exc: last_exc = exc aiohappyeyeballs.pop_addr_infos_interleave(addr_infos, self._interleave) continue diff --git a/docs/client_quickstart.rst b/docs/client_quickstart.rst index 51b2ca1ec6d..f99339cf4a6 100644 --- a/docs/client_quickstart.rst +++ b/docs/client_quickstart.rst @@ -417,7 +417,8 @@ Timeouts Timeout settings are stored in :class:`ClientTimeout` data structure. By default *aiohttp* uses a *total* 300 seconds (5min) timeout, it means that the -whole operation should finish in 5 minutes. +whole operation should finish in 5 minutes. In order to allow time for DNS fallback, +the default ``sock_connect`` timeout is 30 seconds. The value could be overridden by *timeout* parameter for the session (specified in seconds):: diff --git a/docs/client_reference.rst b/docs/client_reference.rst index 7379743ae02..1b582932523 100644 --- a/docs/client_reference.rst +++ b/docs/client_reference.rst @@ -164,10 +164,14 @@ The client session supports the context manager protocol for self closing. overwrite it on a per-request basis. :param timeout: a :class:`ClientTimeout` settings structure, 300 seconds (5min) - total timeout by default. + total timeout, 30 seconds socket connect timeout by default. .. versionadded:: 3.3 + .. versionchanged:: 3.10.9 + + The default value for the ``sock_connect`` timeout has been changed to 30 seconds. + :param bool auto_decompress: Automatically decompress response body (``True`` by default). .. versionadded:: 2.3 @@ -897,7 +901,7 @@ certification chaining. .. versionadded:: 3.7 :param timeout: a :class:`ClientTimeout` settings structure, 300 seconds (5min) - total timeout by default. + total timeout, 30 seconds socket connect timeout by default. :param loop: :ref:`event loop` used for processing HTTP requests. diff --git a/tests/test_client_session.py b/tests/test_client_session.py index 86f3a1b6c6e..dac05ae3eb9 100644 --- a/tests/test_client_session.py +++ b/tests/test_client_session.py @@ -913,7 +913,9 @@ async def test_client_session_timeout_args(loop) -> None: with pytest.warns(DeprecationWarning): session2 = ClientSession(loop=loop, read_timeout=20 * 60, conn_timeout=30 * 60) - assert session2._timeout == client.ClientTimeout(total=20 * 60, connect=30 * 60) + assert session2._timeout == client.ClientTimeout( + total=20 * 60, connect=30 * 60, sock_connect=client.DEFAULT_TIMEOUT.sock_connect + ) with pytest.raises(ValueError): ClientSession( diff --git a/tests/test_connector.py b/tests/test_connector.py index 2e43573db4e..a21dd872993 100644 --- a/tests/test_connector.py +++ b/tests/test_connector.py @@ -10,7 +10,7 @@ from collections import deque from concurrent import futures from contextlib import closing, suppress -from typing import Any, List, Literal, Optional +from typing import Any, List, Literal, Optional, Sequence, Tuple from unittest import mock import pytest @@ -25,6 +25,7 @@ connector as connector_module, web, ) +from aiohttp.client_proto import ResponseHandler from aiohttp.client_reqrep import ConnectionKey from aiohttp.connector import ( _SSL_CONTEXT_UNVERIFIED, @@ -34,6 +35,7 @@ _DNSCacheTable, ) from aiohttp.locks import EventResultOrError +from aiohttp.resolver import ResolveResult from aiohttp.test_utils import make_mocked_coro, unused_port from aiohttp.tracing import Trace @@ -970,7 +972,116 @@ async def create_connection(*args, **kwargs): established_connection.close() -async def test_tcp_connector_resolve_host(loop: Any) -> None: +@pytest.mark.parametrize( + ("request_url"), + [ + ("http://mocked.host"), + ("https://mocked.host"), + ], +) +async def test_tcp_connector_multiple_hosts_one_timeout( + loop: asyncio.AbstractEventLoop, + request_url: str, +) -> None: + conn = aiohttp.TCPConnector() + + ip1 = "192.168.1.1" + ip2 = "192.168.1.2" + ips = [ip1, ip2] + ips_tried = [] + ips_success = [] + timeout_error = False + connected = False + + req = ClientRequest( + "GET", + URL(request_url), + loop=loop, + ) + + async def _resolve_host( + host: str, port: int, traces: object = None + ) -> List[ResolveResult]: + return [ + { + "hostname": host, + "host": ip, + "port": port, + "family": socket.AF_INET6 if ":" in ip else socket.AF_INET, + "proto": 0, + "flags": socket.AI_NUMERICHOST, + } + for ip in ips + ] + + async def start_connection( + addr_infos: Sequence[AddrInfoType], + *, + interleave: Optional[int] = None, + **kwargs: object, + ) -> socket.socket: + nonlocal timeout_error + + addr_info = addr_infos[0] + addr_info_addr = addr_info[-1] + + ip = addr_info_addr[0] + ips_tried.append(ip) + + if ip == ip1: + timeout_error = True + raise asyncio.TimeoutError + + if ip == ip2: + mock_socket = mock.create_autospec( + socket.socket, spec_set=True, instance=True + ) + mock_socket.getpeername.return_value = addr_info_addr + return mock_socket # type: ignore[no-any-return] + + assert False + + async def create_connection( + *args: object, sock: Optional[socket.socket] = None, **kwargs: object + ) -> Tuple[ResponseHandler, ResponseHandler]: + nonlocal connected + + assert isinstance(sock, socket.socket) + addr_info = sock.getpeername() + ip = addr_info[0] + ips_success.append(ip) + connected = True + + # Close the socket since we are not actually connecting + # and we don't want to leak it. + sock.close() + tr = create_mocked_conn(loop) + pr = create_mocked_conn(loop) + return tr, pr + + with mock.patch.object( + conn, "_resolve_host", autospec=True, spec_set=True, side_effect=_resolve_host + ), mock.patch.object( + conn._loop, + "create_connection", + autospec=True, + spec_set=True, + side_effect=create_connection, + ), mock.patch( + "aiohttp.connector.aiohappyeyeballs.start_connection", start_connection + ): + established_connection = await conn.connect(req, [], ClientTimeout()) + + assert ips_tried == ips + assert ips_success == [ip2] + + assert timeout_error + assert connected + + established_connection.close() + + +async def test_tcp_connector_resolve_host(loop: asyncio.AbstractEventLoop) -> None: conn = aiohttp.TCPConnector(use_dns_cache=True) res = await conn._resolve_host("localhost", 8080)