From ac8b43f1efc3bf97fe475e66661c615a8d2a7aa6 Mon Sep 17 00:00:00 2001 From: Pawel Miech Date: Sun, 20 Nov 2016 21:09:36 +0100 Subject: [PATCH] fix HTTP proxying * fix bug that broke HTTP proxy support * add functional tests via mitmdump fixes #1413 --- aiohttp/client_reqrep.py | 5 ++++ requirements-dev.txt | 1 + tests/conftest.py | 51 ++++++++++++++++++++++++++++++++++ tests/mitmdump_script.py | 2 ++ tests/test_proxy_functional.py | 20 +++++++++++++ 5 files changed, 79 insertions(+) create mode 100644 tests/mitmdump_script.py create mode 100644 tests/test_proxy_functional.py diff --git a/aiohttp/client_reqrep.py b/aiohttp/client_reqrep.py index b226c011df8..50195fa5c7d 100644 --- a/aiohttp/client_reqrep.py +++ b/aiohttp/client_reqrep.py @@ -435,6 +435,11 @@ def send(self, writer, reader): path = self.url.raw_path if self.url.raw_query_string: path += '?' + self.url.raw_query_string + if self.proxy: + # path attribute is set by: + # aiohttp.connector._create_proxy_connection and its meaning + # is different. It is actual request.url requested via proxy. + path = self.path request = aiohttp.Request(writer, self.method, path, self.version) diff --git a/requirements-dev.txt b/requirements-dev.txt index e3116116469..f2d61e7f04c 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -3,3 +3,4 @@ ipdb==0.10.1 pytest-sugar==0.7.1 ipython==5.1.0 aiodns==1.1.1 +mitmproxy==0.18.2 diff --git a/tests/conftest.py b/tests/conftest.py index 2cfc3bdd0ca..a2b1f086aa9 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -1,6 +1,11 @@ import collections import logging +import pathlib +import signal +import socket import sys +import time +from subprocess import Popen import pytest @@ -82,3 +87,49 @@ def pytest_ignore_collect(path, config): if 'test_py35' in str(path): if sys.version_info < (3, 5, 0): return True + + +def get_ephemeral_port(): + s = socket.socket() + s.bind(("", 0)) + return s.getsockname()[1] + + +def _wait_for_port(portnum, delay=0.1, attempts=100): + while attempts > 0: + s = socket.socket() + if s.connect_ex(('127.0.0.1', portnum)) == 0: + s.close() + return + time.sleep(delay) + attempts -= 1 + raise RuntimeError("Port %d is not open" % portnum) + + +class FakeProxyProcess(object): + def __init__(self): + self.port = get_ephemeral_port() + script_file = pathlib.Path(__file__).parent / 'mitmdump_script.py' + self.args = ['mitmdump', '-p', str(self.port), '-s', + str(script_file), '--no-http2 '] + + def __enter__(self): + self.proc = Popen(self.args) + self.proc.poll() + _wait_for_port(self.port) + return self + + def __exit__(self, *args): + if self.proc is not None: + self.proc.send_signal(signal.SIGINT) + self.proc.wait() + self.proc = None + + def url(self): + return "http://localhost:{}/".format(self.port) + + +@pytest.yield_fixture +def fake_proxy(): + with FakeProxyProcess() as fake_proxy: + yield fake_proxy diff --git a/tests/mitmdump_script.py b/tests/mitmdump_script.py new file mode 100644 index 00000000000..4f69af6703a --- /dev/null +++ b/tests/mitmdump_script.py @@ -0,0 +1,2 @@ +def response(flow): + flow.response.headers["X-Mitmdump"] = "1" diff --git a/tests/test_proxy_functional.py b/tests/test_proxy_functional.py new file mode 100644 index 00000000000..ef4b08d584d --- /dev/null +++ b/tests/test_proxy_functional.py @@ -0,0 +1,20 @@ +import asyncio + +from aiohttp import web + + +@asyncio.coroutine +def test_proxy_via_mitmdump(loop, test_client, fake_proxy): + @asyncio.coroutine + def handler(request): + return web.Response(text=request.method) + + app = web.Application(loop=loop) + app.router.add_get('/', handler) + + client = yield from test_client(app) + resp = yield from client.get('/', proxy=fake_proxy.url()) + assert 200 == resp.status + assert 'X-Mitmdump' in resp.headers + assert resp.headers['X-Mitmdump'] == '1' + resp.close()