From a4abecc741da8d7b71ce8b2afa7c704d8082aa2a Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Fri, 4 Nov 2016 02:22:38 +0200 Subject: [PATCH 01/12] Extract BaseRequest --- aiohttp/web_reqrep.py | 40 +++++++++++++++++++++++----------------- 1 file changed, 23 insertions(+), 17 deletions(-) diff --git a/aiohttp/web_reqrep.py b/aiohttp/web_reqrep.py index d5c6a3756a9..ca20941cd76 100644 --- a/aiohttp/web_reqrep.py +++ b/aiohttp/web_reqrep.py @@ -22,7 +22,7 @@ from .protocol import HttpVersion10, HttpVersion11 __all__ = ( - 'ContentCoding', 'Request', 'StreamResponse', 'Response', + 'ContentCoding', 'BaseRequest', 'Request', 'StreamResponse', 'Response', 'json_response' ) @@ -44,7 +44,7 @@ class ContentCoding(enum.Enum): ############################################################ -class Request(collections.MutableMapping, HeadersMixin): +class BaseRequest(collections.MutableMapping, HeadersMixin): POST_METHODS = {hdrs.METH_PATCH, hdrs.METH_POST, hdrs.METH_PUT, hdrs.METH_TRACE, hdrs.METH_DELETE} @@ -59,10 +59,6 @@ def __init__(self, message, payload, transport, reader, writer, self._post = None self._post_files_cache = None - # matchdict, route_name, handler - # or information about traversal lookup - self._match_info = None # initialized after route resolving - self._payload = payload self._read_bytes = None @@ -99,7 +95,7 @@ def clone(self, *, method=sentinel, rel_url=sentinel, message = self._message._replace(**dct) - return Request( + return self.__class__( message, self._payload, self._transport, @@ -284,16 +280,6 @@ def keep_alive(self): else: return not self._message.should_close - @property - def match_info(self): - """Result of route resolving.""" - return self._match_info - - @reify - def app(self): - """Application instance.""" - return self._match_info.apps[-1] - @property def transport(self): """Transport used for request processing.""" @@ -440,6 +426,26 @@ def __repr__(self): self.method, ascii_encodable_path) +class Request(BaseRequest): + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + + # matchdict, route_name, handler + # or information about traversal lookup + self._match_info = None # initialized after route resolving + + @property + def match_info(self): + """Result of route resolving.""" + return self._match_info + + @reify + def app(self): + """Application instance.""" + return self._match_info.apps[-1] + + ############################################################ # HTTP Response classes ############################################################ From 9259627ccc57cc8bf431d622f7f978229cdcf151 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Fri, 4 Nov 2016 04:18:55 +0200 Subject: [PATCH 02/12] New server --- aiohttp/web.py | 167 +++++++------------------- aiohttp/web_server.py | 125 +++++++++++++++++++ demos/polls/aiohttpdemo_polls/main.py | 2 +- 3 files changed, 170 insertions(+), 124 deletions(-) create mode 100644 aiohttp/web_server.py diff --git a/aiohttp/web.py b/aiohttp/web.py index 3d76bab02dd..e39d301850c 100644 --- a/aiohttp/web.py +++ b/aiohttp/web.py @@ -9,13 +9,13 @@ from . import hdrs, web_exceptions, web_reqrep, web_urldispatcher, web_ws from .abc import AbstractMatchInfo, AbstractRouter -from .helpers import FrozenList, TimeService, sentinel +from .helpers import FrozenList, sentinel from .log import access_logger, web_logger from .protocol import HttpVersion # noqa -from .server import ServerHttpProtocol from .signals import PostSignal, PreSignal, Signal from .web_exceptions import * # noqa from .web_reqrep import * # noqa +from .web_server import BaseRequestHandler, BaseRequestHandlerFactory from .web_urldispatcher import * # noqa from .web_ws import * # noqa @@ -28,147 +28,68 @@ 'MsgType')) -class RequestHandler(ServerHttpProtocol): +class RequestHandler(BaseRequestHandler): - _request = None - - def __init__(self, manager, app, router, time_service, *, - secure_proxy_ssl_header=None, **kwargs): - super().__init__(**kwargs) + def __init__(self, manager, **kwargs): + super().__init__(manager, **kwargs) self._manager = manager - self._app = app - self._router = router - self._secure_proxy_ssl_header = secure_proxy_ssl_header - self._time_service = time_service - - def __repr__(self): - if self._request is None: - meth = 'none' - path = 'none' - else: - meth = self._request.method - path = self._request.rel_url.raw_path - return "<{} {}:{} {}>".format( - self.__class__.__name__, meth, path, - 'connected' if self.transport is not None else 'disconnected') - - def connection_made(self, transport): - super().connection_made(transport) - - self._manager.connection_made(self, transport) - - def connection_lost(self, exc): - self._manager.connection_lost(self, exc) - - super().connection_lost(exc) - - @asyncio.coroutine - def handle_request(self, message, payload): - self._manager._requests_count += 1 - if self.access_log: - now = self._loop.time() + self._app = manager.app + self._router = self._app.router + self._secure_proxy_ssl_header = manager.secure_proxy_ssl_header - request = web_reqrep.Request( + def make_request(self, message, payload): + return web_reqrep.Request( message, payload, self.transport, self.reader, self.writer, self._time_service, secure_proxy_ssl_header=self._secure_proxy_ssl_header) - self._request = request - try: - match_info = yield from self._router.resolve(request) - assert isinstance(match_info, AbstractMatchInfo), match_info - match_info.add_app(self._app) - match_info.freeze() - - resp = None - request._match_info = match_info - expect = request.headers.get(hdrs.EXPECT) - if expect: - resp = ( - yield from match_info.expect_handler(request)) - - if resp is None: - handler = match_info.handler - for app in match_info.apps: - for factory in reversed(app.middlewares): - handler = yield from factory(app, handler) - resp = yield from handler(request) - - assert isinstance(resp, web_reqrep.StreamResponse), \ - ("Handler {!r} should return response instance, " - "got {!r} [middlewares {!r}]").format( - match_info.handler, type(resp), self._middlewares) - except web_exceptions.HTTPException as exc: - resp = exc - - resp_msg = yield from resp.prepare(request) - yield from resp.write_eof() - - # notify server about keep-alive - self.keep_alive(resp.keep_alive) - - # Restore default state. - # Should be no-op if server code didn't touch these attributes. - self.writer.set_tcp_cork(False) - self.writer.set_tcp_nodelay(True) - - # log access - if self.access_log: - self.log_access(message, None, resp_msg, self._loop.time() - now) - - # for repr - self._request = None - - -class RequestHandlerFactory: + + @asyncio.coroutine + def handle(self, request): + match_info = yield from self._router.resolve(request) + assert isinstance(match_info, AbstractMatchInfo), match_info + match_info.add_app(self._app) + match_info.freeze() + + resp = None + request._match_info = match_info + expect = request.headers.get(hdrs.EXPECT) + if expect: + resp = ( + yield from match_info.expect_handler(request)) + + if resp is None: + handler = match_info.handler + for app in match_info.apps: + for factory in reversed(app.middlewares): + handler = yield from factory(app, handler) + resp = yield from handler(request) + + assert isinstance(resp, web_reqrep.StreamResponse), \ + ("Handler {!r} should return response instance, " + "got {!r} [middlewares {!r}]").format( + match_info.handler, type(resp), self._middlewares) + return resp + + +class RequestHandlerFactory(BaseRequestHandlerFactory): def __init__(self, app, router, *, handler=RequestHandler, loop=None, secure_proxy_ssl_header=None, **kwargs): + kwargs.setdefault('logger', app.logger) + super().__init__(handler=handler, loop=loop, **kwargs) self._app = app - self._router = router - self._handler = handler - self._loop = loop - self._connections = {} self._secure_proxy_ssl_header = secure_proxy_ssl_header - self._kwargs = kwargs - self._kwargs.setdefault('logger', app.logger) - self._requests_count = 0 - self._time_service = TimeService(self._loop) - - @property - def requests_count(self): - """Number of processed requests.""" - return self._requests_count @property def secure_proxy_ssl_header(self): return self._secure_proxy_ssl_header @property - def connections(self): - return list(self._connections.keys()) - - def connection_made(self, handler, transport): - self._connections[handler] = transport - - def connection_lost(self, handler, exc=None): - if handler in self._connections: - del self._connections[handler] - - @asyncio.coroutine - def finish_connections(self, timeout=None): - coros = [conn.shutdown(timeout) for conn in self._connections] - yield from asyncio.gather(*coros, loop=self._loop) - self._connections.clear() - self._time_service.stop() - - def __call__(self): - return self._handler( - self, self._app, self._router, self._time_service, loop=self._loop, - secure_proxy_ssl_header=self._secure_proxy_ssl_header, - **self._kwargs) + def app(self): + return self._app class Application(MutableMapping): diff --git a/aiohttp/web_server.py b/aiohttp/web_server.py new file mode 100644 index 00000000000..354b0680cf5 --- /dev/null +++ b/aiohttp/web_server.py @@ -0,0 +1,125 @@ +"""Low level HTTP server.""" + +import asyncio +from abc import ABC, abstractmethod + +from .helpers import TimeService +from .server import ServerHttpProtocol +from .web_exceptions import HTTPException +from .web_reqrep import BaseRequest + + +class BaseRequestHandler(ServerHttpProtocol, ABC): + _request = None + + def __init__(self, manager, **kwargs): + super().__init__(**kwargs) + self._manager = manager + self._time_service = manager.time_service + + def __repr__(self): + if self._request is None: + meth = 'none' + path = 'none' + else: + meth = self._request.method + path = self._request.rel_url.raw_path + return "<{} {}:{} {}>".format( + self.__class__.__name__, meth, path, + 'connected' if self.transport is not None else 'disconnected') + + def connection_made(self, transport): + super().connection_made(transport) + + self._manager.connection_made(self, transport) + + def connection_lost(self, exc): + self._manager.connection_lost(self, exc) + + super().connection_lost(exc) + + def make_request(self, message, payload): + return BaseRequest( + message, payload, + self.transport, self.reader, self.writer, + self._time_service) + + @abstractmethod + @asyncio.coroutine + def handle(self, request): + pass + + @asyncio.coroutine + def handle_request(self, message, payload): + self._manager._requests_count += 1 + if self.access_log: + now = self._loop.time() + + request = self.make_request(message, payload) + self._request = request + + try: + resp = yield from self.handle(request) + except HTTPException as exc: + resp = exc + + resp_msg = yield from resp.prepare(request) + yield from resp.write_eof() + + # notify server about keep-alive + self.keep_alive(resp.keep_alive) + + # Restore default state. + # Should be no-op if server code didn't touch these attributes. + self.writer.set_tcp_cork(False) + self.writer.set_tcp_nodelay(True) + + # log access + if self.access_log: + self.log_access(message, None, resp_msg, self._loop.time() - now) + + # for repr + self._request = None + + +class BaseRequestHandlerFactory: + + def __init__(self, *, handler=BaseRequestHandler, loop=None, **kwargs): + self._handler = handler + self._loop = loop + self._connections = {} + self._kwargs = kwargs + self._requests_count = 0 + self._time_service = TimeService(self._loop) + + @property + def requests_count(self): + """Number of processed requests.""" + return self._requests_count + + @property + def time_service(self): + return self._time_service + + @property + def connections(self): + return list(self._connections.keys()) + + def connection_made(self, handler, transport): + self._connections[handler] = transport + + def connection_lost(self, handler, exc=None): + if handler in self._connections: + del self._connections[handler] + + @asyncio.coroutine + def finish_connections(self, timeout=None): + coros = [conn.shutdown(timeout) for conn in self._connections] + yield from asyncio.gather(*coros, loop=self._loop) + self._connections.clear() + self._time_service.stop() + + def __call__(self): + return self._handler( + self, loop=self._loop, + **self._kwargs) diff --git a/demos/polls/aiohttpdemo_polls/main.py b/demos/polls/aiohttpdemo_polls/main.py index 21a56ffad13..44e297bf22d 100644 --- a/demos/polls/aiohttpdemo_polls/main.py +++ b/demos/polls/aiohttpdemo_polls/main.py @@ -6,7 +6,7 @@ import aiohttp_jinja2 from aiohttp import web -from aiohttpdemo_polls.db import init_pg, close_pg +from aiohttpdemo_polls.db import close_pg, init_pg from aiohttpdemo_polls.middlewares import setup_middlewares from aiohttpdemo_polls.routes import setup_routes from aiohttpdemo_polls.utils import load_config From f43b488f28aea3c62175efe2b39229b3e6c8ae91 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Fri, 4 Nov 2016 10:48:20 +0200 Subject: [PATCH 03/12] Add no branch instruction --- aiohttp/web_server.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/aiohttp/web_server.py b/aiohttp/web_server.py index 354b0680cf5..ba16afed34f 100644 --- a/aiohttp/web_server.py +++ b/aiohttp/web_server.py @@ -9,6 +9,9 @@ from .web_reqrep import BaseRequest +__all__ = ('BaseRequestHandler', 'BaseRequestHandlerFactory') + + class BaseRequestHandler(ServerHttpProtocol, ABC): _request = None @@ -44,10 +47,10 @@ def make_request(self, message, payload): self.transport, self.reader, self.writer, self._time_service) + @asyncio.coroutine # pragma: no branch @abstractmethod - @asyncio.coroutine def handle(self, request): - pass + """Handle request, return response instance.""" @asyncio.coroutine def handle_request(self, message, payload): From 1236839e398a7755723c14444191288fd0606d39 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Sun, 6 Nov 2016 09:10:44 +0200 Subject: [PATCH 04/12] Work on new server --- aiohttp/web.py | 129 +++++++++++++++++------------------------- aiohttp/web_server.py | 50 +++++++++------- 2 files changed, 81 insertions(+), 98 deletions(-) diff --git a/aiohttp/web.py b/aiohttp/web.py index e39d301850c..0a1ec13eedd 100644 --- a/aiohttp/web.py +++ b/aiohttp/web.py @@ -7,7 +7,8 @@ from yarl import URL -from . import hdrs, web_exceptions, web_reqrep, web_urldispatcher, web_ws +from . import (hdrs, web_exceptions, web_reqrep, web_server, web_urldispatcher, + web_ws) from .abc import AbstractMatchInfo, AbstractRouter from .helpers import FrozenList, sentinel from .log import access_logger, web_logger @@ -15,7 +16,7 @@ from .signals import PostSignal, PreSignal, Signal from .web_exceptions import * # noqa from .web_reqrep import * # noqa -from .web_server import BaseRequestHandler, BaseRequestHandlerFactory +from .web_server import RequestHandlerFactory from .web_urldispatcher import * # noqa from .web_ws import * # noqa @@ -23,89 +24,27 @@ web_exceptions.__all__ + web_urldispatcher.__all__ + web_ws.__all__ + - ('Application', 'RequestHandler', - 'RequestHandlerFactory', 'HttpVersion', - 'MsgType')) - - -class RequestHandler(BaseRequestHandler): - - def __init__(self, manager, **kwargs): - super().__init__(manager, **kwargs) - - self._manager = manager - self._app = manager.app - self._router = self._app.router - self._secure_proxy_ssl_header = manager.secure_proxy_ssl_header - - def make_request(self, message, payload): - return web_reqrep.Request( - message, payload, - self.transport, self.reader, self.writer, - self._time_service, - secure_proxy_ssl_header=self._secure_proxy_ssl_header) - - @asyncio.coroutine - def handle(self, request): - match_info = yield from self._router.resolve(request) - assert isinstance(match_info, AbstractMatchInfo), match_info - match_info.add_app(self._app) - match_info.freeze() - - resp = None - request._match_info = match_info - expect = request.headers.get(hdrs.EXPECT) - if expect: - resp = ( - yield from match_info.expect_handler(request)) - - if resp is None: - handler = match_info.handler - for app in match_info.apps: - for factory in reversed(app.middlewares): - handler = yield from factory(app, handler) - resp = yield from handler(request) - - assert isinstance(resp, web_reqrep.StreamResponse), \ - ("Handler {!r} should return response instance, " - "got {!r} [middlewares {!r}]").format( - match_info.handler, type(resp), self._middlewares) - return resp - - -class RequestHandlerFactory(BaseRequestHandlerFactory): - - def __init__(self, app, router, *, - handler=RequestHandler, loop=None, - secure_proxy_ssl_header=None, **kwargs): - kwargs.setdefault('logger', app.logger) - super().__init__(handler=handler, loop=loop, **kwargs) - self._app = app - self._secure_proxy_ssl_header = secure_proxy_ssl_header - - @property - def secure_proxy_ssl_header(self): - return self._secure_proxy_ssl_header - - @property - def app(self): - return self._app + web_server.__all__ + + ('Application', 'HttpVersion', 'MsgType')) class Application(MutableMapping): def __init__(self, *, logger=web_logger, loop=None, - router=None, handler_factory=RequestHandlerFactory, - middlewares=(), debug=False): + router=None, + middlewares=(), debug=...): if loop is None: loop = asyncio.get_event_loop() if router is None: router = web_urldispatcher.UrlDispatcher(self) assert isinstance(router, AbstractRouter), router + if debug is ...: + debug = loop.get_debug() + self._debug = debug self._router = router - self._handler_factory = handler_factory + self._handler_factory = RequestHandlerFactory self._loop = loop self.logger = logger @@ -222,7 +161,7 @@ def loop(self): def middlewares(self): return self._middlewares - def make_handler(self, **kwargs): + def make_handler(self, *, secure_proxy_ssl_header=None, **kwargs): debug = kwargs.pop('debug', sentinel) if debug is not sentinel: warnings.warn( @@ -239,8 +178,11 @@ def make_handler(self, **kwargs): "web_reference.html#aiohttp.web.Application" ) self.freeze() - return self._handler_factory(self, self.router, debug=self.debug, - loop=self.loop, **kwargs) + self._secure_proxy_ssl_header = secure_proxy_ssl_header + return self._handler_factory(self._handle, + request_factory=self._make_request, + debug=self.debug, loop=self.loop, + **kwargs) @asyncio.coroutine def startup(self): @@ -279,8 +221,41 @@ def register_on_finish(self, func, *args, **kwargs): warnings.warn("Use .on_cleanup.append() instead", DeprecationWarning) self.on_cleanup.append(lambda app: func(app, *args, **kwargs)) - def copy(self): - raise NotImplementedError + def _make_request(self, message, payload, protocol): + return web_reqrep.Request( + message, payload, + protocol.transport, protocol.reader, protocol.writer, + protocol.time_service, + secure_proxy_ssl_header=self._secure_proxy_ssl_header) + + @asyncio.coroutine + def _handle(self, request): + match_info = yield from self._router.resolve(request) + assert isinstance(match_info, AbstractMatchInfo), match_info + match_info.add_app(self) + match_info.freeze() + + resp = None + request._match_info = match_info + expect = request.headers.get(hdrs.EXPECT) + if expect: + resp = ( + yield from match_info.expect_handler(request)) + + if resp is None: + handler = match_info.handler + for app in match_info.apps: + for factory in reversed(app.middlewares): + handler = yield from factory(app, handler) + resp = yield from handler(request) + + assert isinstance(resp, web_reqrep.StreamResponse), \ + ("Handler {!r} should return response instance, " + "got {!r} [middlewares {!r}]").format( + match_info.handler, type(resp), + [middleware for middleware in app.middlewares + for app in match_info.apps]) + return resp def __call__(self): """gunicorn compatibility""" diff --git a/aiohttp/web_server.py b/aiohttp/web_server.py index ba16afed34f..f9ba4808270 100644 --- a/aiohttp/web_server.py +++ b/aiohttp/web_server.py @@ -1,24 +1,24 @@ """Low level HTTP server.""" import asyncio -from abc import ABC, abstractmethod from .helpers import TimeService from .server import ServerHttpProtocol from .web_exceptions import HTTPException from .web_reqrep import BaseRequest +__all__ = ('RequestHandler', 'RequestHandlerFactory') -__all__ = ('BaseRequestHandler', 'BaseRequestHandlerFactory') - -class BaseRequestHandler(ServerHttpProtocol, ABC): +class RequestHandler(ServerHttpProtocol): _request = None def __init__(self, manager, **kwargs): super().__init__(**kwargs) self._manager = manager - self._time_service = manager.time_service + self._request_factory = manager.request_factory + self._handler = manager.handler + self.time_service = manager.time_service def __repr__(self): if self._request is None: @@ -40,17 +40,10 @@ def connection_lost(self, exc): self._manager.connection_lost(self, exc) super().connection_lost(exc) - - def make_request(self, message, payload): - return BaseRequest( - message, payload, - self.transport, self.reader, self.writer, - self._time_service) - - @asyncio.coroutine # pragma: no branch - @abstractmethod - def handle(self, request): - """Handle request, return response instance.""" + self._request_factory = None + self._manager = None + self.time_service = None + self._handler = None @asyncio.coroutine def handle_request(self, message, payload): @@ -58,11 +51,11 @@ def handle_request(self, message, payload): if self.access_log: now = self._loop.time() - request = self.make_request(message, payload) + request = self._request_factory(message, payload, self) self._request = request try: - resp = yield from self.handle(request) + resp = yield from self._handler(request) except HTTPException as exc: resp = exc @@ -85,10 +78,11 @@ def handle_request(self, message, payload): self._request = None -class BaseRequestHandlerFactory: +class RequestHandlerFactory: - def __init__(self, *, handler=BaseRequestHandler, loop=None, **kwargs): + def __init__(self, handler, *, request_factory=None, loop=None, **kwargs): self._handler = handler + self._request_factory = request_factory or self._make_request self._loop = loop self._connections = {} self._kwargs = kwargs @@ -100,6 +94,14 @@ def requests_count(self): """Number of processed requests.""" return self._requests_count + @property + def handler(self): + return self._handler + + @property + def request_factory(self): + return self._request_factory + @property def time_service(self): return self._time_service @@ -115,6 +117,12 @@ def connection_lost(self, handler, exc=None): if handler in self._connections: del self._connections[handler] + def _make_request(self, message, payload, protocol): + return BaseRequest( + message, payload, + protocol.transport, protocol.reader, protocol.writer, + protocol.time_service) + @asyncio.coroutine def finish_connections(self, timeout=None): coros = [conn.shutdown(timeout) for conn in self._connections] @@ -123,6 +131,6 @@ def finish_connections(self, timeout=None): self._time_service.stop() def __call__(self): - return self._handler( + return RequestHandler( self, loop=self._loop, **self._kwargs) From 1534a10598fb8a99095dadb908332a61b42fa58e Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Sun, 6 Nov 2016 19:48:02 +0200 Subject: [PATCH 05/12] Drop useless test --- tests/test_web_application.py | 6 ------ 1 file changed, 6 deletions(-) diff --git a/tests/test_web_application.py b/tests/test_web_application.py index 1a74bcf333b..7af477f66d4 100644 --- a/tests/test_web_application.py +++ b/tests/test_web_application.py @@ -18,12 +18,6 @@ def test_app_call(loop): assert app is app() -def test_app_copy(loop): - app = web.Application(loop=loop) - with pytest.raises(NotImplementedError): - app.copy() - - def test_app_default_loop(loop): asyncio.set_event_loop(loop) app = web.Application() From 0617c460140ce384baac2e578ffb4429b0dc3794 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Sun, 6 Nov 2016 19:58:19 +0200 Subject: [PATCH 06/12] Fix tests --- aiohttp/web.py | 1 + tests/test_web_application.py | 17 ++++++++++++++++- tests/test_web_request_handler.py | 6 ------ 3 files changed, 17 insertions(+), 7 deletions(-) diff --git a/aiohttp/web.py b/aiohttp/web.py index 0a1ec13eedd..a334ad2fb18 100644 --- a/aiohttp/web.py +++ b/aiohttp/web.py @@ -45,6 +45,7 @@ def __init__(self, *, logger=web_logger, loop=None, self._debug = debug self._router = router self._handler_factory = RequestHandlerFactory + self._secure_proxy_ssl_header = None self._loop = loop self.logger = logger diff --git a/tests/test_web_application.py b/tests/test_web_application.py index 7af477f66d4..a95c407ebd4 100644 --- a/tests/test_web_application.py +++ b/tests/test_web_application.py @@ -36,7 +36,9 @@ def test_app_make_handler_debug_exc(loop, mocker, debug): assert 'parameter is deprecated' in exc[0].message.args[0] assert app._handler_factory.call_count == 2 - app._handler_factory.assert_called_with(app, app.router, loop=loop, + app._handler_factory.assert_called_with(app._handle, + request_factory=app._make_request, + loop=loop, debug=debug) with pytest.raises(ValueError) as exc: @@ -169,3 +171,16 @@ def test_app_delitem(loop): assert len(app) == 1 del app['key'] assert len(app) == 0 + + +def test_secure_proxy_ssl_header_default(loop): + app = web.Application(loop=loop) + assert app._secure_proxy_ssl_header is None + + +@asyncio.coroutine +def test_secure_proxy_ssl_header_non_default(loop): + app = web.Application(loop=loop) + hdr = ('X-Forwarded-Proto', 'https') + app.make_handler(secure_proxy_ssl_header=hdr) + assert app._secure_proxy_ssl_header is hdr diff --git a/tests/test_web_request_handler.py b/tests/test_web_request_handler.py index 6c0c7ce912c..b724e474568 100644 --- a/tests/test_web_request_handler.py +++ b/tests/test_web_request_handler.py @@ -64,9 +64,3 @@ def test_finish_connection_timeout(loop): manager.connection_lost(handler, None) assert manager.connections == [] handler.shutdown.assert_called_with(0.1) - - -def test_secure_proxy_ssl_header_default(loop): - app = web.Application(loop=loop) - manager = app.make_handler() - assert manager.secure_proxy_ssl_header is None From 4728e7e90ef2ed6c4e7a5b5382f315956a6c4149 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Sun, 6 Nov 2016 20:27:14 +0200 Subject: [PATCH 07/12] Update doc --- aiohttp/test_utils.py | 2 +- aiohttp/web.py | 2 +- aiohttp/web_server.py | 6 ++++-- aiohttp/worker.py | 2 +- docs/web.rst | 4 ++-- docs/web_reference.rst | 8 +++++++- 6 files changed, 16 insertions(+), 8 deletions(-) diff --git a/aiohttp/test_utils.py b/aiohttp/test_utils.py index 44618e6a2a2..24ab48201b1 100644 --- a/aiohttp/test_utils.py +++ b/aiohttp/test_utils.py @@ -94,7 +94,7 @@ def close(self): self.server.close() yield from self.server.wait_closed() yield from self.app.shutdown() - yield from self.handler.finish_connections() + yield from self.handler.shutdown() yield from self.app.cleanup() self._root = None self.port = None diff --git a/aiohttp/web.py b/aiohttp/web.py index a334ad2fb18..5b1cf684184 100644 --- a/aiohttp/web.py +++ b/aiohttp/web.py @@ -304,7 +304,7 @@ def run_app(app, *, host='0.0.0.0', port=None, srv.close() loop.run_until_complete(srv.wait_closed()) loop.run_until_complete(app.shutdown()) - loop.run_until_complete(handler.finish_connections(shutdown_timeout)) + loop.run_until_complete(handler.shutdown(shutdown_timeout)) loop.run_until_complete(app.cleanup()) loop.close() diff --git a/aiohttp/web_server.py b/aiohttp/web_server.py index f9ba4808270..ef5d0661680 100644 --- a/aiohttp/web_server.py +++ b/aiohttp/web_server.py @@ -78,7 +78,7 @@ def handle_request(self, message, payload): self._request = None -class RequestHandlerFactory: +class Server: def __init__(self, handler, *, request_factory=None, loop=None, **kwargs): self._handler = handler @@ -124,12 +124,14 @@ def _make_request(self, message, payload, protocol): protocol.time_service) @asyncio.coroutine - def finish_connections(self, timeout=None): + def shutdown(self, timeout=None): coros = [conn.shutdown(timeout) for conn in self._connections] yield from asyncio.gather(*coros, loop=self._loop) self._connections.clear() self._time_service.stop() + finish_connections = shutdown + def __call__(self): return RequestHandler( self, loop=self._loop, diff --git a/aiohttp/worker.py b/aiohttp/worker.py index 9d079ccbc92..804d15b4807 100644 --- a/aiohttp/worker.py +++ b/aiohttp/worker.py @@ -73,7 +73,7 @@ def close(self): # stop alive connections tasks = [ - handler.finish_connections( + handler.shutdown( timeout=self.cfg.graceful_timeout / 100 * 95) for handler in servers.values()] yield from asyncio.gather(*tasks, loop=self.loop) diff --git a/docs/web.rst b/docs/web.rst index 20d91d8847a..28be6859e61 100644 --- a/docs/web.rst +++ b/docs/web.rst @@ -1095,7 +1095,7 @@ Proper finalization procedure has three steps: 2. Fire :meth:`Application.shutdown` event. 3. Close accepted connections from clients by - :meth:`RequestHandlerFactory.finish_connections` call with + :meth:`RequestHandlerFactory.shutdown` call with reasonable small delay. 4. Call registered application finalizers by :meth:`Application.cleanup`. @@ -1116,7 +1116,7 @@ finalizing. It's pretty close to :func:`run_app` utility function:: srv.close() loop.run_until_complete(srv.wait_closed()) loop.run_until_complete(app.shutdown()) - loop.run_until_complete(handler.finish_connections(60.0)) + loop.run_until_complete(handler.shutdown(60.0)) loop.run_until_complete(app.cleanup()) loop.close() diff --git a/docs/web_reference.rst b/docs/web_reference.rst index f5283d4e52b..12bc7355fb5 100644 --- a/docs/web_reference.rst +++ b/docs/web_reference.rst @@ -1360,11 +1360,17 @@ RequestHandlerFactory .. versionadded:: 1.0 - .. coroutinemethod:: RequestHandlerFactory.finish_connections(timeout) + .. coroutinemethod:: RequestHandlerFactory.shutdown(timeout) A :ref:`coroutine` that should be called to close all opened connections. + .. coroutinemethod:: RequestHandlerFactory.finish_connections(timeout) + + .. deprecated:: 1.2 + + A deprecated alias for :meth:`shutdown`. + Router ^^^^^^ From ea4db5017079d2ed63a40faefeffd541cfcada8f Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Mon, 7 Nov 2016 12:35:17 +0200 Subject: [PATCH 08/12] Fix tests --- aiohttp/web.py | 12 ++++++------ aiohttp/web_server.py | 4 ++-- tests/test_web_application.py | 12 ++++++------ tests/test_worker.py | 6 +++--- 4 files changed, 17 insertions(+), 17 deletions(-) diff --git a/aiohttp/web.py b/aiohttp/web.py index 5b1cf684184..a1dc06962fc 100644 --- a/aiohttp/web.py +++ b/aiohttp/web.py @@ -16,10 +16,11 @@ from .signals import PostSignal, PreSignal, Signal from .web_exceptions import * # noqa from .web_reqrep import * # noqa -from .web_server import RequestHandlerFactory +from .web_server import WebServer from .web_urldispatcher import * # noqa from .web_ws import * # noqa + __all__ = (web_reqrep.__all__ + web_exceptions.__all__ + web_urldispatcher.__all__ + @@ -44,7 +45,6 @@ def __init__(self, *, logger=web_logger, loop=None, self._debug = debug self._router = router - self._handler_factory = RequestHandlerFactory self._secure_proxy_ssl_header = None self._loop = loop self.logger = logger @@ -180,10 +180,10 @@ def make_handler(self, *, secure_proxy_ssl_header=None, **kwargs): ) self.freeze() self._secure_proxy_ssl_header = secure_proxy_ssl_header - return self._handler_factory(self._handle, - request_factory=self._make_request, - debug=self.debug, loop=self.loop, - **kwargs) + return WebServer(self._handle, + request_factory=self._make_request, + debug=self.debug, loop=self.loop, + **kwargs) @asyncio.coroutine def startup(self): diff --git a/aiohttp/web_server.py b/aiohttp/web_server.py index ef5d0661680..f26fba957f1 100644 --- a/aiohttp/web_server.py +++ b/aiohttp/web_server.py @@ -7,7 +7,7 @@ from .web_exceptions import HTTPException from .web_reqrep import BaseRequest -__all__ = ('RequestHandler', 'RequestHandlerFactory') +__all__ = ('RequestHandler', 'WebServer') class RequestHandler(ServerHttpProtocol): @@ -78,7 +78,7 @@ def handle_request(self, message, payload): self._request = None -class Server: +class WebServer: def __init__(self, handler, *, request_factory=None, loop=None, **kwargs): self._handler = handler diff --git a/tests/test_web_application.py b/tests/test_web_application.py index a95c407ebd4..63950d62c98 100644 --- a/tests/test_web_application.py +++ b/tests/test_web_application.py @@ -28,18 +28,18 @@ def test_app_default_loop(loop): def test_app_make_handler_debug_exc(loop, mocker, debug): app = web.Application(loop=loop, debug=debug) - mocker.spy(app, '_handler_factory') + srv = mocker.patch('aiohttp.web.WebServer') app.make_handler() with pytest.warns(DeprecationWarning) as exc: app.make_handler(debug=debug) assert 'parameter is deprecated' in exc[0].message.args[0] - assert app._handler_factory.call_count == 2 - app._handler_factory.assert_called_with(app._handle, - request_factory=app._make_request, - loop=loop, - debug=debug) + assert srv.call_count == 2 + srv.assert_called_with(app._handle, + request_factory=app._make_request, + loop=loop, + debug=debug) with pytest.raises(ValueError) as exc: app.make_handler(debug=not debug) diff --git a/tests/test_worker.py b/tests/test_worker.py index 944262dfb40..3eee9437c0a 100644 --- a/tests/test_worker.py +++ b/tests/test_worker.py @@ -199,8 +199,8 @@ def test_close(worker, loop): app = worker.wsgi = mock.Mock() app.cleanup = make_mocked_coro(None) handler.connections = [object()] - handler.finish_connections.return_value = helpers.create_future(loop) - handler.finish_connections.return_value.set_result(1) + handler.shutdown.return_value = helpers.create_future(loop) + handler.shutdown.return_value.set_result(1) app.shutdown.return_value = helpers.create_future(loop) app.shutdown.return_value.set_result(None) @@ -208,7 +208,7 @@ def test_close(worker, loop): yield from worker.close() app.shutdown.assert_called_with() app.cleanup.assert_called_with() - handler.finish_connections.assert_called_with(timeout=95.0) + handler.shutdown.assert_called_with(timeout=95.0) srv.close.assert_called_with() assert worker.servers is None From 569a1680d330e97be84070cc79d116e9cb59e516 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Wed, 9 Nov 2016 02:42:19 +0200 Subject: [PATCH 09/12] Add echo printouts --- Makefile | 3 +++ 1 file changed, 3 insertions(+) diff --git a/Makefile b/Makefile index 21eb37e0748..1f98de90345 100644 --- a/Makefile +++ b/Makefile @@ -48,8 +48,11 @@ cov-dev: .develop @echo "open file://`pwd`/coverage/index.html" cov-dev-full: .develop + @echo "Run without extensions" @AIOHTTP_NO_EXTENSIONS=1 py.test --cov=aiohttp tests + @echo "Run in debug mode" @PYTHONASYNCIODEBUG=1 py.test --cov=aiohttp --cov-append tests + @echo "Regular run" @py.test --cov=aiohttp --cov-report=term --cov-report=html --cov-append tests @echo "open file://`pwd`/coverage/index.html" From e6f702a20a0e57b0811ff332af88b7b3fc77a34b Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Wed, 9 Nov 2016 17:20:55 +0200 Subject: [PATCH 10/12] Add test --- aiohttp/pytest_plugin.py | 30 ++++++++++++++- aiohttp/test_utils.py | 80 +++++++++++++++++++++++++++++++++------- aiohttp/web.py | 1 - aiohttp/web_reqrep.py | 13 ++++++- 4 files changed, 106 insertions(+), 18 deletions(-) diff --git a/aiohttp/pytest_plugin.py b/aiohttp/pytest_plugin.py index d0c23f32c70..6a2d81f51fb 100644 --- a/aiohttp/pytest_plugin.py +++ b/aiohttp/pytest_plugin.py @@ -6,7 +6,9 @@ from aiohttp.web import Application from .test_utils import unused_port as _unused_port -from .test_utils import (TestClient, TestServer, loop_context, setup_test_loop, +from .test_utils import (RawTestServer, TestClient, + TestServer, + loop_context, setup_test_loop, teardown_test_loop) @@ -81,6 +83,27 @@ def finalize(): loop.run_until_complete(finalize()) +@pytest.yield_fixture +def raw_test_server(loop): + servers = [] + + @asyncio.coroutine + def go(handler, **kwargs): + server = RawTestServer(handler, loop=loop) + yield from server.start_server(**kwargs) + servers.append(server) + return server + + yield go + + @asyncio.coroutine + def finalize(): + while servers: + yield from servers.pop().close() + + loop.run_until_complete(finalize()) + + @pytest.yield_fixture def test_client(loop): clients = [] @@ -97,6 +120,11 @@ def go(__param, *args, **kwargs): assert __param.app.loop is loop, \ "TestServer is attached to other event loop" client = TestClient(__param, **kwargs) + elif isinstance(__param, RawTestServer): + assert not args, "args should be empty" + assert __param._loop is loop, \ + "TestServer is attached to other event loop" + client = TestClient(__param, **kwargs) else: __param = __param(loop, *args, **kwargs) client = TestClient(__param) diff --git a/aiohttp/test_utils.py b/aiohttp/test_utils.py index 24ab48201b1..3065d275dd8 100644 --- a/aiohttp/test_utils.py +++ b/aiohttp/test_utils.py @@ -7,6 +7,7 @@ import socket import sys import unittest +from abc import ABC, abstractmethod from unittest import mock from multidict import CIMultiDict @@ -19,7 +20,7 @@ from .helpers import sentinel from .protocol import HttpVersion, RawRequestMessage from .signals import Signal -from .web import Application, Request, UrlMappingMatchInfo +from .web import Application, Request, UrlMappingMatchInfo, WebServer PY_35 = sys.version_info >= (3, 5) @@ -39,10 +40,8 @@ def unused_port(): return s.getsockname()[1] -class TestServer: - def __init__(self, app, *, scheme=sentinel, host='127.0.0.1'): - self.app = app - self._loop = app.loop +class BaseTestServer(ABC): + def __init__(self, *, scheme=sentinel, host='127.0.0.1'): self.port = None self.server = None self.handler = None @@ -66,17 +65,30 @@ def start_server(self, **kwargs): self._root = URL('{}://{}:{}'.format(self.scheme, self.host, self.port)) - yield from self.app.startup() - self.handler = self.app.make_handler(**kwargs) - self.server = yield from self._loop.create_server(self.handler, + + handler = yield from self._make_factory(**kwargs) + self.server = yield from self._loop.create_server(handler, self.host, self.port, ssl=self._ssl) + @abstractmethod # pragma: no cover + @asyncio.coroutine + def _make_factory(self): + pass + def make_url(self, path): assert path.startswith('/') return URL(str(self._root) + path) + @property + def started(self): + return self.server is not None + + @property + def closed(self): + return self._closed + @asyncio.coroutine def close(self): """Close all fixtures created by the test client. @@ -90,16 +102,19 @@ def close(self): exit when used as a context manager. """ - if self.server is not None and not self._closed: + if self.started and not self.closed: self.server.close() yield from self.server.wait_closed() - yield from self.app.shutdown() - yield from self.handler.shutdown() - yield from self.app.cleanup() self._root = None self.port = None + yield from self._close_hook() self._closed = True + @abstractmethod + @asyncio.coroutine + def _close_hook(self): + pass + def __enter__(self): self._loop.run_until_complete(self.start_server()) return self @@ -118,6 +133,43 @@ def __aexit__(self, exc_type, exc_value, traceback): yield from self.close() +class TestServer(BaseTestServer): + def __init__(self, app, *, scheme=sentinel, host='127.0.0.1'): + self.app = app + self._loop = app.loop + super().__init__(scheme=scheme, host=host) + + @asyncio.coroutine + def _make_factory(self, **kwargs): + yield from self.app.startup() + self.handler = self.app.make_handler(**kwargs) + return self.handler + + @asyncio.coroutine + def _close_hook(self): + yield from self.app.shutdown() + yield from self.handler.shutdown() + yield from self.app.cleanup() + + +class RawTestServer(BaseTestServer): + def __init__(self, handler, + *, loop=None, scheme=sentinel, host='127.0.0.1'): + if loop is None: + loop = asyncio.get_event_loop() + self._loop = loop + self._handler = handler + super().__init__(scheme=scheme, host=host) + + @asyncio.coroutine + def _make_factory(self, **kwargs): + return WebServer(self._handler, loop=self._loop) + + @asyncio.coroutine + def _close_hook(self): + return + + class TestClient: """ A test client implementation, for a aiohttp.web.Application. @@ -136,7 +188,7 @@ class TestClient: def __init__(self, app_or_server, *, scheme=sentinel, host=sentinel, cookie_jar=None, **kwargs): - if isinstance(app_or_server, TestServer): + if isinstance(app_or_server, BaseTestServer): if scheme is not sentinel or host is not sentinel: raise ValueError("scheme and host are mutable exclusive " "with TestServer parameter") @@ -149,7 +201,7 @@ def __init__(self, app_or_server, *, scheme=sentinel, host=sentinel, else: raise TypeError("app_or_server should be either web.Application " "or TestServer instance") - self._loop = self._server.app.loop + self._loop = self._server._loop if cookie_jar is None: cookie_jar = aiohttp.CookieJar(unsafe=True, loop=self._loop) diff --git a/aiohttp/web.py b/aiohttp/web.py index 44aae14d0a5..135b1220f51 100644 --- a/aiohttp/web.py +++ b/aiohttp/web.py @@ -20,7 +20,6 @@ from .web_urldispatcher import * # noqa from .web_ws import * # noqa - __all__ = (web_reqrep.__all__ + web_exceptions.__all__ + web_urldispatcher.__all__ + diff --git a/aiohttp/web_reqrep.py b/aiohttp/web_reqrep.py index ca20941cd76..f484cc12666 100644 --- a/aiohttp/web_reqrep.py +++ b/aiohttp/web_reqrep.py @@ -425,6 +425,11 @@ def __repr__(self): return "<{} {} {} >".format(self.__class__.__name__, self.method, ascii_encodable_path) + @asyncio.coroutine + def _prepare_hook(self, response): + return + yield # pragma: no cover + class Request(BaseRequest): @@ -445,6 +450,11 @@ def app(self): """Application instance.""" return self._match_info.apps[-1] + @asyncio.coroutine + def _prepare_hook(self, response): + for app in self.match_info.apps: + yield from app.on_response_prepare.send(self, response) + ############################################################ # HTTP Response classes @@ -739,8 +749,7 @@ def prepare(self, request): resp_impl = self._start_pre_check(request) if resp_impl is not None: return resp_impl - for app in request.match_info.apps: - yield from app.on_response_prepare.send(request, self) + yield from request._prepare_hook(self) return self._start(request) From ebcf39fa035f9ef2563ee47248efdec60b314f98 Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Wed, 9 Nov 2016 18:27:12 +0200 Subject: [PATCH 11/12] Add missing test --- tests/test_web_server.py | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) create mode 100644 tests/test_web_server.py diff --git a/tests/test_web_server.py b/tests/test_web_server.py new file mode 100644 index 00000000000..818d84402f2 --- /dev/null +++ b/tests/test_web_server.py @@ -0,0 +1,17 @@ +import asyncio + +from aiohttp import web + + +@asyncio.coroutine +def test_simple_server(raw_test_server, test_client): + @asyncio.coroutine + def handler(request): + return web.Response(text=str(request.rel_url)) + + server = yield from raw_test_server(handler) + client = yield from test_client(server) + resp = yield from client.get('/path/to') + assert resp.status == 200 + txt = yield from resp.text() + assert txt == '/path/to' From f99d9ea2963322726c71652323884c446ff65f2f Mon Sep 17 00:00:00 2001 From: Andrew Svetlov Date: Wed, 9 Nov 2016 19:59:55 +0200 Subject: [PATCH 12/12] Improve coverage --- aiohttp/test_utils.py | 2 +- tests/test_test_utils.py | 11 ++++++++++- 2 files changed, 11 insertions(+), 2 deletions(-) diff --git a/aiohttp/test_utils.py b/aiohttp/test_utils.py index 3065d275dd8..c4061ed1386 100644 --- a/aiohttp/test_utils.py +++ b/aiohttp/test_utils.py @@ -113,7 +113,7 @@ def close(self): @abstractmethod @asyncio.coroutine def _close_hook(self): - pass + pass # pragma: no cover def __enter__(self): self._loop.run_until_complete(self.start_server()) diff --git a/tests/test_test_utils.py b/tests/test_test_utils.py index 668a3d70506..7da535a3765 100644 --- a/tests/test_test_utils.py +++ b/tests/test_test_utils.py @@ -8,7 +8,7 @@ from aiohttp import web, web_reqrep from aiohttp.test_utils import TestClient as _TestClient from aiohttp.test_utils import TestServer as _TestServer -from aiohttp.test_utils import (AioHTTPTestCase, loop_context, +from aiohttp.test_utils import (AioHTTPTestCase, RawTestServer, loop_context, make_mocked_request, setup_test_loop, teardown_test_loop, unittest_run_loop) @@ -266,3 +266,12 @@ def test_client_host_mutually_exclusive_with_server(loop): def test_client_unsupported_arg(): with pytest.raises(TypeError): _TestClient('string') + + +def test_raw_server_implicit_loop(loop): + @asyncio.coroutine + def handler(request): + pass + asyncio.set_event_loop(loop) + srv = RawTestServer(handler) + assert srv._loop is loop