Skip to content

Commit

Permalink
[PR #6002/d66e07c6 backport][3.8] Add xfailing integration tests agai…
Browse files Browse the repository at this point in the history
…nst ``proxy.py`` (#6033)

This patch adds full end-to-end tests for sending requests to HTTP and
HTTPS endpoints through an HTTPS proxy. The first case is currently
supported and the second one is not. This is why the latter test is
marked as expected to fail. The support for TLS-in-TLS in the upstream
stdlib asyncio is currently disabled but is available in Python 3.9
via monkey-patching which is demonstrated in the added tests.

Refs:
* https://bugs.python.org/issue37179
* python/cpython#28073
* #5992

Co-authored-by: bmbouter <[email protected]>
Co-authored-by: Sviatoslav Sydorenko <[email protected]>

PR #6002

(cherry picked from commit d66e07c)
  • Loading branch information
webknjaz authored Oct 3, 2021
1 parent e98c5e8 commit ed8f834
Show file tree
Hide file tree
Showing 5 changed files with 136 additions and 0 deletions.
2 changes: 2 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,8 @@ jobs:
path: ${{ steps.pip-cache.outputs.dir }}
restore-keys: |
pip-ci-${{ runner.os }}-${{ matrix.pyver }}-${{ matrix.no-extensions }}-
- name: Upgrade wheel # Needed for proxy.py install not to explode
run: pip install -U wheel
- name: Cythonize
if: ${{ matrix.no-extensions == '' }}
run: |
Expand Down
2 changes: 2 additions & 0 deletions CHANGES/6002.misc
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
Implemented end-to-end testing of sending HTTP and HTTPS requests
via ``proxy.py``.
3 changes: 3 additions & 0 deletions requirements/dev.txt
Original file line number Diff line number Diff line change
Expand Up @@ -164,6 +164,8 @@ pluggy==0.13.1
# pytest
pre-commit==2.13.0
# via -r requirements/lint.txt
proxy.py==2.3.1
# via -r requirements/test.txt
py==1.10.0
# via
# -r requirements/lint.txt
Expand Down Expand Up @@ -277,6 +279,7 @@ typing-extensions==3.7.4.3
# -r requirements/lint.txt
# async-timeout
# mypy
# proxy.py
uritemplate==3.0.1
# via gidgethub
urllib3==1.26.2
Expand Down
1 change: 1 addition & 0 deletions requirements/test.txt
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ cryptography==3.3.1; platform_machine!="i686" and python_version<"3.9" # no 32-b
freezegun==1.1.0
mypy==0.910; implementation_name=="cpython"
mypy-extensions==0.4.3; implementation_name=="cpython"
proxy.py==2.3.1
pytest==6.1.2
pytest-cov==2.12.1
pytest-mock==3.6.1
Expand Down
128 changes: 128 additions & 0 deletions tests/test_proxy_functional.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,140 @@
import pathlib
from unittest import mock

import proxy
import pytest
from yarl import URL

import aiohttp
from aiohttp import web

ASYNCIO_SUPPORTS_TLS_IN_TLS = hasattr(
asyncio.sslproto._SSLProtocolTransport,
"_start_tls_compatible",
)


@pytest.fixture
def secure_proxy_url(monkeypatch, tls_certificate_pem_path):
"""Return the URL of an instance of a running secure proxy.
This fixture also spawns that instance and tears it down after the test.
"""
proxypy_args = [
"--threadless", # use asyncio
"--num-workers",
"1", # the tests only send one query anyway
"--hostname",
"127.0.0.1", # network interface to listen to
"--port",
0, # ephemeral port, so that kernel allocates a free one
"--cert-file",
tls_certificate_pem_path, # contains both key and cert
"--key-file",
tls_certificate_pem_path, # contains both key and cert
]

class PatchedAccetorPool(proxy.core.acceptor.AcceptorPool):
def listen(self):
super().listen()
self.socket_host, self.socket_port = self.socket.getsockname()[:2]

monkeypatch.setattr(proxy.proxy, "AcceptorPool", PatchedAccetorPool)

with proxy.Proxy(input_args=proxypy_args) as proxy_instance:
yield URL.build(
scheme="https",
host=proxy_instance.acceptors.socket_host,
port=proxy_instance.acceptors.socket_port,
)


@pytest.fixture
def web_server_endpoint_payload():
return "Test message"


@pytest.fixture(params=("http", "https"))
def web_server_endpoint_type(request):
return request.param


@pytest.fixture
async def web_server_endpoint_url(
aiohttp_server,
ssl_ctx,
web_server_endpoint_payload,
web_server_endpoint_type,
):
server_kwargs = (
{
"ssl": ssl_ctx,
}
if web_server_endpoint_type == "https"
else {}
)

async def handler(*args, **kwargs):
return web.Response(text=web_server_endpoint_payload)

app = web.Application()
app.router.add_route("GET", "/", handler)
server = await aiohttp_server(app, **server_kwargs)

return URL.build(
scheme=web_server_endpoint_type,
host=server.host,
port=server.port,
)


@pytest.fixture
def _pretend_asyncio_supports_tls_in_tls(
monkeypatch,
web_server_endpoint_type,
):
if web_server_endpoint_type != "https" or ASYNCIO_SUPPORTS_TLS_IN_TLS:
return

# for https://github.com/python/cpython/pull/28073
# and https://bugs.python.org/issue37179
monkeypatch.setattr(
asyncio.sslproto._SSLProtocolTransport,
"_start_tls_compatible",
True,
raising=False,
)


@pytest.mark.xfail(
reason="https://github.com/aio-libs/aiohttp/pull/5992",
raises=ValueError,
)
@pytest.mark.parametrize("web_server_endpoint_type", ("http", "https"))
@pytest.mark.usefixtures("_pretend_asyncio_supports_tls_in_tls", "loop")
async def test_secure_https_proxy_absolute_path(
client_ssl_ctx,
secure_proxy_url,
web_server_endpoint_url,
web_server_endpoint_payload,
) -> None:
"""Test urls can be requested through a secure proxy."""
conn = aiohttp.TCPConnector()
sess = aiohttp.ClientSession(connector=conn)

response = await sess.get(
web_server_endpoint_url,
proxy=secure_proxy_url,
ssl=client_ssl_ctx, # used for both proxy and endpoint connections
)

assert response.status == 200
assert await response.text() == web_server_endpoint_payload

response.close()
await sess.close()
await conn.close()


@pytest.fixture
def proxy_test_server(aiohttp_raw_server, loop, monkeypatch):
Expand Down

0 comments on commit ed8f834

Please sign in to comment.