Skip to content

Commit

Permalink
Abort SSL connections on close, rather than waiting for remote EOF. (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
tomchristie authored Nov 25, 2020
1 parent 84c7738 commit 6abf663
Showing 1 changed file with 24 additions and 7 deletions.
31 changes: 24 additions & 7 deletions httpcore/_backends/asyncio.py
Original file line number Diff line number Diff line change
Expand Up @@ -165,16 +165,33 @@ async def write(self, data: bytes, timeout: TimeoutDict) -> None:
)

async def aclose(self) -> None:
# NOTE: StreamWriter instances expose a '.wait_closed()' coroutine function,
# but using it has caused compatibility issues with certain sites in
# the past (see https://github.com/encode/httpx/issues/634), which is
# why we don't call it here.
# This is fine, though, because '.aclose()' schedules the actual closing of the
# stream, meaning that at best it will happen during the next event loop
# iteration, and at worst asyncio will take care of it on program exit.
# SSL connections should issue the close and then abort, rather than
# waiting for the remote end of the connection to signal the EOF.
#
# See:
#
# * https://bugs.python.org/issue39758
# * https://github.com/python-trio/trio/blob/
# 31e2ae866ad549f1927d45ce073d4f0ea9f12419/trio/_ssl.py#L779-L829
#
# And related issues caused if we simply omit the 'wait_closed' call,
# without first using `.abort()`
#
# * https://github.com/encode/httpx/issues/825
# * https://github.com/encode/httpx/issues/914
is_ssl = self.stream_writer.get_extra_info("ssl_object") is not None

async with self.write_lock:
with map_exceptions({OSError: CloseError}):
self.stream_writer.close()
if is_ssl:
# Give the connection a chance to write any data in the buffer,
# and then forcibly tear down the SSL connection.
await asyncio.sleep(0)
self.stream_writer.transport.abort() # type: ignore
if hasattr(self.stream_writer, "wait_closed"):
# Python 3.7+ only.
await self.stream_writer.wait_closed() # type: ignore

def is_readable(self) -> bool:
transport = self.stream_reader._transport # type: ignore
Expand Down

0 comments on commit 6abf663

Please sign in to comment.