From e9bfa30c1d97f6290fbb52e0153958728ec43474 Mon Sep 17 00:00:00 2001 From: Eli Uriegas Date: Wed, 18 Jan 2017 23:46:22 -0600 Subject: [PATCH 1/2] Add exception handling for closed transports Adds handling for closed transports in the server for `write_response` as well as `write_error`, letting it all flow to `bail_out` seemed to be a little light handed in terms of telling the logs where the error actually occured. Handling to fix the infinite `write_error` loop is still there and those exceptions will get reported on in the debug logs. --- sanic/server.py | 30 ++++++++++++++++++++---------- 1 file changed, 20 insertions(+), 10 deletions(-) diff --git a/sanic/server.py b/sanic/server.py index a36c47aa..1b574c21 100644 --- a/sanic/server.py +++ b/sanic/server.py @@ -89,15 +89,14 @@ class HttpProtocol(asyncio.Protocol): def connection_lost(self, exc): self.connections.discard(self) self._timeout_handler.cancel() - self.cleanup() def connection_timeout(self): # Check if time_elapsed = current_time - self._last_request_time if time_elapsed < self.request_timeout: time_left = self.request_timeout - time_elapsed - self._timeout_handler = \ - self.loop.call_later(time_left, self.connection_timeout) + self._timeout_handler = ( + self.loop.call_later(time_left, self.connection_timeout)) else: if self._request_handler_task: self._request_handler_task.cancel() @@ -164,8 +163,8 @@ class HttpProtocol(asyncio.Protocol): def write_response(self, response): try: - keep_alive = self.parser.should_keep_alive() \ - and not self.signal.stopped + keep_alive = ( + self.parser.should_keep_alive() and not self.signal.stopped) self.transport.write( response.output( self.request.version, keep_alive, self.request_timeout)) @@ -175,6 +174,10 @@ class HttpProtocol(asyncio.Protocol): # Record that we received data self._last_request_time = current_time self.cleanup() + except RuntimeError: + log.error( + 'Connection lost before response written @ {}'.format( + self.request.ip)) except Exception as e: self.bail_out( "Writing response failed, connection closed {}".format(e)) @@ -185,16 +188,23 @@ class HttpProtocol(asyncio.Protocol): version = self.request.version if self.request else '1.1' self.transport.write(response.output(version)) self.transport.close() + except RuntimeError: + log.error( + 'Connection lost before error written @ {}'.format( + self.request.ip)) except Exception as e: self.bail_out( - "Writing error failed, connection closed {}".format(e)) + "Writing error failed, connection closed {}".format(e), + from_error=True) - def bail_out(self, message): - if self.transport.is_closing(): + def bail_out(self, message, from_error=False): + if from_error and self.transport.is_closing(): log.error( - "Connection closed before error was sent to user @ {}".format( + ("Transport closed @ {} and exception " + "experienced during error handling").format( self.transport.get_extra_info('peername'))) - log.debug('Error experienced:\n{}'.format(traceback.format_exc())) + log.debug( + 'Exception:\n{}'.format(traceback.format_exc())) else: exception = ServerError(message) self.write_error(exception) From ed4752bbc00f420d65a9314f21b380598b801b09 Mon Sep 17 00:00:00 2001 From: Eli Uriegas Date: Thu, 19 Jan 2017 16:35:48 -0600 Subject: [PATCH 2/2] Move transport close to finally statment --- sanic/server.py | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/sanic/server.py b/sanic/server.py index 1b574c21..4b303d41 100644 --- a/sanic/server.py +++ b/sanic/server.py @@ -168,12 +168,6 @@ class HttpProtocol(asyncio.Protocol): self.transport.write( response.output( self.request.version, keep_alive, self.request_timeout)) - if not keep_alive: - self.transport.close() - else: - # Record that we received data - self._last_request_time = current_time - self.cleanup() except RuntimeError: log.error( 'Connection lost before response written @ {}'.format( @@ -181,13 +175,19 @@ class HttpProtocol(asyncio.Protocol): except Exception as e: self.bail_out( "Writing response failed, connection closed {}".format(e)) + finally: + if not keep_alive: + self.transport.close() + else: + # Record that we received data + self._last_request_time = current_time + self.cleanup() def write_error(self, exception): try: response = self.error_handler.response(self.request, exception) version = self.request.version if self.request else '1.1' self.transport.write(response.output(version)) - self.transport.close() except RuntimeError: log.error( 'Connection lost before error written @ {}'.format( @@ -196,6 +196,8 @@ class HttpProtocol(asyncio.Protocol): self.bail_out( "Writing error failed, connection closed {}".format(e), from_error=True) + finally: + self.transport.close() def bail_out(self, message, from_error=False): if from_error and self.transport.is_closing():