From c9c7e88690f6f7ee19ddcc125a767712ebb22679 Mon Sep 17 00:00:00 2001 From: hellysmile Date: Mon, 30 Oct 2017 12:25:55 +0200 Subject: [PATCH 1/2] Fix wrap oserrors. --- CHANGES/2423.bugfix | 1 + aiohttp/connector.py | 15 +++++++++++++-- tests/test_connector.py | 40 +++++++++++++++++++++++++++++++++++++++- tests/test_proxy.py | 2 +- 4 files changed, 54 insertions(+), 4 deletions(-) create mode 100644 CHANGES/2423.bugfix diff --git a/CHANGES/2423.bugfix b/CHANGES/2423.bugfix new file mode 100644 index 00000000000..08823f4f69f --- /dev/null +++ b/CHANGES/2423.bugfix @@ -0,0 +1 @@ +Fix connector convert OSError to ClientConnectorError diff --git a/aiohttp/connector.py b/aiohttp/connector.py index e70072148aa..47f8f3fb95b 100644 --- a/aiohttp/connector.py +++ b/aiohttp/connector.py @@ -785,6 +785,13 @@ def _create_direct_connection(self, req): sslcontext = self._get_ssl_context(req) fingerprint, hashfunc = self._get_fingerprint_and_hashfunc(req) + try: + hosts = yield from self._resolve_host(req.url.raw_host, req.port) + except OSError as exc: + # in case of proxy it is not ClientProxyConnectionError + # it is problem of resolving proxy ip itself + raise ClientConnectorError(req.connection_key, exc) from exc + hosts = yield from self._resolve_host(req.url.raw_host, req.port) for hinfo in hosts: @@ -938,6 +945,10 @@ def path(self): @asyncio.coroutine def _create_connection(self, req): - _, proto = yield from self._loop.create_unix_connection( - self._factory, self._path) + try: + _, proto = yield from self._loop.create_unix_connection( + self._factory, self._path) + except OSError as exc: + raise ClientConnectorError(req.connection_key, exc) from exc + return proto diff --git a/tests/test_connector.py b/tests/test_connector.py index 68167174e68..66d1403c6fb 100644 --- a/tests/test_connector.py +++ b/tests/test_connector.py @@ -9,6 +9,7 @@ import ssl import tempfile import unittest +import uuid from unittest import mock import pytest @@ -18,7 +19,7 @@ from aiohttp import client, helpers, web from aiohttp.client import ClientRequest from aiohttp.connector import Connection, _DNSCacheTable -from aiohttp.test_utils import unused_port +from aiohttp.test_utils import make_mocked_coro, unused_port @pytest.fixture() @@ -507,6 +508,19 @@ def test_tcp_connector_dns_throttle_requests_cancelled_when_close( yield from f +def test_dns_error(loop): + connector = aiohttp.TCPConnector(loop=loop) + connector._resolve_host = make_mocked_coro( + raise_exception=OSError('dont take it serious')) + + req = ClientRequest( + 'GET', URL('http://www.python.org'), + loop=loop, + ) + with pytest.raises(aiohttp.ClientConnectorError): + loop.run_until_complete(connector.connect(req)) + + def test_get_pop_empty_conns(loop): # see issue #473 conn = aiohttp.BaseConnector(loop=loop) @@ -1281,6 +1295,30 @@ def handler(request): assert r.status == 200 +def test_unix_connector_not_found(loop): + connector = aiohttp.UnixConnector('/' + uuid.uuid4().hex, loop=loop) + + req = ClientRequest( + 'GET', URL('http://www.python.org'), + loop=loop, + ) + with pytest.raises(aiohttp.ClientConnectorError): + loop.run_until_complete(connector.connect(req)) + + +def test_unix_connector_permission(loop): + loop.create_unix_connection = make_mocked_coro( + raise_exception=PermissionError()) + connector = aiohttp.UnixConnector('/' + uuid.uuid4().hex, loop=loop) + + req = ClientRequest( + 'GET', URL('http://www.python.org'), + loop=loop, + ) + with pytest.raises(aiohttp.ClientConnectorError): + loop.run_until_complete(connector.connect(req)) + + def test_default_use_dns_cache(loop): conn = aiohttp.TCPConnector(loop=loop) assert conn.use_dns_cache diff --git a/tests/test_proxy.py b/tests/test_proxy.py index 9a0380f520f..ab123c05a56 100644 --- a/tests/test_proxy.py +++ b/tests/test_proxy.py @@ -215,7 +215,7 @@ def _test_connect_request_with_unicode_host(self, Request_mock): Request_mock.assert_called_with(mock.ANY, mock.ANY, "xn--9caa.com:80", mock.ANY, loop=loop) - def test_proxy_connection_error(self): + def test_proxy_dns_error(self): connector = aiohttp.TCPConnector(loop=self.loop) connector._resolve_host = make_mocked_coro( raise_exception=OSError('dont take it serious')) From 7670b7aedef64cbd2bb6863cd240d2f70b41e6da Mon Sep 17 00:00:00 2001 From: hellysmile Date: Mon, 30 Oct 2017 13:28:08 +0200 Subject: [PATCH 2/2] Skip unix socket tests on windows --- tests/test_connector.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/tests/test_connector.py b/tests/test_connector.py index 66d1403c6fb..2970a9aba8e 100644 --- a/tests/test_connector.py +++ b/tests/test_connector.py @@ -1295,6 +1295,8 @@ def handler(request): assert r.status == 200 +@pytest.mark.skipif(not hasattr(socket, 'AF_UNIX'), + reason="requires unix socket") def test_unix_connector_not_found(loop): connector = aiohttp.UnixConnector('/' + uuid.uuid4().hex, loop=loop) @@ -1306,6 +1308,8 @@ def test_unix_connector_not_found(loop): loop.run_until_complete(connector.connect(req)) +@pytest.mark.skipif(not hasattr(socket, 'AF_UNIX'), + reason="requires unix socket") def test_unix_connector_permission(loop): loop.create_unix_connection = make_mocked_coro( raise_exception=PermissionError())